package server import ( "encoding/hex" "errors" "fmt" "net/http" "reflect" "regexp" "strings" tmjson "github.com/tendermint/tendermint/libs/json" "github.com/tendermint/tendermint/libs/log" ctypes "github.com/tendermint/tendermint/rpc/core/types" types "github.com/tendermint/tendermint/rpc/jsonrpc/types" ) // HTTP + URI handler var reInt = regexp.MustCompile(`^-?[0-9]+$`) // convert from a function name to the http handler func makeHTTPHandler(rpcFunc *RPCFunc, logger log.Logger) func(http.ResponseWriter, *http.Request) { // Always return -1 as there's no ID here. dummyID := types.JSONRPCIntID(-1) // URIClientRequestID // Exception for websocket endpoints if rpcFunc.ws { return func(w http.ResponseWriter, r *http.Request) { WriteRPCResponseHTTPError(w, types.RPCMethodNotFoundError(dummyID)) } } // All other endpoints return func(w http.ResponseWriter, r *http.Request) { logger.Debug("HTTP HANDLER", "req", r) ctx := &types.Context{HTTPReq: r} args := []reflect.Value{reflect.ValueOf(ctx)} fnArgs, err := httpParamsToArgs(rpcFunc, r) if err != nil { WriteRPCResponseHTTPError( w, types.RPCInvalidParamsError( dummyID, fmt.Errorf("error converting http params to arguments: %w", err), ), ) return } args = append(args, fnArgs...) returns := rpcFunc.f.Call(args) logger.Debug("HTTPRestRPC", "method", r.URL.Path, "args", args, "returns", returns) result, err := unreflectResult(returns) switch e := err.(type) { // if no error then return a success response case nil: WriteRPCResponseHTTP(w, types.NewRPCSuccessResponse(dummyID, result)) // if this already of type RPC error then forward that error. case *types.RPCError: WriteRPCResponseHTTPError(w, types.NewRPCErrorResponse(dummyID, e.Code, e.Message, e.Data)) default: // we need to unwrap the error and parse it accordingly switch errors.Unwrap(err) { case ctypes.ErrZeroOrNegativeHeight, ctypes.ErrZeroOrNegativePerPage, ctypes.ErrPageOutOfRange, ctypes.ErrInvalidRequest: WriteRPCResponseHTTPError(w, types.RPCInvalidRequestError(dummyID, err)) default: // ctypes.ErrHeightNotAvailable, ctypes.ErrHeightExceedsChainHead: WriteRPCResponseHTTPError(w, types.RPCInternalError(dummyID, err)) } } } } // Covert an http query to a list of properly typed values. // To be properly decoded the arg must be a concrete type from tendermint (if its an interface). func httpParamsToArgs(rpcFunc *RPCFunc, r *http.Request) ([]reflect.Value, error) { // skip types.Context const argsOffset = 1 values := make([]reflect.Value, len(rpcFunc.argNames)) for i, name := range rpcFunc.argNames { argType := rpcFunc.args[i+argsOffset] values[i] = reflect.Zero(argType) // set default for that type arg := getParam(r, name) // log.Notice("param to arg", "argType", argType, "name", name, "arg", arg) if arg == "" { continue } v, ok, err := nonJSONStringToArg(argType, arg) if err != nil { return nil, err } if ok { values[i] = v continue } values[i], err = jsonStringToArg(argType, arg) if err != nil { return nil, err } } return values, nil } func jsonStringToArg(rt reflect.Type, arg string) (reflect.Value, error) { rv := reflect.New(rt) err := tmjson.Unmarshal([]byte(arg), rv.Interface()) if err != nil { return rv, err } rv = rv.Elem() return rv, nil } func nonJSONStringToArg(rt reflect.Type, arg string) (reflect.Value, bool, error) { if rt.Kind() == reflect.Ptr { rv1, ok, err := nonJSONStringToArg(rt.Elem(), arg) switch { case err != nil: return reflect.Value{}, false, err case ok: rv := reflect.New(rt.Elem()) rv.Elem().Set(rv1) return rv, true, nil default: return reflect.Value{}, false, nil } } else { return _nonJSONStringToArg(rt, arg) } } // NOTE: rt.Kind() isn't a pointer. func _nonJSONStringToArg(rt reflect.Type, arg string) (reflect.Value, bool, error) { isIntString := reInt.Match([]byte(arg)) isQuotedString := strings.HasPrefix(arg, `"`) && strings.HasSuffix(arg, `"`) isHexString := strings.HasPrefix(strings.ToLower(arg), "0x") var expectingString, expectingByteSlice, expectingInt bool switch rt.Kind() { case reflect.Int, reflect.Uint, reflect.Int8, reflect.Uint8, reflect.Int16, reflect.Uint16, reflect.Int32, reflect.Uint32, reflect.Int64, reflect.Uint64: expectingInt = true case reflect.String: expectingString = true case reflect.Slice: expectingByteSlice = rt.Elem().Kind() == reflect.Uint8 } if isIntString && expectingInt { qarg := `"` + arg + `"` rv, err := jsonStringToArg(rt, qarg) if err != nil { return rv, false, err } return rv, true, nil } if isHexString { if !expectingString && !expectingByteSlice { err := fmt.Errorf("got a hex string arg, but expected '%s'", rt.Kind().String()) return reflect.ValueOf(nil), false, err } var value []byte value, err := hex.DecodeString(arg[2:]) if err != nil { return reflect.ValueOf(nil), false, err } if rt.Kind() == reflect.String { return reflect.ValueOf(string(value)), true, nil } return reflect.ValueOf(value), true, nil } if isQuotedString && expectingByteSlice { v := reflect.New(reflect.TypeOf("")) err := tmjson.Unmarshal([]byte(arg), v.Interface()) if err != nil { return reflect.ValueOf(nil), false, err } v = v.Elem() return reflect.ValueOf([]byte(v.String())), true, nil } return reflect.ValueOf(nil), false, nil } func getParam(r *http.Request, param string) string { s := r.URL.Query().Get(param) if s == "" { s = r.FormValue(param) } return s }