mirror of https://github.com/ethereum/go-ethereum
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1128 lines
22 KiB
1128 lines
22 KiB
package rpc
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
"math/big"
|
|
|
|
"github.com/ethereum/go-ethereum/common"
|
|
)
|
|
|
|
const (
|
|
defaultLogLimit = 100
|
|
defaultLogOffset = 0
|
|
)
|
|
|
|
func blockHeightFromJson(msg json.RawMessage, number *int64) error {
|
|
var raw interface{}
|
|
if err := json.Unmarshal(msg, &raw); err != nil {
|
|
return NewDecodeParamError(err.Error())
|
|
}
|
|
return blockHeight(raw, number)
|
|
}
|
|
|
|
func blockHeight(raw interface{}, number *int64) error {
|
|
// Parse as integer
|
|
num, ok := raw.(float64)
|
|
if ok {
|
|
*number = int64(num)
|
|
return nil
|
|
}
|
|
|
|
// Parse as string/hexstring
|
|
str, ok := raw.(string)
|
|
if !ok {
|
|
return NewInvalidTypeError("", "not a number or string")
|
|
}
|
|
|
|
switch str {
|
|
case "earliest":
|
|
*number = 0
|
|
case "latest":
|
|
*number = -1
|
|
case "pending":
|
|
*number = -2
|
|
default:
|
|
if common.HasHexPrefix(str) {
|
|
*number = common.String2Big(str).Int64()
|
|
} else {
|
|
return NewInvalidTypeError("blockNumber", "is not a valid string")
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func numString(raw interface{}) (*big.Int, error) {
|
|
var number *big.Int
|
|
// Parse as integer
|
|
num, ok := raw.(float64)
|
|
if ok {
|
|
number = big.NewInt(int64(num))
|
|
return number, nil
|
|
}
|
|
|
|
// Parse as string/hexstring
|
|
str, ok := raw.(string)
|
|
if ok {
|
|
number = common.String2Big(str)
|
|
return number, nil
|
|
}
|
|
|
|
return nil, NewInvalidTypeError("", "not a number or string")
|
|
}
|
|
|
|
// func toNumber(v interface{}) (int64, error) {
|
|
// var str string
|
|
// if v != nil {
|
|
// var ok bool
|
|
// str, ok = v.(string)
|
|
// if !ok {
|
|
// return 0, errors.New("is not a string or undefined")
|
|
// }
|
|
// } else {
|
|
// str = "latest"
|
|
// }
|
|
|
|
// switch str {
|
|
// case "latest":
|
|
// return -1, nil
|
|
// default:
|
|
// return int64(common.Big(v.(string)).Int64()), nil
|
|
// }
|
|
// }
|
|
|
|
// func hashString(raw interface{}, hash *string) error {
|
|
// argstr, ok := raw.(string)
|
|
// if !ok {
|
|
// return NewInvalidTypeError("", "not a string")
|
|
// }
|
|
// v := common.IsHex(argstr)
|
|
// hash = &argstr
|
|
|
|
// return nil
|
|
// }
|
|
|
|
type GetBlockByHashArgs struct {
|
|
BlockHash string
|
|
IncludeTxs bool
|
|
}
|
|
|
|
func (args *GetBlockByHashArgs) UnmarshalJSON(b []byte) (err error) {
|
|
var obj []interface{}
|
|
|
|
if err := json.Unmarshal(b, &obj); err != nil {
|
|
return NewDecodeParamError(err.Error())
|
|
}
|
|
|
|
if len(obj) < 2 {
|
|
return NewInsufficientParamsError(len(obj), 2)
|
|
}
|
|
|
|
argstr, ok := obj[0].(string)
|
|
if !ok {
|
|
return NewInvalidTypeError("blockHash", "not a string")
|
|
}
|
|
args.BlockHash = argstr
|
|
|
|
args.IncludeTxs = obj[1].(bool)
|
|
|
|
return nil
|
|
}
|
|
|
|
type GetBlockByNumberArgs struct {
|
|
BlockNumber int64
|
|
IncludeTxs bool
|
|
}
|
|
|
|
func (args *GetBlockByNumberArgs) UnmarshalJSON(b []byte) (err error) {
|
|
var obj []interface{}
|
|
if err := json.Unmarshal(b, &obj); err != nil {
|
|
return NewDecodeParamError(err.Error())
|
|
}
|
|
|
|
if len(obj) < 2 {
|
|
return NewInsufficientParamsError(len(obj), 2)
|
|
}
|
|
|
|
if err := blockHeight(obj[0], &args.BlockNumber); err != nil {
|
|
return err
|
|
}
|
|
|
|
args.IncludeTxs = obj[1].(bool)
|
|
|
|
return nil
|
|
}
|
|
|
|
type NewTxArgs struct {
|
|
From string
|
|
To string
|
|
Value *big.Int
|
|
Gas *big.Int
|
|
GasPrice *big.Int
|
|
Data string
|
|
|
|
BlockNumber int64
|
|
}
|
|
|
|
func (args *NewTxArgs) UnmarshalJSON(b []byte) (err error) {
|
|
var obj []json.RawMessage
|
|
var ext struct {
|
|
From string
|
|
To string
|
|
Value interface{}
|
|
Gas interface{}
|
|
GasPrice interface{}
|
|
Data string
|
|
}
|
|
|
|
// Decode byte slice to array of RawMessages
|
|
if err := json.Unmarshal(b, &obj); err != nil {
|
|
return NewDecodeParamError(err.Error())
|
|
}
|
|
|
|
// Check for sufficient params
|
|
if len(obj) < 1 {
|
|
return NewInsufficientParamsError(len(obj), 1)
|
|
}
|
|
|
|
// Decode 0th RawMessage to temporary struct
|
|
if err := json.Unmarshal(obj[0], &ext); err != nil {
|
|
return NewDecodeParamError(err.Error())
|
|
}
|
|
|
|
if len(ext.From) == 0 {
|
|
return NewValidationError("from", "is required")
|
|
}
|
|
|
|
args.From = ext.From
|
|
args.To = ext.To
|
|
args.Data = ext.Data
|
|
|
|
var num *big.Int
|
|
if ext.Value == nil {
|
|
num = big.NewInt(0)
|
|
} else {
|
|
num, err = numString(ext.Value)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
args.Value = num
|
|
|
|
num = nil
|
|
if ext.Gas == nil {
|
|
num = big.NewInt(0)
|
|
} else {
|
|
if num, err = numString(ext.Gas); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
args.Gas = num
|
|
|
|
num = nil
|
|
if ext.GasPrice == nil {
|
|
num = big.NewInt(0)
|
|
} else {
|
|
if num, err = numString(ext.GasPrice); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
args.GasPrice = num
|
|
|
|
// Check for optional BlockNumber param
|
|
if len(obj) > 1 {
|
|
if err := blockHeightFromJson(obj[1], &args.BlockNumber); err != nil {
|
|
return err
|
|
}
|
|
} else {
|
|
args.BlockNumber = -1
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
type CallArgs struct {
|
|
From string
|
|
To string
|
|
Value *big.Int
|
|
Gas *big.Int
|
|
GasPrice *big.Int
|
|
Data string
|
|
|
|
BlockNumber int64
|
|
}
|
|
|
|
func (args *CallArgs) UnmarshalJSON(b []byte) (err error) {
|
|
var obj []json.RawMessage
|
|
var ext struct {
|
|
From string
|
|
To string
|
|
Value interface{}
|
|
Gas interface{}
|
|
GasPrice interface{}
|
|
Data string
|
|
}
|
|
|
|
// Decode byte slice to array of RawMessages
|
|
if err := json.Unmarshal(b, &obj); err != nil {
|
|
return NewDecodeParamError(err.Error())
|
|
}
|
|
|
|
// Check for sufficient params
|
|
if len(obj) < 1 {
|
|
return NewInsufficientParamsError(len(obj), 1)
|
|
}
|
|
|
|
// Decode 0th RawMessage to temporary struct
|
|
if err := json.Unmarshal(obj[0], &ext); err != nil {
|
|
return NewDecodeParamError(err.Error())
|
|
}
|
|
|
|
args.From = ext.From
|
|
|
|
if len(ext.To) == 0 {
|
|
return NewValidationError("to", "is required")
|
|
}
|
|
args.To = ext.To
|
|
|
|
var num *big.Int
|
|
if ext.Value == nil {
|
|
num = big.NewInt(0)
|
|
} else {
|
|
if num, err = numString(ext.Value); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
args.Value = num
|
|
|
|
if ext.Gas == nil {
|
|
num = big.NewInt(0)
|
|
} else {
|
|
if num, err = numString(ext.Gas); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
args.Gas = num
|
|
|
|
if ext.GasPrice == nil {
|
|
num = big.NewInt(0)
|
|
} else {
|
|
if num, err = numString(ext.GasPrice); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
args.GasPrice = num
|
|
|
|
args.Data = ext.Data
|
|
|
|
// Check for optional BlockNumber param
|
|
if len(obj) > 1 {
|
|
if err := blockHeightFromJson(obj[1], &args.BlockNumber); err != nil {
|
|
return err
|
|
}
|
|
} else {
|
|
args.BlockNumber = -1
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
type GetStorageArgs struct {
|
|
Address string
|
|
BlockNumber int64
|
|
}
|
|
|
|
func (args *GetStorageArgs) UnmarshalJSON(b []byte) (err error) {
|
|
var obj []interface{}
|
|
if err := json.Unmarshal(b, &obj); err != nil {
|
|
return NewDecodeParamError(err.Error())
|
|
}
|
|
|
|
if len(obj) < 1 {
|
|
return NewInsufficientParamsError(len(obj), 1)
|
|
}
|
|
|
|
addstr, ok := obj[0].(string)
|
|
if !ok {
|
|
return NewInvalidTypeError("address", "not a string")
|
|
}
|
|
args.Address = addstr
|
|
|
|
if len(obj) > 1 {
|
|
if err := blockHeight(obj[1], &args.BlockNumber); err != nil {
|
|
return err
|
|
}
|
|
} else {
|
|
args.BlockNumber = -1
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
type GetStorageAtArgs struct {
|
|
Address string
|
|
Key string
|
|
BlockNumber int64
|
|
}
|
|
|
|
func (args *GetStorageAtArgs) UnmarshalJSON(b []byte) (err error) {
|
|
var obj []interface{}
|
|
if err := json.Unmarshal(b, &obj); err != nil {
|
|
return NewDecodeParamError(err.Error())
|
|
}
|
|
|
|
if len(obj) < 2 {
|
|
return NewInsufficientParamsError(len(obj), 2)
|
|
}
|
|
|
|
addstr, ok := obj[0].(string)
|
|
if !ok {
|
|
return NewInvalidTypeError("address", "not a string")
|
|
}
|
|
args.Address = addstr
|
|
|
|
keystr, ok := obj[1].(string)
|
|
if !ok {
|
|
return NewInvalidTypeError("key", "not a string")
|
|
}
|
|
args.Key = keystr
|
|
|
|
if len(obj) > 2 {
|
|
if err := blockHeight(obj[2], &args.BlockNumber); err != nil {
|
|
return err
|
|
}
|
|
} else {
|
|
args.BlockNumber = -1
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
type GetTxCountArgs struct {
|
|
Address string
|
|
BlockNumber int64
|
|
}
|
|
|
|
func (args *GetTxCountArgs) UnmarshalJSON(b []byte) (err error) {
|
|
var obj []interface{}
|
|
if err := json.Unmarshal(b, &obj); err != nil {
|
|
return NewDecodeParamError(err.Error())
|
|
}
|
|
|
|
if len(obj) < 1 {
|
|
return NewInsufficientParamsError(len(obj), 1)
|
|
}
|
|
|
|
addstr, ok := obj[0].(string)
|
|
if !ok {
|
|
return NewInvalidTypeError("address", "not a string")
|
|
}
|
|
args.Address = addstr
|
|
|
|
if len(obj) > 1 {
|
|
if err := blockHeight(obj[1], &args.BlockNumber); err != nil {
|
|
return err
|
|
}
|
|
} else {
|
|
args.BlockNumber = -1
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
type GetBalanceArgs struct {
|
|
Address string
|
|
BlockNumber int64
|
|
}
|
|
|
|
func (args *GetBalanceArgs) UnmarshalJSON(b []byte) (err error) {
|
|
var obj []interface{}
|
|
if err := json.Unmarshal(b, &obj); err != nil {
|
|
return NewDecodeParamError(err.Error())
|
|
}
|
|
|
|
if len(obj) < 1 {
|
|
return NewInsufficientParamsError(len(obj), 1)
|
|
}
|
|
|
|
addstr, ok := obj[0].(string)
|
|
if !ok {
|
|
return NewInvalidTypeError("address", "not a string")
|
|
}
|
|
args.Address = addstr
|
|
|
|
if len(obj) > 1 {
|
|
if err := blockHeight(obj[1], &args.BlockNumber); err != nil {
|
|
return err
|
|
}
|
|
} else {
|
|
args.BlockNumber = -1
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
type GetDataArgs struct {
|
|
Address string
|
|
BlockNumber int64
|
|
}
|
|
|
|
func (args *GetDataArgs) UnmarshalJSON(b []byte) (err error) {
|
|
var obj []interface{}
|
|
if err := json.Unmarshal(b, &obj); err != nil {
|
|
return NewDecodeParamError(err.Error())
|
|
}
|
|
|
|
if len(obj) < 1 {
|
|
return NewInsufficientParamsError(len(obj), 1)
|
|
}
|
|
|
|
addstr, ok := obj[0].(string)
|
|
if !ok {
|
|
return NewInvalidTypeError("address", "not a string")
|
|
}
|
|
args.Address = addstr
|
|
|
|
if len(obj) > 1 {
|
|
if err := blockHeight(obj[1], &args.BlockNumber); err != nil {
|
|
return err
|
|
}
|
|
} else {
|
|
args.BlockNumber = -1
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
type BlockNumArg struct {
|
|
BlockNumber int64
|
|
}
|
|
|
|
func (args *BlockNumArg) UnmarshalJSON(b []byte) (err error) {
|
|
var obj []interface{}
|
|
if err := json.Unmarshal(b, &obj); err != nil {
|
|
return NewDecodeParamError(err.Error())
|
|
}
|
|
|
|
if len(obj) < 1 {
|
|
return NewInsufficientParamsError(len(obj), 1)
|
|
}
|
|
|
|
if err := blockHeight(obj[0], &args.BlockNumber); err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
type BlockNumIndexArgs struct {
|
|
BlockNumber int64
|
|
Index int64
|
|
}
|
|
|
|
func (args *BlockNumIndexArgs) UnmarshalJSON(b []byte) (err error) {
|
|
var obj []interface{}
|
|
if err := json.Unmarshal(b, &obj); err != nil {
|
|
return NewDecodeParamError(err.Error())
|
|
}
|
|
|
|
if len(obj) < 2 {
|
|
return NewInsufficientParamsError(len(obj), 2)
|
|
}
|
|
|
|
if err := blockHeight(obj[0], &args.BlockNumber); err != nil {
|
|
return err
|
|
}
|
|
|
|
var arg1 *big.Int
|
|
if arg1, err = numString(obj[1]); err != nil {
|
|
return err
|
|
}
|
|
args.Index = arg1.Int64()
|
|
|
|
return nil
|
|
}
|
|
|
|
type HashArgs struct {
|
|
Hash string
|
|
}
|
|
|
|
func (args *HashArgs) UnmarshalJSON(b []byte) (err error) {
|
|
var obj []interface{}
|
|
if err := json.Unmarshal(b, &obj); err != nil {
|
|
return NewDecodeParamError(err.Error())
|
|
}
|
|
|
|
if len(obj) < 1 {
|
|
return NewInsufficientParamsError(len(obj), 1)
|
|
}
|
|
|
|
arg0, ok := obj[0].(string)
|
|
if !ok {
|
|
return NewInvalidTypeError("hash", "not a string")
|
|
}
|
|
args.Hash = arg0
|
|
|
|
return nil
|
|
}
|
|
|
|
type HashIndexArgs struct {
|
|
Hash string
|
|
Index int64
|
|
}
|
|
|
|
func (args *HashIndexArgs) UnmarshalJSON(b []byte) (err error) {
|
|
var obj []interface{}
|
|
if err := json.Unmarshal(b, &obj); err != nil {
|
|
return NewDecodeParamError(err.Error())
|
|
}
|
|
|
|
if len(obj) < 2 {
|
|
return NewInsufficientParamsError(len(obj), 2)
|
|
}
|
|
|
|
arg0, ok := obj[0].(string)
|
|
if !ok {
|
|
return NewInvalidTypeError("hash", "not a string")
|
|
}
|
|
args.Hash = arg0
|
|
|
|
arg1, ok := obj[1].(string)
|
|
if !ok {
|
|
return NewInvalidTypeError("index", "not a string")
|
|
}
|
|
args.Index = common.Big(arg1).Int64()
|
|
|
|
return nil
|
|
}
|
|
|
|
type Sha3Args struct {
|
|
Data string
|
|
}
|
|
|
|
func (args *Sha3Args) UnmarshalJSON(b []byte) (err error) {
|
|
var obj []interface{}
|
|
if err := json.Unmarshal(b, &obj); err != nil {
|
|
return NewDecodeParamError(err.Error())
|
|
}
|
|
|
|
if len(obj) < 1 {
|
|
return NewInsufficientParamsError(len(obj), 1)
|
|
}
|
|
|
|
argstr, ok := obj[0].(string)
|
|
if !ok {
|
|
return NewInvalidTypeError("data", "is not a string")
|
|
}
|
|
args.Data = argstr
|
|
return nil
|
|
}
|
|
|
|
type BlockFilterArgs struct {
|
|
Earliest int64
|
|
Latest int64
|
|
Address []string
|
|
Topics [][]string
|
|
Skip int
|
|
Max int
|
|
}
|
|
|
|
func (args *BlockFilterArgs) UnmarshalJSON(b []byte) (err error) {
|
|
var obj []struct {
|
|
FromBlock interface{} `json:"fromBlock"`
|
|
ToBlock interface{} `json:"toBlock"`
|
|
Limit interface{} `json:"limit"`
|
|
Offset interface{} `json:"offset"`
|
|
Address interface{} `json:"address"`
|
|
Topics interface{} `json:"topics"`
|
|
}
|
|
|
|
if err = json.Unmarshal(b, &obj); err != nil {
|
|
return NewDecodeParamError(err.Error())
|
|
}
|
|
|
|
if len(obj) < 1 {
|
|
return NewInsufficientParamsError(len(obj), 1)
|
|
}
|
|
|
|
// args.Earliest, err = toNumber(obj[0].ToBlock)
|
|
// if err != nil {
|
|
// return NewDecodeParamError(fmt.Sprintf("FromBlock %v", err))
|
|
// }
|
|
// args.Latest, err = toNumber(obj[0].FromBlock)
|
|
// if err != nil {
|
|
// return NewDecodeParamError(fmt.Sprintf("ToBlock %v", err))
|
|
|
|
var num int64
|
|
var numBig *big.Int
|
|
|
|
// if blank then latest
|
|
if obj[0].FromBlock == nil {
|
|
num = -1
|
|
} else {
|
|
if err := blockHeight(obj[0].FromBlock, &num); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
// if -2 or other "silly" number, use latest
|
|
if num < 0 {
|
|
args.Earliest = -1 //latest block
|
|
} else {
|
|
args.Earliest = num
|
|
}
|
|
|
|
// if blank than latest
|
|
if obj[0].ToBlock == nil {
|
|
num = -1
|
|
} else {
|
|
if err := blockHeight(obj[0].ToBlock, &num); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
args.Latest = num
|
|
|
|
if obj[0].Limit == nil {
|
|
numBig = big.NewInt(defaultLogLimit)
|
|
} else {
|
|
if numBig, err = numString(obj[0].Limit); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
args.Max = int(numBig.Int64())
|
|
|
|
if obj[0].Offset == nil {
|
|
numBig = big.NewInt(defaultLogOffset)
|
|
} else {
|
|
if numBig, err = numString(obj[0].Offset); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
args.Skip = int(numBig.Int64())
|
|
|
|
if obj[0].Address != nil {
|
|
marg, ok := obj[0].Address.([]interface{})
|
|
if ok {
|
|
v := make([]string, len(marg))
|
|
for i, arg := range marg {
|
|
argstr, ok := arg.(string)
|
|
if !ok {
|
|
return NewInvalidTypeError(fmt.Sprintf("address[%d]", i), "is not a string")
|
|
}
|
|
v[i] = argstr
|
|
}
|
|
args.Address = v
|
|
} else {
|
|
argstr, ok := obj[0].Address.(string)
|
|
if ok {
|
|
v := make([]string, 1)
|
|
v[0] = argstr
|
|
args.Address = v
|
|
} else {
|
|
return NewInvalidTypeError("address", "is not a string or array")
|
|
}
|
|
}
|
|
}
|
|
|
|
if obj[0].Topics != nil {
|
|
other, ok := obj[0].Topics.([]interface{})
|
|
if ok {
|
|
topicdbl := make([][]string, len(other))
|
|
for i, iv := range other {
|
|
if argstr, ok := iv.(string); ok {
|
|
// Found a string, push into first element of array
|
|
topicsgl := make([]string, 1)
|
|
topicsgl[0] = argstr
|
|
topicdbl[i] = topicsgl
|
|
} else if argarray, ok := iv.([]interface{}); ok {
|
|
// Found an array of other
|
|
topicdbl[i] = make([]string, len(argarray))
|
|
for j, jv := range argarray {
|
|
if v, ok := jv.(string); ok {
|
|
topicdbl[i][j] = v
|
|
} else if jv == nil {
|
|
topicdbl[i][j] = ""
|
|
} else {
|
|
return NewInvalidTypeError(fmt.Sprintf("topic[%d][%d]", i, j), "is not a string")
|
|
}
|
|
}
|
|
} else if iv == nil {
|
|
topicdbl[i] = []string{""}
|
|
} else {
|
|
return NewInvalidTypeError(fmt.Sprintf("topic[%d]", i), "not a string or array")
|
|
}
|
|
}
|
|
args.Topics = topicdbl
|
|
return nil
|
|
} else {
|
|
return NewInvalidTypeError("topic", "is not a string or array")
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
type DbArgs struct {
|
|
Database string
|
|
Key string
|
|
Value []byte
|
|
}
|
|
|
|
func (args *DbArgs) UnmarshalJSON(b []byte) (err error) {
|
|
var obj []interface{}
|
|
if err := json.Unmarshal(b, &obj); err != nil {
|
|
return NewDecodeParamError(err.Error())
|
|
}
|
|
|
|
if len(obj) < 2 {
|
|
return NewInsufficientParamsError(len(obj), 2)
|
|
}
|
|
|
|
var objstr string
|
|
var ok bool
|
|
|
|
if objstr, ok = obj[0].(string); !ok {
|
|
return NewInvalidTypeError("database", "not a string")
|
|
}
|
|
args.Database = objstr
|
|
|
|
if objstr, ok = obj[1].(string); !ok {
|
|
return NewInvalidTypeError("key", "not a string")
|
|
}
|
|
args.Key = objstr
|
|
|
|
if len(obj) > 2 {
|
|
objstr, ok = obj[2].(string)
|
|
if !ok {
|
|
return NewInvalidTypeError("value", "not a string")
|
|
}
|
|
|
|
args.Value = []byte(objstr)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (a *DbArgs) requirements() error {
|
|
if len(a.Database) == 0 {
|
|
return NewValidationError("Database", "cannot be blank")
|
|
}
|
|
if len(a.Key) == 0 {
|
|
return NewValidationError("Key", "cannot be blank")
|
|
}
|
|
return nil
|
|
}
|
|
|
|
type DbHexArgs struct {
|
|
Database string
|
|
Key string
|
|
Value []byte
|
|
}
|
|
|
|
func (args *DbHexArgs) UnmarshalJSON(b []byte) (err error) {
|
|
var obj []interface{}
|
|
if err := json.Unmarshal(b, &obj); err != nil {
|
|
return NewDecodeParamError(err.Error())
|
|
}
|
|
|
|
if len(obj) < 2 {
|
|
return NewInsufficientParamsError(len(obj), 2)
|
|
}
|
|
|
|
var objstr string
|
|
var ok bool
|
|
|
|
if objstr, ok = obj[0].(string); !ok {
|
|
return NewInvalidTypeError("database", "not a string")
|
|
}
|
|
args.Database = objstr
|
|
|
|
if objstr, ok = obj[1].(string); !ok {
|
|
return NewInvalidTypeError("key", "not a string")
|
|
}
|
|
args.Key = objstr
|
|
|
|
if len(obj) > 2 {
|
|
objstr, ok = obj[2].(string)
|
|
if !ok {
|
|
return NewInvalidTypeError("value", "not a string")
|
|
}
|
|
|
|
args.Value = common.FromHex(objstr)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (a *DbHexArgs) requirements() error {
|
|
if len(a.Database) == 0 {
|
|
return NewValidationError("Database", "cannot be blank")
|
|
}
|
|
if len(a.Key) == 0 {
|
|
return NewValidationError("Key", "cannot be blank")
|
|
}
|
|
return nil
|
|
}
|
|
|
|
type WhisperMessageArgs struct {
|
|
Payload string
|
|
To string
|
|
From string
|
|
Topics []string
|
|
Priority uint32
|
|
Ttl uint32
|
|
}
|
|
|
|
func (args *WhisperMessageArgs) UnmarshalJSON(b []byte) (err error) {
|
|
var obj []struct {
|
|
Payload string
|
|
To string
|
|
From string
|
|
Topics []string
|
|
Priority interface{}
|
|
Ttl interface{}
|
|
}
|
|
|
|
if err = json.Unmarshal(b, &obj); err != nil {
|
|
return NewDecodeParamError(err.Error())
|
|
}
|
|
|
|
if len(obj) < 1 {
|
|
return NewInsufficientParamsError(len(obj), 1)
|
|
}
|
|
args.Payload = obj[0].Payload
|
|
args.To = obj[0].To
|
|
args.From = obj[0].From
|
|
args.Topics = obj[0].Topics
|
|
|
|
var num *big.Int
|
|
if num, err = numString(obj[0].Priority); err != nil {
|
|
return err
|
|
}
|
|
args.Priority = uint32(num.Int64())
|
|
|
|
if num, err = numString(obj[0].Ttl); err != nil {
|
|
return err
|
|
}
|
|
args.Ttl = uint32(num.Int64())
|
|
|
|
return nil
|
|
}
|
|
|
|
type CompileArgs struct {
|
|
Source string
|
|
}
|
|
|
|
func (args *CompileArgs) UnmarshalJSON(b []byte) (err error) {
|
|
var obj []interface{}
|
|
if err := json.Unmarshal(b, &obj); err != nil {
|
|
return NewDecodeParamError(err.Error())
|
|
}
|
|
|
|
if len(obj) < 1 {
|
|
return NewInsufficientParamsError(len(obj), 1)
|
|
}
|
|
argstr, ok := obj[0].(string)
|
|
if !ok {
|
|
return NewInvalidTypeError("arg0", "is not a string")
|
|
}
|
|
args.Source = argstr
|
|
|
|
return nil
|
|
}
|
|
|
|
type FilterStringArgs struct {
|
|
Word string
|
|
}
|
|
|
|
func (args *FilterStringArgs) UnmarshalJSON(b []byte) (err error) {
|
|
var obj []interface{}
|
|
if err := json.Unmarshal(b, &obj); err != nil {
|
|
return NewDecodeParamError(err.Error())
|
|
}
|
|
|
|
if len(obj) < 1 {
|
|
return NewInsufficientParamsError(len(obj), 1)
|
|
}
|
|
|
|
var argstr string
|
|
argstr, ok := obj[0].(string)
|
|
if !ok {
|
|
return NewInvalidTypeError("filter", "not a string")
|
|
}
|
|
switch argstr {
|
|
case "latest", "pending":
|
|
break
|
|
default:
|
|
return NewValidationError("Word", "Must be `latest` or `pending`")
|
|
}
|
|
args.Word = argstr
|
|
return nil
|
|
}
|
|
|
|
type FilterIdArgs struct {
|
|
Id int
|
|
}
|
|
|
|
func (args *FilterIdArgs) UnmarshalJSON(b []byte) (err error) {
|
|
var obj []interface{}
|
|
if err := json.Unmarshal(b, &obj); err != nil {
|
|
return NewDecodeParamError(err.Error())
|
|
}
|
|
|
|
if len(obj) < 1 {
|
|
return NewInsufficientParamsError(len(obj), 1)
|
|
}
|
|
|
|
var num *big.Int
|
|
if num, err = numString(obj[0]); err != nil {
|
|
return err
|
|
}
|
|
args.Id = int(num.Int64())
|
|
|
|
return nil
|
|
}
|
|
|
|
type WhisperIdentityArgs struct {
|
|
Identity string
|
|
}
|
|
|
|
func (args *WhisperIdentityArgs) UnmarshalJSON(b []byte) (err error) {
|
|
var obj []interface{}
|
|
if err := json.Unmarshal(b, &obj); err != nil {
|
|
return NewDecodeParamError(err.Error())
|
|
}
|
|
|
|
if len(obj) < 1 {
|
|
return NewInsufficientParamsError(len(obj), 1)
|
|
}
|
|
|
|
argstr, ok := obj[0].(string)
|
|
if !ok {
|
|
return NewInvalidTypeError("arg0", "not a string")
|
|
}
|
|
// if !common.IsHex(argstr) {
|
|
// return NewValidationError("arg0", "not a hexstring")
|
|
// }
|
|
args.Identity = argstr
|
|
|
|
return nil
|
|
}
|
|
|
|
type WhisperFilterArgs struct {
|
|
To string
|
|
From string
|
|
Topics [][]string
|
|
}
|
|
|
|
// UnmarshalJSON implements the json.Unmarshaler interface, invoked to convert a
|
|
// JSON message blob into a WhisperFilterArgs structure.
|
|
func (args *WhisperFilterArgs) UnmarshalJSON(b []byte) (err error) {
|
|
// Unmarshal the JSON message and sanity check
|
|
var obj []struct {
|
|
To interface{} `json:"to"`
|
|
From interface{} `json:"from"`
|
|
Topics interface{} `json:"topics"`
|
|
}
|
|
if err := json.Unmarshal(b, &obj); err != nil {
|
|
return NewDecodeParamError(err.Error())
|
|
}
|
|
if len(obj) < 1 {
|
|
return NewInsufficientParamsError(len(obj), 1)
|
|
}
|
|
// Retrieve the simple data contents of the filter arguments
|
|
if obj[0].To == nil {
|
|
args.To = ""
|
|
} else {
|
|
argstr, ok := obj[0].To.(string)
|
|
if !ok {
|
|
return NewInvalidTypeError("to", "is not a string")
|
|
}
|
|
args.To = argstr
|
|
}
|
|
if obj[0].From == nil {
|
|
args.From = ""
|
|
} else {
|
|
argstr, ok := obj[0].From.(string)
|
|
if !ok {
|
|
return NewInvalidTypeError("from", "is not a string")
|
|
}
|
|
args.From = argstr
|
|
}
|
|
// Construct the nested topic array
|
|
if obj[0].Topics != nil {
|
|
// Make sure we have an actual topic array
|
|
list, ok := obj[0].Topics.([]interface{})
|
|
if !ok {
|
|
return NewInvalidTypeError("topics", "is not an array")
|
|
}
|
|
// Iterate over each topic and handle nil, string or array
|
|
topics := make([][]string, len(list))
|
|
for idx, field := range list {
|
|
switch value := field.(type) {
|
|
case nil:
|
|
topics[idx] = []string{}
|
|
|
|
case string:
|
|
topics[idx] = []string{value}
|
|
|
|
case []interface{}:
|
|
topics[idx] = make([]string, len(value))
|
|
for i, nested := range value {
|
|
switch value := nested.(type) {
|
|
case nil:
|
|
topics[idx][i] = ""
|
|
|
|
case string:
|
|
topics[idx][i] = value
|
|
|
|
default:
|
|
return NewInvalidTypeError(fmt.Sprintf("topic[%d][%d]", idx, i), "is not a string")
|
|
}
|
|
}
|
|
default:
|
|
return NewInvalidTypeError(fmt.Sprintf("topic[%d]", idx), "not a string or array")
|
|
}
|
|
}
|
|
args.Topics = topics
|
|
}
|
|
return nil
|
|
}
|
|
|
|
type SubmitWorkArgs struct {
|
|
Nonce uint64
|
|
Header string
|
|
Digest string
|
|
}
|
|
|
|
func (args *SubmitWorkArgs) UnmarshalJSON(b []byte) (err error) {
|
|
var obj []interface{}
|
|
if err = json.Unmarshal(b, &obj); err != nil {
|
|
return NewDecodeParamError(err.Error())
|
|
}
|
|
|
|
if len(obj) < 3 {
|
|
return NewInsufficientParamsError(len(obj), 3)
|
|
}
|
|
|
|
var objstr string
|
|
var ok bool
|
|
if objstr, ok = obj[0].(string); !ok {
|
|
return NewInvalidTypeError("nonce", "not a string")
|
|
}
|
|
|
|
args.Nonce = common.String2Big(objstr).Uint64()
|
|
if objstr, ok = obj[1].(string); !ok {
|
|
return NewInvalidTypeError("header", "not a string")
|
|
}
|
|
|
|
args.Header = objstr
|
|
|
|
if objstr, ok = obj[2].(string); !ok {
|
|
return NewInvalidTypeError("digest", "not a string")
|
|
}
|
|
|
|
args.Digest = objstr
|
|
|
|
return nil
|
|
}
|
|
|