mirror of
https://github.com/go-sylixos/elvish.git
synced 2024-12-14 02:57:52 +08:00
116 lines
2.2 KiB
Go
116 lines
2.2 KiB
Go
package eval
|
|
|
|
import (
|
|
"errors"
|
|
"unicode/utf8"
|
|
|
|
"github.com/elves/elvish/parse"
|
|
"github.com/xiaq/persistent/hash"
|
|
)
|
|
|
|
// String is just a string.
|
|
type String string
|
|
|
|
var (
|
|
_ Value = String("")
|
|
_ ListLike = String("")
|
|
)
|
|
|
|
var ErrReplacementMustBeString = errors.New("replacement must be string")
|
|
|
|
func (String) Kind() string {
|
|
return "string"
|
|
}
|
|
|
|
func (s String) Repr(int) string {
|
|
return quote(string(s))
|
|
}
|
|
|
|
func (s String) Equal(rhs interface{}) bool {
|
|
return s == rhs
|
|
}
|
|
|
|
func (s String) Hash() uint32 {
|
|
return hash.String(string(s))
|
|
}
|
|
|
|
func (s String) String() string {
|
|
return string(s)
|
|
}
|
|
|
|
func (s String) Len() int {
|
|
return len(string(s))
|
|
}
|
|
|
|
func (s String) IndexOne(idx Value) Value {
|
|
i, j := s.index(idx)
|
|
return s[i:j]
|
|
}
|
|
|
|
func (s String) Assoc(idx, v Value) Value {
|
|
i, j := s.index(idx)
|
|
repl, ok := v.(String)
|
|
if !ok {
|
|
throw(ErrReplacementMustBeString)
|
|
}
|
|
return s[:i] + repl + s[j:]
|
|
}
|
|
|
|
func (s String) index(idx Value) (int, int) {
|
|
slice, i, j := ParseAndFixListIndex(ToString(idx), len(s))
|
|
r, size := utf8.DecodeRuneInString(string(s[i:]))
|
|
if r == utf8.RuneError {
|
|
throw(ErrBadIndex)
|
|
}
|
|
if slice {
|
|
if r, _ := utf8.DecodeLastRuneInString(string(s[:j])); r == utf8.RuneError {
|
|
throw(ErrBadIndex)
|
|
}
|
|
return i, j
|
|
}
|
|
return i, i + size
|
|
}
|
|
|
|
func (s String) Iterate(f func(v Value) bool) {
|
|
for _, r := range s {
|
|
b := f(String(string(r)))
|
|
if !b {
|
|
break
|
|
}
|
|
}
|
|
}
|
|
|
|
// Call resolves a command name to either a Fn variable or external command and
|
|
// calls it.
|
|
func (s String) Call(ec *EvalCtx, args []Value, opts map[string]Value) {
|
|
resolve(string(s), ec).Call(ec, args, opts)
|
|
}
|
|
|
|
func resolve(s string, ec *EvalCtx) CallableValue {
|
|
// Try variable
|
|
explode, ns, name := ParseAndFixVariable(string(s))
|
|
if !explode {
|
|
if v := ec.ResolveVar(ns, FnPrefix+name); v != nil {
|
|
if caller, ok := v.Get().(CallableValue); ok {
|
|
return caller
|
|
}
|
|
}
|
|
}
|
|
|
|
// External command
|
|
return ExternalCmd{string(s)}
|
|
}
|
|
|
|
// ToString converts a Value to String. When the Value type implements
|
|
// String(), it is used. Otherwise Repr(NoPretty) is used.
|
|
func ToString(v Value) string {
|
|
if s, ok := v.(Stringer); ok {
|
|
return s.String()
|
|
}
|
|
return v.Repr(NoPretty)
|
|
}
|
|
|
|
func quote(s string) string {
|
|
return parse.Quote(s)
|
|
}
|