2017-12-15 19:28:06 +01:00
|
|
|
package di
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"reflect"
|
|
|
|
)
|
|
|
|
|
|
|
|
type BindType uint32
|
|
|
|
|
|
|
|
const (
|
|
|
|
Static BindType = iota // simple assignable value, a static value.
|
|
|
|
Dynamic // dynamic value, depends on some input arguments from the caller.
|
|
|
|
)
|
|
|
|
|
2017-12-19 22:40:42 +01:00
|
|
|
func bindTypeString(typ BindType) string {
|
|
|
|
switch typ {
|
|
|
|
case Dynamic:
|
|
|
|
return "Dynamic"
|
|
|
|
default:
|
|
|
|
return "Static"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-12-15 19:28:06 +01:00
|
|
|
type BindObject struct {
|
|
|
|
Type reflect.Type // the Type of 'Value' or the type of the returned 'ReturnValue' .
|
|
|
|
Value reflect.Value
|
|
|
|
|
|
|
|
BindType BindType
|
|
|
|
ReturnValue func([]reflect.Value) reflect.Value
|
|
|
|
}
|
|
|
|
|
|
|
|
func MakeBindObject(v reflect.Value, goodFunc TypeChecker) (b BindObject, err error) {
|
|
|
|
if IsFunc(v) {
|
|
|
|
b.BindType = Dynamic
|
|
|
|
b.ReturnValue, b.Type, err = MakeReturnValue(v, goodFunc)
|
|
|
|
} else {
|
|
|
|
b.BindType = Static
|
|
|
|
b.Type = v.Type()
|
|
|
|
b.Value = v
|
|
|
|
}
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
var errBad = errors.New("bad")
|
|
|
|
|
|
|
|
// MakeReturnValue takes any function
|
|
|
|
// that accept custom values and returns something,
|
|
|
|
// it returns a binder function, which accepts a slice of reflect.Value
|
|
|
|
// and returns a single one reflect.Value for that.
|
|
|
|
// It's being used to resolve the input parameters on a "x" consumer faster.
|
|
|
|
//
|
|
|
|
// The "fn" can have the following form:
|
|
|
|
// `func(myService) MyViewModel`.
|
|
|
|
//
|
|
|
|
// The return type of the "fn" should be a value instance, not a pointer, for your own protection.
|
|
|
|
// The binder function should return only one value.
|
|
|
|
func MakeReturnValue(fn reflect.Value, goodFunc TypeChecker) (func([]reflect.Value) reflect.Value, reflect.Type, error) {
|
|
|
|
typ := IndirectType(fn.Type())
|
|
|
|
|
|
|
|
// invalid if not a func.
|
|
|
|
if typ.Kind() != reflect.Func {
|
|
|
|
return nil, typ, errBad
|
|
|
|
}
|
|
|
|
|
|
|
|
// invalid if not returns one single value.
|
|
|
|
if typ.NumOut() != 1 {
|
|
|
|
return nil, typ, errBad
|
|
|
|
}
|
|
|
|
|
|
|
|
if goodFunc != nil {
|
|
|
|
if !goodFunc(typ) {
|
|
|
|
return nil, typ, errBad
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
outTyp := typ.Out(0)
|
|
|
|
zeroOutVal := reflect.New(outTyp).Elem()
|
|
|
|
|
|
|
|
bf := func(ctxValue []reflect.Value) reflect.Value {
|
|
|
|
results := fn.Call(ctxValue)
|
|
|
|
if len(results) == 0 {
|
|
|
|
return zeroOutVal
|
|
|
|
}
|
|
|
|
|
|
|
|
v := results[0]
|
|
|
|
if !v.IsValid() {
|
|
|
|
return zeroOutVal
|
|
|
|
}
|
|
|
|
return v
|
|
|
|
}
|
|
|
|
|
|
|
|
return bf, outTyp, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *BindObject) IsAssignable(to reflect.Type) bool {
|
|
|
|
return equalTypes(b.Type, to)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *BindObject) Assign(ctx []reflect.Value, toSetter func(reflect.Value)) {
|
|
|
|
if b.BindType == Dynamic {
|
|
|
|
toSetter(b.ReturnValue(ctx))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
toSetter(b.Value)
|
|
|
|
}
|