2016-02-20 00:21:51 +08:00
|
|
|
package eval
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bufio"
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"os"
|
|
|
|
"path"
|
|
|
|
"strings"
|
2017-08-13 06:03:28 +08:00
|
|
|
"sync"
|
2016-02-20 00:21:51 +08:00
|
|
|
|
2018-01-01 04:31:45 +08:00
|
|
|
"github.com/elves/elvish/eval/types"
|
2016-02-20 00:21:51 +08:00
|
|
|
"github.com/elves/elvish/glob"
|
|
|
|
"github.com/elves/elvish/parse"
|
2018-01-21 09:14:39 +08:00
|
|
|
"github.com/elves/elvish/util"
|
2016-02-20 00:21:51 +08:00
|
|
|
)
|
|
|
|
|
2016-02-22 02:25:49 +08:00
|
|
|
var outputCaptureBufferSize = 16
|
|
|
|
|
2018-01-21 06:34:38 +08:00
|
|
|
// ValuesOp is an operation on an Frame that produce Value's.
|
2016-02-20 07:48:13 +08:00
|
|
|
type ValuesOp struct {
|
2018-01-21 21:20:06 +08:00
|
|
|
Body ValuesOpBody
|
2016-02-20 07:48:13 +08:00
|
|
|
Begin, End int
|
|
|
|
}
|
|
|
|
|
2018-01-21 21:20:06 +08:00
|
|
|
// ValuesOpBody is the body of ValuesOp.
|
|
|
|
type ValuesOpBody interface {
|
2018-01-30 01:39:41 +08:00
|
|
|
Invoke(*Frame) ([]interface{}, error)
|
2018-01-21 21:20:06 +08:00
|
|
|
}
|
2017-06-11 07:04:53 +08:00
|
|
|
|
|
|
|
// Exec executes a ValuesOp and produces Value's.
|
2018-01-30 01:39:41 +08:00
|
|
|
func (op ValuesOp) Exec(ec *Frame) ([]interface{}, error) {
|
2016-02-20 07:48:13 +08:00
|
|
|
ec.begin, ec.end = op.Begin, op.End
|
2018-01-21 21:20:06 +08:00
|
|
|
return op.Body.Invoke(ec)
|
2016-02-20 07:48:13 +08:00
|
|
|
}
|
|
|
|
|
2018-01-21 21:20:06 +08:00
|
|
|
func (cp *compiler) compound(n *parse.Compound) ValuesOpBody {
|
2016-02-20 00:21:51 +08:00
|
|
|
if len(n.Indexings) == 0 {
|
|
|
|
return literalStr("")
|
|
|
|
}
|
|
|
|
|
|
|
|
tilde := false
|
|
|
|
indexings := n.Indexings
|
|
|
|
|
|
|
|
if n.Indexings[0].Head.Type == parse.Tilde {
|
|
|
|
// A lone ~.
|
|
|
|
if len(n.Indexings) == 1 {
|
2018-01-30 01:39:41 +08:00
|
|
|
return funcValuesOp(func(ec *Frame) ([]interface{}, error) {
|
2018-01-21 09:14:39 +08:00
|
|
|
home, err := util.GetHome("")
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2018-01-30 01:39:41 +08:00
|
|
|
return []interface{}{home}, nil
|
2018-01-21 21:20:06 +08:00
|
|
|
})
|
2016-02-20 00:21:51 +08:00
|
|
|
}
|
|
|
|
tilde = true
|
|
|
|
indexings = indexings[1:]
|
|
|
|
}
|
|
|
|
|
2018-01-21 21:20:06 +08:00
|
|
|
return compoundOp{tilde, cp.indexingOps(indexings)}
|
|
|
|
}
|
|
|
|
|
|
|
|
type compoundOp struct {
|
|
|
|
tilde bool
|
|
|
|
subops []ValuesOp
|
|
|
|
}
|
|
|
|
|
2018-01-30 01:39:41 +08:00
|
|
|
func (op compoundOp) Invoke(ec *Frame) ([]interface{}, error) {
|
2018-01-21 21:20:06 +08:00
|
|
|
// Accumulator.
|
|
|
|
vs, err := op.subops[0].Exec(ec)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2016-02-20 00:21:51 +08:00
|
|
|
|
2018-01-21 21:20:06 +08:00
|
|
|
for _, subop := range op.subops[1:] {
|
|
|
|
us, err := subop.Exec(ec)
|
2018-01-21 09:14:39 +08:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2018-01-21 21:20:06 +08:00
|
|
|
vs, err = outerProduct(vs, us, cat)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
2016-02-20 00:21:51 +08:00
|
|
|
}
|
2018-01-21 21:20:06 +08:00
|
|
|
}
|
|
|
|
if op.tilde {
|
2018-01-30 01:39:41 +08:00
|
|
|
newvs := make([]interface{}, len(vs))
|
2018-01-21 21:20:06 +08:00
|
|
|
for i, v := range vs {
|
|
|
|
newvs[i] = doTilde(v)
|
2016-02-20 00:21:51 +08:00
|
|
|
}
|
2018-01-21 21:20:06 +08:00
|
|
|
vs = newvs
|
|
|
|
}
|
|
|
|
hasGlob := false
|
|
|
|
for _, v := range vs {
|
|
|
|
if _, ok := v.(GlobPattern); ok {
|
|
|
|
hasGlob = true
|
|
|
|
break
|
2016-02-20 00:21:51 +08:00
|
|
|
}
|
2018-01-21 21:20:06 +08:00
|
|
|
}
|
|
|
|
if hasGlob {
|
2018-01-30 01:39:41 +08:00
|
|
|
newvs := make([]interface{}, 0, len(vs))
|
2018-01-21 21:20:06 +08:00
|
|
|
for _, v := range vs {
|
|
|
|
if gp, ok := v.(GlobPattern); ok {
|
|
|
|
// Logger.Printf("globbing %v", gp)
|
|
|
|
newvs = append(newvs, doGlob(gp, ec.Interrupts())...)
|
|
|
|
} else {
|
|
|
|
newvs = append(newvs, v)
|
2016-02-20 00:21:51 +08:00
|
|
|
}
|
|
|
|
}
|
2018-01-21 21:20:06 +08:00
|
|
|
vs = newvs
|
2016-02-20 00:21:51 +08:00
|
|
|
}
|
2018-01-21 21:20:06 +08:00
|
|
|
return vs, nil
|
2016-02-20 00:21:51 +08:00
|
|
|
}
|
|
|
|
|
2018-01-30 01:39:41 +08:00
|
|
|
func cat(lhs, rhs interface{}) (interface{}, error) {
|
2016-02-20 00:21:51 +08:00
|
|
|
switch lhs := lhs.(type) {
|
2018-01-25 09:40:15 +08:00
|
|
|
case string:
|
2016-02-20 00:21:51 +08:00
|
|
|
switch rhs := rhs.(type) {
|
2018-01-25 09:40:15 +08:00
|
|
|
case string:
|
2018-01-21 19:17:17 +08:00
|
|
|
return lhs + rhs, nil
|
2016-02-20 00:21:51 +08:00
|
|
|
case GlobPattern:
|
2018-01-25 09:40:15 +08:00
|
|
|
segs := stringToSegments(lhs)
|
2016-02-20 00:21:51 +08:00
|
|
|
// We know rhs contains exactly one segment.
|
|
|
|
segs = append(segs, rhs.Segments[0])
|
2018-01-21 19:17:17 +08:00
|
|
|
return GlobPattern{glob.Pattern{segs, ""}, rhs.Flags, rhs.Buts}, nil
|
2016-02-20 00:21:51 +08:00
|
|
|
}
|
|
|
|
case GlobPattern:
|
|
|
|
// NOTE Modifies lhs in place.
|
|
|
|
switch rhs := rhs.(type) {
|
2018-01-25 09:40:15 +08:00
|
|
|
case string:
|
|
|
|
lhs.append(stringToSegments(rhs)...)
|
2018-01-21 19:17:17 +08:00
|
|
|
return lhs, nil
|
2016-02-20 00:21:51 +08:00
|
|
|
case GlobPattern:
|
|
|
|
// We know rhs contains exactly one segment.
|
|
|
|
lhs.append(rhs.Segments[0])
|
2017-02-03 05:23:20 +08:00
|
|
|
lhs.Flags |= rhs.Flags
|
2017-02-16 12:10:31 +08:00
|
|
|
lhs.Buts = append(lhs.Buts, rhs.Buts...)
|
2018-01-21 19:17:17 +08:00
|
|
|
return lhs, nil
|
2016-02-20 00:21:51 +08:00
|
|
|
}
|
|
|
|
}
|
2018-01-25 07:57:58 +08:00
|
|
|
return nil, fmt.Errorf("unsupported concat: %s and %s",
|
|
|
|
types.Kind(lhs), types.Kind(rhs))
|
2016-02-20 00:21:51 +08:00
|
|
|
}
|
|
|
|
|
2018-01-30 01:39:41 +08:00
|
|
|
func outerProduct(vs []interface{}, us []interface{}, f func(interface{}, interface{}) (interface{}, error)) ([]interface{}, error) {
|
|
|
|
ws := make([]interface{}, len(vs)*len(us))
|
2016-02-20 00:21:51 +08:00
|
|
|
nu := len(us)
|
|
|
|
for i, v := range vs {
|
|
|
|
for j, u := range us {
|
2018-01-21 19:17:17 +08:00
|
|
|
var err error
|
|
|
|
ws[i*nu+j], err = f(v, u)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2016-02-20 00:21:51 +08:00
|
|
|
}
|
|
|
|
}
|
2018-01-21 19:17:17 +08:00
|
|
|
return ws, nil
|
2016-02-20 00:21:51 +08:00
|
|
|
}
|
|
|
|
|
2017-06-11 07:04:53 +08:00
|
|
|
// Errors thrown when globbing.
|
2016-02-22 23:05:14 +08:00
|
|
|
var (
|
|
|
|
ErrBadGlobPattern = errors.New("bad GlobPattern; elvish bug")
|
|
|
|
ErrCannotDetermineUsername = errors.New("cannot determine user name from glob pattern")
|
|
|
|
)
|
|
|
|
|
2018-01-30 01:39:41 +08:00
|
|
|
func doTilde(v interface{}) interface{} {
|
2016-02-20 00:21:51 +08:00
|
|
|
switch v := v.(type) {
|
2018-01-25 09:40:15 +08:00
|
|
|
case string:
|
|
|
|
s := v
|
2016-02-20 00:21:51 +08:00
|
|
|
i := strings.Index(s, "/")
|
|
|
|
var uname, rest string
|
|
|
|
if i == -1 {
|
|
|
|
uname = s
|
|
|
|
} else {
|
|
|
|
uname = s[:i]
|
|
|
|
rest = s[i+1:]
|
|
|
|
}
|
|
|
|
dir := mustGetHome(uname)
|
2018-01-25 09:40:15 +08:00
|
|
|
return path.Join(dir, rest)
|
2016-02-20 00:21:51 +08:00
|
|
|
case GlobPattern:
|
2016-02-22 23:05:14 +08:00
|
|
|
if len(v.Segments) == 0 {
|
|
|
|
throw(ErrBadGlobPattern)
|
2016-02-20 00:21:51 +08:00
|
|
|
}
|
2017-02-03 03:15:42 +08:00
|
|
|
switch seg := v.Segments[0].(type) {
|
2016-02-22 23:05:14 +08:00
|
|
|
case glob.Literal:
|
2017-02-03 03:15:42 +08:00
|
|
|
s := seg.Data
|
2016-02-22 23:05:14 +08:00
|
|
|
// Find / in the first segment to determine the username.
|
|
|
|
i := strings.Index(s, "/")
|
|
|
|
if i == -1 {
|
|
|
|
throw(ErrCannotDetermineUsername)
|
|
|
|
}
|
|
|
|
uname := s[:i]
|
|
|
|
dir := mustGetHome(uname)
|
|
|
|
// Replace ~uname in first segment with the found path.
|
2017-02-03 03:15:42 +08:00
|
|
|
v.Segments[0] = glob.Literal{dir + s[i:]}
|
2016-02-22 23:05:14 +08:00
|
|
|
case glob.Slash:
|
|
|
|
v.DirOverride = mustGetHome("")
|
|
|
|
default:
|
|
|
|
throw(ErrCannotDetermineUsername)
|
2016-02-20 00:21:51 +08:00
|
|
|
}
|
|
|
|
return v
|
|
|
|
default:
|
2018-01-25 07:57:58 +08:00
|
|
|
throw(fmt.Errorf("tilde doesn't work on value of type %s", types.Kind(v)))
|
2016-02-20 00:21:51 +08:00
|
|
|
panic("unreachable")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-21 21:20:06 +08:00
|
|
|
func (cp *compiler) array(n *parse.Array) ValuesOpBody {
|
|
|
|
return seqValuesOp{cp.compoundOps(n.Compounds)}
|
2016-02-20 00:21:51 +08:00
|
|
|
}
|
|
|
|
|
2018-01-21 21:20:06 +08:00
|
|
|
func (cp *compiler) indexing(n *parse.Indexing) ValuesOpBody {
|
2016-02-20 00:21:51 +08:00
|
|
|
if len(n.Indicies) == 0 {
|
|
|
|
return cp.primary(n.Head)
|
|
|
|
}
|
|
|
|
|
2018-01-21 21:20:06 +08:00
|
|
|
return &indexingOp{cp.primaryOp(n.Head), cp.arrayOps(n.Indicies)}
|
2016-02-20 00:21:51 +08:00
|
|
|
}
|
|
|
|
|
2018-01-21 21:20:06 +08:00
|
|
|
type indexingOp struct {
|
|
|
|
headOp ValuesOp
|
|
|
|
indexOps []ValuesOp
|
2016-02-20 00:21:51 +08:00
|
|
|
}
|
|
|
|
|
2018-01-30 01:39:41 +08:00
|
|
|
func (op *indexingOp) Invoke(ec *Frame) ([]interface{}, error) {
|
2018-01-21 21:20:06 +08:00
|
|
|
vs, err := op.headOp.Exec(ec)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
for _, indexOp := range op.indexOps {
|
|
|
|
indicies, err := indexOp.Exec(ec)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
2016-02-20 00:21:51 +08:00
|
|
|
}
|
2018-01-30 01:39:41 +08:00
|
|
|
newvs := make([]interface{}, 0, len(vs)*len(indicies))
|
2018-01-21 21:20:06 +08:00
|
|
|
for _, v := range vs {
|
|
|
|
for _, index := range indicies {
|
2018-01-25 08:26:46 +08:00
|
|
|
result, err := types.Index(v, index)
|
2018-01-21 21:20:06 +08:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
newvs = append(newvs, result)
|
2016-02-20 00:21:51 +08:00
|
|
|
}
|
|
|
|
}
|
2018-01-21 21:20:06 +08:00
|
|
|
vs = newvs
|
2016-02-20 00:21:51 +08:00
|
|
|
}
|
2018-01-21 21:20:06 +08:00
|
|
|
return vs, nil
|
2016-02-20 00:21:51 +08:00
|
|
|
}
|
|
|
|
|
2018-01-21 21:20:06 +08:00
|
|
|
func (cp *compiler) primary(n *parse.Primary) ValuesOpBody {
|
2016-02-20 00:21:51 +08:00
|
|
|
switch n.Type {
|
|
|
|
case parse.Bareword, parse.SingleQuoted, parse.DoubleQuoted:
|
|
|
|
return literalStr(n.Value)
|
|
|
|
case parse.Variable:
|
2018-01-30 01:29:38 +08:00
|
|
|
explode, ns, name := ParseVariableRef(n.Value)
|
2018-01-21 21:20:06 +08:00
|
|
|
if !cp.registerVariableGet(ns, name) {
|
2016-02-20 04:16:23 +08:00
|
|
|
cp.errorf("variable $%s not found", n.Value)
|
2016-02-20 00:21:51 +08:00
|
|
|
}
|
2018-01-21 21:20:06 +08:00
|
|
|
return &variableOp{explode, ns, name}
|
2016-02-20 00:21:51 +08:00
|
|
|
case parse.Wildcard:
|
2017-06-27 01:47:10 +08:00
|
|
|
seg, err := wildcardToSegment(n.SourceText())
|
|
|
|
if err != nil {
|
|
|
|
cp.errorf("%s", err)
|
|
|
|
}
|
2018-01-30 01:39:41 +08:00
|
|
|
vs := []interface{}{
|
2017-06-27 01:47:10 +08:00
|
|
|
GlobPattern{glob.Pattern{[]glob.Segment{seg}, ""}, 0, nil}}
|
2018-01-21 09:14:39 +08:00
|
|
|
return literalValues(vs...)
|
2016-02-20 00:21:51 +08:00
|
|
|
case parse.Tilde:
|
2016-02-20 04:16:23 +08:00
|
|
|
cp.errorf("compiler bug: Tilde not handled in .compound")
|
2016-02-20 00:21:51 +08:00
|
|
|
return literalStr("~")
|
2017-02-18 12:43:27 +08:00
|
|
|
case parse.ExceptionCapture:
|
2018-01-21 21:20:06 +08:00
|
|
|
return exceptionCaptureOp{cp.chunkOp(n.Chunk)}
|
2016-02-20 00:21:51 +08:00
|
|
|
case parse.OutputCapture:
|
2018-01-21 21:20:06 +08:00
|
|
|
return outputCaptureOp{cp.chunkOp(n.Chunk)}
|
2016-02-20 00:21:51 +08:00
|
|
|
case parse.List:
|
2017-08-06 03:49:50 +08:00
|
|
|
return cp.list(n)
|
2016-02-20 00:21:51 +08:00
|
|
|
case parse.Lambda:
|
|
|
|
return cp.lambda(n)
|
|
|
|
case parse.Map:
|
|
|
|
return cp.map_(n)
|
|
|
|
case parse.Braced:
|
|
|
|
return cp.braced(n)
|
|
|
|
default:
|
2016-02-20 04:16:23 +08:00
|
|
|
cp.errorf("bad PrimaryType; parser bug")
|
2016-02-20 00:21:51 +08:00
|
|
|
return literalStr(n.SourceText())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-21 21:20:06 +08:00
|
|
|
type variableOp struct {
|
|
|
|
explode bool
|
|
|
|
ns string
|
|
|
|
name string
|
|
|
|
}
|
|
|
|
|
2018-01-30 01:39:41 +08:00
|
|
|
func (op variableOp) Invoke(ec *Frame) ([]interface{}, error) {
|
2018-01-21 21:20:06 +08:00
|
|
|
variable := ec.ResolveVar(op.ns, op.name)
|
|
|
|
if variable == nil {
|
|
|
|
return nil, fmt.Errorf("variable $%s:%s not found", op.ns, op.name)
|
|
|
|
}
|
|
|
|
value := variable.Get()
|
|
|
|
if op.explode {
|
2018-01-25 08:48:31 +08:00
|
|
|
return types.Collect(value)
|
2016-03-08 09:01:58 +08:00
|
|
|
}
|
2018-01-30 01:39:41 +08:00
|
|
|
return []interface{}{value}, nil
|
2016-03-08 09:01:58 +08:00
|
|
|
}
|
|
|
|
|
2018-01-21 21:20:06 +08:00
|
|
|
func (cp *compiler) list(n *parse.Primary) ValuesOpBody {
|
2018-01-21 21:24:00 +08:00
|
|
|
return listOp{cp.compoundOps(n.Elements)}
|
2018-01-21 21:20:06 +08:00
|
|
|
}
|
|
|
|
|
2018-01-21 21:24:00 +08:00
|
|
|
type listOp struct{ subops []ValuesOp }
|
2018-01-21 21:20:06 +08:00
|
|
|
|
2018-01-30 01:39:41 +08:00
|
|
|
func (op listOp) Invoke(fm *Frame) ([]interface{}, error) {
|
2018-01-28 01:26:22 +08:00
|
|
|
list := types.EmptyList
|
2018-01-21 21:24:00 +08:00
|
|
|
for _, subop := range op.subops {
|
|
|
|
moreValues, err := subop.Exec(fm)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
for _, moreValue := range moreValues {
|
2018-01-28 01:26:22 +08:00
|
|
|
list = list.Cons(moreValue)
|
2018-01-21 21:24:00 +08:00
|
|
|
}
|
2016-02-20 00:21:51 +08:00
|
|
|
}
|
2018-01-30 01:39:41 +08:00
|
|
|
return []interface{}{list}, nil
|
2016-02-20 00:21:51 +08:00
|
|
|
}
|
|
|
|
|
2018-01-21 21:20:06 +08:00
|
|
|
type exceptionCaptureOp struct{ subop Op }
|
|
|
|
|
2018-01-30 01:39:41 +08:00
|
|
|
func (op exceptionCaptureOp) Invoke(fm *Frame) ([]interface{}, error) {
|
2018-01-21 21:20:06 +08:00
|
|
|
err := fm.PEval(op.subop)
|
|
|
|
if err == nil {
|
2018-01-30 01:39:41 +08:00
|
|
|
return []interface{}{OK}, nil
|
2016-02-20 00:21:51 +08:00
|
|
|
}
|
2018-01-30 01:39:41 +08:00
|
|
|
return []interface{}{err.(*Exception)}, nil
|
2016-02-20 00:21:51 +08:00
|
|
|
}
|
|
|
|
|
2018-01-21 21:20:06 +08:00
|
|
|
type outputCaptureOp struct{ subop Op }
|
|
|
|
|
2018-01-30 01:39:41 +08:00
|
|
|
func (op outputCaptureOp) Invoke(fm *Frame) ([]interface{}, error) {
|
2018-01-21 21:20:06 +08:00
|
|
|
return pcaptureOutput(fm, op.subop)
|
2016-10-22 21:02:27 +08:00
|
|
|
}
|
|
|
|
|
2018-01-30 01:39:41 +08:00
|
|
|
func pcaptureOutput(ec *Frame, op Op) ([]interface{}, error) {
|
|
|
|
vs := []interface{}{}
|
2017-08-13 06:03:28 +08:00
|
|
|
var m sync.Mutex
|
2018-01-30 01:39:41 +08:00
|
|
|
valueCb := func(ch <-chan interface{}) {
|
2016-02-20 00:21:51 +08:00
|
|
|
for v := range ch {
|
2017-08-13 06:03:28 +08:00
|
|
|
m.Lock()
|
2016-02-20 00:21:51 +08:00
|
|
|
vs = append(vs, v)
|
2017-08-13 06:03:28 +08:00
|
|
|
m.Unlock()
|
2016-02-20 00:21:51 +08:00
|
|
|
}
|
2017-08-13 06:03:28 +08:00
|
|
|
}
|
|
|
|
bytesCb := func(r *os.File) {
|
|
|
|
buffered := bufio.NewReader(r)
|
2016-02-20 00:21:51 +08:00
|
|
|
for {
|
2017-08-13 06:03:28 +08:00
|
|
|
line, err := buffered.ReadString('\n')
|
2017-07-11 01:00:32 +08:00
|
|
|
if line != "" {
|
2018-01-25 09:40:15 +08:00
|
|
|
v := strings.TrimSuffix(line, "\n")
|
2017-08-13 06:03:28 +08:00
|
|
|
m.Lock()
|
|
|
|
vs = append(vs, v)
|
|
|
|
m.Unlock()
|
2017-07-11 01:00:32 +08:00
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
if err != io.EOF {
|
|
|
|
logger.Println("error on reading:", err)
|
|
|
|
}
|
2016-02-20 00:21:51 +08:00
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
2017-08-13 06:03:28 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
err := pcaptureOutputInner(ec, op, valueCb, bytesCb)
|
|
|
|
return vs, err
|
|
|
|
}
|
|
|
|
|
2018-01-30 01:39:41 +08:00
|
|
|
func pcaptureOutputInner(ec *Frame, op Op, valuesCb func(<-chan interface{}), bytesCb func(*os.File)) error {
|
2017-08-13 06:03:28 +08:00
|
|
|
|
2017-09-23 05:06:27 +08:00
|
|
|
newEc := ec.fork("[output capture]")
|
2017-08-13 06:03:28 +08:00
|
|
|
|
2018-01-30 01:39:41 +08:00
|
|
|
ch := make(chan interface{}, outputCaptureBufferSize)
|
2017-08-13 06:03:28 +08:00
|
|
|
pipeRead, pipeWrite, err := os.Pipe()
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("failed to create pipe: %v", err)
|
|
|
|
}
|
|
|
|
newEc.ports[1] = &Port{
|
|
|
|
Chan: ch, CloseChan: true,
|
|
|
|
File: pipeWrite, CloseFile: true,
|
|
|
|
}
|
|
|
|
|
|
|
|
bytesCollected := make(chan struct{})
|
|
|
|
chCollected := make(chan struct{})
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
valuesCb(ch)
|
|
|
|
close(chCollected)
|
|
|
|
}()
|
|
|
|
go func() {
|
|
|
|
bytesCb(pipeRead)
|
|
|
|
pipeRead.Close()
|
|
|
|
close(bytesCollected)
|
2016-02-20 00:21:51 +08:00
|
|
|
}()
|
|
|
|
|
2016-10-22 21:02:27 +08:00
|
|
|
err = newEc.PEval(op)
|
2016-02-20 00:21:51 +08:00
|
|
|
|
2017-08-13 06:03:28 +08:00
|
|
|
ClosePorts(newEc.ports)
|
2016-02-20 00:21:51 +08:00
|
|
|
<-bytesCollected
|
|
|
|
<-chCollected
|
|
|
|
|
2017-08-13 06:03:28 +08:00
|
|
|
return err
|
2016-02-20 00:21:51 +08:00
|
|
|
}
|
|
|
|
|
2018-01-21 21:20:06 +08:00
|
|
|
func (cp *compiler) lambda(n *parse.Primary) ValuesOpBody {
|
2017-08-06 04:27:44 +08:00
|
|
|
// Parse signature.
|
|
|
|
var (
|
|
|
|
argNames []string
|
|
|
|
restArgName string
|
|
|
|
optNames []string
|
|
|
|
optDefaultOps []ValuesOp
|
|
|
|
)
|
2017-08-06 03:49:50 +08:00
|
|
|
if len(n.Elements) > 0 {
|
2017-08-06 04:27:44 +08:00
|
|
|
// Argument list.
|
2017-08-06 03:49:50 +08:00
|
|
|
argNames = make([]string, len(n.Elements))
|
|
|
|
for i, arg := range n.Elements {
|
2017-08-06 04:27:44 +08:00
|
|
|
qname := mustString(cp, arg, "argument name must be literal string")
|
2018-01-30 01:29:38 +08:00
|
|
|
explode, ns, name := ParseVariableRef(qname)
|
2016-02-26 08:54:27 +08:00
|
|
|
if ns != "" {
|
2017-08-06 04:27:44 +08:00
|
|
|
cp.errorpf(arg.Begin(), arg.End(), "argument name must be unqualified")
|
2016-02-26 08:54:27 +08:00
|
|
|
}
|
|
|
|
if name == "" {
|
|
|
|
cp.errorpf(arg.Begin(), arg.End(), "argument name must not be empty")
|
|
|
|
}
|
2016-10-13 14:10:10 +08:00
|
|
|
if explode {
|
2017-08-06 03:49:50 +08:00
|
|
|
if i != len(n.Elements)-1 {
|
2016-02-26 08:54:27 +08:00
|
|
|
cp.errorpf(arg.Begin(), arg.End(), "only the last argument may have @")
|
|
|
|
}
|
2017-08-06 04:27:44 +08:00
|
|
|
restArgName = name
|
2016-02-26 08:54:27 +08:00
|
|
|
argNames = argNames[:i]
|
|
|
|
} else {
|
|
|
|
argNames[i] = name
|
|
|
|
}
|
2016-02-20 00:21:51 +08:00
|
|
|
}
|
|
|
|
}
|
2017-08-06 04:27:44 +08:00
|
|
|
if len(n.MapPairs) > 0 {
|
|
|
|
optNames = make([]string, len(n.MapPairs))
|
|
|
|
optDefaultOps = make([]ValuesOp, len(n.MapPairs))
|
|
|
|
for i, opt := range n.MapPairs {
|
|
|
|
qname := mustString(cp, opt.Key, "option name must be literal string")
|
2018-01-30 01:29:38 +08:00
|
|
|
_, ns, name := ParseVariableRef(qname)
|
2017-08-06 04:27:44 +08:00
|
|
|
if ns != "" {
|
|
|
|
cp.errorpf(opt.Key.Begin(), opt.Key.End(), "option name must be unqualified")
|
|
|
|
}
|
|
|
|
if name == "" {
|
|
|
|
cp.errorpf(opt.Key.Begin(), opt.Key.End(), "option name must not be empty")
|
|
|
|
}
|
|
|
|
optNames[i] = name
|
|
|
|
if opt.Value == nil {
|
|
|
|
cp.errorpf(opt.End(), opt.End(), "option must have default value")
|
|
|
|
} else {
|
|
|
|
optDefaultOps[i] = cp.compoundOp(opt.Value)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-02-20 00:21:51 +08:00
|
|
|
|
|
|
|
thisScope := cp.pushScope()
|
|
|
|
for _, argName := range argNames {
|
2017-12-24 07:51:32 +08:00
|
|
|
thisScope.set(argName)
|
2016-02-20 00:21:51 +08:00
|
|
|
}
|
2017-08-06 04:27:44 +08:00
|
|
|
if restArgName != "" {
|
2017-12-24 07:51:32 +08:00
|
|
|
thisScope.set(restArgName)
|
2017-08-06 04:27:44 +08:00
|
|
|
}
|
|
|
|
for _, optName := range optNames {
|
2017-12-24 07:51:32 +08:00
|
|
|
thisScope.set(optName)
|
2016-02-26 08:54:27 +08:00
|
|
|
}
|
2017-12-24 07:51:32 +08:00
|
|
|
thisScope.set("opts")
|
2017-08-06 04:27:44 +08:00
|
|
|
|
2018-01-21 21:20:06 +08:00
|
|
|
subop := cp.chunkOp(n.Chunk)
|
2017-08-06 04:27:44 +08:00
|
|
|
|
|
|
|
// XXX The fiddlings with cp.capture is error-prone.
|
2016-02-20 00:21:51 +08:00
|
|
|
capture := cp.capture
|
2017-12-24 07:51:32 +08:00
|
|
|
cp.capture = make(staticNs)
|
2016-02-20 00:21:51 +08:00
|
|
|
cp.popScope()
|
|
|
|
|
2017-12-24 07:51:32 +08:00
|
|
|
for name := range capture {
|
|
|
|
cp.registerVariableGetQname(name)
|
2017-09-21 07:10:21 +08:00
|
|
|
}
|
2016-02-20 00:21:51 +08:00
|
|
|
|
2018-01-21 21:20:06 +08:00
|
|
|
return &lambdaOp{argNames, restArgName, optNames, optDefaultOps, capture, subop, cp.srcMeta}
|
|
|
|
}
|
2016-10-11 21:37:27 +08:00
|
|
|
|
2018-01-21 21:20:06 +08:00
|
|
|
type lambdaOp struct {
|
|
|
|
argNames []string
|
|
|
|
restArgName string
|
|
|
|
optNames []string
|
|
|
|
optDefaultOps []ValuesOp
|
|
|
|
capture staticNs
|
|
|
|
subop Op
|
|
|
|
srcMeta *Source
|
|
|
|
}
|
|
|
|
|
2018-01-30 01:39:41 +08:00
|
|
|
func (op *lambdaOp) Invoke(fm *Frame) ([]interface{}, error) {
|
2018-01-21 21:20:06 +08:00
|
|
|
evCapture := make(Ns)
|
|
|
|
for name := range op.capture {
|
|
|
|
evCapture[name] = fm.ResolveVar("", name)
|
|
|
|
}
|
2018-01-30 01:39:41 +08:00
|
|
|
optDefaults := make([]interface{}, len(op.optDefaultOps))
|
2018-01-21 21:20:06 +08:00
|
|
|
for i, op := range op.optDefaultOps {
|
|
|
|
defaultValue := fm.ExecAndUnwrap("option default value", op).One().Any()
|
|
|
|
optDefaults[i] = defaultValue
|
2016-02-20 00:21:51 +08:00
|
|
|
}
|
2018-01-21 21:20:06 +08:00
|
|
|
// XXX(xiaq): Capture uses.
|
2018-01-30 01:39:41 +08:00
|
|
|
return []interface{}{&Closure{op.argNames, op.restArgName, op.optNames, optDefaults, op.subop, evCapture, op.srcMeta}}, nil
|
2016-02-20 00:21:51 +08:00
|
|
|
}
|
|
|
|
|
2018-01-21 21:20:06 +08:00
|
|
|
func (cp *compiler) map_(n *parse.Primary) ValuesOpBody {
|
2016-09-15 05:34:10 +08:00
|
|
|
return cp.mapPairs(n.MapPairs)
|
|
|
|
}
|
|
|
|
|
2018-01-21 21:20:06 +08:00
|
|
|
func (cp *compiler) mapPairs(pairs []*parse.MapPair) ValuesOpBody {
|
2016-09-15 05:34:10 +08:00
|
|
|
npairs := len(pairs)
|
2016-02-20 00:21:51 +08:00
|
|
|
keysOps := make([]ValuesOp, npairs)
|
|
|
|
valuesOps := make([]ValuesOp, npairs)
|
2016-02-20 07:48:13 +08:00
|
|
|
begins, ends := make([]int, npairs), make([]int, npairs)
|
2016-09-15 05:34:10 +08:00
|
|
|
for i, pair := range pairs {
|
2016-02-20 07:48:13 +08:00
|
|
|
keysOps[i] = cp.compoundOp(pair.Key)
|
2016-02-20 00:21:51 +08:00
|
|
|
if pair.Value == nil {
|
2016-02-20 07:48:13 +08:00
|
|
|
p := pair.End()
|
2018-01-27 23:58:27 +08:00
|
|
|
valuesOps[i] = ValuesOp{literalValues(true), p, p}
|
2016-02-20 00:21:51 +08:00
|
|
|
} else {
|
2016-09-15 05:34:10 +08:00
|
|
|
valuesOps[i] = cp.compoundOp(pairs[i].Value)
|
2016-02-20 00:21:51 +08:00
|
|
|
}
|
2016-02-20 07:48:13 +08:00
|
|
|
begins[i], ends[i] = pair.Begin(), pair.End()
|
2016-02-20 00:21:51 +08:00
|
|
|
}
|
2018-01-21 21:20:06 +08:00
|
|
|
return &mapPairsOp{keysOps, valuesOps, begins, ends}
|
|
|
|
}
|
|
|
|
|
|
|
|
type mapPairsOp struct {
|
|
|
|
keysOps []ValuesOp
|
|
|
|
valuesOps []ValuesOp
|
|
|
|
begins []int
|
|
|
|
ends []int
|
|
|
|
}
|
|
|
|
|
2018-01-30 01:39:41 +08:00
|
|
|
func (op *mapPairsOp) Invoke(fm *Frame) ([]interface{}, error) {
|
2018-01-29 22:36:43 +08:00
|
|
|
m := types.EmptyMap
|
2018-01-21 21:20:06 +08:00
|
|
|
for i := range op.keysOps {
|
|
|
|
keys, err := op.keysOps[i].Exec(fm)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
values, err := op.valuesOps[i].Exec(fm)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if len(keys) != len(values) {
|
|
|
|
fm.errorpf(op.begins[i], op.ends[i],
|
|
|
|
"%d keys but %d values", len(keys), len(values))
|
|
|
|
}
|
|
|
|
for j, key := range keys {
|
|
|
|
m = m.Assoc(key, values[j])
|
2016-02-20 00:21:51 +08:00
|
|
|
}
|
|
|
|
}
|
2018-01-30 01:39:41 +08:00
|
|
|
return []interface{}{m}, nil
|
2016-02-20 00:21:51 +08:00
|
|
|
}
|
|
|
|
|
2018-01-21 21:20:06 +08:00
|
|
|
func (cp *compiler) braced(n *parse.Primary) ValuesOpBody {
|
2016-02-20 07:48:13 +08:00
|
|
|
ops := cp.compoundOps(n.Braced)
|
2016-02-20 00:21:51 +08:00
|
|
|
// TODO: n.IsRange
|
|
|
|
// isRange := n.IsRange
|
2018-01-21 21:20:06 +08:00
|
|
|
return seqValuesOp{ops}
|
2018-01-21 09:14:39 +08:00
|
|
|
}
|
|
|
|
|
2018-01-30 01:39:41 +08:00
|
|
|
type literalValuesOp struct{ values []interface{} }
|
2018-01-21 21:20:06 +08:00
|
|
|
|
2018-01-30 01:39:41 +08:00
|
|
|
func (op literalValuesOp) Invoke(*Frame) ([]interface{}, error) {
|
2018-01-21 21:20:06 +08:00
|
|
|
return op.values, nil
|
|
|
|
}
|
|
|
|
|
2018-01-30 01:39:41 +08:00
|
|
|
func literalValues(v ...interface{}) ValuesOpBody {
|
2018-01-21 21:20:06 +08:00
|
|
|
return literalValuesOp{v}
|
|
|
|
}
|
|
|
|
|
|
|
|
func literalStr(text string) ValuesOpBody {
|
2018-01-25 09:40:15 +08:00
|
|
|
return literalValues(text)
|
2018-01-21 21:20:06 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
type seqValuesOp struct{ subops []ValuesOp }
|
|
|
|
|
2018-01-30 01:39:41 +08:00
|
|
|
func (op seqValuesOp) Invoke(ec *Frame) ([]interface{}, error) {
|
|
|
|
var values []interface{}
|
2018-01-21 21:20:06 +08:00
|
|
|
for _, subop := range op.subops {
|
|
|
|
moreValues, err := subop.Exec(ec)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
2018-01-21 09:14:39 +08:00
|
|
|
}
|
2018-01-21 21:20:06 +08:00
|
|
|
values = append(values, moreValues...)
|
2018-01-21 09:14:39 +08:00
|
|
|
}
|
2018-01-21 21:20:06 +08:00
|
|
|
return values, nil
|
2016-02-20 00:21:51 +08:00
|
|
|
}
|
2018-01-21 21:20:06 +08:00
|
|
|
|
2018-01-30 01:39:41 +08:00
|
|
|
type funcValuesOp func(*Frame) ([]interface{}, error)
|
2018-01-21 21:20:06 +08:00
|
|
|
|
2018-01-30 01:39:41 +08:00
|
|
|
func (op funcValuesOp) Invoke(fm *Frame) ([]interface{}, error) { return op(fm) }
|