156 lines
5.3 KiB
Go
156 lines
5.3 KiB
Go
|
package handler
|
||
|
|
||
|
import (
|
||
|
"errors"
|
||
|
"fmt"
|
||
|
"reflect"
|
||
|
)
|
||
|
|
||
|
// NewPos adapts a function to a jrpc2.Handler. The concrete value of fn must
|
||
|
// be a function accepted by Positional. The resulting Func will handle JSON
|
||
|
// encoding and decoding, call fn, and report appropriate errors.
|
||
|
//
|
||
|
// NewPos is intended for use during program initialization, and will panic if
|
||
|
// the type of fn does not have one of the accepted forms. Programs that need
|
||
|
// to check for possible errors should call handler.Positional directly, and
|
||
|
// use the Wrap method of the resulting FuncInfo to obtain the wrapper.
|
||
|
func NewPos(fn interface{}, names ...string) Func {
|
||
|
fi, err := Positional(fn, names...)
|
||
|
if err != nil {
|
||
|
panic(err)
|
||
|
}
|
||
|
return fi.Wrap()
|
||
|
}
|
||
|
|
||
|
// Positional checks whether fn can serve as a jrpc2.Handler. The concrete
|
||
|
// value of fn must be a function with one of the following type signature
|
||
|
// schemes:
|
||
|
//
|
||
|
// func(context.Context, X1, x2, ..., Xn) (Y, error)
|
||
|
// func(context.Context, X1, x2, ..., Xn) Y
|
||
|
// func(context.Context, X1, x2, ..., Xn) error
|
||
|
//
|
||
|
// For JSON-marshalable types X_i and Y. If fn does not have one of these
|
||
|
// forms, Positional reports an error. The given names must match the number of
|
||
|
// non-context arguments exactly. Variadic functions are not supported.
|
||
|
//
|
||
|
// In contrast to Check, this function allows any number of arguments, but the
|
||
|
// caller must provide names for them. Positional creates an anonymous struct
|
||
|
// type whose fields correspond to the non-context arguments of fn. The names
|
||
|
// are used as the JSON field keys for the corresponding parameters.
|
||
|
//
|
||
|
// When converted into a handler.Func, the wrapped function accepts a JSON
|
||
|
// object with the field keys named. For example, given:
|
||
|
//
|
||
|
// func add(ctx context.Context, x, y int) int { return x + y }
|
||
|
//
|
||
|
// fi, err := handler.Positional(add, "first", "second")
|
||
|
// // ...
|
||
|
// call := fi.Wrap()
|
||
|
//
|
||
|
// the resulting JSON-RPC handler accepts a parameter object like:
|
||
|
//
|
||
|
// {"first": 17, "second": 23}
|
||
|
//
|
||
|
// where "first" is mapped to argument x and "second" to argument y. Unknown
|
||
|
// field keys generate an error. The field names are not required to match the
|
||
|
// parameter names declared by the function; it is the names assigned here that
|
||
|
// determine which object keys are accepted.
|
||
|
func Positional(fn interface{}, names ...string) (*FuncInfo, error) {
|
||
|
if fn == nil {
|
||
|
return nil, errors.New("nil function")
|
||
|
}
|
||
|
|
||
|
fv := reflect.ValueOf(fn)
|
||
|
if fv.Kind() != reflect.Func {
|
||
|
return nil, errors.New("not a function")
|
||
|
}
|
||
|
ft := fv.Type()
|
||
|
if np := ft.NumIn(); np == 0 {
|
||
|
return nil, errors.New("wrong number of parameters")
|
||
|
} else if ft.In(0) != ctxType {
|
||
|
return nil, errors.New("first parameter is not context.Context")
|
||
|
} else if np == 1 {
|
||
|
// If the context is the only argument, there is nothing to do.
|
||
|
return Check(fn)
|
||
|
} else if ft.IsVariadic() {
|
||
|
return nil, errors.New("variadic functions are not supported")
|
||
|
}
|
||
|
|
||
|
// Reaching here, we have at least one non-context argument.
|
||
|
atype, err := makeArgType(ft, names)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
fi, err := Check(makeCaller(ft, fv, atype))
|
||
|
if err == nil {
|
||
|
fi.strictFields = true
|
||
|
}
|
||
|
return fi, err
|
||
|
}
|
||
|
|
||
|
// makeArgType creates a struct type whose fields match the parameters of t,
|
||
|
// with JSON struct tags corresponding to the given names.
|
||
|
//
|
||
|
// Preconditions: t is a function with len(names)+1 arguments.
|
||
|
func makeArgType(t reflect.Type, names []string) (reflect.Type, error) {
|
||
|
if t.NumIn()-1 != len(names) {
|
||
|
return nil, fmt.Errorf("got %d names for %d inputs", len(names), t.NumIn()-1)
|
||
|
}
|
||
|
|
||
|
// TODO(creachadair): I wanted to implement the strictFielder interface on
|
||
|
// the generated struct instead of having extra magic in the wrapper.
|
||
|
// However, it is not now possible to add methods to a type constructed by
|
||
|
// reflection.
|
||
|
//
|
||
|
// Embedding an anonymous field that exposes the method doesn't work for
|
||
|
// JSON unmarshaling: The base struct will have the method, but its pointer
|
||
|
// will not, probably related to https://github.com/golang/go/issues/15924.
|
||
|
// JSON unmarshaling requires a pointer to its argument.
|
||
|
//
|
||
|
// For now, I worked around this by adding a hook into the wrapper compiler.
|
||
|
|
||
|
var fields []reflect.StructField
|
||
|
for i, name := range names {
|
||
|
tag := `json:"-"`
|
||
|
if name != "" && name != "-" {
|
||
|
tag = fmt.Sprintf(`json:"%s,omitempty"`, name)
|
||
|
}
|
||
|
fields = append(fields, reflect.StructField{
|
||
|
Name: fmt.Sprintf("P_%d", i+1),
|
||
|
Type: t.In(i + 1),
|
||
|
Tag: reflect.StructTag(tag),
|
||
|
})
|
||
|
}
|
||
|
return reflect.StructOf(fields), nil
|
||
|
}
|
||
|
|
||
|
// makeCaller creates a wrapper function that takes a context and an atype as
|
||
|
// arguments, and calls fv with the context and the struct fields unpacked into
|
||
|
// positional arguments.
|
||
|
//
|
||
|
// Preconditions: fv is a function and atype is its argument struct.
|
||
|
func makeCaller(ft reflect.Type, fv reflect.Value, atype reflect.Type) interface{} {
|
||
|
atypes := []reflect.Type{ctxType, atype}
|
||
|
|
||
|
otypes := make([]reflect.Type, ft.NumOut())
|
||
|
for i := 0; i < ft.NumOut(); i++ {
|
||
|
otypes[i] = ft.Out(i)
|
||
|
}
|
||
|
|
||
|
call := fv.Call
|
||
|
wtype := reflect.FuncOf(atypes, otypes, false)
|
||
|
wrap := reflect.MakeFunc(wtype, func(args []reflect.Value) []reflect.Value {
|
||
|
st := args[1]
|
||
|
cargs := make([]reflect.Value, st.NumField()+1)
|
||
|
cargs[0] = args[0] // ctx
|
||
|
|
||
|
// Unpack the struct fields into positional arguments.
|
||
|
for i := 0; i < st.NumField(); i++ {
|
||
|
cargs[i+1] = st.Field(i)
|
||
|
}
|
||
|
return call(cargs)
|
||
|
})
|
||
|
return wrap.Interface()
|
||
|
}
|