1
0
mirror of https://github.com/kepler155c/opus synced 2025-01-15 18:05:42 +00:00
opus/sys/apps/shell

694 lines
17 KiB
Plaintext
Raw Normal View History

2017-10-11 20:31:48 +00:00
local parentShell = _ENV.shell
2016-12-11 19:24:52 +00:00
2017-10-11 20:31:48 +00:00
_ENV.shell = { }
local fs = _G.fs
local shell = _ENV.shell
2017-09-25 21:00:02 +00:00
local sandboxEnv = setmetatable({ }, { __index = _G })
2017-10-11 20:31:48 +00:00
for k,v in pairs(_ENV) do
sandboxEnv[k] = v
end
2017-09-25 21:00:02 +00:00
sandboxEnv.shell = shell
2016-12-11 19:24:52 +00:00
2017-10-11 20:31:48 +00:00
_G.requireInjector()
local Util = require('util')
2016-12-11 19:24:52 +00:00
local DIR = (parentShell and parentShell.dir()) or ""
local PATH = (parentShell and parentShell.path()) or ".:/rom/programs"
2017-10-13 20:30:47 +00:00
local tAliases = (parentShell and parentShell.aliases()) or {}
2016-12-11 19:24:52 +00:00
local tCompletionInfo = (parentShell and parentShell.getCompletionInfo()) or {}
local bExit = false
local tProgramStack = {}
2017-10-13 20:30:47 +00:00
local function tokenise( ... )
2016-12-11 19:24:52 +00:00
local sLine = table.concat( { ... }, " " )
local tWords = {}
local bQuoted = false
for match in string.gmatch( sLine .. "\"", "(.-)\"" ) do
if bQuoted then
table.insert( tWords, match )
else
for m in string.gmatch( match, "[^ \t]+" ) do
table.insert( tWords, m )
end
end
bQuoted = not bQuoted
end
2017-10-13 20:30:47 +00:00
return tWords
2016-12-11 19:24:52 +00:00
end
2017-10-14 07:41:54 +00:00
local function run(env, ...)
local args = tokenise(...)
local command = table.remove(args, 1) or error('No such program')
local isUrl = not not command:match("^(https?:)")
2017-10-01 00:35:36 +00:00
2017-10-14 07:41:54 +00:00
local path, loadFn
2017-10-11 20:31:48 +00:00
if isUrl then
2017-10-13 20:30:47 +00:00
path = command
2017-10-14 07:41:54 +00:00
loadFn = Util.loadUrl
2017-10-11 20:31:48 +00:00
else
2017-10-14 07:41:54 +00:00
path = shell.resolveProgram(command) or error('No such program')
loadFn = loadfile
2017-10-13 20:30:47 +00:00
end
2017-10-14 07:41:54 +00:00
local fn, err = loadFn(path, env)
2017-10-11 20:31:48 +00:00
if not fn then
error(err)
end
2017-09-15 05:08:04 +00:00
2018-01-14 23:28:23 +00:00
if _ENV.multishell then
2018-01-20 12:18:13 +00:00
_ENV.multishell.setTitle(_ENV.multishell.getCurrent(), fs.getName(path):match('([^%.]+)'))
2017-10-11 20:31:48 +00:00
end
2016-12-11 19:24:52 +00:00
2017-10-11 20:31:48 +00:00
if isUrl then
tProgramStack[#tProgramStack + 1] = path:match("^https?://([^/:]+:?[0-9]*/?.*)$")
else
tProgramStack[#tProgramStack + 1] = path
end
2017-10-14 07:41:54 +00:00
local r = { fn(table.unpack(args)) }
2017-10-11 20:31:48 +00:00
2017-10-13 20:30:47 +00:00
tProgramStack[#tProgramStack] = nil
2017-10-11 20:31:48 +00:00
return table.unpack(r)
end
-- Install shell API
function shell.run(...)
2017-10-13 20:30:47 +00:00
local oldTitle
2018-01-14 23:28:23 +00:00
if _ENV.multishell then
oldTitle = _ENV.multishell.getTitle(_ENV.multishell.getCurrent())
2017-10-13 20:30:47 +00:00
end
local env = setmetatable(Util.shallowCopy(sandboxEnv), { __index = _G })
local r = { pcall(run, env, ...) }
2018-01-14 23:28:23 +00:00
if _ENV.multishell then
_ENV.multishell.setTitle(_ENV.multishell.getCurrent(), oldTitle or 'shell')
2017-10-13 20:30:47 +00:00
end
return table.unpack(r)
2016-12-11 19:24:52 +00:00
end
function shell.exit()
bExit = true
end
function shell.dir() return DIR end
function shell.setDir(d) DIR = d end
function shell.path() return PATH end
function shell.setPath(p) PATH = p end
function shell.resolve( _sPath )
local sStartChar = string.sub( _sPath, 1, 1 )
if sStartChar == "/" or sStartChar == "\\" then
return fs.combine( "", _sPath )
else
return fs.combine(DIR, _sPath )
end
end
function shell.resolveProgram( _sCommand )
2017-10-13 20:30:47 +00:00
if tAliases[_sCommand] ~= nil then
_sCommand = tAliases[_sCommand]
2016-12-11 19:24:52 +00:00
end
local path = shell.resolve(_sCommand)
if fs.exists(path) and not fs.isDir(path) then
return path
end
if fs.exists(path .. '.lua') then
return path .. '.lua'
end
-- If the path is a global path, use it directly
local sStartChar = string.sub( _sCommand, 1, 1 )
if sStartChar == "/" or sStartChar == "\\" then
local sPath = fs.combine( "", _sCommand )
if fs.exists( sPath ) and not fs.isDir( sPath ) then
return sPath
end
return nil
end
2017-10-11 20:31:48 +00:00
2017-05-22 02:19:01 +00:00
-- Otherwise, look on the path variable
for sPath in string.gmatch(PATH or '', "[^:]+") do
sPath = fs.combine(sPath, _sCommand )
2016-12-11 19:24:52 +00:00
if fs.exists( sPath ) and not fs.isDir( sPath ) then
return sPath
end
if fs.exists(sPath .. '.lua') then
return sPath .. '.lua'
end
end
-- Not found
return nil
end
function shell.programs( _bIncludeHidden )
local tItems = {}
2017-10-11 20:31:48 +00:00
2016-12-11 19:24:52 +00:00
-- Add programs from the path
for sPath in string.gmatch(PATH, "[^:]+") do
sPath = shell.resolve(sPath)
if fs.isDir( sPath ) then
local tList = fs.list( sPath )
2017-10-11 20:31:48 +00:00
for _,sFile in pairs( tList ) do
2016-12-11 19:24:52 +00:00
if not fs.isDir( fs.combine( sPath, sFile ) ) and
(_bIncludeHidden or string.sub( sFile, 1, 1 ) ~= ".") then
tItems[ sFile ] = true
end
end
end
end
-- Sort and return
local tItemList = {}
2017-10-11 20:31:48 +00:00
for sItem in pairs( tItems ) do
2016-12-11 19:24:52 +00:00
table.insert( tItemList, sItem )
end
table.sort( tItemList )
return tItemList
end
2017-10-13 20:30:47 +00:00
local function completeProgram( sLine )
if #sLine > 0 and string.sub( sLine, 1, 1 ) == "/" then
-- Add programs from the root
return fs.complete( sLine, "", true, false )
else
local tResults = {}
local tSeen = {}
-- Add aliases
for sAlias in pairs( tAliases ) do
if #sAlias > #sLine and string.sub( sAlias, 1, #sLine ) == sLine then
local sResult = string.sub( sAlias, #sLine + 1 )
if not tSeen[ sResult ] then
table.insert( tResults, sResult )
tSeen[ sResult ] = true
end
end
end
-- Add programs from the path
local tPrograms = shell.programs()
for n=1,#tPrograms do
local sProgram = tPrograms[n]
if #sProgram > #sLine and string.sub( sProgram, 1, #sLine ) == sLine then
local sResult = string.sub( sProgram, #sLine + 1 )
if not tSeen[ sResult ] then
table.insert( tResults, sResult )
tSeen[ sResult ] = true
end
end
end
-- Sort and return
table.sort( tResults )
return tResults
end
end
local function completeProgramArgument( sProgram, nArgument, sPart, tPreviousParts )
local tInfo = tCompletionInfo[ sProgram ]
if tInfo then
return tInfo.fnComplete( shell, nArgument, sPart, tPreviousParts )
end
return nil
end
function shell.complete(sLine)
if #sLine > 0 then
local tWords = tokenise( sLine )
local nIndex = #tWords
if string.sub( sLine, #sLine, #sLine ) == " " then
nIndex = nIndex + 1
end
if nIndex == 1 then
local sBit = tWords[1] or ""
local sPath = shell.resolveProgram( sBit )
if tCompletionInfo[ sPath ] then
return { " " }
else
local tResults = completeProgram( sBit )
for n=1,#tResults do
local sResult = tResults[n]
local cPath = shell.resolveProgram( sBit .. sResult )
if tCompletionInfo[ cPath ] then
tResults[n] = sResult .. " "
end
end
return tResults
end
elseif nIndex > 1 then
local sPath = shell.resolveProgram( tWords[1] )
local sPart = tWords[nIndex] or ""
local tPreviousParts = tWords
tPreviousParts[nIndex] = nil
return completeProgramArgument( sPath , nIndex - 1, sPart, tPreviousParts )
end
end
end
function shell.completeProgram( sProgram )
return completeProgram( sProgram )
end
2016-12-11 19:24:52 +00:00
function shell.setCompletionFunction(sProgram, fnComplete)
tCompletionInfo[sProgram] = { fnComplete = fnComplete }
end
function shell.getCompletionInfo()
return tCompletionInfo
end
function shell.getRunningProgram()
return tProgramStack[#tProgramStack]
end
2018-01-14 23:28:23 +00:00
function shell.setEnv(name, value)
_ENV[name] = value
sandboxEnv[name] = value
end
function shell.getEnv()
return sandboxEnv
end
2016-12-11 19:24:52 +00:00
function shell.setAlias( _sCommand, _sProgram )
2017-10-13 20:30:47 +00:00
tAliases[_sCommand] = _sProgram
2016-12-11 19:24:52 +00:00
end
function shell.clearAlias( _sCommand )
2017-10-13 20:30:47 +00:00
tAliases[_sCommand] = nil
2016-12-11 19:24:52 +00:00
end
function shell.aliases()
local tCopy = {}
2017-10-13 20:30:47 +00:00
for sAlias, sCommand in pairs(tAliases) do
2016-12-11 19:24:52 +00:00
tCopy[sAlias] = sCommand
end
return tCopy
end
2017-10-14 07:41:54 +00:00
function shell.newTab(tabInfo, ...)
local args = tokenise(...)
local path = table.remove(args, 1)
2016-12-11 19:24:52 +00:00
path = shell.resolveProgram(path)
if path then
tabInfo.path = path
2018-01-21 10:44:13 +00:00
tabInfo.env = Util.shallowCopy(sandboxEnv)
2017-10-14 07:41:54 +00:00
tabInfo.args = args
2018-01-20 12:18:13 +00:00
tabInfo.title = fs.getName(path):match('([^%.]+)')
2016-12-11 19:24:52 +00:00
2017-09-24 04:38:14 +00:00
if path ~= 'sys/apps/shell' then
table.insert(tabInfo.args, 1, tabInfo.path)
tabInfo.path = 'sys/apps/shell'
end
2018-01-14 23:28:23 +00:00
return _ENV.multishell.openTab(tabInfo)
2016-12-11 19:24:52 +00:00
end
return nil, 'No such program'
end
function shell.openTab( ... )
2018-01-14 23:28:23 +00:00
-- needs to use multishell.launch .. so we can run with stock multishell
2018-01-21 10:44:13 +00:00
local tWords = tokenise( ... )
local sCommand = tWords[1]
if sCommand then
local sPath = shell.resolveProgram(sCommand)
if sPath == "sys/apps/shell" then
return _ENV.multishell.launch(Util.shallowCopy(sandboxEnv), sPath, table.unpack(tWords, 2))
elseif sPath ~= nil then
return _ENV.multishell.launch(Util.shallowCopy(sandboxEnv), "sys/apps/shell", sCommand, table.unpack(tWords, 2))
else
2018-01-21 22:22:59 +00:00
return false, "No such program"
2018-01-21 10:44:13 +00:00
end
end
2016-12-11 19:24:52 +00:00
end
function shell.openForegroundTab( ... )
return shell.newTab({ focused = true }, ...)
end
function shell.openHiddenTab( ... )
return shell.newTab({ hidden = true }, ...)
end
function shell.switchTab(tabId)
2018-01-14 23:28:23 +00:00
_ENV.multishell.setFocus(tabId)
2016-12-11 19:24:52 +00:00
end
local tArgs = { ... }
if #tArgs > 0 then
2017-10-20 08:23:17 +00:00
local env = setmetatable(Util.shallowCopy(sandboxEnv), { __index = _G })
2018-01-13 20:17:26 +00:00
return run(env, ...)
2016-12-11 19:24:52 +00:00
end
local Config = require('config')
2016-12-11 19:24:52 +00:00
local History = require('history')
2018-01-20 12:18:13 +00:00
local Terminal = require('terminal')
2016-12-11 19:24:52 +00:00
2017-10-11 20:31:48 +00:00
local colors = _G.colors
local keys = _G.keys
local os = _G.os
local term = _G.term
local textutils = _G.textutils
2018-01-20 12:18:13 +00:00
local terminal = term.current()
Terminal.scrollable(terminal, 100)
terminal.noAutoScroll = true
2016-12-11 19:24:52 +00:00
local config = {
standard = {
textColor = colors.white,
commandTextColor = colors.lightGray,
directoryTextColor = colors.gray,
directoryBackgroundColor = colors.black,
promptTextColor = colors.gray,
promptBackgroundColor = colors.black,
directoryColor = colors.gray,
},
color = {
textColor = colors.white,
commandTextColor = colors.yellow,
directoryTextColor = colors.orange,
directoryBackgroundColor = colors.black,
promptTextColor = colors.blue,
promptBackgroundColor = colors.black,
directoryColor = colors.green,
},
displayDirectory = true,
}
2017-10-11 20:31:48 +00:00
Config.load('shellprompt', config)
2016-12-11 19:24:52 +00:00
local _colors = config.standard
if term.isColor() then
_colors = config.color
end
2017-10-13 20:30:47 +00:00
local function autocompleteArgument(program, words)
2016-12-11 19:24:52 +00:00
local word = ''
if #words > 1 then
word = words[#words]
end
local tInfo = tCompletionInfo[program]
2017-10-13 20:30:47 +00:00
return tInfo.fnComplete(shell, #words - 1, word, words)
end
local function autocompleteAnything(line, words)
local results = shell.complete(line)
if results and #results == 0 and #words == 1 then
results = nil
2016-12-11 19:24:52 +00:00
end
2017-10-13 20:30:47 +00:00
if not results then
results = fs.complete(words[#words] or '', shell.dir(), true, false)
end
return results
2016-12-11 19:24:52 +00:00
end
2017-10-13 20:30:47 +00:00
local function autocomplete(line)
2016-12-11 19:24:52 +00:00
local words = { }
for word in line:gmatch("%S+") do
table.insert(words, word)
end
if line:match(' $') then
table.insert(words, '')
end
2017-10-13 20:30:47 +00:00
if #words == 0 then
words = { '' }
end
2016-12-11 19:24:52 +00:00
2017-10-13 20:30:47 +00:00
local results
2016-12-11 19:24:52 +00:00
2017-10-13 20:30:47 +00:00
local program = shell.resolveProgram(words[1])
if tCompletionInfo[program] then
results = autocompleteArgument(program, words) or { }
2016-12-11 19:24:52 +00:00
else
2017-10-13 20:30:47 +00:00
results = autocompleteAnything(line, words) or { }
2016-12-11 19:24:52 +00:00
end
2017-10-13 20:30:47 +00:00
Util.filterInplace(results, function(f)
return not Util.key(results, f .. '/')
end)
local w = words[#words] or ''
for k,arg in pairs(results) do
results[k] = w .. arg
2016-12-11 19:24:52 +00:00
end
2017-10-13 20:30:47 +00:00
if #results == 1 then
words[#words] = results[1]
2016-12-11 19:24:52 +00:00
return table.concat(words, ' ')
2017-10-13 20:30:47 +00:00
elseif #results > 1 then
local function someComplete()
-- ugly (complete as much as possible)
local word = words[#words] or ''
local i = #word + 1
while true do
local ch
for _,f in ipairs(results) do
if #f < i then
words[#words] = string.sub(f, 1, i - 1)
return table.concat(words, ' ')
end
if not ch then
ch = string.sub(f, i, i)
elseif string.sub(f, i, i) ~= ch then
if i == #word + 1 then
return
end
words[#words] = string.sub(f, 1, i - 1)
return table.concat(words, ' ')
end
end
i = i + 1
end
end
local t = someComplete()
if t then
return t
end
2016-12-11 19:24:52 +00:00
print()
local word = words[#words] or ''
local prefix = word:match("(.*/)") or ''
if #prefix > 0 then
2017-10-13 20:30:47 +00:00
for _,f in ipairs(results) do
2016-12-11 19:24:52 +00:00
if f:match("^" .. prefix) ~= prefix then
prefix = ''
break
end
end
end
local tDirs, tFiles = { }, { }
2017-10-13 20:30:47 +00:00
for _,f in ipairs(results) do
if fs.isDir(shell.resolve(f)) then
2016-12-11 19:24:52 +00:00
f = f:gsub(prefix, '', 1)
table.insert(tDirs, f)
else
f = f:gsub(prefix, '', 1)
table.insert(tFiles, f)
end
end
table.sort(tDirs)
table.sort(tFiles)
if #tDirs > 0 and #tDirs < #tFiles then
2017-10-13 20:30:47 +00:00
local tw = term.getSize()
local nMaxLen = tw / 8
for _,sItem in pairs(results) do
2016-12-11 19:24:52 +00:00
nMaxLen = math.max(string.len(sItem) + 1, nMaxLen)
end
local nCols = math.floor(w / nMaxLen)
if #tDirs < nCols then
2017-10-11 20:31:48 +00:00
for _ = #tDirs + 1, nCols do
2016-12-11 19:24:52 +00:00
table.insert(tDirs, '')
end
end
end
if #tDirs > 0 then
textutils.tabulate(_colors.directoryColor, tDirs, colors.white, tFiles)
else
textutils.tabulate(colors.white, tFiles)
end
term.setTextColour(_colors.promptTextColor)
term.setBackgroundColor(_colors.promptBackgroundColor)
2017-10-11 20:31:48 +00:00
term.write("$ " )
2016-12-11 19:24:52 +00:00
term.setTextColour(_colors.commandTextColor)
term.setBackgroundColor(colors.black)
return line
end
end
2017-10-03 04:50:54 +00:00
local function shellRead(history)
2016-12-11 19:24:52 +00:00
term.setCursorBlink( true )
local sLine = ""
local nPos = 0
local w = term.getSize()
local sx = term.getCursorPos()
2017-10-03 04:50:54 +00:00
history:reset()
2016-12-11 19:24:52 +00:00
local function redraw( sReplace )
local nScroll = 0
if sx + nPos >= w then
nScroll = (sx + nPos) - w
end
2017-10-11 20:31:48 +00:00
local _,cy = term.getCursorPos()
2016-12-11 19:24:52 +00:00
term.setCursorPos( sx, cy )
if sReplace then
term.write( string.rep( sReplace, math.max( string.len(sLine) - nScroll, 0 ) ) )
else
term.write( string.sub( sLine, nScroll + 1 ) )
end
term.setCursorPos( sx + nPos - nScroll, cy )
end
while true do
2017-10-11 20:31:48 +00:00
local sEvent, param = os.pullEventRaw()
2016-12-11 19:24:52 +00:00
if sEvent == "char" then
sLine = string.sub( sLine, 1, nPos ) .. param .. string.sub( sLine, nPos + 1 )
nPos = nPos + 1
redraw()
elseif sEvent == "paste" then
sLine = string.sub( sLine, 1, nPos ) .. param .. string.sub( sLine, nPos + 1 )
nPos = nPos + string.len( param )
redraw()
elseif sEvent == 'mouse_click' and param == 2 then
redraw(string.rep(' ', #sLine))
sLine = ''
nPos = 0
redraw()
2018-01-20 12:18:13 +00:00
elseif sEvent == 'mouse_scroll' then
if param == -1 then
terminal.scrollUp()
else
terminal.scrollDown()
end
2016-12-11 19:24:52 +00:00
elseif sEvent == 'terminate' then
bExit = true
break
elseif sEvent == "key" then
if param == keys.enter then
-- Enter
break
elseif param == keys.tab then
if nPos == #sLine then
2017-10-13 20:30:47 +00:00
local cline = autocomplete(sLine)
2016-12-11 19:24:52 +00:00
if cline then
sLine = cline
nPos = #sLine
redraw()
end
end
elseif param == keys.left then
if nPos > 0 then
nPos = nPos - 1
redraw()
end
elseif param == keys.right then
if nPos < string.len(sLine) then
redraw(" ")
nPos = nPos + 1
redraw()
end
elseif param == keys.up or param == keys.down then
2017-10-03 04:50:54 +00:00
redraw(" ")
if param == keys.up then
sLine = history:back()
else
sLine = history:forward()
2016-12-11 19:24:52 +00:00
end
2017-10-03 04:50:54 +00:00
if sLine then
nPos = string.len(sLine)
else
sLine = ""
nPos = 0
end
redraw()
2016-12-11 19:24:52 +00:00
elseif param == keys.backspace then
if nPos > 0 then
redraw(" ")
sLine = string.sub( sLine, 1, nPos - 1 ) .. string.sub( sLine, nPos + 1 )
2017-10-11 20:31:48 +00:00
nPos = nPos - 1
2016-12-11 19:24:52 +00:00
redraw()
end
elseif param == keys.home then
redraw(" ")
nPos = 0
redraw()
elseif param == keys.delete then
if nPos < string.len(sLine) then
redraw(" ")
sLine = string.sub( sLine, 1, nPos ) .. string.sub( sLine, nPos + 2 )
redraw()
end
elseif param == keys["end"] then
redraw(" ")
nPos = string.len(sLine)
redraw()
end
elseif sEvent == "term_resize" then
w = term.getSize()
redraw()
end
end
2017-10-11 20:31:48 +00:00
local _, cy = term.getCursorPos()
2016-12-11 19:24:52 +00:00
term.setCursorPos( w + 1, cy )
print()
term.setCursorBlink( false )
return sLine
end
2017-05-20 22:27:26 +00:00
local history = History.load('usr/.shell_history', 25)
2016-12-11 19:24:52 +00:00
while not bExit do
if config.displayDirectory then
term.setTextColour(_colors.directoryTextColor)
term.setBackgroundColor(_colors.directoryBackgroundColor)
print('==' .. os.getComputerLabel() .. ':/' .. DIR)
end
term.setTextColour(_colors.promptTextColor)
term.setBackgroundColor(_colors.promptBackgroundColor)
2017-10-11 20:31:48 +00:00
term.write("$ " )
2016-12-11 19:24:52 +00:00
term.setTextColour(_colors.commandTextColor)
term.setBackgroundColor(colors.black)
2017-10-03 04:50:54 +00:00
local sLine = shellRead(history)
2016-12-11 19:24:52 +00:00
if bExit then -- terminated
break
end
sLine = Util.trim(sLine)
if #sLine > 0 and sLine ~= 'exit' then
2017-10-03 04:50:54 +00:00
history:add(sLine)
2016-12-11 19:24:52 +00:00
end
term.setTextColour(_colors.textColor)
if #sLine > 0 then
2017-10-14 07:41:54 +00:00
local result, err = shell.run(sLine)
2017-07-28 23:01:59 +00:00
if not result and err then
2017-10-11 20:31:48 +00:00
_G.printError(err)
2016-12-11 19:24:52 +00:00
end
end
2017-10-14 07:41:54 +00:00
end