2017-09-29 20:20:26 +08:00
|
|
|
package eval
|
|
|
|
|
|
|
|
// This file implements facilities for "scanning" arguments and options.
|
|
|
|
|
|
|
|
import (
|
|
|
|
"reflect"
|
|
|
|
|
|
|
|
"github.com/elves/elvish/parse"
|
|
|
|
"github.com/elves/elvish/util"
|
|
|
|
)
|
|
|
|
|
|
|
|
// ScanArgs scans arguments into pointers to supported argument types. If the
|
|
|
|
// arguments cannot be scanned, an error is thrown.
|
2017-09-29 21:35:05 +08:00
|
|
|
func ScanArgs(src []Value, dstPtrs ...interface{}) {
|
|
|
|
if len(src) != len(dstPtrs) {
|
|
|
|
throwf("arity mistmatch: want %d arguments, got %d", len(dstPtrs), len(src))
|
2017-09-29 20:20:26 +08:00
|
|
|
}
|
2017-09-29 21:09:36 +08:00
|
|
|
for i, value := range src {
|
2017-10-01 08:45:16 +08:00
|
|
|
scanValueToGo(value, dstPtrs[i])
|
2017-09-29 20:20:26 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ScanArgsVariadic is like ScanArgs, but the last element of args should be a
|
|
|
|
// pointer to a slice, and the rest of arguments will be scanned into it.
|
2017-09-29 21:35:05 +08:00
|
|
|
func ScanArgsVariadic(src []Value, dstPtrs ...interface{}) {
|
|
|
|
if len(src) < len(dstPtrs)-1 {
|
|
|
|
throwf("arity mistmatch: want at least %d arguments, got %d", len(dstPtrs)-1, len(src))
|
2017-09-29 20:20:26 +08:00
|
|
|
}
|
2017-09-29 21:35:05 +08:00
|
|
|
ScanArgs(src[:len(dstPtrs)-1], dstPtrs[:len(dstPtrs)-1]...)
|
2017-09-29 20:20:26 +08:00
|
|
|
|
|
|
|
// Scan the rest of arguments into a slice.
|
2017-09-29 21:35:05 +08:00
|
|
|
rest := src[len(dstPtrs)-1:]
|
|
|
|
restDst := reflect.ValueOf(dstPtrs[len(dstPtrs)-1])
|
2017-09-29 21:09:36 +08:00
|
|
|
if restDst.Kind() != reflect.Ptr || restDst.Elem().Kind() != reflect.Slice {
|
2017-09-29 21:35:05 +08:00
|
|
|
throwf("internal bug: %T to ScanArgsVariadic, need pointer to slice", dstPtrs[len(dstPtrs)-1])
|
2017-09-29 20:20:26 +08:00
|
|
|
}
|
2017-09-29 21:09:36 +08:00
|
|
|
scanned := reflect.MakeSlice(restDst.Elem().Type(), len(rest), len(rest))
|
2017-09-29 20:20:26 +08:00
|
|
|
for i, value := range rest {
|
2017-10-01 08:45:16 +08:00
|
|
|
scanValueToGo(value, scanned.Index(i).Addr().Interface())
|
2017-09-29 20:20:26 +08:00
|
|
|
}
|
2017-09-29 21:09:36 +08:00
|
|
|
reflect.Indirect(restDst).Set(scanned)
|
2017-09-29 20:20:26 +08:00
|
|
|
}
|
|
|
|
|
2017-09-29 21:09:36 +08:00
|
|
|
// ScanArgsOptionalInput is like ScanArgs, but the argument can contain an
|
|
|
|
// optional iterable value at the end containing inputs to the function. The
|
|
|
|
// return value is a function that iterates the iterable value if it exists, or
|
|
|
|
// the input otherwise.
|
2017-09-29 21:35:05 +08:00
|
|
|
func ScanArgsOptionalInput(ec *EvalCtx, src []Value, dstArgs ...interface{}) func(func(Value)) {
|
2017-09-29 21:09:36 +08:00
|
|
|
switch len(src) {
|
2017-09-29 21:35:05 +08:00
|
|
|
case len(dstArgs):
|
|
|
|
ScanArgs(src, dstArgs...)
|
2017-09-29 20:20:26 +08:00
|
|
|
return ec.IterateInputs
|
2017-09-29 21:35:05 +08:00
|
|
|
case len(dstArgs) + 1:
|
|
|
|
ScanArgs(src[:len(dstArgs)], dstArgs...)
|
|
|
|
value := src[len(dstArgs)]
|
2017-09-29 20:20:26 +08:00
|
|
|
iterable, ok := value.(Iterable)
|
|
|
|
if !ok {
|
|
|
|
throwf("need iterable argument, got %s", value.Kind())
|
|
|
|
}
|
|
|
|
return func(f func(Value)) {
|
|
|
|
iterable.Iterate(func(v Value) bool {
|
|
|
|
f(v)
|
|
|
|
return true
|
|
|
|
})
|
|
|
|
}
|
|
|
|
default:
|
2017-09-29 21:35:05 +08:00
|
|
|
throwf("arity mistmatch: want %d or %d arguments, got %d", len(dstArgs), len(dstArgs)+1, len(src))
|
2017-09-29 20:20:26 +08:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-29 21:09:36 +08:00
|
|
|
// OptToScan is a data structure for an option that is intended to be used in
|
|
|
|
// ScanOpts.
|
|
|
|
type OptToScan struct {
|
2017-09-29 20:20:26 +08:00
|
|
|
Name string
|
|
|
|
Ptr interface{}
|
|
|
|
Default Value
|
|
|
|
}
|
|
|
|
|
|
|
|
// ScanOpts scans options from a map.
|
2017-09-29 21:09:36 +08:00
|
|
|
func ScanOpts(m map[string]Value, opts ...OptToScan) {
|
2017-09-29 20:20:26 +08:00
|
|
|
scanned := make(map[string]bool)
|
|
|
|
for _, opt := range opts {
|
|
|
|
a := opt.Ptr
|
|
|
|
value, ok := m[opt.Name]
|
|
|
|
if !ok {
|
|
|
|
value = opt.Default
|
|
|
|
}
|
2017-10-01 08:45:16 +08:00
|
|
|
scanValueToGo(value, a)
|
2017-09-29 20:20:26 +08:00
|
|
|
scanned[opt.Name] = true
|
|
|
|
}
|
|
|
|
for key := range m {
|
|
|
|
if !scanned[key] {
|
|
|
|
throwf("unknown option %s", parse.Quote(key))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ScanOptsToStruct scan options from a map like ScanOpts except the destination
|
|
|
|
// is a struct whose fields correspond to the options to be parsed. A field
|
|
|
|
// named FieldName corresponds to the option named field-name, unless the field
|
|
|
|
// has a explicit "name" tag.
|
|
|
|
func ScanOptsToStruct(m map[string]Value, structPtr interface{}) {
|
|
|
|
ptrValue := reflect.ValueOf(structPtr)
|
|
|
|
if ptrValue.Kind() != reflect.Ptr || ptrValue.Elem().Kind() != reflect.Struct {
|
|
|
|
throwf("internal bug: need struct ptr for ScanOptsToStruct, got %T", structPtr)
|
|
|
|
}
|
|
|
|
struc := ptrValue.Elem()
|
|
|
|
|
|
|
|
// fieldIdxForOpt maps option name to the index of field in struc.
|
|
|
|
fieldIdxForOpt := make(map[string]int)
|
|
|
|
for i := 0; i < struc.Type().NumField(); i++ {
|
|
|
|
// ignore unexported fields
|
|
|
|
if !struc.Field(i).CanSet() {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
f := struc.Type().Field(i)
|
|
|
|
optName := f.Tag.Get("name")
|
|
|
|
if optName == "" {
|
|
|
|
optName = util.CamelToDashed(f.Name)
|
|
|
|
}
|
|
|
|
fieldIdxForOpt[optName] = i
|
|
|
|
}
|
|
|
|
|
|
|
|
for k, v := range m {
|
|
|
|
fieldIdx, ok := fieldIdxForOpt[k]
|
|
|
|
if !ok {
|
|
|
|
throwf("unknown option %s", parse.Quote(k))
|
|
|
|
}
|
2017-10-01 08:45:16 +08:00
|
|
|
scanValueToGo(v, struc.Field(fieldIdx).Addr().Interface())
|
2017-09-29 20:20:26 +08:00
|
|
|
}
|
|
|
|
}
|