mirror of
https://github.com/Hilbis/Hilbish
synced 2025-04-04 20:53:24 +00:00
refactor: port exec code to lua
This commit is contained in:
parent
789c2cc714
commit
6c51f1e374
2
api.go
2
api.go
@ -300,7 +300,7 @@ hilbish.multiprompt '-->'
|
|||||||
*/
|
*/
|
||||||
func hlmultiprompt(t *rt.Thread, c *rt.GoCont) (rt.Cont, error) {
|
func hlmultiprompt(t *rt.Thread, c *rt.GoCont) (rt.Cont, error) {
|
||||||
if err := c.Check1Arg(); err != nil {
|
if err := c.Check1Arg(); err != nil {
|
||||||
return nil, err
|
return c.PushingNext1(t.Runtime, rt.StringValue(multilinePrompt)), nil
|
||||||
}
|
}
|
||||||
prompt, err := c.StringArg(0)
|
prompt, err := c.StringArg(0)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
132
exec.go
132
exec.go
@ -3,12 +3,9 @@ package main
|
|||||||
import (
|
import (
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
|
||||||
"os"
|
"os"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"hilbish/util"
|
|
||||||
|
|
||||||
rt "github.com/arnodel/golua/runtime"
|
rt "github.com/arnodel/golua/runtime"
|
||||||
//"github.com/yuin/gopher-lua/parse"
|
//"github.com/yuin/gopher-lua/parse"
|
||||||
)
|
)
|
||||||
@ -19,100 +16,11 @@ var runnerMode rt.Value = rt.NilValue
|
|||||||
|
|
||||||
func runInput(input string, priv bool) {
|
func runInput(input string, priv bool) {
|
||||||
running = true
|
running = true
|
||||||
cmdString := aliases.Resolve(input)
|
runnerRun := hshMod.Get(rt.StringValue("runner")).AsTable().Get(rt.StringValue("run"))
|
||||||
hooks.Emit("command.preexec", input, cmdString)
|
_, err := rt.Call1(l.MainThread(), runnerRun, rt.StringValue(input), rt.BoolValue(priv))
|
||||||
|
|
||||||
// save incase it changes while prompting (For some reason)
|
|
||||||
currentRunner := runnerMode
|
|
||||||
|
|
||||||
rerun:
|
|
||||||
var exitCode uint8
|
|
||||||
var cont bool
|
|
||||||
var newline bool
|
|
||||||
input, exitCode, cont, newline, runnerErr, err := runLuaRunner(currentRunner, input)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Fprintln(os.Stderr, err)
|
fmt.Fprintln(os.Stderr, err)
|
||||||
cmdFinish(124, input, priv)
|
|
||||||
return
|
|
||||||
}
|
}
|
||||||
// we only use `err` to check for lua eval error
|
|
||||||
// our actual error should only be a runner provided error at this point
|
|
||||||
// command not found type, etc
|
|
||||||
err = runnerErr
|
|
||||||
|
|
||||||
if cont {
|
|
||||||
input, err = continuePrompt(input, newline)
|
|
||||||
if err == nil {
|
|
||||||
goto rerun
|
|
||||||
} else if err == io.EOF {
|
|
||||||
lr.SetPrompt(fmtPrompt(prompt))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if err != nil && err != io.EOF {
|
|
||||||
if exErr, ok := util.IsExecError(err); ok {
|
|
||||||
hooks.Emit("command." + exErr.Typ, exErr.Cmd)
|
|
||||||
} else {
|
|
||||||
fmt.Fprintln(os.Stderr, err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
cmdFinish(exitCode, input, priv)
|
|
||||||
}
|
|
||||||
|
|
||||||
func reprompt(input string, newline bool) (string, error) {
|
|
||||||
for {
|
|
||||||
/*
|
|
||||||
if strings.HasSuffix(input, "\\") {
|
|
||||||
input = strings.TrimSuffix(input, "\\") + "\n"
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
in, err := continuePrompt(input, newline)
|
|
||||||
if err != nil {
|
|
||||||
lr.SetPrompt(fmtPrompt(prompt))
|
|
||||||
return input, err
|
|
||||||
}
|
|
||||||
|
|
||||||
return in, nil
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func runLuaRunner(runr rt.Value, userInput string) (input string, exitCode uint8, continued bool, newline bool, runnerErr, err error) {
|
|
||||||
term := rt.NewTerminationWith(l.MainThread().CurrentCont(), 3, false)
|
|
||||||
err = rt.Call(l.MainThread(), runr, []rt.Value{rt.StringValue(userInput)}, term)
|
|
||||||
if err != nil {
|
|
||||||
return "", 124, false, false, nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
var runner *rt.Table
|
|
||||||
var ok bool
|
|
||||||
runnerRet := term.Get(0)
|
|
||||||
if runner, ok = runnerRet.TryTable(); !ok {
|
|
||||||
fmt.Fprintln(os.Stderr, "runner did not return a table")
|
|
||||||
exitCode = 125
|
|
||||||
input = userInput
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
if code, ok := runner.Get(rt.StringValue("exitCode")).TryInt(); ok {
|
|
||||||
exitCode = uint8(code)
|
|
||||||
}
|
|
||||||
|
|
||||||
if inp, ok := runner.Get(rt.StringValue("input")).TryString(); ok {
|
|
||||||
input = inp
|
|
||||||
}
|
|
||||||
|
|
||||||
if errStr, ok := runner.Get(rt.StringValue("err")).TryString(); ok {
|
|
||||||
runnerErr = fmt.Errorf("%s", errStr)
|
|
||||||
}
|
|
||||||
|
|
||||||
if c, ok := runner.Get(rt.StringValue("continue")).TryBool(); ok {
|
|
||||||
continued = c
|
|
||||||
}
|
|
||||||
|
|
||||||
if nl, ok := runner.Get(rt.StringValue("newline")).TryBool(); ok {
|
|
||||||
newline = nl
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func handleLua(input string) (string, uint8, error) {
|
func handleLua(input string) (string, uint8, error) {
|
||||||
@ -142,34 +50,6 @@ func handleLua(input string) (string, uint8, error) {
|
|||||||
return cmdString, 125, err
|
return cmdString, 125, err
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
func execSh(cmdString string) (input string, exitcode uint8, cont bool, newline bool, e error) {
|
|
||||||
_, _, err := execCommand(cmdString, nil)
|
|
||||||
if err != nil {
|
|
||||||
// If input is incomplete, start multiline prompting
|
|
||||||
if syntax.IsIncomplete(err) {
|
|
||||||
if !interactive {
|
|
||||||
return cmdString, 126, false, false, err
|
|
||||||
}
|
|
||||||
|
|
||||||
newline := false
|
|
||||||
if strings.Contains(err.Error(), "unclosed here-document") {
|
|
||||||
newline = true
|
|
||||||
}
|
|
||||||
return cmdString, 126, true, newline, err
|
|
||||||
} else {
|
|
||||||
if code, ok := interp.IsExitStatus(err); ok {
|
|
||||||
return cmdString, code, false, false, nil
|
|
||||||
} else {
|
|
||||||
return cmdString, 126, false, false, err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return cmdString, 0, false, false, nil
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
|
|
||||||
func splitInput(input string) ([]string, string) {
|
func splitInput(input string) ([]string, string) {
|
||||||
// end my suffering
|
// end my suffering
|
||||||
// TODO: refactor this garbage
|
// TODO: refactor this garbage
|
||||||
@ -203,11 +83,3 @@ func splitInput(input string) ([]string, string) {
|
|||||||
|
|
||||||
return cmdArgs, cmdstr.String()
|
return cmdArgs, cmdstr.String()
|
||||||
}
|
}
|
||||||
|
|
||||||
func cmdFinish(code uint8, cmdstr string, private bool) {
|
|
||||||
util.SetField(l, hshMod, "exitCode", rt.IntValue(int64(code)))
|
|
||||||
// using AsValue (to convert to lua type) on an interface which is an int
|
|
||||||
// results in it being unknown in lua .... ????
|
|
||||||
// so we allow the hook handler to take lua runtime Values
|
|
||||||
hooks.Emit("command.exit", rt.IntValue(int64(code)), cmdstr, private)
|
|
||||||
}
|
|
||||||
|
@ -72,10 +72,8 @@ end
|
|||||||
--- Sets Hilbish's runner mode by name.
|
--- Sets Hilbish's runner mode by name.
|
||||||
--- @param name string
|
--- @param name string
|
||||||
function hilbish.runner.setCurrent(name)
|
function hilbish.runner.setCurrent(name)
|
||||||
local r = hilbish.runner.get(name)
|
hilbish.runner.get(name) -- throws if it doesnt exist.
|
||||||
currentRunner = name
|
currentRunner = name
|
||||||
|
|
||||||
hilbish.runner.setMode(r.run)
|
|
||||||
end
|
end
|
||||||
|
|
||||||
--- Returns the current runner by name.
|
--- Returns the current runner by name.
|
||||||
@ -84,6 +82,67 @@ function hilbish.runner.getCurrent()
|
|||||||
return currentRunner
|
return currentRunner
|
||||||
end
|
end
|
||||||
|
|
||||||
|
local function finishExec(exitCode, input, priv)
|
||||||
|
hilbish.exitCode = exitCode
|
||||||
|
bait.throw('command.exit', exitCode, input, priv)
|
||||||
|
end
|
||||||
|
|
||||||
|
local function continuePrompt(prev, newline)
|
||||||
|
local multilinePrompt = hilbish.multiprompt()
|
||||||
|
-- the return of hilbish.read is nil when error or ctrl-d
|
||||||
|
local cont = hilbish.read(multilinePrompt)
|
||||||
|
if not cont then
|
||||||
|
return
|
||||||
|
end
|
||||||
|
|
||||||
|
if newline then
|
||||||
|
cont = '\n' .. cont
|
||||||
|
end
|
||||||
|
|
||||||
|
if cont:match '\\$' then
|
||||||
|
cont = cont:gsub('\\$', '') .. '\n'
|
||||||
|
end
|
||||||
|
|
||||||
|
return prev .. cont
|
||||||
|
end
|
||||||
|
|
||||||
|
--- Runs `input` with the currently set Hilbish runner.
|
||||||
|
--- This method is how Hilbish executes commands.
|
||||||
|
--- `priv` is an optional boolean used to state if the input should be saved to history.
|
||||||
|
-- @param input string
|
||||||
|
-- @param priv bool
|
||||||
|
function hilbish.runner.run(input, priv)
|
||||||
|
local command = hilbish.aliases.resolve(input)
|
||||||
|
bait.throw('command.preexec', input, command)
|
||||||
|
|
||||||
|
::rerun::
|
||||||
|
local runner = hilbish.runner.get(currentRunner)
|
||||||
|
local ok, out = pcall(runner.run, input)
|
||||||
|
if not ok then
|
||||||
|
io.stderr:write(out .. '\n')
|
||||||
|
finishExec(124, out.input, priv)
|
||||||
|
return
|
||||||
|
end
|
||||||
|
|
||||||
|
if out.continue then
|
||||||
|
local contInput = continuePrompt(input, out.newline)
|
||||||
|
if contInput then
|
||||||
|
input = contInput
|
||||||
|
goto rerun
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
if out.err then
|
||||||
|
local fields = string.split(out.err, ': ')
|
||||||
|
if fields[2] == 'not-found' or fields[2] == 'not-executable' then
|
||||||
|
bait.throw('command.' .. fields[2], fields[1])
|
||||||
|
else
|
||||||
|
io.stderr:write(out.err .. '\n')
|
||||||
|
end
|
||||||
|
end
|
||||||
|
finishExec(out.exitCode, out.input, priv)
|
||||||
|
end
|
||||||
|
|
||||||
function hilbish.runner.sh(input)
|
function hilbish.runner.sh(input)
|
||||||
return hilbish.snail:run(input)
|
return hilbish.snail:run(input)
|
||||||
end
|
end
|
||||||
|
Loading…
x
Reference in New Issue
Block a user