package main import ( "bytes" "errors" "fmt" "io" "os" "os/exec" "sync" "syscall" "hilbish/util" rt "github.com/arnodel/golua/runtime" ) var jobs *jobHandler var jobMetaKey = rt.StringValue("hshjob") // #type // #interface jobs // #property cmd The user entered command string for the job. // #property running Whether the job is running or not. // #property id The ID of the job in the job table // #property pid The Process ID // #property exitCode The last exit code of the job. // #property stdout The standard output of the job. This just means the normal logs of the process. // #property stderr The standard error stream of the process. This (usually) includes error messages of the job. // The Job type describes a Hilbish job. type job struct { cmd string running bool id int pid int exitCode int once bool args []string // save path for a few reasons, one being security (lmao) while the other // would just be so itll be the same binary command always (path changes) path string handle *exec.Cmd cmdout io.Writer cmderr io.Writer stdout *bytes.Buffer stderr *bytes.Buffer ud *rt.UserData } func (j *job) start() error { if j.handle == nil || j.once { // cmd cant be reused so make a new one cmd := exec.Cmd{ Path: j.path, Args: j.args, } j.setHandle(&cmd) } // bgProcAttr is defined in job_.go, it holds a procattr struct // in a simple explanation, it makes signals from hilbish (like sigint) // not go to it (child process) j.handle.SysProcAttr = bgProcAttr // reset output buffers j.stdout.Reset() j.stderr.Reset() // make cmd write to both standard output and output buffers for lua access j.handle.Stdout = io.MultiWriter(j.cmdout, j.stdout) j.handle.Stderr = io.MultiWriter(j.cmderr, j.stderr) if !j.once { j.once = true } err := j.handle.Start() proc := j.getProc() j.pid = proc.Pid j.running = true hooks.Emit("job.start", rt.UserDataValue(j.ud)) return err } func (j *job) stop() { // finish will be called in exec handle proc := j.getProc() if proc != nil { proc.Kill() } } func (j *job) finish() { j.running = false hooks.Emit("job.done", rt.UserDataValue(j.ud)) } func (j *job) wait() { j.handle.Wait() } func (j *job) setHandle(handle *exec.Cmd) { j.handle = handle j.args = handle.Args j.path = handle.Path if handle.Stdout != nil { j.cmdout = handle.Stdout } if handle.Stderr != nil { j.cmderr = handle.Stderr } } func (j *job) getProc() *os.Process { handle := j.handle if handle != nil { return handle.Process } return nil } // #interface jobs // #member // start() // Starts running the job. func luaStartJob(t *rt.Thread, c *rt.GoCont) (rt.Cont, error) { if err := c.Check1Arg(); err != nil { return nil, err } j, err := jobArg(c, 0) if err != nil { return nil, err } if !j.running { err := j.start() exit := util.HandleExecErr(err) j.exitCode = int(exit) j.finish() } return c.Next(), nil } // #interface jobs // #member // stop() // Stops the job from running. func luaStopJob(t *rt.Thread, c *rt.GoCont) (rt.Cont, error) { if err := c.Check1Arg(); err != nil { return nil, err } j, err := jobArg(c, 0) if err != nil { return nil, err } if j.running { j.stop() j.finish() } return c.Next(), nil } // #interface jobs // #member // foreground() // Puts a job in the foreground. This will cause it to run like it was // executed normally and wait for it to complete. func luaForegroundJob(t *rt.Thread, c *rt.GoCont) (rt.Cont, error) { if err := c.Check1Arg(); err != nil { return nil, err } j, err := jobArg(c, 0) if err != nil { return nil, err } if !j.running { return nil, errors.New("job not running") } // lua code can run in other threads and goroutines, so this exists jobs.foreground = true // this is kinda funny // background continues the process incase it got suspended err = j.background() if err != nil { return nil, err } err = j.foreground() if err != nil { return nil, err } jobs.foreground = false return c.Next(), nil } // #interface jobs // #member // background() // Puts a job in the background. This acts the same as initially running a job. func luaBackgroundJob(t *rt.Thread, c *rt.GoCont) (rt.Cont, error) { if err := c.Check1Arg(); err != nil { return nil, err } j, err := jobArg(c, 0) if err != nil { return nil, err } if !j.running { return nil, errors.New("job not running") } err = j.background() if err != nil { return nil, err } return c.Next(), nil } type jobHandler struct { jobs map[int]*job latestID int foreground bool // if job currently in the foreground mu *sync.RWMutex } func newJobHandler() *jobHandler { return &jobHandler{ jobs: make(map[int]*job), latestID: 0, mu: &sync.RWMutex{}, } } func (j *jobHandler) add(cmd string, args []string, path string) *job { j.mu.Lock() defer j.mu.Unlock() j.latestID++ jb := &job{ cmd: cmd, running: false, id: j.latestID, args: args, path: path, cmdout: os.Stdout, cmderr: os.Stderr, stdout: &bytes.Buffer{}, stderr: &bytes.Buffer{}, } jb.ud = jobUserData(jb) j.jobs[j.latestID] = jb hooks.Emit("job.add", rt.UserDataValue(jb.ud)) return jb } func (j *jobHandler) getLatest() *job { j.mu.RLock() defer j.mu.RUnlock() return j.jobs[j.latestID] } func (j *jobHandler) disown(id int) error { j.mu.RLock() if j.jobs[id] == nil { return errors.New("job doesnt exist") } j.mu.RUnlock() j.mu.Lock() delete(j.jobs, id) j.mu.Unlock() return nil } func (j *jobHandler) stopAll() { j.mu.RLock() defer j.mu.RUnlock() for _, jb := range j.jobs { // on exit, unix shell should send sighup to all jobs if jb.running { proc := jb.getProc() proc.Signal(syscall.SIGHUP) jb.wait() // waits for program to exit due to sighup } } } // #interface jobs // background job management /* Manage interactive jobs in Hilbish via Lua. Jobs are the name of background tasks/commands. A job can be started via interactive usage or with the functions defined below for use in external runners. */ func (j *jobHandler) loader(rtm *rt.Runtime) *rt.Table { jobMethods := rt.NewTable() jFuncs := map[string]util.LuaExport{ "stop": {luaStopJob, 1, false}, "start": {luaStartJob, 1, false}, "foreground": {luaForegroundJob, 1, false}, "background": {luaBackgroundJob, 1, false}, } util.SetExports(l, jobMethods, jFuncs) jobMeta := rt.NewTable() jobIndex := func(t *rt.Thread, c *rt.GoCont) (rt.Cont, error) { j, _ := jobArg(c, 0) arg := c.Arg(1) val := jobMethods.Get(arg) if val != rt.NilValue { return c.PushingNext1(t.Runtime, val), nil } keyStr, _ := arg.TryString() switch keyStr { case "cmd": val = rt.StringValue(j.cmd) case "running": val = rt.BoolValue(j.running) case "id": val = rt.IntValue(int64(j.id)) case "pid": val = rt.IntValue(int64(j.pid)) case "exitCode": val = rt.IntValue(int64(j.exitCode)) case "stdout": val = rt.StringValue(string(j.stdout.Bytes())) case "stderr": val = rt.StringValue(string(j.stderr.Bytes())) } return c.PushingNext1(t.Runtime, val), nil } jobMeta.Set(rt.StringValue("__index"), rt.FunctionValue(rt.NewGoFunction(jobIndex, "__index", 2, false))) l.SetRegistry(jobMetaKey, rt.TableValue(jobMeta)) jobFuncs := map[string]util.LuaExport{ "all": {j.luaAllJobs, 0, false}, "last": {j.luaLastJob, 0, false}, "get": {j.luaGetJob, 1, false}, "add": {j.luaAddJob, 3, false}, "disown": {j.luaDisownJob, 1, false}, } luaJob := rt.NewTable() util.SetExports(rtm, luaJob, jobFuncs) return luaJob } func jobArg(c *rt.GoCont, arg int) (*job, error) { j, ok := valueToJob(c.Arg(arg)) if !ok { return nil, fmt.Errorf("#%d must be a job", arg + 1) } return j, nil } func valueToJob(val rt.Value) (*job, bool) { u, ok := val.TryUserData() if !ok { return nil, false } j, ok := u.Value().(*job) return j, ok } func jobUserData(j *job) *rt.UserData { jobMeta := l.Registry(jobMetaKey) return rt.NewUserData(j, jobMeta.AsTable()) } // #interface jobs // get(id) -> @Job // Get a job object via its ID. // --- @param id number // --- @returns Job func (j *jobHandler) luaGetJob(t *rt.Thread, c *rt.GoCont) (rt.Cont, error) { j.mu.RLock() defer j.mu.RUnlock() if err := c.Check1Arg(); err != nil { return nil, err } jobID, err := c.IntArg(0) if err != nil { return nil, err } job := j.jobs[int(jobID)] if job == nil { return c.Next(), nil } return c.PushingNext(t.Runtime, rt.UserDataValue(job.ud)), nil } // #interface jobs // add(cmdstr, args, execPath) // Creates a new job. This function does not run the job. This function is intended to be // used by runners, but can also be used to create jobs via Lua. Commanders cannot be ran as jobs. // #param cmdstr string String that a user would write for the job // #param args table Arguments for the commands. Has to include the name of the command. // #param execPath string Binary to use to run the command. Needs to be an absolute path. /* #example hilbish.jobs.add('go build', {'go', 'build'}, '/usr/bin/go') #example */ func (j *jobHandler) luaAddJob(t *rt.Thread, c *rt.GoCont) (rt.Cont, error) { if err := c.CheckNArgs(3); err != nil { return nil, err } cmd, err := c.StringArg(0) if err != nil { return nil, err } largs, err := c.TableArg(1) if err != nil { return nil, err } execPath, err := c.StringArg(2) if err != nil { return nil, err } var args []string util.ForEach(largs, func(k rt.Value, v rt.Value) { if v.Type() == rt.StringType { args = append(args, v.AsString()) } }) jb := j.add(cmd, args, execPath) return c.PushingNext1(t.Runtime, rt.UserDataValue(jb.ud)), nil } // #interface jobs // all() -> table[@Job] // Returns a table of all job objects. // #returns table[Job] func (j *jobHandler) luaAllJobs(t *rt.Thread, c *rt.GoCont) (rt.Cont, error) { j.mu.RLock() defer j.mu.RUnlock() jobTbl := rt.NewTable() for id, job := range j.jobs { jobTbl.Set(rt.IntValue(int64(id)), rt.UserDataValue(job.ud)) } return c.PushingNext1(t.Runtime, rt.TableValue(jobTbl)), nil } // #interface jobs // disown(id) // Disowns a job. This simply deletes it from the list of jobs without stopping it. // #param id number func (j *jobHandler) luaDisownJob(t *rt.Thread, c *rt.GoCont) (rt.Cont, error) { if err := c.Check1Arg(); err != nil { return nil, err } jobID, err := c.IntArg(0) if err != nil { return nil, err } err = j.disown(int(jobID)) if err != nil { return nil, err } return c.Next(), nil } // #interface jobs // last() -> @Job // Returns the last added job to the table. // #returns Job func (j *jobHandler) luaLastJob(t *rt.Thread, c *rt.GoCont) (rt.Cont, error) { j.mu.RLock() defer j.mu.RUnlock() job := j.jobs[j.latestID] if job == nil { // incase we dont have any jobs yet return c.Next(), nil } return c.PushingNext1(t.Runtime, rt.UserDataValue(job.ud)), nil }