1
0
mirror of https://github.com/kepler155c/opus synced 2025-01-17 19:02:53 +00:00
opus/sys/apps/shell.lua

679 lines
15 KiB
Lua
Raw Normal View History

2017-10-11 20:31:48 +00:00
local parentShell = _ENV.shell
_ENV.shell = { }
local trace = require('opus.trace')
2020-06-13 18:18:04 +00:00
local Util = require('opus.util')
local fs = _G.fs
local settings = _G.settings
local shell = _ENV.shell
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 = {}
2020-06-13 18:18:04 +00:00
local function tokenise(...)
local sLine = table.concat({ ... }, ' ')
local tWords = { }
2018-01-24 22:39:38 +00:00
local bQuoted = false
2020-06-13 18:18:04 +00:00
for match in string.gmatch(sLine .. "\"", "(.-)\"") do
2018-01-24 22:39:38 +00:00
if bQuoted then
2020-06-13 18:18:04 +00:00
table.insert(tWords, match)
2018-01-24 22:39:38 +00:00
else
2020-06-13 18:18:04 +00:00
for m in string.gmatch(match, "[^ \t]+") do
table.insert(tWords, m)
2018-01-24 22:39:38 +00:00
end
end
bQuoted = not bQuoted
end
return tWords
2016-12-11 19:24:52 +00:00
end
local defaultHandlers = {
2020-06-13 18:18:04 +00:00
function(env, command, args)
return command:match("^(https?:)") and {
title = fs.getName(command),
path = command,
args = args,
load = Util.loadUrl,
2020-06-13 18:18:04 +00:00
env = env,
}
end,
2020-06-13 18:18:04 +00:00
function(env, command, args)
command = env.shell.resolveProgram(command)
2020-06-11 19:28:43 +00:00
or error('No such program')
2020-06-13 18:18:04 +00:00
_G.requireInjector(env, fs.getDir(command))
return {
title = fs.getName(command):match('([^%.]+)'),
2020-06-11 19:28:43 +00:00
path = command,
args = args,
load = loadfile,
2020-06-13 18:18:04 +00:00
env = env,
}
end,
}
function shell.getHandlers()
if parentShell and parentShell.getHandlers then
return parentShell.getHandlers()
end
return defaultHandlers
end
local handlers = shell.getHandlers()
2017-10-01 00:35:36 +00:00
function shell.registerHandler(fn)
table.insert(handlers, 1, fn)
end
2020-06-13 18:18:04 +00:00
local function handleCommand(env, command, args)
for _,v in pairs(handlers) do
2020-06-13 18:18:04 +00:00
local pi = v(env, command, args)
if pi then
return pi
end
end
end
local function run(...)
local args = tokenise(...)
if #args == 0 then
error('No such program')
2018-01-24 22:39:38 +00:00
end
2017-10-13 20:30:47 +00:00
2020-06-13 18:18:04 +00:00
local pi = handleCommand(shell.makeEnv(_ENV), table.remove(args, 1), args)
local O_v_O, err = pi.load(pi.path, pi.env)
if not O_v_O then
error(err, -1)
2018-01-24 22:39:38 +00:00
end
2017-09-15 05:08:04 +00:00
2018-01-24 22:39:38 +00:00
if _ENV.multishell then
_ENV.multishell.setTitle(_ENV.multishell.getCurrent(), pi.title)
2018-01-24 22:39:38 +00:00
end
2016-12-11 19:24:52 +00:00
tProgramStack[#tProgramStack + 1] = pi
2017-10-11 20:31:48 +00:00
pi.env[ "arg" ] = { [0] = pi.path, table.unpack(pi.args) }
local r = { O_v_O(table.unpack(pi.args)) }
2017-10-11 20:31:48 +00:00
2018-01-24 22:39:38 +00:00
tProgramStack[#tProgramStack] = nil
2017-10-11 20:31:48 +00:00
2018-01-24 22:39:38 +00:00
return table.unpack(r)
2017-10-11 20:31:48 +00:00
end
-- Install shell API
function shell.run(...)
2018-01-24 22:39:38 +00:00
local oldTitle
2017-10-13 20:30:47 +00:00
2018-01-24 22:39:38 +00:00
if _ENV.multishell then
oldTitle = _ENV.multishell.getTitle(_ENV.multishell.getCurrent())
end
2017-10-13 20:30:47 +00:00
2020-05-05 23:25:58 +00:00
local r = { trace(run, ...) }
2017-10-13 20:30:47 +00:00
2018-01-24 22:39:38 +00:00
if _ENV.multishell then
_ENV.multishell.setTitle(_ENV.multishell.getCurrent(), oldTitle or 'shell')
end
2017-10-13 20:30:47 +00:00
2018-01-24 22:39:38 +00:00
return table.unpack(r)
2016-12-11 19:24:52 +00:00
end
function shell.exit()
2018-01-24 22:39:38 +00:00
bExit = true
2016-12-11 19:24:52 +00:00
end
function shell.dir() return DIR end
function shell.setDir(d)
d = fs.combine(d, '')
if not fs.isDir(d) then
error("Not a directory", 2)
end
DIR = d
end
2016-12-11 19:24:52 +00:00
function shell.path() return PATH end
function shell.setPath(p) PATH = p end
function shell.resolve( _sPath )
2018-01-24 22:39:38 +00:00
local sStartChar = string.sub( _sPath, 1, 1 )
if sStartChar == "/" or sStartChar == "\\" then
return fs.combine( "", _sPath )
else
return fs.combine(DIR, _sPath )
end
2016-12-11 19:24:52 +00:00
end
2020-06-13 18:18:04 +00:00
function shell.resolveProgram(_sCommand)
2018-01-24 22:39:38 +00:00
if tAliases[_sCommand] ~= nil then
_sCommand = tAliases[_sCommand]
end
2020-06-14 02:26:17 +00:00
local function check(f)
return fs.exists(f) and not fs.isDir(f) and f
2018-01-24 22:39:38 +00:00
end
2020-06-14 02:26:17 +00:00
local function inPath()
-- Otherwise, look on the path variable
for sPath in string.gmatch(PATH or '', "[^:]+") do
sPath = fs.combine(sPath, _sCommand )
if check(sPath) then
return sPath
end
if check(sPath .. '.lua') then
return sPath .. '.lua'
2020-06-13 18:18:04 +00:00
end
2018-01-24 22:39:38 +00:00
end
end
2020-06-14 02:26:17 +00:00
if not _sCommand:find('/') then
return inPath()
end
-- so... even if you are in the rom directory and you run:
-- 'packages/common/edit.lua', allow this even though it
-- does not use a leading slash. Ideally, fs.combine would
-- provide the leading slash... but it does not.
return check(shell.resolve(_sCommand))
2020-06-14 02:26:17 +00:00
or check(shell.resolve(_sCommand) .. '.lua')
or check(_sCommand)
or check(_sCommand .. '.lua')
2016-12-11 19:24:52 +00:00
end
2020-06-13 18:18:04 +00:00
function shell.programs(_bIncludeHidden)
local tItems = { }
2018-01-24 22:39:38 +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 )
for _,sFile in pairs( tList ) do
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
2020-06-13 18:18:04 +00:00
local tItemList = { }
for sItem in pairs(tItems) do
table.insert(tItemList, sItem)
2018-01-24 22:39:38 +00:00
end
2020-06-13 18:18:04 +00:00
table.sort(tItemList)
2018-01-24 22:39:38 +00:00
return tItemList
2016-12-11 19:24:52 +00:00
end
2020-06-13 18:18:04 +00:00
function shell.completeProgram(sLine)
if #sLine > 0 and string.sub(sLine, 1, 1) == '/' then
2018-01-24 22:39:38 +00:00
-- Add programs from the root
2020-06-13 18:18:04 +00:00
return fs.complete(sLine, '', true, false)
end
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
2018-01-24 22:39:38 +00:00
end
end
2020-06-13 18:18:04 +00:00
end
2018-01-24 22:39:38 +00:00
2020-06-13 18:18:04 +00:00
-- 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
2018-01-24 22:39:38 +00:00
end
end
end
2017-10-13 20:30:47 +00:00
2020-06-13 18:18:04 +00:00
-- Sort and return
table.sort(tResults)
return tResults
2017-10-13 20:30:47 +00:00
end
function shell.complete(sLine)
2020-06-13 18:18:04 +00:00
local tWords = tokenise(sLine)
local nIndex = #tWords
if string.sub(sLine, #sLine, #sLine) == ' ' and #Util.trim(sLine) > 0 then
nIndex = nIndex + 1
end
if nIndex == 0 then
return fs.complete('', shell.dir(), true, false)
elseif nIndex == 1 then
local results = shell.completeProgram(tWords[1] or '')
for _, v in pairs(fs.complete(table.concat(tWords, ' '), shell.dir(), true, false)) do
table.insert(results, v)
2018-01-24 22:39:38 +00:00
end
2020-06-13 18:18:04 +00:00
return results
2018-01-24 22:39:38 +00:00
2020-06-13 18:18:04 +00:00
else
local sPath = shell.resolveProgram(tWords[1])
local sPart = tWords[nIndex] or ''
local tPreviousParts = tWords
tPreviousParts[nIndex] = nil
local results
local tInfo = tCompletionInfo[sPath]
if tInfo then
results = tInfo.fnComplete(shell, nIndex - 1, sPart, tPreviousParts)
2018-01-24 22:39:38 +00:00
end
2020-06-13 18:18:04 +00:00
return results and #results > 0 and results
or fs.complete(sPart, shell.dir(), true, false)
2018-01-24 22:39:38 +00:00
end
2017-10-13 20:30:47 +00:00
end
2016-12-11 19:24:52 +00:00
function shell.setCompletionFunction(sProgram, fnComplete)
2018-01-24 22:39:38 +00:00
tCompletionInfo[sProgram] = { fnComplete = fnComplete }
2016-12-11 19:24:52 +00:00
end
function shell.getCompletionInfo()
2018-01-24 22:39:38 +00:00
return tCompletionInfo
2016-12-11 19:24:52 +00:00
end
function shell.getRunningProgram()
return tProgramStack[#tProgramStack] and tProgramStack[#tProgramStack].path
end
function shell.getRunningInfo()
2018-01-24 22:39:38 +00:00
return tProgramStack[#tProgramStack]
2016-12-11 19:24:52 +00:00
end
-- convenience function for making a runnable env
2020-06-11 19:28:43 +00:00
function shell.makeEnv(env, dir)
env = setmetatable(Util.shallowCopy(env), { __index = _G })
2020-06-11 19:28:43 +00:00
_G.requireInjector(env, dir)
2020-05-05 03:06:23 +00:00
return env
2018-01-14 23:28:23 +00:00
end
2020-06-13 18:18:04 +00:00
function shell.setAlias(_sCommand, _sProgram)
2018-01-24 22:39:38 +00:00
tAliases[_sCommand] = _sProgram
2016-12-11 19:24:52 +00:00
end
2020-06-13 18:18:04 +00:00
function shell.clearAlias(_sCommand)
2018-01-24 22:39:38 +00:00
tAliases[_sCommand] = nil
2016-12-11 19:24:52 +00:00
end
function shell.aliases()
2018-01-24 22:39:38 +00:00
local tCopy = {}
for sAlias, sCommand in pairs(tAliases) do
tCopy[sAlias] = sCommand
end
return tCopy
2016-12-11 19:24:52 +00:00
end
2017-10-14 07:41:54 +00:00
function shell.newTab(tabInfo, ...)
2018-01-24 22:39:38 +00:00
local args = tokenise(...)
local path = table.remove(args, 1)
path = shell.resolveProgram(path)
2016-12-11 19:24:52 +00:00
2018-01-24 22:39:38 +00:00
if path then
tabInfo.path = path
tabInfo.args = args
tabInfo.title = fs.getName(path):match('([^%.]+)')
2016-12-11 19:24:52 +00:00
2019-02-12 22:03:50 +00:00
if path ~= 'sys/apps/shell.lua' then
2018-01-24 22:39:38 +00:00
table.insert(tabInfo.args, 1, tabInfo.path)
2019-02-12 22:03:50 +00:00
tabInfo.path = 'sys/apps/shell.lua'
2018-01-24 22:39:38 +00:00
end
return _ENV.multishell.openTab(_ENV, tabInfo)
2018-01-24 22:39:38 +00:00
end
return nil, 'No such program'
2016-12-11 19:24:52 +00:00
end
if not _ENV.multishell then
function shell.newTab()
error('Multishell is not available')
2018-01-24 22:39:38 +00:00
end
2016-12-11 19:24:52 +00:00
end
function shell.openTab(...)
return shell.newTab({ }, ...)
end
2016-12-11 19:24:52 +00:00
function shell.openForegroundTab( ... )
2018-01-24 22:39:38 +00:00
return shell.newTab({ focused = true }, ...)
2016-12-11 19:24:52 +00:00
end
function shell.openHiddenTab( ... )
2018-01-24 22:39:38 +00:00
return shell.newTab({ hidden = true }, ...)
2016-12-11 19:24:52 +00:00
end
function shell.switchTab(tabId)
2018-01-24 22:39:38 +00:00
_ENV.multishell.setFocus(tabId)
2016-12-11 19:24:52 +00:00
end
local tArgs = { ... }
if #tArgs > 0 then
2020-05-05 23:25:58 +00:00
return run(...)
2016-12-11 19:24:52 +00:00
end
local Config = require('opus.config')
local Entry = require('opus.entry')
local History = require('opus.history')
local Input = require('opus.input')
local Sound = require('opus.sound')
local Terminal = require('opus.terminal')
2016-12-11 19:24:52 +00:00
2017-10-11 20:31:48 +00:00
local colors = _G.colors
local os = _G.os
local term = _G.term
local textutils = _G.textutils
2019-02-06 04:03:57 +00:00
local oldTerm
2019-03-29 13:56:56 +00:00
local terminal = term.current()
2020-06-13 18:18:04 +00:00
local _len = string.len
2019-03-29 13:56:56 +00:00
local _rep = string.rep
local _sub = string.sub
2019-02-06 04:03:57 +00:00
2016-12-11 19:24:52 +00:00
local config = {
2018-01-24 22:39:38 +00:00
color = {
textColor = colors.white,
commandTextColor = colors.yellow,
directoryTextColor = colors.orange,
promptTextColor = colors.blue,
directoryColor = colors.green,
fileColor = colors.white,
backgroundColor = colors.black,
2018-01-24 22:39:38 +00:00
},
displayDirectory = true,
2016-12-11 19:24:52 +00:00
}
2017-10-11 20:31:48 +00:00
Config.load('shellprompt', config)
2016-12-11 19:24:52 +00:00
local _colors = config.color
2019-03-28 11:39:34 +00:00
-- temp
if not _colors.backgroundColor then
_colors.backgroundColor = colors.black
_colors.fileColor = colors.white
end
2020-06-13 18:18:04 +00:00
if not terminal.scrollUp then
terminal = Terminal.window(term.current())
terminal.setMaxScroll(200)
oldTerm = term.redirect(terminal)
term.setBackgroundColor(_colors.backgroundColor)
term.clear()
2017-10-13 20:30:47 +00:00
end
2020-06-13 18:18:04 +00:00
local palette = terminal.canvas.palette
2016-12-11 19:24:52 +00:00
2017-10-13 20:30:47 +00:00
local function autocomplete(line)
2018-01-24 22:39:38 +00:00
local words = { }
for word in line:gmatch("%S+") do
table.insert(words, word)
end
if line:match(' $') then
table.insert(words, '')
end
if #words == 0 then
words = { '' }
end
2020-06-13 18:18:04 +00:00
local results = shell.complete(line) or { }
2018-01-24 22:39:38 +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
end
if #results == 1 then
words[#words] = results[1]
return table.concat(words, ' ')
2020-06-13 18:18:04 +00:00
elseif #results > 1 then
2018-01-24 22:39:38 +00:00
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
2020-06-13 18:18:04 +00:00
words[#words] = _sub(f, 1, i - 1)
2018-01-24 22:39:38 +00:00
return table.concat(words, ' ')
end
if not ch then
2020-06-13 18:18:04 +00:00
ch = _sub(f, i, i)
elseif _sub(f, i, i) ~= ch then
2018-01-24 22:39:38 +00:00
if i == #word + 1 then
return
end
2020-06-13 18:18:04 +00:00
words[#words] = _sub(f, 1, i - 1)
2018-01-24 22:39:38 +00:00
return table.concat(words, ' ')
end
end
i = i + 1
end
end
local t = someComplete()
if t then
return t
end
print()
local word = words[#words] or ''
local prefix = word:match("(.*/)") or ''
if #prefix > 0 then
for _,f in ipairs(results) do
if f:match("^" .. prefix) ~= prefix then
prefix = ''
break
end
end
end
local tDirs, tFiles = { }, { }
for _,f in ipairs(results) do
if fs.isDir(shell.resolve(f)) then
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
local tw = term.getSize()
local nMaxLen = tw / 8
for _,sItem in pairs(results) do
2020-06-13 18:18:04 +00:00
nMaxLen = math.max(_len(sItem) + 1, nMaxLen)
2018-01-24 22:39:38 +00:00
end
2020-06-13 18:18:04 +00:00
local nCols = math.floor(tw / nMaxLen)
2018-01-24 22:39:38 +00:00
if #tDirs < nCols then
for _ = #tDirs + 1, nCols do
table.insert(tDirs, '')
end
end
end
if #tDirs > 0 then
textutils.tabulate(_colors.directoryColor, tDirs, _colors.fileColor, tFiles)
2018-01-24 22:39:38 +00:00
else
textutils.tabulate(_colors.fileColor, tFiles)
2018-01-24 22:39:38 +00:00
end
term.setTextColour(_colors.promptTextColor)
term.write("$ " )
term.setTextColour(_colors.commandTextColor)
return line
end
2016-12-11 19:24:52 +00:00
end
2017-10-03 04:50:54 +00:00
local function shellRead(history)
2018-01-24 22:39:38 +00:00
local lastLen = 0
local entry = Entry({
2019-03-29 13:56:56 +00:00
width = term.getSize() - 3,
offset = 3,
2018-01-24 22:39:38 +00:00
})
history:reset()
term.setCursorBlink(true)
2019-03-29 13:56:56 +00:00
local function updateCursor()
term.setCursorPos(3 + entry.pos - entry.scroll, select(2, term.getCursorPos()))
end
2018-01-24 22:39:38 +00:00
local function redraw()
2019-02-06 04:03:57 +00:00
if terminal.scrollBottom then
terminal.scrollBottom()
end
2018-01-24 22:39:38 +00:00
local _,cy = term.getCursorPos()
term.setCursorPos(3, cy)
2019-11-13 21:38:24 +00:00
entry.value = entry.value or ''
2018-01-24 22:39:38 +00:00
local filler = #entry.value < lastLen
2020-06-13 18:18:04 +00:00
and _rep(' ', lastLen - #entry.value)
2018-01-24 22:39:38 +00:00
or ''
2020-06-13 18:18:04 +00:00
local str = _sub(entry.value, entry.scroll + 1, entry.width + entry.scroll) .. filler
2019-03-29 13:56:56 +00:00
local fg = _rep(palette[_colors.commandTextColor], #str)
local bg = _rep(palette[_colors.backgroundColor], #str)
if entry.mark.active then
2020-06-13 18:18:04 +00:00
bg = _rep('f', entry.mark.x) ..
_rep('7', entry.mark.ex - entry.mark.x) ..
_rep('f', #entry.value - entry.mark.ex + #filler + 1)
bg = _sub(bg, entry.scroll + 1, entry.scroll + #str)
2019-03-29 13:56:56 +00:00
end
term.blit(str, fg, bg)
updateCursor()
2018-01-24 22:39:38 +00:00
lastLen = #entry.value
end
while true do
local event, p1, p2, p3 = os.pullEventRaw()
local ie = Input:translate(event, p1, p2, p3)
if ie then
2019-02-06 04:03:57 +00:00
if ie.code == 'scroll_up' and terminal.scrollUp then
terminal.scrollUp()
2018-01-24 22:39:38 +00:00
2019-02-06 04:03:57 +00:00
elseif ie.code == 'scroll_down' and terminal.scrollDown then
terminal.scrollDown()
2018-01-24 22:39:38 +00:00
elseif ie.code == 'terminate' then
bExit = true
break
elseif ie.code == 'enter' then
break
elseif ie.code == 'up' or ie.code == 'control-p' or
ie.code == 'down' or ie.code == 'control-n' then
2019-03-29 13:56:56 +00:00
entry:reset()
if ie.code == 'up' or ie.code == 'control-p' then
2018-01-24 22:39:38 +00:00
entry.value = history:back() or ''
else
entry.value = history:forward() or ''
end
2019-03-29 13:56:56 +00:00
entry.pos = #entry.value
2018-01-24 22:39:38 +00:00
entry:updateScroll()
redraw()
elseif ie.code == 'tab' then
2019-11-13 21:38:24 +00:00
entry.value = entry.value or ''
2018-01-24 22:39:38 +00:00
if entry.pos == #entry.value then
local cline = autocomplete(entry.value)
if cline then
entry.value = cline
entry.pos = #entry.value
2019-03-29 13:56:56 +00:00
entry:unmark()
2018-01-24 22:39:38 +00:00
entry:updateScroll()
redraw()
2019-03-29 14:44:21 +00:00
else
Sound.play('entity.villager.no')
2018-01-24 22:39:38 +00:00
end
end
2019-03-29 13:56:56 +00:00
else
entry:process(ie)
2019-11-13 21:38:24 +00:00
entry.value = entry.value or ''
2019-03-29 13:56:56 +00:00
if entry.textChanged then
redraw()
elseif entry.posChanged then
updateCursor()
end
2018-01-24 22:39:38 +00:00
end
elseif event == "term_resize" then
2020-05-10 20:04:20 +00:00
terminal.reposition(1, 1, oldTerm.getSize())
2018-01-24 22:39:38 +00:00
entry.width = term.getSize() - 3
2019-03-29 13:56:56 +00:00
entry:updateScroll()
2018-01-24 22:39:38 +00:00
redraw()
end
end
print()
2020-06-13 18:18:04 +00:00
term.setCursorBlink(false)
2019-11-13 21:38:24 +00:00
return entry.value or ''
2016-12-11 19:24:52 +00:00
end
local history = History.load('usr/.shell_history', 100)
2016-12-11 19:24:52 +00:00
term.setBackgroundColor(_colors.backgroundColor)
2020-06-13 18:18:04 +00:00
if settings.get("motd.enable") then
2019-08-13 17:14:11 +00:00
shell.run("motd")
end
2016-12-11 19:24:52 +00:00
while not bExit do
2018-01-24 22:39:38 +00:00
if config.displayDirectory then
term.setTextColour(_colors.directoryTextColor)
print('==' .. os.getComputerLabel() .. ':/' .. DIR)
end
term.setTextColour(_colors.promptTextColor)
term.write("$ " )
term.setTextColour(_colors.commandTextColor)
local sLine = shellRead(history)
if bExit then -- terminated
break
end
sLine = Util.trim(sLine)
if #sLine > 0 and sLine ~= 'exit' then
history:add(sLine)
end
term.setTextColour(_colors.textColor)
if #sLine > 0 then
local result, err = shell.run(sLine)
2020-06-13 18:18:04 +00:00
local cx = term.getCursorPos()
if cx ~= 1 then
print()
end
term.setBackgroundColor(_colors.backgroundColor)
2018-01-24 22:39:38 +00:00
if not result and err then
_G.printError(err)
end
end
2017-10-14 07:41:54 +00:00
end
2019-02-06 04:03:57 +00:00
if oldTerm then
term.redirect(oldTerm)
end