aboutsummaryrefslogblamecommitdiffstats
path: root/rpc/util.go
blob: b3d83409fb7c1ed4d7c25158451764d895c66838 (plain) (tree)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16















                                                                            



                       
             
                  
                 
              
 
                                                
                                                
                                               
                                              

 
                                       
 






                                                                                                           
                                                       

         




                                                     




























                                                                                                           
                                                               






                                                                             
                                                               





                  
                          
                                                         

 






                                         
                                       
                                      
                                        






                                         
                                                           
                                               
                                                 
                                       
                                                    
                                                





                         



                                      
                    














                                                          
                   











                                            
/*
    This file is part of go-ethereum

    go-ethereum is free software: you can redistribute it and/or modify
    it under the terms of the GNU General Public License as published by
    the Free Software Foundation, either version 3 of the License, or
    (at your option) any later version.

    go-ethereum is distributed in the hope that it will be useful,
    but WITHOUT ANY WARRANTY; without even the implied warranty of
    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  GNU General Public License for more details.

    You should have received a copy of the GNU General Public License
    along with go-ethereum.  If not, see <http://www.gnu.org/licenses/>.
*/
package rpc

import (
    "encoding/json"
    "fmt"
    "math/big"
    "reflect"
    "time"

    "github.com/ethereum/go-ethereum/common"
    "github.com/ethereum/go-ethereum/logger"
    "github.com/ethereum/go-ethereum/state"
    "github.com/ethereum/go-ethereum/xeth"
)

var rpclogger = logger.NewLogger("RPC")

// Unmarshal state is a helper method which has the ability to decode messsages
// that use the `defaultBlock` (https://github.com/ethereum/wiki/wiki/JSON-RPC#the-default-block-parameter)
// For example a `call`: [{to: "0x....", data:"0x..."}, "latest"]. The first argument is the transaction
// message and the second one refers to the block height (or state) to which to apply this `call`.
func UnmarshalRawMessages(b []byte, iface interface{}, number *int64) (err error) {
    var data []json.RawMessage
    if err = json.Unmarshal(b, &data); err != nil && len(data) == 0 {
        return NewDecodeParamError(err.Error())
    }

    // Hrm... Occurs when no params
    if len(data) == 0 {
        return NewDecodeParamError("No data")
    }

    // Number index determines the index in the array for a possible block number
    numberIndex := 0

    value := reflect.ValueOf(iface)
    rvalue := reflect.Indirect(value)

    switch rvalue.Kind() {
    case reflect.Slice:
        // This is a bit of a cheat, but `data` is expected to be larger than 2 if iface is a slice
        if number != nil {
            numberIndex = len(data) - 1
        } else {
            numberIndex = len(data)
        }

        slice := reflect.MakeSlice(rvalue.Type(), numberIndex, numberIndex)
        for i, raw := range data[0:numberIndex] {
            v := slice.Index(i).Interface()
            if err = json.Unmarshal(raw, &v); err != nil {
                fmt.Println(err, v)
                return err
            }
            slice.Index(i).Set(reflect.ValueOf(v))
        }
        reflect.Indirect(rvalue).Set(slice) //value.Set(slice)
    case reflect.Struct:
        fallthrough
    default:
        if err = json.Unmarshal(data[0], iface); err != nil {
            return NewDecodeParamError(err.Error())
        }
        numberIndex = 1
    }

    // <0 index means out of bound for block number
    if numberIndex >= 0 && len(data) > numberIndex {
        if err = blockNumber(data[numberIndex], number); err != nil {
            return NewDecodeParamError(err.Error())
        }
    }

    return nil
}

func i2hex(n int) string {
    return common.ToHex(big.NewInt(int64(n)).Bytes())
}

type RpcServer interface {
    Start()
    Stop()
}

type Log struct {
    Address string   `json:"address"`
    Topic   []string `json:"topic"`
    Data    string   `json:"data"`
    Number  uint64   `json:"number"`
}

func toLogs(logs state.Logs) (ls []Log) {
    ls = make([]Log, len(logs))

    for i, log := range logs {
        var l Log
        l.Topic = make([]string, len(log.Topics()))
        l.Address = log.Address().Hex()
        l.Data = common.ToHex(log.Data())
        l.Number = log.Number()
        for j, topic := range log.Topics() {
            l.Topic[j] = topic.Hex()
        }
        ls[i] = l
    }

    return
}

type whisperFilter struct {
    messages []xeth.WhisperMessage
    timeout  time.Time
    id       int
}

func (w *whisperFilter) add(msgs ...xeth.WhisperMessage) {
    w.messages = append(w.messages, msgs...)
}
func (w *whisperFilter) get() []xeth.WhisperMessage {
    w.timeout = time.Now()
    tmp := w.messages
    w.messages = nil
    return tmp
}

type logFilter struct {
    logs    state.Logs
    timeout time.Time
    id      int
}

func (l *logFilter) add(logs ...state.Log) {
    l.logs = append(l.logs, logs...)
}

func (l *logFilter) get() state.Logs {
    l.timeout = time.Now()
    tmp := l.logs
    l.logs = nil
    return tmp
}