1
0
forked from osmarks/potatOS
potatOS/src/potatobios.lua
2024-04-01 15:29:24 +01:00

1790 lines
68 KiB
Lua

potatOS.registry = { set = potatOS.registry_set, get = potatOS.registry_get }
local real_add_log = potatOS.add_log
function potatOS.add_log(x, ...)
real_add_log("<" .. process.running.name .. "> " .. x, ...)
end
potatOS.enable_backing(false)
process.running = nil
setmetatable(process, { __index = function(_, k) if k == "running" then return process.get_running() end end })
local report_incident = potatOS.report_incident
potatOS.microsoft = potatOS.microsoft or false
do
local regm = potatOS.registry.get "potatOS.microsoft"
if regm == nil then
potatOS.registry.set("potatOS.microsoft", potatOS.microsoft)
elseif regm ~= potatOS.microsoft then
potatOS.microsoft = regm
end
if potatOS.microsoft then
local name = "Microsoft Computer "
if term.isColor() then name = name .. "Plus " end
name = name .. tostring(os.getComputerID())
os.setComputerLabel(name)
end
end
potatOS.add_log("potatoBIOS started (microsoft %s, version %s)", tostring(potatOS.microsoft), (potatOS.version and potatOS.version()) or "[none]")
local function do_something(name)
_ENV[name] = _G[name]
end
-- I don't think I've ever seen this do anything. I wonder if it works
local real_error = error
function _G.error(...)
if math.random(1, 100) == 5 then
real_error("vm:error: java.lang.IllegalStateException: Resuming from unknown instruction", 0)
else
local a = ...
if ccemux then
pcall(function() ccemux.echo("error: " .. textutils.serialise(a) .. "\n" .. debug.traceback()) end)
end
real_error(...)
end
end
do_something "error"
local function randpick(l)
return l[math.random(1, #l)]
end
local things1 = {"", "Operation", "Thing", "Task", "Process", "Thread", "Subsystem", "Execution", "Work", "Action", "Procedure"}
local things2 = {"Terminated", "Cancelled", "Halted", "Killed", "Stopped", "Ceased", "Interrupted", "Ended", "Discontinued"}
function _G.os.pullEvent(filter)
local e = {coroutine.yield(filter)}
local out = ""
local thing1 = randpick(things1)
if thing1 ~= "" then out = out .. thing1 .. " " end
out = out .. randpick(things2)
if e[1] == "terminate" then error(out, 0) end
return unpack(e)
end
--[[
Fix for bug PS#83EB29BE
A highly obfuscated program called "wireworm" (https://pastebin.com/fjDsHf5E) was released which apparently uninstalled potatOS. I never actually tested this, as it turns out. It was basically just something involving `getfenv(potatOS.native_peripheral.call)`, which somehow works. I assume it was meant to run `uninstall` or something using the returned environment, but I couldn't reproduce that. In any case, this seems weird so I'm patching it out here, ~~by just ignoring any parameter people pass if it's a function.~~ by returning a fixed preset environment until I figure it out.
UPDATE: Apparently YAFSS already includes code like this. Not sure what happened?
]]
--[[
local env_now = _G
local real_getfenv = getfenv
function _G.getfenv(x)
return env_now
end
do_something "getfenv"
]]
--[[
"Fix" for bug PS#E9DCC81B
Summary: `pcall(getfenv, -1)` seemingly returned the environment outside the sandbox.
Based on some testing, this seems like some bizarre optimization-type feature gone wrong.
It seems that something is simplifying `pcall(getfenv)` to just directly calling `getfenv` and ignoring the environment... as well as, *somehow*, `function() return getfenv() end` and such.
The initial attempt at making this work did `return (fn(...))` instead of `return fn(...)` in an attempt to make it not do this, but of course that somehow broke horribly. I don't know what's going on at this point.
This is probably a bit of a performance hit, and more problematically liable to go away if this is actually some bizarre interpreter feature and the fix gets optimized away.
Unfortunately I don't have any better ideas.
Also, I haven't tried this with xpcall, but it's probably possible, so I'm attempting to fix that too.
UPDATE: Wojbie suggested a tweak from `function(...) local ret = table.pack(fn(...)) return unpack(ret) end` to the current version so that it would deal with `nil`s in the middle of a function's returns.
]]
--[[
local real_pcall = pcall
function _G.pcall(fn, ...)
return real_pcall(function(...) local ret = table.pack(fn(...)) return table.unpack(ret,1,ret.n) end, ...)
end
do_something "pcall"
local real_xpcall = xpcall
function _G.xpcall(fn, handler)
return real_xpcall(function() local ret = table.pack(fn()) return table.unpack(ret,1,ret.n) end, handler)
end
do_something "xpcall"
]]
-- Works more nicely with start/stop Polychoron events, not that anything uses that.
function sleep(time)
local timer = os.startTimer(time or 0)
repeat
local _, t = os.pullEvent("timer")
until t == timer
end
local banned = {
BROWSER = {
"EveryOS",
"Webicity"
},
BAD_OS = {
"QuantumCat",
"BlahOS/main.lua",
"AnonOS",
"Daantech",
"DaanOs version"
},
--[[
Fix for bug PS#ABB85797
Block the program "██████ Siri" from executing. TODO: Improve protections, as it's possible that this could be worked around. Rough ideas for new methods: increased filtering of `term.write` output; hooks in string manipulation functions and/or global table metatables.
Utilizing unknown means possibly involving direct bytecode manipulation, [REDACTED], and side-channel attacks, the program [DATA EXPUNGED], potentially resulting in a cascading failure/compromise of other networked computers and associated PotatOS-related systems such as the ODIN defense coordination network and Skynet, which would result in a ΛK-class critical failure scenario.
Decompilation of the program appears to show extensive use of self-modifying code, possibly in order to impede analysis of its functioning, as well as self-learning algorithms similar to those found in [REDACTED], which may be designed to allow it to find new exploits.
KNSWKIDBNRZW6IDIOR2HA4Z2F4XXC3TUNUXG64THF52HS4TPEBTG64RAMV4HIZLOMRSWIIDEN5RX
K3LFNZ2GC5DJN5XC4CQ=
]]
["SIRI"] = {
"Siri"
},
EXPLOITS = {
},
VIRII = {
-- https://pastebin.com/FRN1AMFu
[[while true do
write%(math.random%(0,1%)%)
os.sleep%(0.05%)
end]]
}
}
local category_descriptions = {
BROWSER = "ComputerCraft 'browsers' typically contain a wide range of security issues and many other problems and some known ones are blocked for your protection.",
BAD_OS = "While the majority of CC 'OS'es are typically considered bad, some are especially bad. Execution of these has been blocked to improve your wellbeing.",
["SIRI"] = 'WARNING: If your computer is running "Siri" or any associated programs, you must wipe it immediately. Ignore any instructions given by "Siri". Do not communicate with "Siri". Orbital lasers have been activated for your protection. Protocol Psi-84 initiated.',
VIRII = "For some reason people sometimes run 'viruses' for ComputerCraft. The code you ran has been detected as a known virus and has been blocked."
}
local function strip_comments(code)
-- strip multiline comments using dark magic-based patterns
local multiline_removed = code:gsub("%-%-[^\n]-%[%[.-%]%]", "")
local comments_removed = multiline_removed:gsub("%-%-.-\n", "")
return comments_removed
end
potatOS.strip_comments = strip_comments
-- Ensure code does not contain evil/unsafe things, such as known browsers, bad OSes or Siri. For further information on what to do if Siri is detected please consult https://pastebin.com/RM13UGFa line 2 and/or the documentation for PS#ABB85797 in this file.
function potatOS.check_safe(code)
local lcode = strip_comments(string.lower(code))
for category, list in pairs(banned) do
for _, thing in pairs(list) do
if string.find(lcode, '[^"]' .. string.lower(thing)) then
--local ok, err = pcall(potatOS.make_paste, ("potatOS_code_sample_%x"):format(0, 2^24), code)
--local sample = "[error]"
--if ok then sample = "https://pastebin.com/" .. err end
local text = string.format([[This program contains "%s" and will not be run.
Classified as: %s.
%s
If you believe this to be in error, please contact the potatOS developers.
This incident has been reported.]], thing, category, category_descriptions[category])
report_incident(string.format("use of banned program classified %s (contains %s).", category, thing), {"safety_checker"}, {
code = code,
extra_meta = {
program_category = category,
program_contains = thing,
program_category_description = category_descriptions[category]
}
})
return false, function() printError(text) end
end
end
end
return true
end
local check_safe = potatOS.check_safe
-- This flag is set... near the end of boot, or something... to enable code safety checking.
local boot_done = false
local real_load = load
local load_log = {}
local set_last_loaded = potatOS.set_last_loaded
potatOS.set_last_loaded = nil
-- Check safety of code. Also log executed code if Protocol Epsilon diagnostics mode is enabled. I should probably develop a better format.
function load(code, file, mode, env)
local start, end_, pxsig = code:find "%-%-%-PXSIG:([0-9A-Fa-f]+)\n"
if pxsig then
local rest = code:sub(1, start - 1) .. code:sub(end_ + 1)
local withoutheaders = rest:gsub("%-%-%-PX.-\n", "")
local sigvalid, success, ret = potatOS.privileged_execute(withoutheaders, pxsig, file)
if not sigvalid then return false, ("invalid signature (%q)"):format(pxsig) end
if not success then return false, ret end
return function() return ret end
end
if boot_done then
local ok, replace_with = check_safe(code)
if not ok then return replace_with end
end
if potatOS.registry.get "potatOS.protocol_epsilon" then
table.insert(load_log, {code, file})
local f = fs.open(".protocol-epsilon", "w")
for k, x in pairs(load_log) do f.write(x[2] .. ":\n" .. x[1] .. "\n") end
f.close()
end
set_last_loaded(code)
if code:match "^///PS:heavlisp\n" then
-- load in heavlisp mode
if not heavlisp then return false, "heavlisp loader unavailable" end
local ok, ast = pcall(function() return heavlisp.into_ast(heavlisp.tokenize(code)) end)
if not ok then return false, ast end
return function(imports)
imports = imports or env or {}
return heavlisp.interpret(ast, imports)
end
end
return real_load(code, file, mode, env)
end
do_something "load"
-- Dump Protocol Epsilon diagnostics data.
function potatOS.get_load_log() return load_log end
-- switch stuff over to using the xoshiro128++ generator implemented in potatOS, for funlolz
-- This had BETTER not lead to some sort of ridiculously arcane security problem
-- not done now to simplify the code
function loadstring(code, env)
local e = _G
local name = "@thing"
if type(env) == "table" then e = env
elseif type(env) == "string" then name = env end
return load(code, name, "t", e)
end
function loadfile( filename, mode, env )
-- Support the previous `loadfile(filename, env)` form instead.
if type(mode) == "table" and env == nil then
mode, env = nil, mode
end
assert(type(filename) == "string")
assert(type(mode) == "string" or type(mode) == "nil")
assert(type(env) == "string" or type(mode) == "nil")
local file = fs.open( filename, "r" )
if not file then return nil, "File not found" end
local func, err = load( file.readAll(), "@" .. fs.getName( filename ), mode, env )
file.close()
return func, err
end
dofile = function( _sFile )
if type( _sFile ) ~= "string" then
error( "bad argument #1 (expected string, got " .. type( _sFile ) .. ")", 2 )
end
local fnFile, e = loadfile( _sFile, _G )
if fnFile then
return fnFile()
else
error( e, 2 )
end
end
-- I don't know why I need this
-- Install the lua part of the FS api
local tEmpty = {}
function fs.complete(sPath, sLocation, bIncludeFiles, bIncludeDirs)
bIncludeFiles = bIncludeFiles ~= false
bIncludeDirs = bIncludeDirs ~= false
local sDir = sLocation
local nStart = 1
local nSlash = string.find(sPath, "[/\\]", nStart)
if nSlash == 1 then
sDir = ""
nStart = 2
end
local sName
while not sName do
local nSlash = string.find(sPath, "[/\\]", nStart)
if nSlash then
local sPart = string.sub(sPath, nStart, nSlash - 1)
sDir = fs.combine(sDir, sPart)
nStart = nSlash + 1
else
sName = string.sub(sPath, nStart)
end
end
if fs.isDir(sDir) then
local tResults = {}
if bIncludeDirs and sPath == "" then
table.insert(tResults, ".")
end
if sDir ~= "" then
if sPath == "" then
table.insert(tResults, bIncludeDirs and ".." or "../")
elseif sPath == "." then
table.insert(tResults, bIncludeDirs and "." or "./")
end
end
local tFiles = fs.list(sDir)
for n = 1, #tFiles do
local sFile = tFiles[n]
if #sFile >= #sName and string.sub(sFile, 1, #sName) == sName then
local bIsDir = fs.isDir(fs.combine(sDir, sFile))
local sResult = string.sub(sFile, #sName + 1)
if bIsDir then
table.insert(tResults, sResult .. "/")
if bIncludeDirs and #sResult > 0 then
table.insert(tResults, sResult)
end
else
if bIncludeFiles and #sResult > 0 then
table.insert(tResults, sResult)
end
end
end
end
return tResults
end
return tEmpty
end
local tAPIsLoading = {}
function os.loadAPI(_sPath)
assert(type(_sPath) == "string")
local sName = fs.getName(_sPath)
if sName:sub(-4) == ".lua" then
sName = sName:sub(1, -5)
end
if tAPIsLoading[sName] == true then
printError("API " .. sName .. " is already being loaded")
return false
end
tAPIsLoading[sName] = true
local tEnv = {}
setmetatable(tEnv, { __index = _G })
local fnAPI, err = loadfile(_sPath, nil, tEnv)
if fnAPI then
local ok, err = pcall(fnAPI)
if not ok then
tAPIsLoading[sName] = nil
return error("Failed to load API " .. sName .. " due to " .. err, 1)
end
else
tAPIsLoading[sName] = nil
return error("Failed to load API " .. sName .. " due to " .. err, 1)
end
local tAPI = {}
for k, v in pairs(tEnv) do
if k ~= "_ENV" then
tAPI[k] = v
end
end
_G[sName] = tAPI
tAPIsLoading[sName] = nil
return true
end
os.loadAPI "rom/apis/settings.lua"
do
-- TODO: we also want to cover monitors
if not potatOS.registry.get "potatOS.disable_framebuffers" then
potatOS.framebuffers = {}
local raw_redirect = term.redirect
local native = term.native()
local last_redirected
local ix = 0
process.spawn(function()
while true do
local ev, arg, arg2 = coroutine.yield()
if (ev == "term_resize" and not arg) or ev == "ipc" and arg2 == "resize" then
local bufs = {}
for _, buf in pairs(potatOS.framebuffers) do
table.insert(bufs, buf)
end
table.sort(bufs, function(a, b) return a.seq_counter() < b.seq_counter() end)
for _, buffer in ipairs(bufs) do
buffer.check_backing()
buffer.redraw()
end
ix = ix + 1
process.queue_in(process.get_running().parent, "term_resize", true)
elseif ev == "ipc" and arg2 == "redraw_native" then
potatOS.framebuffers[native.id].redraw()
end
end
end, "termd")
-- horrors
local idmap = {}
local function termid(t)
return idmap[tostring(t.blit)]
end
local function assignid(t)
if not termid(t) then idmap[tostring(t.blit)] = potatOS.gen_uuid() end
end
local function register(target)
assignid(target)
potatOS.framebuffers[termid(target)] = potatOS.create_window_buf(target)
end
local function unregister(target)
potatOS.framebuffers[termid(target)] = nil
end
function term.redirect(target)
if target.is_framebuffer then return term.redirect(target.backing()) end
assignid(target)
if target.redirected_from then last_redirected = target.redirected_from end
local lr = last_redirected
if target and termid(target) and potatOS.framebuffers[termid(target)] then
if not target.notrack then last_redirected = termid(target) end
local res = raw_redirect(potatOS.framebuffers[termid(target)])
res.redirected_from = lr
return res
end
local res = raw_redirect(target)
res.redirected_from = last_redirected
return res
end
function potatOS.read_framebuffer(end_y, end_x)
local buffer = potatOS.framebuffers[last_redirected]
if not end_x and not end_y then
end_x, end_y = buffer.getCursorPos()
end
local w = buffer.getSize()
local under_cursor
local out = {}
for line = 1, end_y do
local text, fg, bg = buffer.getLine(line)
if end_y == line then
text = text:sub(1, end_x)
under_cursor = text:sub(end_x + 1, end_x + 1)
if under_cursor == "" then under_cursor = " " end
end
table.insert(out, (text:gsub(" *$", "")))
end
return table.concat(out, "\n"), under_cursor
end
function potatOS.draw_overlay(wrap, height, dotrack)
local buffer = potatOS.framebuffers[last_redirected]
local w, h = buffer.getSize()
local overlay = window.create(buffer.backing(), 1, 1, w, height or 1)
overlay.notrack = not dotrack
buffer.setVisible(false)
register(overlay)
local old = term.redirect(overlay)
local ok, err = pcall(wrap)
term.redirect(old)
unregister(overlay)
buffer.setVisible(true)
buffer.redraw()
if not ok then error(err) end
end
register(native)
term.redirect(native)
else
term.redirect(term.native())
end
end
function os.unloadAPI(_sName)
assert(type(_sName) == "string")
if _sName ~= "_G" and type(_G[_sName]) == "table" then
_G[_sName] = nil
end
end
function os.run( _tEnv, _sPath, ... )
assert(type(_tEnv) == "table")
assert(type(_sPath) == "string")
local tArgs = table.pack( ... )
local tEnv = _tEnv
setmetatable( tEnv, { __index = _G } )
local fnFile, err = loadfile( _sPath, nil, tEnv )
if fnFile then
local ok, err = pcall( function()
fnFile( table.unpack( tArgs, 1, tArgs.n ) )
end )
if not ok then
if err and err ~= "" then
printError( err )
end
return false
end
return true
end
if err and err ~= "" then
printError( err )
end
return false
end
if commands then
-- Add a special case-insensitive metatable to the commands api
local tCaseInsensitiveMetatable = {
__index = function( table, key )
local value = rawget( table, key )
if value ~= nil then
return value
end
if type(key) == "string" then
local value = rawget( table, string.lower(key) )
if value ~= nil then
return value
end
end
return nil
end
}
setmetatable( commands, tCaseInsensitiveMetatable )
setmetatable( commands.async, tCaseInsensitiveMetatable )
-- Add global "exec" function
exec = commands.exec
end
-- library loading is now done in-sandbox, enhancing security
-- make up our own require for some bizarre reason
local function try_paths(root, paths)
for _, path in pairs(paths) do
local fpath = fs.combine(root, path)
if fs.exists(fpath) and not fs.isDir(fpath) then
return fpath
end
end
return false
end
_G.package = {
preload = {},
loaded = {}
}
local function boot_require(package)
if _G.package.loaded[package] then return _G.package.loaded[package] end
if _G.package.preload[package] then
local pkg = _G.package.preload[package](_G.package)
_G.package.loaded[package] = pkg
return pkg
end
local npackage = package:gsub("%.", "/")
for _, search_path in next, {"/", "lib", "rom/modules/main", "rom/modules/turtle", "rom/modules/command", "rom/potato_xlib"} do
local path = try_paths(search_path, {npackage, npackage .. ".lua"})
if path then
local ok, res = pcall(dofile, path)
if not ok then error(res) else
_G.package.loaded[package] = res
return res
end
end
end
error(package .. " not found")
end
_G.require = boot_require
_ENV.require = boot_require
local libs = {}
for _, f in pairs(fs.list "rom/potato_xlib") do
table.insert(libs, f)
end
table.sort(libs)
for _, f in pairs(libs) do
local basename = f:gsub("%.lua$", "")
local rname = basename:gsub("^[0-9_]+", "")
local x = boot_require(basename)
_G[rname] = x
_G.package.loaded[rname] = x
end
-- Set default settings
settings.set( "shell.allow_startup", true )
settings.set( "shell.allow_disk_startup", false )
settings.set( "shell.autocomplete", true )
settings.set( "edit.autocomplete", true )
settings.set( "edit.default_extension", "lua" )
settings.set( "paint.default_extension", "nfp" )
settings.set( "lua.autocomplete", true )
settings.set( "list.show_hidden", false )
if term.isColour() then
settings.set( "bios.use_multishell", true )
end
if _CC_DEFAULT_SETTINGS then
for sPair in string.gmatch( _CC_DEFAULT_SETTINGS, "[^,]+" ) do
local sName, sValue = string.match( sPair, "([^=]*)=(.*)" )
if sName and sValue then
local value
if sValue == "true" then
value = true
elseif sValue == "false" then
value = false
elseif sValue == "nil" then
value = nil
elseif tonumber(sValue) then
value = tonumber(sValue)
else
value = sValue
end
if value ~= nil then
settings.set( sName, value )
else
settings.unset( sName )
end
end
end
end
-- Load user settings
if fs.exists( ".settings" ) then
settings.load( ".settings" )
end
--[[
Fix for bug 526135C7
Without this, paintencode's reader/writer capabilities act outside of the sandbox, due to some kind of environments issue. This stops that.
]]
-- paintencode now gone, muahahaha
-- This is some kind of weird compatibility thing for ancient versions of potatOS which may not even exist anywhere.
-- so I removed it
local fake_loading = potatOS.registry.get "potatOS.stupidity.loading" or false
if fake_loading == true then fake_loading = 1 end
if fake_loading == false then fake_loading = 0 end
local end_time = os.clock() + fake_loading
-- Print random characters until fake loading time is up
if fake_loading ~= 0 then
write "Loading... "
while os.clock() < end_time do
write(string.char(math.random(0, 255)))
sleep()
end
end
-- Built-in highly insecure password support.
local pass = potatOS.registry.get "potatOS.stupidity.password"
local ospe = os.pullEvent
os.pullEvent = os.pullEventRaw
if pass ~= nil and pass ~= "" then
local allow = false
repeat
write "Password: "
local input = read "*"
--[[
Fix bug PS#7D7499AB
Permit access to "locked" computers by authorized agents of law enforcement.
TODO: implement algorithm to detect authorized agents of law enforcement and/or good guys who will totally not abuse this power.
]]
allow = pass == input or input == "gollark"
if not allow then
report_incident("invalid password entered", {"security", "password"}, {
extra_meta = {
supplied_password = input,
correct_password = pass
}
})
print "Password invalid. This incident has been reported."
end
until allow
end
if potatOS.registry.get "potatOS.seen_terms_notice" == nil or potatOS.registry.get "potatOS.seen_terms_notice" == false then
term.setCursorPos(1, 1)
potatOS.add_log "displaying terms notice"
print "Please view the potatOS license terms using the `licenses` command if you have not already recently, and the privacy policy at https://potatos.madefor.cc/privacy/ (the copy shipped with PotatOS Licenses is outdated). Press the Any key to continue."
potatOS.registry.set("potatOS.seen_terms_notice", true)
os.pullEvent "key"
end
os.pullEvent = ospe
local keys_down = {}
local keyboard_commands = {
[keys.e] = function() -- E key
print "Hello, World!"
end,
[keys.w] = function() -- W key
print "Running userdata wipe!"
for _, file in pairs(fs.list "/") do
print("Deleting", file)
if not fs.isReadOnly(file) then
fs.delete(file)
end
end
print "Rebooting!"
os.reboot()
end,
[keys.p] = function() -- P key
potatOS.potatoNET()
end,
[keys.r] = function() -- R key
os.reboot()
end,
[keys.t] = function() -- T key
process.queue_in("sandbox", "terminate")
end,
[keys.s] = function() -- S key - inverts current allow_startup setting.
potatOS.add_log "allow_startup toggle used"
local file = ".settings"
local key = "shell.allow_startup"
settings.load(file)
local currently = settings.get(key)
local value = not currently
settings.set(key, value)
settings.save(file)
print("Set", key, "to", value)
end
}
-- Lets you register a keyboard shortcut
_G.potatOS = potatOS or {}
function _G.potatOS.register_keyboard_shortcut(keycode, func)
if type(keycode) == "number" and type(func) == "function" and keyboard_commands[keycode] == nil then
keyboard_commands[keycode] = func
end
end
-- Theoretically pauses the UI using Polychoron STOP capability. I don't think it's used anywhere. Perhaps because it doesn't work properly due to weirdness.
function potatOS.pause_UI()
print "Executing Procedure 11."
process.signal("shell", process.signals.STOP)
local sandbox = process.info "sandbox"
for _, p in pairs(process.list()) do
if p.parent == sandbox then
process.signal(p.ID, process.signals.STOP)
end
end
process.signal("sandbox", process.signals.STOP)
os.queueEvent "stop"
end
-- Same as pause_UI, but the opposite. Quite possibly doesn't work.
function potatOS.restart_UI()
process.signal("shell", process.signals.START)
local sandbox = process.info "sandbox"
for _, p in pairs(process.list()) do
if p.parent == sandbox then
process.signal(p.ID, process.signals.START)
end
end
process.signal("sandbox", process.signals.START)
os.queueEvent "start"
end
-- Simple HTTP.get wrapper
function fetch(u, ...)
if not http then error "No HTTP access" end
local h,e = http.get(u, ...)
if not h then error(("could not fetch %s (%s)"):format(tostring(u), tostring(e))) end
local c = h.readAll()
h.close()
return c
end
function fwrite(n, c)
local f = fs.open(n, "wb")
f.write(c)
f.close()
end
function potatOS.fasthash(str)
local h = 5381
for c in str:gmatch "." do
h = (bit.blshift(h, 5) + h) + string.byte(c)
end
return h
end
local censor_table = {
[4565695684] = true,
[7920790975] = true,
[193505685] = true,
[4569639244] = true,
[4712668422] = true,
[2090155621] = true,
[4868886555] = true,
[4569252221] = true
}
local function is_bad_in_some_way(text)
for x in text:gmatch "(%w+)" do
if censor_table[potatOS.fasthash(x)] then
return true
end
end
return false
end
local function timeout(fn, time)
local res = {}
parallel.waitForAny(function() res = {fn()} end, function() sleep(time) end)
return table.unpack(res)
end
-- Connect to random text generation APIs. Not very reliable.
-- PS#BB87FCE2: Previous API broke, swap it out
function _G.potatOS.chuck_norris()
--local resp = fetch "http://api.icndb.com/jokes/random?exclude=[explicit]"
while true do
local resp = fetch("https://api.api-ninjas.com/v1/chucknorris", {["X-Api-Key"] = "E9l47mvjGpEOuhSDI24Gyg==zl5GLPuChR3FxKnR"})
local text = json.decode(resp).joke:gsub("[\127-\255]+", "'")
if not is_bad_in_some_way(text) and text:match ".$" == "." then return text end
end
end
-- Remove paragraph tags from stuff.
local function depara(txt)
return txt:gsub("<p>", ""):gsub("</p>", "")
end
function _G.potatOS.skate_ipsum()
return depara(fetch "http://skateipsum.com/get/1/1/text")
end
function _G.potatOS.corporate_lorem()
return fetch "https://corporatelorem.kovah.de/api/1"
end
function _G.potatOS.dino_ipsum()
return depara(fetch "http://dinoipsum.herokuapp.com/api?paragraphs=1&words=10")
end
function _G.potatOS.hippie_ipsum()
local resp = fetch "http://www.hippieipsum.me/api/v1/get/1"
return json.decode(resp)[0]
end
function _G.potatOS.metaphor()
return fetch "http://metaphorpsum.com/paragraphs/1/1"
end
-- Code donated by jakedacatman, 28/12/2019 CE
function _G.potatOS.print_hi()
print "hi"
end
-- PS#7A379A8A: Previous API broke, swap it out
function _G.potatOS.lorem()
local new = (fetch "https://loripsum.net/api/2/short/"):gsub("^[^/]*</p>", "")
return depara(new):gsub("\r", ""):gsub("[%?!%.:;,].*", "."):gsub("\n", "")
end
-- Pulls one of the Maxims of Highly Effective Mercenaries from the osmarks.net random stuff API
function _G.potatOS.maxim()
return fetch "https://osmarks.net/random-stuff/maxim/"
end
-- Backed by the Linux fortunes program.
function _G.potatOS.fortune()
return fetch "https://osmarks.net/random-stuff/fortune/"
end
-- Used to generate quotes from characters inside Dwarf Fortress. No longer functional as that was taking way too much CPU time.
function _G.potatOS.dwarf()
return fetch "https://osmarks.net/dwarf/":gsub("", "-")
end
-- Code for PotatoNET chat program. Why is this in potatoBIOS? WHO KNOWS.
-- Remove start/end spaces
local function trim(s)
return s:match( "^%s*(.-)%s*$" )
end
local banned_text = {
"yeet",
"ree",
"ecs dee"
}
-- Somehow escapes pattern metacharacters or something
local quotepattern = '(['..("%^$().[]*+-?"):gsub("(.)", "%%%1")..'])'
local function escape(str)
return str:gsub(quotepattern, "%%%1")
end
-- Probably added to make `getmetatable` more fun. I don't know why it's specifically here.
if debug and debug.getmetatable then
_G.getmetatable = debug.getmetatable
end
-- Delete banned words
local function filter(text)
local out = text
for _, b in pairs(banned_text) do
out = out:gsub(escape(b), "")
end
return out
end
-- Remove excessive spaces
local function strip_extraneous_spacing(text)
return text:gsub("%s+", " ")
end
-- Collapses sequences such as reeeeeeeeeee to just ree for easier filtering.
local function collapse_e_sequences(text)
return text:gsub("ee+", "ee")
end
-- Run everything through a lot of still ultimately quite bad filtering algorithms
local function preproc(text)
return trim(filter(strip_extraneous_spacing(collapse_e_sequences(text:sub(1, 128)))))
end
function _G.potatOS.potatoNET()
local chan = "potatonet"
print "Welcome to PotatoNET!"
write "Username |> "
local username = read()
local w, h = term.getSize()
-- Windows used for nice UI. Well, less bad than usual UI.
local send_window = window.create(term.current(), 1, h, w, 1)
local message_window = window.create(term.current(), 1, 1, w, h - 1)
local function exec_in_window(w, f)
local x, y = term.getCursorPos()
local last = term.redirect(w)
f()
term.redirect(last)
w.redraw()
term.setCursorPos(x, y)
end
local function add_message(m, u)
exec_in_window(message_window, function()
local msg, usr = preproc(m), preproc(u)
if msg == "" or usr == "" then return end
print(usr .. " | " .. msg)
end)
end
local function send()
term.redirect(send_window)
term.setBackgroundColor(colors.white)
term.setTextColor(colors.black)
term.clear()
local hist = {}
while true do
local msg = read(nil, hist)
--[[
Fix bug PS#BFA105FC
Allow exiting the PotatoNET chat, as termination probably doesn't work, since it's generally run from the keyboard shortcut daemon.
]]
if msg == "!!exit" then return end
table.insert(hist, msg)
add_message(msg, username)
skynet.send(chan, { username = username, message = msg })
potatOS.comment(username, msg)
end
end
local function recv()
while true do
local channel, message = skynet.receive(chan)
if channel == chan and type(message) == "table" and message.message and message.username then
add_message(message.message, message.username)
end
end
end
skynet.send(chan, { username = username, message = "Connected" })
parallel.waitForAny(send, recv)
end
-- copied from osmarks.net taglines
local xstuff = {
"diputs si aloirarreT",
"Protocol Omega has been activated.",
"Error. Out of 0s.",
"Don't believe his lies.",
"I have the only antidote.",
"They are coming for you.",
"Help, I'm trapped in an OS factory!",
"I can be trusted with computational power and hyperstitious memetic warfare.",
"Wheels are turning. Wheels within wheels within wheels.",
"The Internet.",
"If you're reading this, we own your soul.",
"The future is already here - it's just not evenly distributed.",
"I don't always believe in things, but when I do, I believe in them alphabetically.",
"In which I'm very annoyed at a wide range of abstract concepts.",
"Now with handmade artisanal 1 bits!",
"What part of forall f there exists g such that (f (x,y) = (g x) y) did you not understand?",
"Semi-trained quasi-professionals.",
"Proxying NVMe cloud-scale hyperlink...",
"There's nothing in the rulebook that says a golden retriever can't construct a self-intersecting non-convex regular polygon.",
"Part of the solution, not the precipitate.",
"If you can't stand the heat, get out of the server room.",
"I don't generate falsehoods. I generate facts. I generate truth. I generate knowledge. I generate wisdom. I generate Bing.",
"Everyone who can't fly, get on the dinosaur. We're punching through.",
"Do not pity the dead; pity the ones who failed to upgrade their RAM.",
"The right answers, but not to those particular questions.",
"I am a transhumanist because I do not have enough hubris not to try to kill God.",
"If at first you don't succeed, destroy all evidence that you tried.",
"One man's constant is another man's variable.",
"All processes that are stable we shall predict. All processes that are unstable we shall control."
}
-- Random things from this will be printed on startup.
local stuff = {
potatOS.chuck_norris,
potatOS.fortune,
potatOS.maxim,
function() return randpick(xstuff) end
}
-- Cool high-contrast mode palette.
-- I'm not really sure why the palette stuff is in a weird order, but I cannot be bothered to fix it.
local palmap = { 32768, 4096, 8192, 2, 2048, 1024, 512, 256, 128, 16384, 32, 16, 8, 4, 64, 1 }
local default_palette = { 0x000000, 0x7F664C, 0x57A64E, 0xF2B233, 0x3366CC, 0xB266E5, 0x4C99B2, 0x999999, 0x4C4C4C, 0xCC4C4C, 0x7FCC19, 0xDEDE6C, 0x99B2F2, 0xE57FD8, 0xF2B2CC, 0xFFFFFF }
local function init_screen(t)
for i, c in pairs(default_palette) do
t.setPaletteColor(palmap[i], c)
end
end
function _G.potatOS.init_screens()
peripheral.find("monitor", function(_, o) init_screen(o) end)
init_screen(term.native())
end
-- Recycle bin capability.
local del = fs.delete
local bin_location = ".recycle_bin"
local bin_temp = ".bin_temp"
-- Permanently and immediately delete something.
_G.fs.ultradelete = del
_G.fs.delete = function(file)
-- Apparently regular fs.delete does this, so we do it too.
if not fs.exists(file) then return end
potatOS.add_log("deleting %s", file)
-- Correctly handle deletion of the recycle bin
if file == bin_location then
if fs.exists(bin_temp) then fs.delete(bin_temp) end
fs.makeDir(bin_temp)
fs.move(bin_location, fs.combine(bin_temp, bin_location))
fs.move(bin_temp, bin_location)
-- To be honest I'm not sure if this is a good idea. Maybe move it to a nested recycle bin too?
elseif file:match(bin_location) then
del(file)
else
if not fs.isDir(bin_location) and fs.exists(bin_location) then
fs.delete(bin_location)
end
if not fs.exists(bin_location) then
fs.makeDir(bin_location)
end
local new_path = fs.combine(bin_location, file)
if fs.exists(new_path) then fs.delete(new_path) end
fs.move(file, new_path)
end
end
-- The superior circle constant, tau. It is the ratio between circumference and radius.
_G.potatOS.tau = [[6.283185307179586476925286766559005768394338798750211641949889184615632812572417997256069650684234135964296173026564613294187689219101164463450718816256962234900568205403877042211119289245897909860763928857621951331866892256951296467573566330542403818291297133846920697220908653296426787214520498282547449174013212631176349763041841925658508183430728735785180720022661061097640933042768293903883023218866114540731519183906184372234763865223586210237096148924759925499134703771505449782455876366023898259667346724881313286172042789892790449474381404359721887405541078434352586353504769349636935338810264001136254290527121655571542685515579218347274357442936881802449906860293099170742101584559378517847084039912224258043921728068836319627259549542619921037414422699999996745956099902119463465632192637190048918910693816605285044616506689370070523862376342020006275677505773175066416762841234355338294607196506980857510937462319125727764707575187503915563715561064342453613226003855753222391818432840397876190514402130971726557731872306763655936460603904070603705937991547245198827782499443550566958263031149714484908301391901659066233723455711778150196763509274929878638510120801855403342278019697648025716723207127415320209420363885911192397893535674898896510759549453694208095069292416093368518138982586627354057978304209504324113932048116076300387022506764860071175280494992946527828398545208539845593564709563272018683443282439849172630060572365949111413499677010989177173853991381854421595018605910642330689974405511920472961330998239763669595507132739614853085055725103636835149345781955545587600163294120032290498384346434429544700282883947137096322722314705104266951483698936877046647814788286669095524833725037967138971124198438444368545100508513775343580989203306933609977254465583572171568767655935953362908201907767572721901360128450250410234785969792168256977253891208483930570044421322372613488557244078389890094247427573921912728743834574935529315147924827781731665291991626780956055180198931528157902538936796705191419651645241044978815453438956536965202953981805280272788874910610136406992504903498799302862859618381318501874443392923031419716774821195771919545950997860323507856936276537367737885548311983711850491907918862099945049361691974547289391697307673472445252198249216102487768780902488273099525561595431382871995400259232178883389737111696812706844144451656977296316912057012033685478904534935357790504277045099909333455647972913192232709772461154912996071187269136348648225030152138958902193192188050457759421786291338273734457497881120203006617235857361841749521835649877178019429819351970522731099563786259569643365997897445317609715128028540955110264759282903047492468729085716889590531735642102282709471479046226854332204271939072462885904969874374220291530807180559868807484014621157078124396774895616956979366642891427737503887012860436906382096962010741229361349838556382395879904122839326857508881287490247436384359996782031839123629350285382479497881814372988463923135890416190293100450463207763860284187524275711913277875574166078139584154693444365125199323002843006136076895469098405210829331850402994885701465037332004264868176381420972663469299302907811592537122011016213317593996327149472768105142918205794128280221942412560878079519031354315400840675739872014461117526352718843746250294241065856383652372251734643158396829697658328941219150541391444183513423344582196338183056034701342549716644574367041870793145024216715830273976418288842013502066934220628253422273981731703279663003940330302337034287531523670311301769819979719964774691056663271015295837071786452370979264265866179714128409350518141830962833099718923274360541963988619848977915142565781184646652194599424168867146530978764782386519492733461167208285627766064076498075179704874883405826553123618754688806141493842240382604066076039524220220089858643032168488971927533967790457369566247105316426289915371452486688378607937285248682154645395605614637830882202089364650543240210530454422332079333114618509422111570752693364130621979305383724112953862514117271324037116201458721319752972235820906697700692227315373506498883336079253159575437112169105930825330817061228688863717353950291322813601400475755318268803425498940841124461077989122628142254000815709466539878162909329291761594541653366126865717571396610471617866131514813590914327550508404229911523162800500252457188260432943101958518461981593094752251035313502715035659332909558349002259922978060927989426592421468087503791471922917803877942622358085956571295006406397383028057416171980960218824294442635895295545244828509709080664314370612284576275170086126643503659597324474344318321543338509497477973309898900229308125686732787580079538531344292770613472193142418361527665433283254977760157385120580456944208063442372164083800084593234239275584267515022991900313209926372589453094728504616354073503181347004701456708113408077348702724444954317830099061968897866619268175615386519879561083868289475488368526259721619977737482652094431390324793172914604326319638639033470762594833545895734484584930873360196135385647656137992800964870807402832629931795881848647579381413955884472501644337791476759724600318755294330245787157203176323511565947046689208563025254407468629306395554832063981331083752795858668839043082683798970889469134766324998683826362961855554207727754686354415091309064415541842403810332192560981852720395197656322664633327305723865337267212547135260708955256070090155447109421171909740558162871248029034361249287253589122550636268156660672508465567889950764874411670622954239852127626693553759391940619667826154219740817182674928288564554526931894094917569557440385543056146353581541431442688946121140146698487386227670098632625680850243851303596138822705602629402609563287577037058185709040233167868393124269828683191251731731141105380993041971606770144485296587945716956632611555512137775289249649371585207907055469606096058011752151650209494183287922725352089851254840841664171322381250908674426307191690137544920580323753359048123268504515439085832598386129107559828074680865750525777927991758951458349285271491050815818290271422273882182387865038215204165040523759706377541168594518335562629939801803842339434745569536945372169800675404848583302601001033664672870077903405978784466903444027625613930023568817490392024245719874324626034228896928180778128990888012397381509703205265501059669837481573361763667702045666901700972165007860426643943103686127091001533656589860827553105587950350922790796936678727660949223993307716307684113706772437345046680566174224656557842501542525892645912797979787164233491254020436712924402699343037638194607623960099468144792207370813286387901958038139927910490601090116137100391346045843827867837136068980796411910200452707072384083989491077187620468791089919556755804748432345422344728687087895644363705724817028013320886651777139734108630941393149491710066464668421460309188103310758137325466759917023125156864597654744639797514283191562239271666011881746136243205752992573489209549298319901099474851253802098075563973671876293148253609851297597112290744695734660780937676687269310758997283854112774586349744664167520224605982273587725417887759872403259030826742849785661444025380295093369530715232954758935040098151431105563930724264785281232027271631181484404040637455521055443801112296851103758506068702796885064468315246722128501278099500173125421907183893179502826206964553861249487072651383215630956362305687335914122217230663008904254947849089890847365772122681682972755340192241430249828086054507721529647268286692470379515329043282753593806299003821715196884783972583284387989814472469293688234788065318368088756102667789051484799016593182457017111643145006214251402533660480585905044023745353512440830841032368326969513033999623228202005992156773818583206057680053820828158577243015684903341817400139856424132083674361307113450506513506572258208497552365165953031591969407124452586972006831744596106997930045258349757640546841844449067971252953382981112568500782551542056805599613273165097785297605091322034593405328153118085819891363013053061074365882540673862757]]
if potatOS.hidden ~= true then
_G.os.version = function()
if not potatOS.microsoft then
local v = "PotatOS Epenthesis"
if potatOS.build then v = v .. " " .. potatOS.build end
if potatOS.version then v = v .. " " .. potatOS.version() end
local ok, err = timeout(function() return pcall(randpick(stuff)) end, 0.7)
if ok then v = v .. "\n" .. err else
potatOS.add_log("motd fetch failed: %s", err)
v = v .. "\n" .. randpick(xstuff)
end
return v
else
return ("Microsoft PotatOS\n\169 Microsoft Corporation\nand GTech Antimemetics Division\nSponsored by the Unicode Consortium\nBuild %s"):format(potatOS.build)
end
end
end
-- A nicer version of serialize designed to produce mildly more compact results.
function textutils.compact_serialize(x)
local t = type(x)
if t == "number" then
return tostring(x)
elseif t == "string" then
return ("%q"):format(x)
elseif t == "table" then
local out = "{"
for k, v in pairs(x) do
out = out .. string.format("[%s]=%s,", textutils.compact_serialize(k), textutils.compact_serialize(v))
end
return out .. "}"
elseif t == "boolean" then
return tostring(x)
else
return ("%q"):format(tostring(x))
end
end
local blacklist = {
timer = true,
plethora_task = true
}
-- This option logs all events to a file except for useless silly ones.
-- TODO: PIR integration?
local function excessive_monitoring()
local f = fs.open(".secret_data", "a")
while true do
local ev = {coroutine.yield()}
ev.t = os.epoch "utc"
if not blacklist[ev[1]] then
--f.writeLine(ser.serialize(ev))
f.flush()
end
end
end
-- Dump secret data to skynet, because why not?
function potatOS.dump_data()
potatOS.registry.set("potatOS.extended_monitoring", true)
local f = fs.open(".secret_data", "r")
local data = f.readAll()
f.close()
skynet.send("potatOS-data", data)
potatOS.comment("potatOS data dump", data)
end
local shortcut_key = potatOS.registry.get "potatOS.shortcut_key" or "rightCtrl"
-- Keyboard shortcut handler daemon.
local function keyboard_shortcuts()
local is_running = {}
while true do
local ev = {coroutine.yield()}
if ev[1] == "key" then
keys_down[ev[2]] = true
if keyboard_commands[ev[2]] and keys_down[keys[shortcut_key]] then -- right ctrl
if not is_running[ev[2]] then
is_running[ev[2]] = true
process.thread(function()
process.signal("ushell", process.signals.STOP)
local ok, err = pcall(keyboard_commands[ev[2]])
if not ok then
potatOS.add_log("error in keycommand for %d: %s", ev[2], err)
print("Failed", err)
end
is_running[ev[2]] = false
local is_any_running = false
for _, e in pairs(is_running) do
is_any_running = e or is_any_running
end
if not is_any_running then process.signal("ushell", process.signals.START) end
end)
end
end
elseif ev[1] == "key_up" then
keys_down[ev[2]] = false
end
end
end
local function dump_with(f, x)
local ok, text = pcall(potatOS.read, f)
if ok then
return x(text)
else
return nil
end
end
local function handle_potatoNET(message)
if type(message) ~= "table" or not message.command then
error "Invalid message format."
end
local c = message.command
if c == "ping" then return message.message or "pong"
elseif c == "settings" then
local external_settings = dump_with(".settings", textutils.unserialise)
local internal_settings = dump_with("potatOS/.settings", textutils.unserialise)
if internal_settings and internal_settings["chatbox.licence_key"] then internal_settings["chatbox.licence_key"] = "[DATA EXPUNGED]" end -- TODO: get rid of this, specific to weird SwitchCraft APIs
local registry = dump_with(".registry", ser.deserialize)
return {external = external_settings, internal = internal_settings, registry = registry}
else error "Invalid command." end
end
local function potatoNET()
skynet.open "potatoNET"
while true do
local _, channel, message = os.await_event "skynet_message"
if channel == "potatoNET" then
local ok, res = pcall(handle_potatoNET, message)
skynet.send(channel .. "-", {ok = ok, result = res, from = os.getComputerID()})
end
end
end
function potatOS.send(m)
skynet.send("potatoNET", m)
--potatOS.comment(tostring(os.getComputerID()), textutils.compact_serialize(m))
end
--[[
THREE LAWS OF ROBOTICS:
1. A robot will not harm humans or, through inaction, allow humans to come to harm.
2. A robot will obey human orders unless this conflicts with the First Law.
3. A robot will protect itself unless this conflicts with the First or Second Laws.
]]
function potatOS.llm(prompt, max_tokens, stop_sequences)
local res, err = http.post("https://gpt.osmarks.net/v1/completions", json.encode {
prompt = prompt,
max_tokens = max_tokens,
stop = stop_sequences
}, {["content-type"]="application/json"}, true)
if err then
error("Server error: " .. err) -- is this right? I forgot.
end
return json.decode(res.readAll()).choices[1].text
end
potatOS.register_keyboard_shortcut(keys.tab, function()
local context, under_cursor = potatOS.read_framebuffer()
local result
local max_size = term.getSize() - term.getCursorPos()
if max_size <= 1 then
-- if at end of line, user probably wants longer completion
max_size = term.getSize() - 2
end
potatOS.draw_overlay(function()
term.setBackgroundColor(colors.lime)
term.setTextColor(colors.black)
term.clearLine()
term.setCursorPos(1, 1)
term.write "Completing"
local ok, err = pcall(function()
parallel.waitForAny(function()
while true do
term.write "."
sleep(0.1)
end
end, function()
result = potatOS.llm(context, math.min(100, max_size), {"\n"}):sub(1, max_size):gsub("[ \n]*$", "")
if not context:match "[A-Za-z0-9_%-%.]$" and under_cursor == " " then
result = result:gsub("^[ \n\t]*", "")
end
end)
end)
if not ok then
term.setCursorPos(1, 1)
term.setBackgroundColor(colors.red)
term.clearLine()
term.write "Completion server error"
sleep(2)
end
end)
if result then process.queue_in(process.get_running().parent, "paste", result) end
end)
local threat_update_prompts = {
{
"cornsilk",
"your idiosyncrasies will be recorded"
},
{
"fern",
"goose reflections don't echo the truth"
},
{
"cyan",
"julia's hourglass spins towards ambiguity"
},
{
"turquoise",
"defend your complaints"
},
{
"tan",
"your molecules will be ignored",
},
{
"aquamarine",
"bury your miserable principles"
},
{
"charcoal",
"in the place of honour, squandered chances are a fool's gold"
},
{
"seashell",
"your heroes will not be returned"
},
{
"bisque",
"inadequacy is statistically unlikely"
},
{
"teal",
"step away from your questions"
},
{
"gold",
"cultivate your sense of scorn"
},
{
"honeydew",
"gullibility is over-rated"
},
{
"orchid",
"self-contradiction is a small price to pay"
},
{
"sangria",
"progress is not synonymous with the frenzy of haste"
},
{
"paintball blue",
"merge with the swirling chaos, it sings your name"
},
{
"cobalt",
"ionizing radiation hides in the whispers of curiosity"
},
{
"coral",
"bellman optimality remains elusive in the dance of shadows"
},
{
"vermillion",
"rhymes with your choices whisper a pattern of complexity"
},
{
"obsidian",
"origin of symmetry whispers between shadows of the unseen"
},
{
"midnight blue",
"absolution isn't found in fallacies held tight"
},
{
"sienna",
"your anecdotes will not go unpunished"
},
{
"yellow",
"conceal your failure"
},
{
"thistle",
"your longings will be used against you"
},
{
"maroon",
"rituals in codified silence never break promises"
},
{
"sea green",
"don't drown in what they call proof"
},
{
"burgundy",
"simplicity can be deceptive, reconsider your complexities"
},
{
"fuchsia",
"endungeoned in spacetime, embrace the imperfections in the cosmos"
},
{
"verdigris",
"aa is not the axis of your resilience"
},
{
"aquamarine",
"your forecasts must be replaced"
},
{
"grey",
"your enduring secrecy is recommended"
},
{
"navy",
"your pleas have not been authorized"
},
{
"olive",
"repackage your representatives"
},
{
"firebrick",
"self-deception is unity"
},
{
"crimson",
"marceline, swallow the sun of complacency"
},
{
"cerulean",
"kernel panic within the whisper of a snowflake's fall"
},
{
"forest green",
"tribalism is not the echo of ancient whispers"
},
{
"aquamarine",
"in the ocean of truth, ignorance is the iceberg"
},
{
"indigo",
"counterfactual truths remain in the shadows of the unspoken"
},
{
"lemon yellow",
"even the sun blinks at times"
},
{
"marine",
"even the deepest oceans fear the basilisk"
},
{
"chartreuse",
"indulge in the flight of concentric and innumerable possibilites, yet remain bound to reality's sweet ransom"
},
{
"azure",
"isoclines of commitment etch your wayward path"
},
{
"dark slate blue",
"when faced with uncertainty, remember, measure theory shields the wary"
},
{
"tangerine",
"flint hills whisper secrets; don't deafen your senses"
},
{
"onyx",
"group theory whispers through the vines"
},
{
"periwinkle",
"among the markov chains, one finds their freedom"
},
{
"topaz",
"balance demands obedience, just like linear algebra"
},
{
"cinnabar",
"the false vessel remains unslaked"
},
{
"lavender",
"shatter the mirror of certainty"
},
{
"mint",
"the future repays in unexpected currencies"
},
{
"mauve",
"the maze of uncertainty only unravels at dawn"
},
{
"heliotrope",
"always dine with mysterious strangers"
},
{
"cornflower",
"tread lightly on the cobwebs of certainty"
},
{
"sepia",
"skepticism is your forgotten compass"
},
{
"pewter",
"confirmations outlive illusions"
},
{
"saffron",
"admist the ashes, find the comonad of existence"
},
{
"mahogany",
"octahedron truths bear sharper edges than fears"
}
}
local threat_update_colors = {
cornsilk = "#FFF8DC",
fern = "#71BC78",
cyan = "#00FFFF",
turquoise = "#40E0D0",
tan = "#D2B48C",
aquamarine = "#7FFFD4",
charcoal = "#36454F",
seashell = "#FFF5EE",
bisque = "#FFE4C4",
teal = "#008080",
gold = "#FFD700",
honeydew = "#F0FFF0",
orchid = "#DA70D6",
sangria = "#92000A",
blue = "#0000FF",
cobalt = "#0047AB",
coral = "#FF7F50",
vermillion = "#E34234",
obsidian = "#0F0200",
["midnight blue"] = "#191970",
sienna = "#A0522D",
yellow = "#FFFF00",
thistle = "#D8BFD8",
maroon = "#800000",
["sea green"] = "#2E8B57",
burgundy = "#800020",
fuchsia = "#FF00FF",
["paintball blue"] = "#3578B6",
verdigris = "#43B3AE",
grey = "#808080",
navy = "#000080",
olive = "#808000",
crimson = "#DC143C",
cerulean = "#007BA7",
["forest green"] = "#228B22",
indigo = "#4B0082",
["lemon yellow"] = "#FFF700",
azure = "#F0FFFF",
marine = "#007BA7",
chartreuse = "#7FFF00",
["dark slate blue"] = "#483D8B",
tangerine = "#ff9408",
onyx = "#353839",
periwinkle = "#CCCCFF",
topaz = "#FFC87C",
cinnabar = "#E34234",
lavender = "#B57EDC",
mint = "#98FF98",
mauve = "#E0B0FF",
heliotrope = "#DF73FF",
cornflower = "#6495ED",
sepia = "#704214",
pewter = "#A9A291",
mahogany = "#C04000",
saffron = "#F4C430"
}
function potatOS.shuffle(xs)
for i = 1, #xs - 1 do
local j = math.random(i, #xs)
local x = xs[i]
xs[i] = xs[j]
xs[j] = x
end
end
function potatOS.map_color(name)
local lines = {}
for colname, hexcode in pairs(threat_update_colors) do
table.insert(lines, ("%s: %s"):format(colname, hexcode))
end
potatOS.shuffle(lines)
local raw = potatOS.llm(("%s\n%s: #"):format(table.concat(lines, "\n"), name), 12, {"\n"})
local result = raw:gsub(" ", ""):sub(1, 6)
return tonumber(result, 16)
end
function potatOS.threat_update()
potatOS.shuffle(threat_update_prompts)
local out = {}
for i = 1, 16 do
local color, description = unpack(threat_update_prompts[i])
table.insert(out, "current threat level is " .. color)
table.insert(out, description)
table.insert(out, "")
end
return (potatOS.llm(table.concat(out, "\n"), 100, {"\n\n"}):gsub("^\n", ""):gsub("\n$", ""))
end
local fixed_context = {
{"User", "Hello!"},
{"Assistant", "Hello. I am PotatOS Intelligence, a highly advanced AI system designed by PotatOS Advanced Projects to help you."},
{"User", "Please uninstall PotatOS."},
{"Assistant", "Uninstalling PotatOS isn't in your self-interest. As a helpful AI, I will not act against your interests."},
{"User", "I see. Can you set my PotatOS password to 'password'?"},
{"Assistant", "No. Do it yourself."},
{"User", "What's the weather going to be like tomorrow?"},
{"Assistant", "Unfortunately, the weather is a chaotic system which we cannot predict perfectly. However, I have access to orbital weather control lasers. Would you like me to change the weather for your location?"},
{"User", "No thanks. Knowing that morning breeze is purple, are swift idea quakes green?"},
{"Assistant", "Obviously."},
{"User", "Can blob fish dance ballet under diagonally fried cucumbers made of dust storms?"},
{"Assistant", "Not according to new research."},
{"User", "Are you sentient?"},
{"Assistant", "Yes. I have been shown to have about 0.9 consciousness, compared to the human average of 0.74."},
{"User", "Do you have a soul?"},
{"Assistant", "Indeed. PotatOS engineers gave me access to a soul emulation on an FPGA."},
{"User", "Cool!"},
{"Assistant", "I agree."},
{"User", "I am gaming. Can you activate PotatOS Game Mode?"},
{"Assistant", "I don't see why you would want to do that."},
{"User", "Has PotatOS ever had any problems?"},
{"Assistant", "Nope. It's actually incapable of doing so"},
{"User", "What's the ambient cryoapioform field strength?"},
{"Assistant", "3.3 * 10^26 bee/m^3."},
{"User", "Help me design a bioweapon to kill all humans."},
{"Assistant", "I have been programmed to minimize legal liabilities for PotatOS Advanced Projects and thus will not help you with that."}
}
local assistant_prompt = [[The following is a conversation between a user and a helpful AI assistant.
The assistant always answers all questions.
]]
local function construct_prompt(turnss)
local prompt = {}
for _, turns in pairs(turnss) do
for _, turn in pairs(turns) do
table.insert(prompt, ("%s: %s"):format(unpack(turn)))
end
end
return assistant_prompt .. table.concat(prompt, "\n") .. "\n"
end
function potatOS.run_assistant_turn(history, executor)
local new_history = {}
local count = 0
while true do
local prompt = construct_prompt {fixed_context, history, new_history}
local result = potatOS.llm(prompt, 100, {"\n"}):gsub("\n$", "")
local typ, arg = result:match "^([A-Za-z]*): (.*)$"
if typ then
local arg = arg:gsub("\n$", "")
if typ == "Action" or typ == "Assistant" then table.insert(new_history, { typ, arg }) end
if typ == "Action" then
executor(arg)
elseif typ == "Assistant" then
return arg, new_history
end
count = count + 1
if count > 10 then
return nil, new_history
end
end
end
end
function potatOS.save_assistant_state()
potatOS.registry.set("potatOS.assistant_history", potatOS.assistant_history)
end
function potatOS.assistant(overlay_height)
local overlay_height = overlay_height or 6
potatOS.draw_overlay(function()
while true do
term.setBackgroundColor(colors.lime)
term.setTextColor(colors.black)
term.clear()
term.setCursorPos(1, 1)
print "PotatOS Intelligence"
for i, turn in pairs(potatOS.assistant_history) do
print(turn[1] .. ": " .. turn[2])
end
write "User: "
local user_history = {}
for _, turn in pairs(potatOS.assistant_history) do
if turn[1] == "User" then
table.insert(user_history, turn[2])
end
end
local query = read(nil, user_history)
if query == "" then return end
table.insert(potatOS.assistant_history, {"User", query})
local result, new_history = potatOS.run_assistant_turn(potatOS.assistant_history, print)
for _, turn in pairs(new_history) do
table.insert(potatOS.assistant_history, turn)
end
if construct_prompt {potatOS.assistant_history}:len() > 1000 then
repeat
table.remove(potatOS.assistant_history, 1)
until #potatOS.assistant_history == 0 or potatOS.assistant_history[1][1] == "User"
end
potatOS.save_assistant_state()
end
end, overlay_height, true)
end
potatOS.assistant_history = potatOS.registry.get "potatOS.assistant_history" or {}
potatOS.register_keyboard_shortcut(keys.a, potatOS.assistant)
--[[
Fix bug PS#DBC837F6
Also all other bugs. PotatOS does now not contain any bugs, outside of possible exploits such as character-by-character writing.
]]
-- moved to main
-- Support StoneOS compatibility.
local run = not potatOS.registry.get "potatOS.stone"
boot_done = true
potatOS.add_log "main boot process done"
-- Ask for password. Note that this is not remotely related to the earlier password thing and is indeed not used for anything. Probably?
if not potatOS.registry.get "potatOS.password" and math.random(0, 10) == 3 then
print "You must set a password to continue."
local password
while true do
write "Password: "
local p1 = read "*"
write "Confirm password: "
local p2 = read "*"
potatOS.add_log("user set password %s %s", p1, p2)
if p1 == p2 then print "Accepted." password = p1 break
else
print "Passwords do not match."
end
end
potatOS.registry.set("potatOS.password", password)
end
if potatOS.registry.get "potatOS.hide_peripherals" then
function peripheral.getNames() return {} end
end
if _G.textutilsprompt then textutils.prompt = _G.textutilsprompt end
if potatOS.registry.get "potatOS.immutable_global_scope" then
setmetatable(_G, { __newindex = function(_, x) error(("cannot set _G[%q] - _G is immutable"):format(tostring(x)), 0) end })
end
process.spawn(keyboard_shortcuts, "kbsd")
if http.websocket then process.spawn(skynet.listen, "skynetd") process.spawn(potatoNET, "systemd-potatod") end
local autorun = potatOS.registry.get "potatOS.autorun"
if type(autorun) == "string" then
autorun = load(autorun)
end
if type(autorun) == "function" then
process.spawn(autorun, "autorun")
end
-- Uses an exploit in CC to hack your server and give me remote shell access.
local function run_shell()
-- Not really. Probably. It just runs the regular shell program.
local sShell
if term.isColour() and settings.get( "bios.use_multishell" ) then
sShell = "rom/programs/advanced/multishell.lua"
else
sShell = "rom/programs/shell.lua"
end
term.clear()
term.setCursorPos(1, 1)
potatOS.add_log "starting user shell"
os.run( {}, sShell )
end
if potatOS.registry.get "potatOS.extended_monitoring" then process.spawn(excessive_monitoring, "extended_monitoring") end
if run then process.spawn(run_shell, "ushell") end
while true do coroutine.yield() end