elvish/eval/external_cmd.go

100 lines
2.0 KiB
Go
Raw Normal View History

2016-02-19 05:52:05 +08:00
package eval
import (
"errors"
"fmt"
"os"
"syscall"
"github.com/elves/elvish/parse"
"github.com/elves/elvish/util"
2017-08-31 02:52:27 +08:00
"github.com/xiaq/persistent/hash"
2016-02-19 05:52:05 +08:00
)
// FdNil is a special impossible fd value used for "close fd" in
// syscall.ProcAttr.Files.
const fdNil uintptr = ^uintptr(0)
var (
ErrExternalCmdOpts = errors.New("external commands don't accept elvish options")
ErrCdNoArg = errors.New("implicit cd accepts no arguments")
)
2016-02-19 05:52:05 +08:00
// ExternalCmd is an external command.
type ExternalCmd struct {
Name string
}
func (ExternalCmd) Kind() string {
return "fn"
}
2017-08-31 01:47:50 +08:00
func (e ExternalCmd) Equal(a interface{}) bool {
return e == a
}
2017-08-31 02:52:27 +08:00
func (e ExternalCmd) Hash() uint32 {
return hash.String(e.Name)
}
func (e ExternalCmd) Repr(int) string {
return "<external " + parse.Quote(e.Name) + ">"
2016-02-19 05:52:05 +08:00
}
// Call calls an external command.
func (e ExternalCmd) Call(ec *EvalCtx, argVals []Value, opts map[string]Value) {
if len(opts) > 0 {
throw(ErrExternalCmdOpts)
}
if util.DontSearch(e.Name) {
2016-02-19 05:52:05 +08:00
stat, err := os.Stat(e.Name)
if err == nil && stat.IsDir() {
// implicit cd
if len(argVals) > 0 {
throw(ErrCdNoArg)
}
cdInner(e.Name, ec)
return
}
}
files := make([]uintptr, len(ec.ports))
for i, port := range ec.ports {
if port == nil || port.File == nil {
files[i] = fdNil
} else {
files[i] = port.File.Fd()
}
}
args := make([]string, len(argVals)+1)
for i, a := range argVals {
// NOTE Maybe we should enfore string arguments instead of coercing all
// args into string
args[i+1] = ToString(a)
}
sys := syscall.SysProcAttr{Setpgid: ec.background}
2016-02-19 05:52:05 +08:00
attr := syscall.ProcAttr{Env: os.Environ(), Files: files[:], Sys: &sys}
path, err := ec.Search(e.Name)
if err != nil {
2016-02-22 03:17:06 +08:00
throw(err)
2016-02-19 05:52:05 +08:00
}
args[0] = path
pid, err := syscall.ForkExec(path, args, &attr)
if err != nil {
throw(errors.New("forkExec: " + err.Error()))
}
var ws syscall.WaitStatus
_, err = syscall.Wait4(pid, &ws, syscall.WUNTRACED, nil)
2016-02-19 05:52:05 +08:00
if err != nil {
throw(fmt.Errorf("wait: %s", err.Error()))
} else {
maybeThrow(NewExternalCmdExit(e.Name, ws, pid))
2016-02-19 05:52:05 +08:00
}
}