mirror of
https://github.com/go-sylixos/elvish.git
synced 2024-12-15 11:57:55 +08:00
670 lines
15 KiB
Go
670 lines
15 KiB
Go
package eval
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"os"
|
|
"strings"
|
|
"sync"
|
|
|
|
"github.com/elves/elvish/pkg/diag"
|
|
"github.com/elves/elvish/pkg/eval/errs"
|
|
"github.com/elves/elvish/pkg/eval/vals"
|
|
"github.com/elves/elvish/pkg/eval/vars"
|
|
"github.com/elves/elvish/pkg/parse"
|
|
"github.com/elves/elvish/pkg/util"
|
|
"github.com/xiaq/persistent/hashmap"
|
|
)
|
|
|
|
func (cp *compiler) chunkOp(n *parse.Chunk) effectOp {
|
|
return makeEffectOp(n, chunkOp{cp.pipelineOps(n.Pipelines)})
|
|
}
|
|
|
|
type chunkOp struct {
|
|
subops []effectOp
|
|
}
|
|
|
|
func (op chunkOp) invoke(fm *Frame) error {
|
|
for _, subop := range op.subops {
|
|
err := subop.exec(fm)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
// Check for interrupts after the chunk.
|
|
// We also check for interrupts before each pipeline, so there is no
|
|
// need to check it before the chunk or after each pipeline.
|
|
if fm.IsInterrupted() {
|
|
return ErrInterrupted
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (cp *compiler) pipelineOp(n *parse.Pipeline) effectOp {
|
|
saveNewLocals := cp.newLocals
|
|
cp.newLocals = nil
|
|
|
|
formOps := cp.formOps(n.Forms)
|
|
|
|
newLocals := cp.newLocals
|
|
cp.newLocals = saveNewLocals
|
|
|
|
return makeEffectOp(n,
|
|
&pipelineOp{n.Background, n.SourceText(), formOps, newLocals})
|
|
}
|
|
|
|
func (cp *compiler) pipelineOps(ns []*parse.Pipeline) []effectOp {
|
|
ops := make([]effectOp, len(ns))
|
|
for i, n := range ns {
|
|
ops[i] = cp.pipelineOp(n)
|
|
}
|
|
return ops
|
|
}
|
|
|
|
type pipelineOp struct {
|
|
bg bool
|
|
source string
|
|
subops []effectOp
|
|
newLocals []string
|
|
}
|
|
|
|
const pipelineChanBufferSize = 32
|
|
|
|
func (op *pipelineOp) invoke(fm *Frame) error {
|
|
if fm.IsInterrupted() {
|
|
return ErrInterrupted
|
|
}
|
|
|
|
for _, name := range op.newLocals {
|
|
var variable vars.Var
|
|
if strings.HasSuffix(name, FnSuffix) {
|
|
val := Callable(nil)
|
|
variable = vars.FromPtr(&val)
|
|
} else if strings.HasSuffix(name, NsSuffix) {
|
|
val := Ns(nil)
|
|
variable = vars.FromPtr(&val)
|
|
} else {
|
|
variable = vars.FromInit(nil)
|
|
}
|
|
fm.local[name] = variable
|
|
}
|
|
|
|
if op.bg {
|
|
fm = fm.fork("background job" + op.source)
|
|
fm.intCh = nil
|
|
fm.background = true
|
|
fm.Evaler.state.addNumBgJobs(1)
|
|
|
|
if fm.Editor != nil {
|
|
// TODO: Redirect output in interactive mode so that the line
|
|
// editor does not get messed up.
|
|
}
|
|
}
|
|
|
|
nforms := len(op.subops)
|
|
|
|
var wg sync.WaitGroup
|
|
wg.Add(nforms)
|
|
errors := make([]*Exception, nforms)
|
|
|
|
var nextIn *Port
|
|
|
|
// For each form, create a dedicated evalCtx and run asynchronously
|
|
for i, op := range op.subops {
|
|
hasChanInput := i > 0
|
|
newFm := fm.fork("[form op]")
|
|
if i > 0 {
|
|
newFm.ports[0] = nextIn
|
|
}
|
|
if i < nforms-1 {
|
|
// Each internal port pair consists of a (byte) pipe pair and a
|
|
// channel.
|
|
// os.Pipe sets O_CLOEXEC, which is what we want.
|
|
reader, writer, e := os.Pipe()
|
|
if e != nil {
|
|
return fmt.Errorf("failed to create pipe: %s", e)
|
|
}
|
|
ch := make(chan interface{}, pipelineChanBufferSize)
|
|
newFm.ports[1] = &Port{
|
|
File: writer, Chan: ch, CloseFile: true, CloseChan: true}
|
|
nextIn = &Port{
|
|
File: reader, Chan: ch, CloseFile: true, CloseChan: false}
|
|
}
|
|
thisOp := op
|
|
thisError := &errors[i]
|
|
go func() {
|
|
err := thisOp.exec(newFm)
|
|
newFm.Close()
|
|
if err != nil {
|
|
*thisError = err.(*Exception)
|
|
}
|
|
wg.Done()
|
|
if hasChanInput {
|
|
// If the command has channel input, drain it. This
|
|
// mitigates the effect of erroneous pipelines like
|
|
// "range 100 | cat"; without draining the pipeline will
|
|
// lock up.
|
|
for range newFm.ports[0].Chan {
|
|
}
|
|
}
|
|
}()
|
|
}
|
|
|
|
if op.bg {
|
|
// Background job, wait for form termination asynchronously.
|
|
go func() {
|
|
wg.Wait()
|
|
fm.Evaler.state.addNumBgJobs(-1)
|
|
msg := "job " + op.source + " finished"
|
|
err := makePipelineError(errors)
|
|
if err != nil {
|
|
msg += ", errors = " + err.Error()
|
|
}
|
|
if fm.Evaler.state.getNotifyBgJobSuccess() || err != nil {
|
|
if fm.Editor != nil {
|
|
fm.Editor.Notify("%s", msg)
|
|
} else {
|
|
fm.ports[2].File.WriteString(msg + "\n")
|
|
}
|
|
}
|
|
}()
|
|
return nil
|
|
}
|
|
wg.Wait()
|
|
return makePipelineError(errors)
|
|
}
|
|
|
|
func (cp *compiler) formOp(n *parse.Form) effectOp {
|
|
var saveVarsOps []lvaluesOp
|
|
var assignmentOps []effectOp
|
|
if len(n.Assignments) > 0 {
|
|
assignmentOps = cp.assignmentOps(n.Assignments)
|
|
if n.Head == nil && n.Vars == nil {
|
|
// Permanent assignment.
|
|
return makeEffectOp(n, seqOp{assignmentOps})
|
|
}
|
|
for _, a := range n.Assignments {
|
|
v, r := cp.lvaluesOp(a.Left)
|
|
saveVarsOps = append(saveVarsOps, v, r)
|
|
}
|
|
logger.Println("temporary assignment of", len(n.Assignments), "pairs")
|
|
}
|
|
|
|
// Depending on the type of the form, exactly one of the three below will be
|
|
// set.
|
|
var (
|
|
specialOpFunc effectOpBody
|
|
headOp valuesOp
|
|
spaceyAssignOp effectOp
|
|
)
|
|
|
|
// Forward declaration; needed when compiling assignment forms.
|
|
var argOps []valuesOp
|
|
|
|
if n.Head != nil {
|
|
headStr, ok := oneString(n.Head)
|
|
if ok {
|
|
compileForm, ok := builtinSpecials[headStr]
|
|
if ok {
|
|
// Special form.
|
|
specialOpFunc = compileForm(cp, n)
|
|
} else {
|
|
var headOpFunc valuesOpBody
|
|
sigil, qname := SplitVariableRef(headStr)
|
|
if sigil == "" && cp.registerVariableGet(qname+FnSuffix) {
|
|
// $head~ resolves.
|
|
headOpFunc = variableOp{false, qname + FnSuffix}
|
|
} else {
|
|
// Fall back to $e:head~.
|
|
headOpFunc = literalValues(ExternalCmd{headStr})
|
|
}
|
|
headOp = valuesOp{headOpFunc, n.Head.Range()}
|
|
}
|
|
} else {
|
|
// Head exists and is not a literal string. Evaluate as a normal
|
|
// expression.
|
|
headOp = cp.compoundOp(n.Head)
|
|
}
|
|
argOps = cp.compoundOps(n.Args)
|
|
} else {
|
|
// Assignment form.
|
|
varsOp, restOp := cp.lvaluesMulti(n.Vars)
|
|
argOps = cp.compoundOps(n.Args)
|
|
valuesOp := valuesOp{body: seqValuesOp{argOps}}
|
|
if len(argOps) > 0 {
|
|
valuesOp.From = argOps[0].From
|
|
valuesOp.To = argOps[len(argOps)-1].To
|
|
} else {
|
|
valuesOp.From = n.Range().To
|
|
valuesOp.To = n.Range().To
|
|
}
|
|
spaceyAssignOp = effectOp{
|
|
&assignmentOp{varsOp, restOp, valuesOp}, n.Range(),
|
|
}
|
|
}
|
|
|
|
optsOp := cp.mapPairs(n.Opts)
|
|
redirOps := cp.redirOps(n.Redirs)
|
|
// TODO: n.ErrorRedir
|
|
|
|
return makeEffectOp(n, &formOp{n.Range(), saveVarsOps, assignmentOps, redirOps, specialOpFunc, headOp, argOps, optsOp, spaceyAssignOp})
|
|
}
|
|
|
|
func (cp *compiler) formOps(ns []*parse.Form) []effectOp {
|
|
ops := make([]effectOp, len(ns))
|
|
for i, n := range ns {
|
|
ops[i] = cp.formOp(n)
|
|
}
|
|
return ops
|
|
}
|
|
|
|
type formOp struct {
|
|
diag.Ranging
|
|
saveVarsOps []lvaluesOp
|
|
assignmentOps []effectOp
|
|
redirOps []effectOp
|
|
specialOpBody effectOpBody
|
|
headOp valuesOp
|
|
argOps []valuesOp
|
|
optsOp valuesOpBody
|
|
spaceyAssignOp effectOp
|
|
}
|
|
|
|
func (op *formOp) invoke(fm *Frame) (errRet error) {
|
|
// fm here is always a sub-frame created in compiler.pipeline, so it can
|
|
// be safely modified.
|
|
|
|
// Temporary assignment.
|
|
if len(op.saveVarsOps) > 0 {
|
|
// There is a temporary assignment.
|
|
// Save variables.
|
|
var saveVars []vars.Var
|
|
var saveVals []interface{}
|
|
for _, op := range op.saveVarsOps {
|
|
moreSaveVars, err := op.exec(fm)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
saveVars = append(saveVars, moreSaveVars...)
|
|
}
|
|
for i, v := range saveVars {
|
|
// XXX(xiaq): If the variable to save is a elemVariable, save
|
|
// the outermost variable instead.
|
|
if u := vars.HeadOfElement(v); u != nil {
|
|
v = u
|
|
saveVars[i] = v
|
|
}
|
|
val := v.Get()
|
|
saveVals = append(saveVals, val)
|
|
logger.Printf("saved %s = %s", v, val)
|
|
}
|
|
// Do assignment.
|
|
for _, subop := range op.assignmentOps {
|
|
err := subop.exec(fm)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
// Defer variable restoration. Will be executed even if an error
|
|
// occurs when evaling other part of the form.
|
|
defer func() {
|
|
for i, v := range saveVars {
|
|
val := saveVals[i]
|
|
if val == nil {
|
|
// XXX Old value is nonexistent. We should delete the
|
|
// variable. However, since the compiler now doesn't delete
|
|
// it, we don't delete it in the evaler either.
|
|
val = ""
|
|
}
|
|
err := v.Set(val)
|
|
if err != nil {
|
|
errRet = err
|
|
}
|
|
logger.Printf("restored %s = %s", v, val)
|
|
}
|
|
}()
|
|
}
|
|
|
|
// redirs
|
|
for _, redirOp := range op.redirOps {
|
|
err := redirOp.exec(fm)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
if op.specialOpBody != nil {
|
|
return op.specialOpBody.invoke(fm)
|
|
}
|
|
var headFn Callable
|
|
var args []interface{}
|
|
if op.headOp.body != nil {
|
|
var err error
|
|
// head
|
|
headFn, err = evalForCommand(fm, op.headOp, "command")
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// args
|
|
for _, argOp := range op.argOps {
|
|
moreArgs, err := argOp.exec(fm)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
args = append(args, moreArgs...)
|
|
}
|
|
}
|
|
|
|
// opts
|
|
// XXX This conversion should be avoided.
|
|
optValues, err := op.optsOp.invoke(fm)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
opts := optValues[0].(hashmap.Map)
|
|
convertedOpts := make(map[string]interface{})
|
|
for it := opts.Iterator(); it.HasElem(); it.Next() {
|
|
k, v := it.Elem()
|
|
if ks, ok := k.(string); ok {
|
|
convertedOpts[ks] = v
|
|
} else {
|
|
// TODO(xiaq): Point to the particular key.
|
|
return errs.BadValue{
|
|
What: "option key", Valid: "string", Actual: vals.Kind(k)}
|
|
}
|
|
}
|
|
|
|
if headFn != nil {
|
|
fm.traceback = fm.addTraceback(op)
|
|
err := headFn.Call(fm, args, convertedOpts)
|
|
if _, ok := err.(*Exception); ok {
|
|
return err
|
|
}
|
|
return &Exception{err, fm.traceback}
|
|
}
|
|
return op.spaceyAssignOp.exec(fm)
|
|
}
|
|
|
|
func evalForCommand(fm *Frame, op valuesOp, what string) (Callable, error) {
|
|
value, err := evalForValue(fm, op, what)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
switch value := value.(type) {
|
|
case Callable:
|
|
return value, nil
|
|
case string:
|
|
if util.DontSearch(value) {
|
|
return ExternalCmd{value}, nil
|
|
}
|
|
}
|
|
return nil, fm.errorp(op, errs.BadValue{
|
|
What: what,
|
|
Valid: "callable or string containing relative path",
|
|
Actual: vals.Kind(value)})
|
|
}
|
|
|
|
func allTrue(vs []interface{}) bool {
|
|
for _, v := range vs {
|
|
if !vals.Bool(v) {
|
|
return false
|
|
}
|
|
}
|
|
return true
|
|
}
|
|
|
|
func (cp *compiler) assignmentOp(n *parse.Assignment) effectOp {
|
|
valuesOp := cp.compoundOp(n.Right)
|
|
variablesOp, restOp := cp.lvaluesOp(n.Left)
|
|
return makeEffectOp(n, &assignmentOp{variablesOp, restOp, valuesOp})
|
|
}
|
|
|
|
func (cp *compiler) assignmentOps(ns []*parse.Assignment) []effectOp {
|
|
ops := make([]effectOp, len(ns))
|
|
for i, n := range ns {
|
|
ops[i] = cp.assignmentOp(n)
|
|
}
|
|
return ops
|
|
}
|
|
|
|
// ErrMoreThanOneRest is returned when the LHS of an assignment contains more
|
|
// than one rest variables.
|
|
var ErrMoreThanOneRest = errors.New("more than one @ lvalue")
|
|
|
|
type assignmentOp struct {
|
|
variablesOp lvaluesOp
|
|
restOp lvaluesOp
|
|
valuesOp valuesOp
|
|
}
|
|
|
|
func (op *assignmentOp) invoke(fm *Frame) (errRet error) {
|
|
variables, err := op.variablesOp.exec(fm)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
rest, err := op.restOp.exec(fm)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
values, err := op.valuesOp.exec(fm)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if len(rest) > 1 {
|
|
return ErrMoreThanOneRest
|
|
}
|
|
if len(rest) == 1 {
|
|
if len(values) < len(variables) {
|
|
return errs.ArityMismatch{
|
|
What: "assignment right-hand-side",
|
|
ValidLow: len(variables), ValidHigh: -1, Actual: len(values)}
|
|
}
|
|
} else {
|
|
if len(variables) != len(values) {
|
|
return errs.ArityMismatch{
|
|
What: "assignment right-hand-side",
|
|
ValidLow: len(variables), ValidHigh: len(variables), Actual: len(values)}
|
|
}
|
|
}
|
|
|
|
for i, variable := range variables {
|
|
err := variable.Set(values[i])
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
if len(rest) == 1 {
|
|
err := rest[0].Set(vals.MakeList(values[len(variables):]...))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (cp *compiler) literal(n *parse.Primary, msg string) string {
|
|
switch n.Type {
|
|
case parse.Bareword, parse.SingleQuoted, parse.DoubleQuoted:
|
|
return n.Value
|
|
default:
|
|
cp.errorpf(n, msg)
|
|
return ""
|
|
}
|
|
}
|
|
|
|
const defaultFileRedirPerm = 0644
|
|
|
|
// redir compiles a Redir into a op.
|
|
func (cp *compiler) redirOp(n *parse.Redir) effectOp {
|
|
var dstOp valuesOp
|
|
if n.Left != nil {
|
|
dstOp = cp.compoundOp(n.Left)
|
|
}
|
|
flag := makeFlag(n.Mode)
|
|
if flag == -1 {
|
|
// TODO: Record and get redirection sign position
|
|
cp.errorpf(n, "bad redirection sign")
|
|
}
|
|
return makeEffectOp(n,
|
|
&redirOp{dstOp, cp.compoundOp(n.Right), n.RightIsFd, n.Mode, flag})
|
|
}
|
|
|
|
func (cp *compiler) redirOps(ns []*parse.Redir) []effectOp {
|
|
ops := make([]effectOp, len(ns))
|
|
for i, n := range ns {
|
|
ops[i] = cp.redirOp(n)
|
|
}
|
|
return ops
|
|
}
|
|
|
|
func makeFlag(m parse.RedirMode) int {
|
|
switch m {
|
|
case parse.Read:
|
|
return os.O_RDONLY
|
|
case parse.Write:
|
|
return os.O_WRONLY | os.O_CREATE | os.O_TRUNC
|
|
case parse.ReadWrite:
|
|
return os.O_RDWR | os.O_CREATE
|
|
case parse.Append:
|
|
return os.O_WRONLY | os.O_CREATE | os.O_APPEND
|
|
default:
|
|
return -1
|
|
}
|
|
}
|
|
|
|
type redirOp struct {
|
|
dstOp valuesOp
|
|
srcOp valuesOp
|
|
srcIsFd bool
|
|
mode parse.RedirMode
|
|
flag int
|
|
}
|
|
|
|
type invalidFD struct{ fd int }
|
|
|
|
func (err invalidFD) Error() string { return fmt.Sprintf("invalid fd: %d", err.fd) }
|
|
|
|
func (op *redirOp) invoke(fm *Frame) error {
|
|
var dst int
|
|
if op.dstOp.body == nil {
|
|
// use default dst fd
|
|
switch op.mode {
|
|
case parse.Read:
|
|
dst = 0
|
|
case parse.Write, parse.ReadWrite, parse.Append:
|
|
dst = 1
|
|
default:
|
|
return fmt.Errorf("bad RedirMode; parser bug")
|
|
}
|
|
} else {
|
|
var err error
|
|
// dst must be a valid fd
|
|
dst, err = evalForFd(fm, op.dstOp, false, "redirection destination")
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
fm.growPorts(dst + 1)
|
|
fm.ports[dst].Close()
|
|
|
|
if op.srcIsFd {
|
|
src, err := evalForFd(fm, op.srcOp, true, "redirection source")
|
|
if err != nil {
|
|
return err
|
|
}
|
|
switch {
|
|
case src == -1:
|
|
// close
|
|
fm.ports[dst] = &Port{}
|
|
case src >= len(fm.ports) || fm.ports[src] == nil:
|
|
return invalidFD{src}
|
|
default:
|
|
fm.ports[dst] = fm.ports[src].Fork()
|
|
}
|
|
return nil
|
|
}
|
|
src, err := evalForValue(fm, op.srcOp, "redirection source")
|
|
if err != nil {
|
|
return err
|
|
}
|
|
switch src := src.(type) {
|
|
case string:
|
|
f, err := os.OpenFile(src, op.flag, defaultFileRedirPerm)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to open file %s: %s", vals.Repr(src, vals.NoPretty), err)
|
|
}
|
|
fm.ports[dst] = &Port{
|
|
File: f, Chan: BlackholeChan,
|
|
CloseFile: true,
|
|
}
|
|
case vals.File:
|
|
fm.ports[dst] = &Port{
|
|
File: src, Chan: BlackholeChan,
|
|
CloseFile: false,
|
|
}
|
|
case vals.Pipe:
|
|
var f *os.File
|
|
switch op.mode {
|
|
case parse.Read:
|
|
f = src.ReadEnd
|
|
case parse.Write:
|
|
f = src.WriteEnd
|
|
default:
|
|
return errors.New("can only use < or > with pipes")
|
|
}
|
|
fm.ports[dst] = &Port{
|
|
File: f, Chan: BlackholeChan,
|
|
CloseFile: false,
|
|
}
|
|
default:
|
|
return fm.errorp(op.srcOp, errs.BadValue{
|
|
What: "redirection source",
|
|
Valid: "string, file or pipe", Actual: vals.Kind(src)})
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func evalForFd(fm *Frame, op valuesOp, closeOK bool, what string) (int, error) {
|
|
value, err := evalForValue(fm, op, what)
|
|
if err != nil {
|
|
return -1, err
|
|
}
|
|
switch value {
|
|
case "stdin":
|
|
return 0, nil
|
|
case "stdout":
|
|
return 1, nil
|
|
case "stderr":
|
|
return 2, nil
|
|
}
|
|
var fd int
|
|
if vals.ScanToGo(value, &fd) == nil {
|
|
return fd, nil
|
|
} else if value == "-" && closeOK {
|
|
return -1, nil
|
|
}
|
|
valid := "fd name or number"
|
|
if closeOK {
|
|
valid = "fd name or number or '-'"
|
|
}
|
|
return -1, fm.errorp(op, errs.BadValue{
|
|
What: what, Valid: valid, Actual: vals.Repr(value, vals.NoPretty)})
|
|
}
|
|
|
|
type seqOp struct{ subops []effectOp }
|
|
|
|
func (op seqOp) invoke(fm *Frame) error {
|
|
for _, subop := range op.subops {
|
|
err := subop.exec(fm)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|