mirror of
https://github.com/go-sylixos/elvish.git
synced 2024-12-04 10:57:50 +08:00
73 lines
1.9 KiB
Go
73 lines
1.9 KiB
Go
package vals
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"math/big"
|
|
)
|
|
|
|
// Concatter wraps the Concat method. See Concat for how it is used.
|
|
type Concatter interface {
|
|
// Concat concatenates the receiver with another value, the receiver being
|
|
// the left operand. If concatenation is not supported for the given value,
|
|
// the method can return the special error type ErrCatNotImplemented.
|
|
Concat(v any) (any, error)
|
|
}
|
|
|
|
// RConcatter wraps the RConcat method. See Concat for how it is used.
|
|
type RConcatter interface {
|
|
RConcat(v any) (any, error)
|
|
}
|
|
|
|
// ErrConcatNotImplemented is a special error value used to signal that
|
|
// concatenation is not implemented. See Concat for how it is used.
|
|
var ErrConcatNotImplemented = errors.New("concat not implemented")
|
|
|
|
type cannotConcat struct {
|
|
lhsKind string
|
|
rhsKind string
|
|
}
|
|
|
|
func (err cannotConcat) Error() string {
|
|
return fmt.Sprintf("cannot concatenate %s and %s", err.lhsKind, err.rhsKind)
|
|
}
|
|
|
|
// Concat concatenates two values. If both operands are strings, it returns lhs
|
|
// + rhs, nil. If the left operand implements Concatter, it calls
|
|
// lhs.Concat(rhs). If lhs doesn't implement the interface or returned
|
|
// ErrConcatNotImplemented, it then calls rhs.RConcat(lhs). If all attempts
|
|
// fail, it returns nil and an error.
|
|
func Concat(lhs, rhs any) (any, error) {
|
|
if v, ok := tryConcatBuiltins(lhs, rhs); ok {
|
|
return v, nil
|
|
}
|
|
|
|
if lhs, ok := lhs.(Concatter); ok {
|
|
v, err := lhs.Concat(rhs)
|
|
if err != ErrConcatNotImplemented {
|
|
return v, err
|
|
}
|
|
}
|
|
|
|
if rhs, ok := rhs.(RConcatter); ok {
|
|
v, err := rhs.RConcat(lhs)
|
|
if err != ErrConcatNotImplemented {
|
|
return v, err
|
|
}
|
|
}
|
|
|
|
return nil, cannotConcat{Kind(lhs), Kind(rhs)}
|
|
}
|
|
|
|
func tryConcatBuiltins(lhs, rhs any) (any, bool) {
|
|
switch lhs := lhs.(type) {
|
|
case string, int, *big.Int, *big.Rat, float64:
|
|
switch rhs := rhs.(type) {
|
|
case string, int, *big.Int, *big.Rat, float64:
|
|
return ToString(lhs) + ToString(rhs), true
|
|
}
|
|
}
|
|
|
|
return nil, false
|
|
}
|