elvish/eval/value.go
2017-09-16 17:48:53 +01:00

218 lines
4.8 KiB
Go

package eval
import (
"fmt"
"github.com/elves/elvish/util"
"github.com/xiaq/persistent/hashmap"
)
// Definitions for Value interfaces, some simple Value types and some common
// Value helpers.
const (
NoPretty = util.MinInt
)
// Value is an elvish value.
type Value interface {
Kinder
Equaler
Hasher
Reprer
}
// Kinder is anything with kind string.
type Kinder interface {
Kind() string
}
// Reprer is anything with a Repr method.
type Reprer interface {
// Repr returns a string that represents a Value. The string either be a
// literal of that Value that is preferably deep-equal to it (like `[a b c]`
// for a list), or a string enclosed in "<>" containing the kind and
// identity of the Value(like `<fn 0xdeadcafe>`).
//
// If indent is at least 0, it should be pretty-printed with the current
// indentation level of indent; the indent of the first line has already
// been written and shall not be written in Repr. The returned string
// should never contain a trailing newline.
Repr(indent int) string
}
// Equaler is anything that knows how to compare itself against other values.
type Equaler interface {
Equal(interface{}) bool
}
// Hasher is anything that knows how to compute its hash code.
type Hasher interface {
Hash() uint32
}
// Booler is anything that can be converted to a bool.
type Booler interface {
Bool() bool
}
// Stringer is anything that can be converted to a string.
type Stringer interface {
String() string
}
// Lener is anything with a length.
type Lener interface {
Len() int
}
// Iterable is anything that can be iterated.
type Iterable interface {
Iterate(func(Value) bool)
}
type IterableValue interface {
Iterable
Value
}
func collectFromIterable(it Iterable) []Value {
var vs []Value
if lener, ok := it.(Lener); ok {
vs = make([]Value, 0, lener.Len())
}
it.Iterate(func(v Value) bool {
vs = append(vs, v)
return true
})
return vs
}
// IterateKeyer is anything with keys that can be iterated.
type IterateKeyer interface {
IterateKey(func(Value) bool)
}
// IteratePairer is anything with key-value pairs that can be iterated.
type IteratePairer interface {
IteratePair(func(Value, Value) bool)
}
var (
NoArgs = []Value{}
NoOpts = map[string]Value{}
)
// Callable is anything may be called on an evalCtx with a list of Value's.
type Callable interface {
Call(ec *EvalCtx, args []Value, opts map[string]Value)
}
type CallableValue interface {
Value
Callable
}
func mustFn(v Value) Callable {
fn, ok := v.(Callable)
if !ok {
throw(fmt.Errorf("a %s is not callable", v.Kind()))
}
return fn
}
// Indexer is anything that can be indexed by Values and yields Values.
type Indexer interface {
Index(idx []Value) []Value
}
// IndexOneer is anything that can be indexed by one Value and yields one Value.
type IndexOneer interface {
IndexOne(idx Value) Value
}
func mustIndexer(v Value, ec *EvalCtx) Indexer {
indexer, ok := getIndexer(v, ec)
if !ok {
throw(fmt.Errorf("a %s is not indexable", v.Kind()))
}
return indexer
}
// getIndexer adapts a Value to an Indexer if there is an adapter. It adapts a
// Fn if ec is not nil.
func getIndexer(v Value, ec *EvalCtx) (Indexer, bool) {
if indexer, ok := v.(Indexer); ok {
return indexer, true
}
if indexOneer, ok := v.(IndexOneer); ok {
return IndexOneerIndexer{indexOneer}, true
}
return nil, false
}
// IndexOneerIndexer adapts an IndexOneer to an Indexer by calling all the
// indicies on the IndexOner and collect the results.
type IndexOneerIndexer struct {
IndexOneer
}
func (ioi IndexOneerIndexer) Index(vs []Value) []Value {
results := make([]Value, len(vs))
for i, v := range vs {
results[i] = ioi.IndexOneer.IndexOne(v)
}
return results
}
// Assocer is anything tha can return a slightly modified version of itself as a
// new Value.
type Assocer interface {
Assoc(k, v Value) Value
}
// Assocer is anything tha can return a slightly modified version of itself with
// the specified key removed, as a new value.
type Dissocer interface {
Dissoc(k Value) Value
}
// IndexOneAssocer combines IndexOneer and Assocer.
type IndexOneAssocer interface {
IndexOneer
Assocer
}
// FromJSONInterface converts a interface{} that results from json.Unmarshal to
// a Value.
func FromJSONInterface(v interface{}) Value {
if v == nil {
// TODO Use a more appropriate type
return String("")
}
switch v.(type) {
case bool:
return Bool(v.(bool))
case float64, string:
// TODO Use a numeric type for float64
return String(fmt.Sprint(v))
case []interface{}:
a := v.([]interface{})
vs := make([]Value, len(a))
for i, v := range a {
vs[i] = FromJSONInterface(v)
}
return NewList(vs...)
case map[string]interface{}:
m := v.(map[string]interface{})
mv := hashmap.Empty
for k, v := range m {
mv = mv.Assoc(String(k), FromJSONInterface(v))
}
return NewMap(mv)
default:
throw(fmt.Errorf("unexpected json type: %T", v))
return nil // not reached
}
}