Merge pull request #3402 from fjl/ethclient-api-fixes
eth/filters, ethclient, ethereum: API improvements
This commit is contained in:
commit
7f79d249a6
@ -988,7 +988,7 @@ func (self *BlockChain) InsertChain(chain types.Blocks) (int, error) {
|
|||||||
glog.Infof("inserted forked block #%d [%x…] (TD=%v) in %9v: %3d txs %d uncles.", block.Number(), block.Hash().Bytes()[0:4], block.Difficulty(), common.PrettyDuration(time.Since(bstart)), len(block.Transactions()), len(block.Uncles()))
|
glog.Infof("inserted forked block #%d [%x…] (TD=%v) in %9v: %3d txs %d uncles.", block.Number(), block.Hash().Bytes()[0:4], block.Difficulty(), common.PrettyDuration(time.Since(bstart)), len(block.Transactions()), len(block.Uncles()))
|
||||||
}
|
}
|
||||||
blockInsertTimer.UpdateSince(bstart)
|
blockInsertTimer.UpdateSince(bstart)
|
||||||
events = append(events, ChainSideEvent{block, logs})
|
events = append(events, ChainSideEvent{block})
|
||||||
|
|
||||||
case SplitStatTy:
|
case SplitStatTy:
|
||||||
events = append(events, ChainSplitEvent{block, logs})
|
events = append(events, ChainSplitEvent{block, logs})
|
||||||
@ -1069,17 +1069,18 @@ func (self *BlockChain) reorg(oldBlock, newBlock *types.Block) error {
|
|||||||
newStart = newBlock
|
newStart = newBlock
|
||||||
deletedTxs types.Transactions
|
deletedTxs types.Transactions
|
||||||
deletedLogs vm.Logs
|
deletedLogs vm.Logs
|
||||||
deletedLogsByHash = make(map[common.Hash]vm.Logs)
|
|
||||||
// collectLogs collects the logs that were generated during the
|
// collectLogs collects the logs that were generated during the
|
||||||
// processing of the block that corresponds with the given hash.
|
// processing of the block that corresponds with the given hash.
|
||||||
// These logs are later announced as deleted.
|
// These logs are later announced as deleted.
|
||||||
collectLogs = func(h common.Hash) {
|
collectLogs = func(h common.Hash) {
|
||||||
// Coalesce logs
|
// Coalesce logs and set 'Removed'.
|
||||||
receipts := GetBlockReceipts(self.chainDb, h, self.hc.GetBlockNumber(h))
|
receipts := GetBlockReceipts(self.chainDb, h, self.hc.GetBlockNumber(h))
|
||||||
for _, receipt := range receipts {
|
for _, receipt := range receipts {
|
||||||
deletedLogs = append(deletedLogs, receipt.Logs...)
|
for _, log := range receipt.Logs {
|
||||||
|
del := *log
|
||||||
deletedLogsByHash[h] = receipt.Logs
|
del.Removed = true
|
||||||
|
deletedLogs = append(deletedLogs, &del)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
@ -1173,7 +1174,7 @@ func (self *BlockChain) reorg(oldBlock, newBlock *types.Block) error {
|
|||||||
if len(oldChain) > 0 {
|
if len(oldChain) > 0 {
|
||||||
go func() {
|
go func() {
|
||||||
for _, block := range oldChain {
|
for _, block := range oldChain {
|
||||||
self.eventMux.Post(ChainSideEvent{Block: block, Logs: deletedLogsByHash[block.Hash()]})
|
self.eventMux.Post(ChainSideEvent{Block: block})
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
}
|
}
|
||||||
|
@ -61,7 +61,6 @@ type ChainEvent struct {
|
|||||||
|
|
||||||
type ChainSideEvent struct {
|
type ChainSideEvent struct {
|
||||||
Block *types.Block
|
Block *types.Block
|
||||||
Logs vm.Logs
|
|
||||||
}
|
}
|
||||||
|
|
||||||
type PendingBlockEvent struct {
|
type PendingBlockEvent struct {
|
||||||
|
128
core/vm/log.go
128
core/vm/log.go
@ -29,20 +29,42 @@ import (
|
|||||||
|
|
||||||
var errMissingLogFields = errors.New("missing required JSON log fields")
|
var errMissingLogFields = errors.New("missing required JSON log fields")
|
||||||
|
|
||||||
// Log represents a contract log event. These events are generated by the LOG
|
// Log represents a contract log event. These events are generated by the LOG opcode and
|
||||||
// opcode and stored/indexed by the node.
|
// stored/indexed by the node.
|
||||||
type Log struct {
|
type Log struct {
|
||||||
// Consensus fields.
|
// Consensus fields.
|
||||||
Address common.Address // address of the contract that generated the event
|
Address common.Address // address of the contract that generated the event
|
||||||
Topics []common.Hash // list of topics provided by the contract.
|
Topics []common.Hash // list of topics provided by the contract.
|
||||||
Data []byte // supplied by the contract, usually ABI-encoded
|
Data []byte // supplied by the contract, usually ABI-encoded
|
||||||
|
|
||||||
// Derived fields (don't reorder!).
|
// Derived fields. These fields are filled in by the node
|
||||||
|
// but not secured by consensus.
|
||||||
BlockNumber uint64 // block in which the transaction was included
|
BlockNumber uint64 // block in which the transaction was included
|
||||||
TxHash common.Hash // hash of the transaction
|
TxHash common.Hash // hash of the transaction
|
||||||
TxIndex uint // index of the transaction in the block
|
TxIndex uint // index of the transaction in the block
|
||||||
BlockHash common.Hash // hash of the block in which the transaction was included
|
BlockHash common.Hash // hash of the block in which the transaction was included
|
||||||
Index uint // index of the log in the receipt
|
Index uint // index of the log in the receipt
|
||||||
|
|
||||||
|
// The Removed field is true if this log was reverted due to a chain reorganisation.
|
||||||
|
// You must pay attention to this field if you receive logs through a filter query.
|
||||||
|
Removed bool
|
||||||
|
}
|
||||||
|
|
||||||
|
type rlpLog struct {
|
||||||
|
Address common.Address
|
||||||
|
Topics []common.Hash
|
||||||
|
Data []byte
|
||||||
|
}
|
||||||
|
|
||||||
|
type rlpStorageLog struct {
|
||||||
|
Address common.Address
|
||||||
|
Topics []common.Hash
|
||||||
|
Data []byte
|
||||||
|
BlockNumber uint64
|
||||||
|
TxHash common.Hash
|
||||||
|
TxIndex uint
|
||||||
|
BlockHash common.Hash
|
||||||
|
Index uint
|
||||||
}
|
}
|
||||||
|
|
||||||
type jsonLog struct {
|
type jsonLog struct {
|
||||||
@ -54,73 +76,115 @@ type jsonLog struct {
|
|||||||
TxHash *common.Hash `json:"transactionHash"`
|
TxHash *common.Hash `json:"transactionHash"`
|
||||||
BlockHash *common.Hash `json:"blockHash"`
|
BlockHash *common.Hash `json:"blockHash"`
|
||||||
Index *hexutil.Uint `json:"logIndex"`
|
Index *hexutil.Uint `json:"logIndex"`
|
||||||
|
Removed bool `json:"removed"`
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewLog(address common.Address, topics []common.Hash, data []byte, number uint64) *Log {
|
func NewLog(address common.Address, topics []common.Hash, data []byte, number uint64) *Log {
|
||||||
return &Log{Address: address, Topics: topics, Data: data, BlockNumber: number}
|
return &Log{Address: address, Topics: topics, Data: data, BlockNumber: number}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// EncodeRLP implements rlp.Encoder.
|
||||||
func (l *Log) EncodeRLP(w io.Writer) error {
|
func (l *Log) EncodeRLP(w io.Writer) error {
|
||||||
return rlp.Encode(w, []interface{}{l.Address, l.Topics, l.Data})
|
return rlp.Encode(w, rlpLog{Address: l.Address, Topics: l.Topics, Data: l.Data})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// DecodeRLP implements rlp.Decoder.
|
||||||
func (l *Log) DecodeRLP(s *rlp.Stream) error {
|
func (l *Log) DecodeRLP(s *rlp.Stream) error {
|
||||||
var log struct {
|
var dec rlpLog
|
||||||
Address common.Address
|
err := s.Decode(&dec)
|
||||||
Topics []common.Hash
|
if err == nil {
|
||||||
Data []byte
|
l.Address, l.Topics, l.Data = dec.Address, dec.Topics, dec.Data
|
||||||
}
|
}
|
||||||
if err := s.Decode(&log); err != nil {
|
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
l.Address, l.Topics, l.Data = log.Address, log.Topics, log.Data
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (l *Log) String() string {
|
func (l *Log) String() string {
|
||||||
return fmt.Sprintf(`log: %x %x %x %x %d %x %d`, l.Address, l.Topics, l.Data, l.TxHash, l.TxIndex, l.BlockHash, l.Index)
|
return fmt.Sprintf(`log: %x %x %x %x %d %x %d`, l.Address, l.Topics, l.Data, l.TxHash, l.TxIndex, l.BlockHash, l.Index)
|
||||||
}
|
}
|
||||||
|
|
||||||
// MarshalJSON implements json.Marshaler.
|
// MarshalJSON implements json.Marshaler.
|
||||||
func (r *Log) MarshalJSON() ([]byte, error) {
|
func (l *Log) MarshalJSON() ([]byte, error) {
|
||||||
return json.Marshal(&jsonLog{
|
jslog := &jsonLog{
|
||||||
Address: &r.Address,
|
Address: &l.Address,
|
||||||
Topics: &r.Topics,
|
Topics: &l.Topics,
|
||||||
Data: (*hexutil.Bytes)(&r.Data),
|
Data: (*hexutil.Bytes)(&l.Data),
|
||||||
BlockNumber: (*hexutil.Uint64)(&r.BlockNumber),
|
TxIndex: (*hexutil.Uint)(&l.TxIndex),
|
||||||
TxIndex: (*hexutil.Uint)(&r.TxIndex),
|
TxHash: &l.TxHash,
|
||||||
TxHash: &r.TxHash,
|
Index: (*hexutil.Uint)(&l.Index),
|
||||||
BlockHash: &r.BlockHash,
|
Removed: l.Removed,
|
||||||
Index: (*hexutil.Uint)(&r.Index),
|
}
|
||||||
})
|
// Set block information for mined logs.
|
||||||
|
if (l.BlockHash != common.Hash{}) {
|
||||||
|
jslog.BlockHash = &l.BlockHash
|
||||||
|
jslog.BlockNumber = (*hexutil.Uint64)(&l.BlockNumber)
|
||||||
|
}
|
||||||
|
return json.Marshal(jslog)
|
||||||
}
|
}
|
||||||
|
|
||||||
// UnmarshalJSON implements json.Umarshaler.
|
// UnmarshalJSON implements json.Umarshaler.
|
||||||
func (r *Log) UnmarshalJSON(input []byte) error {
|
func (l *Log) UnmarshalJSON(input []byte) error {
|
||||||
var dec jsonLog
|
var dec jsonLog
|
||||||
if err := json.Unmarshal(input, &dec); err != nil {
|
if err := json.Unmarshal(input, &dec); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
if dec.Address == nil || dec.Topics == nil || dec.Data == nil || dec.BlockNumber == nil ||
|
if dec.Address == nil || dec.Topics == nil || dec.Data == nil ||
|
||||||
dec.TxIndex == nil || dec.TxHash == nil || dec.BlockHash == nil || dec.Index == nil {
|
dec.TxIndex == nil || dec.TxHash == nil || dec.Index == nil {
|
||||||
return errMissingLogFields
|
return errMissingLogFields
|
||||||
}
|
}
|
||||||
*r = Log{
|
declog := Log{
|
||||||
Address: *dec.Address,
|
Address: *dec.Address,
|
||||||
Topics: *dec.Topics,
|
Topics: *dec.Topics,
|
||||||
Data: *dec.Data,
|
Data: *dec.Data,
|
||||||
BlockNumber: uint64(*dec.BlockNumber),
|
|
||||||
TxHash: *dec.TxHash,
|
TxHash: *dec.TxHash,
|
||||||
TxIndex: uint(*dec.TxIndex),
|
TxIndex: uint(*dec.TxIndex),
|
||||||
BlockHash: *dec.BlockHash,
|
|
||||||
Index: uint(*dec.Index),
|
Index: uint(*dec.Index),
|
||||||
|
Removed: dec.Removed,
|
||||||
}
|
}
|
||||||
|
// Block information may be missing if the log is received through
|
||||||
|
// the pending log filter, so it's handled specially here.
|
||||||
|
if dec.BlockHash != nil && dec.BlockNumber != nil {
|
||||||
|
declog.BlockHash = *dec.BlockHash
|
||||||
|
declog.BlockNumber = uint64(*dec.BlockNumber)
|
||||||
|
}
|
||||||
|
*l = declog
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
type Logs []*Log
|
type Logs []*Log
|
||||||
|
|
||||||
// LogForStorage is a wrapper around a Log that flattens and parses the entire
|
// LogForStorage is a wrapper around a Log that flattens and parses the entire content of
|
||||||
// content of a log, as opposed to only the consensus fields originally (by hiding
|
// a log including non-consensus fields.
|
||||||
// the rlp interface methods).
|
|
||||||
type LogForStorage Log
|
type LogForStorage Log
|
||||||
|
|
||||||
|
// EncodeRLP implements rlp.Encoder.
|
||||||
|
func (l *LogForStorage) EncodeRLP(w io.Writer) error {
|
||||||
|
return rlp.Encode(w, rlpStorageLog{
|
||||||
|
Address: l.Address,
|
||||||
|
Topics: l.Topics,
|
||||||
|
Data: l.Data,
|
||||||
|
BlockNumber: l.BlockNumber,
|
||||||
|
TxHash: l.TxHash,
|
||||||
|
TxIndex: l.TxIndex,
|
||||||
|
BlockHash: l.BlockHash,
|
||||||
|
Index: l.Index,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// DecodeRLP implements rlp.Decoder.
|
||||||
|
func (l *LogForStorage) DecodeRLP(s *rlp.Stream) error {
|
||||||
|
var dec rlpStorageLog
|
||||||
|
err := s.Decode(&dec)
|
||||||
|
if err == nil {
|
||||||
|
*l = LogForStorage{
|
||||||
|
Address: dec.Address,
|
||||||
|
Topics: dec.Topics,
|
||||||
|
Data: dec.Data,
|
||||||
|
BlockNumber: dec.BlockNumber,
|
||||||
|
TxHash: dec.TxHash,
|
||||||
|
TxIndex: dec.TxIndex,
|
||||||
|
BlockHash: dec.BlockHash,
|
||||||
|
Index: dec.Index,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
@ -18,18 +18,81 @@ package vm
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
|
"reflect"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
|
"github.com/davecgh/go-spew/spew"
|
||||||
|
"github.com/ethereum/go-ethereum/common"
|
||||||
|
"github.com/ethereum/go-ethereum/common/hexutil"
|
||||||
)
|
)
|
||||||
|
|
||||||
var unmarshalLogTests = map[string]struct {
|
var unmarshalLogTests = map[string]struct {
|
||||||
input string
|
input string
|
||||||
|
want *Log
|
||||||
wantError error
|
wantError error
|
||||||
}{
|
}{
|
||||||
"ok": {
|
"ok": {
|
||||||
input: `{"address":"0xecf8f87f810ecf450940c9f60066b4a7a501d6a7","blockHash":"0x656c34545f90a730a19008c0e7a7cd4fb3895064b48d6d69761bd5abad681056","blockNumber":"0x1ecfa4","data":"0x000000000000000000000000000000000000000000000001a055690d9db80000","logIndex":"0x2","topics":["0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef","0x00000000000000000000000080b2c9d7cbbf30a1b0fc8983c647d754c6525615","0x000000000000000000000000f9dff387dcb5cc4cca5b91adb07a95f54e9f1bb6"],"transactionHash":"0x3b198bfd5d2907285af009e9ae84a0ecd63677110d89d7e030251acb87f6487e","transactionIndex":"0x3"}`,
|
input: `{"address":"0xecf8f87f810ecf450940c9f60066b4a7a501d6a7","blockHash":"0x656c34545f90a730a19008c0e7a7cd4fb3895064b48d6d69761bd5abad681056","blockNumber":"0x1ecfa4","data":"0x000000000000000000000000000000000000000000000001a055690d9db80000","logIndex":"0x2","topics":["0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef","0x00000000000000000000000080b2c9d7cbbf30a1b0fc8983c647d754c6525615"],"transactionHash":"0x3b198bfd5d2907285af009e9ae84a0ecd63677110d89d7e030251acb87f6487e","transactionIndex":"0x3"}`,
|
||||||
|
want: &Log{
|
||||||
|
Address: common.HexToAddress("0xecf8f87f810ecf450940c9f60066b4a7a501d6a7"),
|
||||||
|
BlockHash: common.HexToHash("0x656c34545f90a730a19008c0e7a7cd4fb3895064b48d6d69761bd5abad681056"),
|
||||||
|
BlockNumber: 2019236,
|
||||||
|
Data: hexutil.MustDecode("0x000000000000000000000000000000000000000000000001a055690d9db80000"),
|
||||||
|
Index: 2,
|
||||||
|
TxIndex: 3,
|
||||||
|
TxHash: common.HexToHash("0x3b198bfd5d2907285af009e9ae84a0ecd63677110d89d7e030251acb87f6487e"),
|
||||||
|
Topics: []common.Hash{
|
||||||
|
common.HexToHash("0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef"),
|
||||||
|
common.HexToHash("0x00000000000000000000000080b2c9d7cbbf30a1b0fc8983c647d754c6525615"),
|
||||||
|
},
|
||||||
|
},
|
||||||
},
|
},
|
||||||
"empty data": {
|
"empty data": {
|
||||||
input: `{"address":"0xecf8f87f810ecf450940c9f60066b4a7a501d6a7","blockHash":"0x656c34545f90a730a19008c0e7a7cd4fb3895064b48d6d69761bd5abad681056","blockNumber":"0x1ecfa4","data":"0x","logIndex":"0x2","topics":["0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef","0x00000000000000000000000080b2c9d7cbbf30a1b0fc8983c647d754c6525615","0x000000000000000000000000f9dff387dcb5cc4cca5b91adb07a95f54e9f1bb6"],"transactionHash":"0x3b198bfd5d2907285af009e9ae84a0ecd63677110d89d7e030251acb87f6487e","transactionIndex":"0x3"}`,
|
input: `{"address":"0xecf8f87f810ecf450940c9f60066b4a7a501d6a7","blockHash":"0x656c34545f90a730a19008c0e7a7cd4fb3895064b48d6d69761bd5abad681056","blockNumber":"0x1ecfa4","data":"0x","logIndex":"0x2","topics":["0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef","0x00000000000000000000000080b2c9d7cbbf30a1b0fc8983c647d754c6525615"],"transactionHash":"0x3b198bfd5d2907285af009e9ae84a0ecd63677110d89d7e030251acb87f6487e","transactionIndex":"0x3"}`,
|
||||||
|
want: &Log{
|
||||||
|
Address: common.HexToAddress("0xecf8f87f810ecf450940c9f60066b4a7a501d6a7"),
|
||||||
|
BlockHash: common.HexToHash("0x656c34545f90a730a19008c0e7a7cd4fb3895064b48d6d69761bd5abad681056"),
|
||||||
|
BlockNumber: 2019236,
|
||||||
|
Data: []byte{},
|
||||||
|
Index: 2,
|
||||||
|
TxIndex: 3,
|
||||||
|
TxHash: common.HexToHash("0x3b198bfd5d2907285af009e9ae84a0ecd63677110d89d7e030251acb87f6487e"),
|
||||||
|
Topics: []common.Hash{
|
||||||
|
common.HexToHash("0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef"),
|
||||||
|
common.HexToHash("0x00000000000000000000000080b2c9d7cbbf30a1b0fc8983c647d754c6525615"),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"missing block fields (pending logs)": {
|
||||||
|
input: `{"address":"0xecf8f87f810ecf450940c9f60066b4a7a501d6a7","data":"0x","logIndex":"0x0","topics":["0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef"],"transactionHash":"0x3b198bfd5d2907285af009e9ae84a0ecd63677110d89d7e030251acb87f6487e","transactionIndex":"0x3"}`,
|
||||||
|
want: &Log{
|
||||||
|
Address: common.HexToAddress("0xecf8f87f810ecf450940c9f60066b4a7a501d6a7"),
|
||||||
|
BlockHash: common.Hash{},
|
||||||
|
BlockNumber: 0,
|
||||||
|
Data: []byte{},
|
||||||
|
Index: 0,
|
||||||
|
TxIndex: 3,
|
||||||
|
TxHash: common.HexToHash("0x3b198bfd5d2907285af009e9ae84a0ecd63677110d89d7e030251acb87f6487e"),
|
||||||
|
Topics: []common.Hash{
|
||||||
|
common.HexToHash("0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef"),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"Removed: true": {
|
||||||
|
input: `{"address":"0xecf8f87f810ecf450940c9f60066b4a7a501d6a7","blockHash":"0x656c34545f90a730a19008c0e7a7cd4fb3895064b48d6d69761bd5abad681056","blockNumber":"0x1ecfa4","data":"0x","logIndex":"0x2","topics":["0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef"],"transactionHash":"0x3b198bfd5d2907285af009e9ae84a0ecd63677110d89d7e030251acb87f6487e","transactionIndex":"0x3","removed":true}`,
|
||||||
|
want: &Log{
|
||||||
|
Address: common.HexToAddress("0xecf8f87f810ecf450940c9f60066b4a7a501d6a7"),
|
||||||
|
BlockHash: common.HexToHash("0x656c34545f90a730a19008c0e7a7cd4fb3895064b48d6d69761bd5abad681056"),
|
||||||
|
BlockNumber: 2019236,
|
||||||
|
Data: []byte{},
|
||||||
|
Index: 2,
|
||||||
|
TxIndex: 3,
|
||||||
|
TxHash: common.HexToHash("0x3b198bfd5d2907285af009e9ae84a0ecd63677110d89d7e030251acb87f6487e"),
|
||||||
|
Topics: []common.Hash{
|
||||||
|
common.HexToHash("0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef"),
|
||||||
|
},
|
||||||
|
Removed: true,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
"missing data": {
|
"missing data": {
|
||||||
input: `{"address":"0xecf8f87f810ecf450940c9f60066b4a7a501d6a7","blockHash":"0x656c34545f90a730a19008c0e7a7cd4fb3895064b48d6d69761bd5abad681056","blockNumber":"0x1ecfa4","logIndex":"0x2","topics":["0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef","0x00000000000000000000000080b2c9d7cbbf30a1b0fc8983c647d754c6525615","0x000000000000000000000000f9dff387dcb5cc4cca5b91adb07a95f54e9f1bb6"],"transactionHash":"0x3b198bfd5d2907285af009e9ae84a0ecd63677110d89d7e030251acb87f6487e","transactionIndex":"0x3"}`,
|
input: `{"address":"0xecf8f87f810ecf450940c9f60066b4a7a501d6a7","blockHash":"0x656c34545f90a730a19008c0e7a7cd4fb3895064b48d6d69761bd5abad681056","blockNumber":"0x1ecfa4","logIndex":"0x2","topics":["0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef","0x00000000000000000000000080b2c9d7cbbf30a1b0fc8983c647d754c6525615","0x000000000000000000000000f9dff387dcb5cc4cca5b91adb07a95f54e9f1bb6"],"transactionHash":"0x3b198bfd5d2907285af009e9ae84a0ecd63677110d89d7e030251acb87f6487e","transactionIndex":"0x3"}`,
|
||||||
@ -38,10 +101,16 @@ var unmarshalLogTests = map[string]struct {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestUnmarshalLog(t *testing.T) {
|
func TestUnmarshalLog(t *testing.T) {
|
||||||
|
dumper := spew.ConfigState{DisableMethods: true, Indent: " "}
|
||||||
for name, test := range unmarshalLogTests {
|
for name, test := range unmarshalLogTests {
|
||||||
var log *Log
|
var log *Log
|
||||||
err := json.Unmarshal([]byte(test.input), &log)
|
err := json.Unmarshal([]byte(test.input), &log)
|
||||||
checkError(t, name, err, test.wantError)
|
checkError(t, name, err, test.wantError)
|
||||||
|
if test.wantError == nil && err == nil {
|
||||||
|
if !reflect.DeepEqual(log, test.want) {
|
||||||
|
t.Errorf("test %q:\nGOT %sWANT %s", name, dumper.Sdump(log), dumper.Sdump(test.want))
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -29,6 +29,7 @@ import (
|
|||||||
"github.com/ethereum/go-ethereum/common"
|
"github.com/ethereum/go-ethereum/common"
|
||||||
"github.com/ethereum/go-ethereum/common/hexutil"
|
"github.com/ethereum/go-ethereum/common/hexutil"
|
||||||
"github.com/ethereum/go-ethereum/core/types"
|
"github.com/ethereum/go-ethereum/core/types"
|
||||||
|
"github.com/ethereum/go-ethereum/core/vm"
|
||||||
"github.com/ethereum/go-ethereum/ethdb"
|
"github.com/ethereum/go-ethereum/ethdb"
|
||||||
"github.com/ethereum/go-ethereum/event"
|
"github.com/ethereum/go-ethereum/event"
|
||||||
"github.com/ethereum/go-ethereum/rpc"
|
"github.com/ethereum/go-ethereum/rpc"
|
||||||
@ -45,7 +46,7 @@ type filter struct {
|
|||||||
deadline *time.Timer // filter is inactiv when deadline triggers
|
deadline *time.Timer // filter is inactiv when deadline triggers
|
||||||
hashes []common.Hash
|
hashes []common.Hash
|
||||||
crit FilterCriteria
|
crit FilterCriteria
|
||||||
logs []Log
|
logs []*vm.Log
|
||||||
s *Subscription // associated subscription in event system
|
s *Subscription // associated subscription in event system
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -241,7 +242,7 @@ func (api *PublicFilterAPI) Logs(ctx context.Context, crit FilterCriteria) (*rpc
|
|||||||
|
|
||||||
var (
|
var (
|
||||||
rpcSub = notifier.CreateSubscription()
|
rpcSub = notifier.CreateSubscription()
|
||||||
matchedLogs = make(chan []Log)
|
matchedLogs = make(chan []*vm.Log)
|
||||||
)
|
)
|
||||||
|
|
||||||
logsSub, err := api.events.SubscribeLogs(crit, matchedLogs)
|
logsSub, err := api.events.SubscribeLogs(crit, matchedLogs)
|
||||||
@ -292,14 +293,14 @@ type FilterCriteria struct {
|
|||||||
//
|
//
|
||||||
// https://github.com/ethereum/wiki/wiki/JSON-RPC#eth_newfilter
|
// https://github.com/ethereum/wiki/wiki/JSON-RPC#eth_newfilter
|
||||||
func (api *PublicFilterAPI) NewFilter(crit FilterCriteria) (rpc.ID, error) {
|
func (api *PublicFilterAPI) NewFilter(crit FilterCriteria) (rpc.ID, error) {
|
||||||
logs := make(chan []Log)
|
logs := make(chan []*vm.Log)
|
||||||
logsSub, err := api.events.SubscribeLogs(crit, logs)
|
logsSub, err := api.events.SubscribeLogs(crit, logs)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return rpc.ID(""), err
|
return rpc.ID(""), err
|
||||||
}
|
}
|
||||||
|
|
||||||
api.filtersMu.Lock()
|
api.filtersMu.Lock()
|
||||||
api.filters[logsSub.ID] = &filter{typ: LogsSubscription, crit: crit, deadline: time.NewTimer(deadline), logs: make([]Log, 0), s: logsSub}
|
api.filters[logsSub.ID] = &filter{typ: LogsSubscription, crit: crit, deadline: time.NewTimer(deadline), logs: make([]*vm.Log, 0), s: logsSub}
|
||||||
api.filtersMu.Unlock()
|
api.filtersMu.Unlock()
|
||||||
|
|
||||||
go func() {
|
go func() {
|
||||||
@ -326,7 +327,7 @@ func (api *PublicFilterAPI) NewFilter(crit FilterCriteria) (rpc.ID, error) {
|
|||||||
// GetLogs returns logs matching the given argument that are stored within the state.
|
// GetLogs returns logs matching the given argument that are stored within the state.
|
||||||
//
|
//
|
||||||
// https://github.com/ethereum/wiki/wiki/JSON-RPC#eth_getlogs
|
// https://github.com/ethereum/wiki/wiki/JSON-RPC#eth_getlogs
|
||||||
func (api *PublicFilterAPI) GetLogs(ctx context.Context, crit FilterCriteria) ([]Log, error) {
|
func (api *PublicFilterAPI) GetLogs(ctx context.Context, crit FilterCriteria) ([]*vm.Log, error) {
|
||||||
if crit.FromBlock == nil {
|
if crit.FromBlock == nil {
|
||||||
crit.FromBlock = big.NewInt(rpc.LatestBlockNumber.Int64())
|
crit.FromBlock = big.NewInt(rpc.LatestBlockNumber.Int64())
|
||||||
}
|
}
|
||||||
@ -365,7 +366,7 @@ func (api *PublicFilterAPI) UninstallFilter(id rpc.ID) bool {
|
|||||||
// If the filter could not be found an empty array of logs is returned.
|
// If the filter could not be found an empty array of logs is returned.
|
||||||
//
|
//
|
||||||
// https://github.com/ethereum/wiki/wiki/JSON-RPC#eth_getfilterlogs
|
// https://github.com/ethereum/wiki/wiki/JSON-RPC#eth_getfilterlogs
|
||||||
func (api *PublicFilterAPI) GetFilterLogs(ctx context.Context, id rpc.ID) ([]Log, error) {
|
func (api *PublicFilterAPI) GetFilterLogs(ctx context.Context, id rpc.ID) ([]*vm.Log, error) {
|
||||||
api.filtersMu.Lock()
|
api.filtersMu.Lock()
|
||||||
f, found := api.filters[id]
|
f, found := api.filters[id]
|
||||||
api.filtersMu.Unlock()
|
api.filtersMu.Unlock()
|
||||||
@ -440,9 +441,9 @@ func returnHashes(hashes []common.Hash) []common.Hash {
|
|||||||
|
|
||||||
// returnLogs is a helper that will return an empty log array in case the given logs array is nil,
|
// returnLogs is a helper that will return an empty log array in case the given logs array is nil,
|
||||||
// otherwise the given logs array is returned.
|
// otherwise the given logs array is returned.
|
||||||
func returnLogs(logs []Log) []Log {
|
func returnLogs(logs []*vm.Log) []*vm.Log {
|
||||||
if logs == nil {
|
if logs == nil {
|
||||||
return []Log{}
|
return []*vm.Log{}
|
||||||
}
|
}
|
||||||
return logs
|
return logs
|
||||||
}
|
}
|
||||||
|
@ -25,6 +25,7 @@ import (
|
|||||||
"github.com/ethereum/go-ethereum/common"
|
"github.com/ethereum/go-ethereum/common"
|
||||||
"github.com/ethereum/go-ethereum/core"
|
"github.com/ethereum/go-ethereum/core"
|
||||||
"github.com/ethereum/go-ethereum/core/types"
|
"github.com/ethereum/go-ethereum/core/types"
|
||||||
|
"github.com/ethereum/go-ethereum/core/vm"
|
||||||
"github.com/ethereum/go-ethereum/ethdb"
|
"github.com/ethereum/go-ethereum/ethdb"
|
||||||
"github.com/ethereum/go-ethereum/event"
|
"github.com/ethereum/go-ethereum/event"
|
||||||
"github.com/ethereum/go-ethereum/rpc"
|
"github.com/ethereum/go-ethereum/rpc"
|
||||||
@ -38,7 +39,7 @@ type Backend interface {
|
|||||||
GetReceipts(ctx context.Context, blockHash common.Hash) (types.Receipts, error)
|
GetReceipts(ctx context.Context, blockHash common.Hash) (types.Receipts, error)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Filter can be used to retrieve and filter logs
|
// Filter can be used to retrieve and filter logs.
|
||||||
type Filter struct {
|
type Filter struct {
|
||||||
backend Backend
|
backend Backend
|
||||||
useMipMap bool
|
useMipMap bool
|
||||||
@ -85,7 +86,7 @@ func (f *Filter) SetTopics(topics [][]common.Hash) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Run filters logs with the current parameters set
|
// Run filters logs with the current parameters set
|
||||||
func (f *Filter) Find(ctx context.Context) ([]Log, error) {
|
func (f *Filter) Find(ctx context.Context) ([]*vm.Log, error) {
|
||||||
head, _ := f.backend.HeaderByNumber(ctx, rpc.LatestBlockNumber)
|
head, _ := f.backend.HeaderByNumber(ctx, rpc.LatestBlockNumber)
|
||||||
if head == nil {
|
if head == nil {
|
||||||
return nil, nil
|
return nil, nil
|
||||||
@ -110,7 +111,7 @@ func (f *Filter) Find(ctx context.Context) ([]Log, error) {
|
|||||||
return f.mipFind(beginBlockNo, endBlockNo, 0), nil
|
return f.mipFind(beginBlockNo, endBlockNo, 0), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (f *Filter) mipFind(start, end uint64, depth int) (logs []Log) {
|
func (f *Filter) mipFind(start, end uint64, depth int) (logs []*vm.Log) {
|
||||||
level := core.MIPMapLevels[depth]
|
level := core.MIPMapLevels[depth]
|
||||||
// normalise numerator so we can work in level specific batches and
|
// normalise numerator so we can work in level specific batches and
|
||||||
// work with the proper range checks
|
// work with the proper range checks
|
||||||
@ -141,7 +142,7 @@ func (f *Filter) mipFind(start, end uint64, depth int) (logs []Log) {
|
|||||||
return logs
|
return logs
|
||||||
}
|
}
|
||||||
|
|
||||||
func (f *Filter) getLogs(ctx context.Context, start, end uint64) (logs []Log, err error) {
|
func (f *Filter) getLogs(ctx context.Context, start, end uint64) (logs []*vm.Log, err error) {
|
||||||
for i := start; i <= end; i++ {
|
for i := start; i <= end; i++ {
|
||||||
header, err := f.backend.HeaderByNumber(ctx, rpc.BlockNumber(i))
|
header, err := f.backend.HeaderByNumber(ctx, rpc.BlockNumber(i))
|
||||||
if header == nil || err != nil {
|
if header == nil || err != nil {
|
||||||
@ -156,13 +157,9 @@ func (f *Filter) getLogs(ctx context.Context, start, end uint64) (logs []Log, er
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
var unfiltered []Log
|
var unfiltered []*vm.Log
|
||||||
for _, receipt := range receipts {
|
for _, receipt := range receipts {
|
||||||
rl := make([]Log, len(receipt.Logs))
|
unfiltered = append(unfiltered, ([]*vm.Log)(receipt.Logs)...)
|
||||||
for i, l := range receipt.Logs {
|
|
||||||
rl[i] = Log{l, false}
|
|
||||||
}
|
|
||||||
unfiltered = append(unfiltered, rl...)
|
|
||||||
}
|
}
|
||||||
logs = append(logs, filterLogs(unfiltered, nil, nil, f.addresses, f.topics)...)
|
logs = append(logs, filterLogs(unfiltered, nil, nil, f.addresses, f.topics)...)
|
||||||
}
|
}
|
||||||
@ -181,15 +178,15 @@ func includes(addresses []common.Address, a common.Address) bool {
|
|||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
func filterLogs(logs []Log, fromBlock, toBlock *big.Int, addresses []common.Address, topics [][]common.Hash) []Log {
|
// filterLogs creates a slice of logs matching the given criteria.
|
||||||
var ret []Log
|
func filterLogs(logs []*vm.Log, fromBlock, toBlock *big.Int, addresses []common.Address, topics [][]common.Hash) []*vm.Log {
|
||||||
// Filter the logs for interesting stuff
|
var ret []*vm.Log
|
||||||
Logs:
|
Logs:
|
||||||
for _, log := range logs {
|
for _, log := range logs {
|
||||||
if fromBlock != nil && fromBlock.Int64() >= 0 && uint64(fromBlock.Int64()) > log.BlockNumber {
|
if fromBlock != nil && fromBlock.Int64() >= 0 && fromBlock.Uint64() > log.BlockNumber {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
if toBlock != nil && toBlock.Int64() >= 0 && uint64(toBlock.Int64()) < log.BlockNumber {
|
if toBlock != nil && toBlock.Int64() >= 0 && toBlock.Uint64() < log.BlockNumber {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -19,7 +19,6 @@
|
|||||||
package filters
|
package filters
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/json"
|
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"sync"
|
"sync"
|
||||||
@ -60,42 +59,12 @@ var (
|
|||||||
ErrInvalidSubscriptionID = errors.New("invalid id")
|
ErrInvalidSubscriptionID = errors.New("invalid id")
|
||||||
)
|
)
|
||||||
|
|
||||||
// Log is a helper that can hold additional information about vm.Log
|
|
||||||
// necessary for the RPC interface.
|
|
||||||
type Log struct {
|
|
||||||
*vm.Log
|
|
||||||
Removed bool `json:"removed"`
|
|
||||||
}
|
|
||||||
|
|
||||||
// MarshalJSON returns *l as the JSON encoding of l.
|
|
||||||
func (l *Log) MarshalJSON() ([]byte, error) {
|
|
||||||
fields := map[string]interface{}{
|
|
||||||
"address": l.Address,
|
|
||||||
"data": fmt.Sprintf("0x%x", l.Data),
|
|
||||||
"blockNumber": nil,
|
|
||||||
"logIndex": fmt.Sprintf("%#x", l.Index),
|
|
||||||
"blockHash": nil,
|
|
||||||
"transactionHash": l.TxHash,
|
|
||||||
"transactionIndex": fmt.Sprintf("%#x", l.TxIndex),
|
|
||||||
"topics": l.Topics,
|
|
||||||
"removed": l.Removed,
|
|
||||||
}
|
|
||||||
|
|
||||||
// mined logs
|
|
||||||
if l.BlockHash != (common.Hash{}) {
|
|
||||||
fields["blockNumber"] = fmt.Sprintf("%#x", l.BlockNumber)
|
|
||||||
fields["blockHash"] = l.BlockHash
|
|
||||||
}
|
|
||||||
|
|
||||||
return json.Marshal(fields)
|
|
||||||
}
|
|
||||||
|
|
||||||
type subscription struct {
|
type subscription struct {
|
||||||
id rpc.ID
|
id rpc.ID
|
||||||
typ Type
|
typ Type
|
||||||
created time.Time
|
created time.Time
|
||||||
logsCrit FilterCriteria
|
logsCrit FilterCriteria
|
||||||
logs chan []Log
|
logs chan []*vm.Log
|
||||||
hashes chan common.Hash
|
hashes chan common.Hash
|
||||||
headers chan *types.Header
|
headers chan *types.Header
|
||||||
installed chan struct{} // closed when the filter is installed
|
installed chan struct{} // closed when the filter is installed
|
||||||
@ -182,7 +151,7 @@ func (es *EventSystem) subscribe(sub *subscription) *Subscription {
|
|||||||
// SubscribeLogs creates a subscription that will write all logs matching the
|
// SubscribeLogs creates a subscription that will write all logs matching the
|
||||||
// given criteria to the given logs channel. Default value for the from and to
|
// given criteria to the given logs channel. Default value for the from and to
|
||||||
// block is "latest". If the fromBlock > toBlock an error is returned.
|
// block is "latest". If the fromBlock > toBlock an error is returned.
|
||||||
func (es *EventSystem) SubscribeLogs(crit FilterCriteria, logs chan []Log) (*Subscription, error) {
|
func (es *EventSystem) SubscribeLogs(crit FilterCriteria, logs chan []*vm.Log) (*Subscription, error) {
|
||||||
var from, to rpc.BlockNumber
|
var from, to rpc.BlockNumber
|
||||||
if crit.FromBlock == nil {
|
if crit.FromBlock == nil {
|
||||||
from = rpc.LatestBlockNumber
|
from = rpc.LatestBlockNumber
|
||||||
@ -220,7 +189,7 @@ func (es *EventSystem) SubscribeLogs(crit FilterCriteria, logs chan []Log) (*Sub
|
|||||||
|
|
||||||
// subscribeMinedPendingLogs creates a subscription that returned mined and
|
// subscribeMinedPendingLogs creates a subscription that returned mined and
|
||||||
// pending logs that match the given criteria.
|
// pending logs that match the given criteria.
|
||||||
func (es *EventSystem) subscribeMinedPendingLogs(crit FilterCriteria, logs chan []Log) *Subscription {
|
func (es *EventSystem) subscribeMinedPendingLogs(crit FilterCriteria, logs chan []*vm.Log) *Subscription {
|
||||||
sub := &subscription{
|
sub := &subscription{
|
||||||
id: rpc.NewID(),
|
id: rpc.NewID(),
|
||||||
typ: MinedAndPendingLogsSubscription,
|
typ: MinedAndPendingLogsSubscription,
|
||||||
@ -238,7 +207,7 @@ func (es *EventSystem) subscribeMinedPendingLogs(crit FilterCriteria, logs chan
|
|||||||
|
|
||||||
// subscribeLogs creates a subscription that will write all logs matching the
|
// subscribeLogs creates a subscription that will write all logs matching the
|
||||||
// given criteria to the given logs channel.
|
// given criteria to the given logs channel.
|
||||||
func (es *EventSystem) subscribeLogs(crit FilterCriteria, logs chan []Log) *Subscription {
|
func (es *EventSystem) subscribeLogs(crit FilterCriteria, logs chan []*vm.Log) *Subscription {
|
||||||
sub := &subscription{
|
sub := &subscription{
|
||||||
id: rpc.NewID(),
|
id: rpc.NewID(),
|
||||||
typ: LogsSubscription,
|
typ: LogsSubscription,
|
||||||
@ -256,7 +225,7 @@ func (es *EventSystem) subscribeLogs(crit FilterCriteria, logs chan []Log) *Subs
|
|||||||
|
|
||||||
// subscribePendingLogs creates a subscription that writes transaction hashes for
|
// subscribePendingLogs creates a subscription that writes transaction hashes for
|
||||||
// transactions that enter the transaction pool.
|
// transactions that enter the transaction pool.
|
||||||
func (es *EventSystem) subscribePendingLogs(crit FilterCriteria, logs chan []Log) *Subscription {
|
func (es *EventSystem) subscribePendingLogs(crit FilterCriteria, logs chan []*vm.Log) *Subscription {
|
||||||
sub := &subscription{
|
sub := &subscription{
|
||||||
id: rpc.NewID(),
|
id: rpc.NewID(),
|
||||||
typ: PendingLogsSubscription,
|
typ: PendingLogsSubscription,
|
||||||
@ -279,7 +248,7 @@ func (es *EventSystem) SubscribeNewHeads(headers chan *types.Header) *Subscripti
|
|||||||
id: rpc.NewID(),
|
id: rpc.NewID(),
|
||||||
typ: BlocksSubscription,
|
typ: BlocksSubscription,
|
||||||
created: time.Now(),
|
created: time.Now(),
|
||||||
logs: make(chan []Log),
|
logs: make(chan []*vm.Log),
|
||||||
hashes: make(chan common.Hash),
|
hashes: make(chan common.Hash),
|
||||||
headers: headers,
|
headers: headers,
|
||||||
installed: make(chan struct{}),
|
installed: make(chan struct{}),
|
||||||
@ -296,7 +265,7 @@ func (es *EventSystem) SubscribePendingTxEvents(hashes chan common.Hash) *Subscr
|
|||||||
id: rpc.NewID(),
|
id: rpc.NewID(),
|
||||||
typ: PendingTransactionsSubscription,
|
typ: PendingTransactionsSubscription,
|
||||||
created: time.Now(),
|
created: time.Now(),
|
||||||
logs: make(chan []Log),
|
logs: make(chan []*vm.Log),
|
||||||
hashes: hashes,
|
hashes: hashes,
|
||||||
headers: make(chan *types.Header),
|
headers: make(chan *types.Header),
|
||||||
installed: make(chan struct{}),
|
installed: make(chan struct{}),
|
||||||
@ -319,7 +288,7 @@ func (es *EventSystem) broadcast(filters filterIndex, ev *event.Event) {
|
|||||||
if len(e) > 0 {
|
if len(e) > 0 {
|
||||||
for _, f := range filters[LogsSubscription] {
|
for _, f := range filters[LogsSubscription] {
|
||||||
if ev.Time.After(f.created) {
|
if ev.Time.After(f.created) {
|
||||||
if matchedLogs := filterLogs(convertLogs(e, false), f.logsCrit.FromBlock, f.logsCrit.ToBlock, f.logsCrit.Addresses, f.logsCrit.Topics); len(matchedLogs) > 0 {
|
if matchedLogs := filterLogs(e, f.logsCrit.FromBlock, f.logsCrit.ToBlock, f.logsCrit.Addresses, f.logsCrit.Topics); len(matchedLogs) > 0 {
|
||||||
f.logs <- matchedLogs
|
f.logs <- matchedLogs
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -328,7 +297,7 @@ func (es *EventSystem) broadcast(filters filterIndex, ev *event.Event) {
|
|||||||
case core.RemovedLogsEvent:
|
case core.RemovedLogsEvent:
|
||||||
for _, f := range filters[LogsSubscription] {
|
for _, f := range filters[LogsSubscription] {
|
||||||
if ev.Time.After(f.created) {
|
if ev.Time.After(f.created) {
|
||||||
if matchedLogs := filterLogs(convertLogs(e.Logs, true), f.logsCrit.FromBlock, f.logsCrit.ToBlock, f.logsCrit.Addresses, f.logsCrit.Topics); len(matchedLogs) > 0 {
|
if matchedLogs := filterLogs(e.Logs, f.logsCrit.FromBlock, f.logsCrit.ToBlock, f.logsCrit.Addresses, f.logsCrit.Topics); len(matchedLogs) > 0 {
|
||||||
f.logs <- matchedLogs
|
f.logs <- matchedLogs
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -336,7 +305,7 @@ func (es *EventSystem) broadcast(filters filterIndex, ev *event.Event) {
|
|||||||
case core.PendingLogsEvent:
|
case core.PendingLogsEvent:
|
||||||
for _, f := range filters[PendingLogsSubscription] {
|
for _, f := range filters[PendingLogsSubscription] {
|
||||||
if ev.Time.After(f.created) {
|
if ev.Time.After(f.created) {
|
||||||
if matchedLogs := filterLogs(convertLogs(e.Logs, false), nil, f.logsCrit.ToBlock, f.logsCrit.Addresses, f.logsCrit.Topics); len(matchedLogs) > 0 {
|
if matchedLogs := filterLogs(e.Logs, nil, f.logsCrit.ToBlock, f.logsCrit.Addresses, f.logsCrit.Topics); len(matchedLogs) > 0 {
|
||||||
f.logs <- matchedLogs
|
f.logs <- matchedLogs
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -401,25 +370,22 @@ func (es *EventSystem) lightFilterNewHead(newHeader *types.Header, callBack func
|
|||||||
}
|
}
|
||||||
|
|
||||||
// filter logs of a single header in light client mode
|
// filter logs of a single header in light client mode
|
||||||
func (es *EventSystem) lightFilterLogs(header *types.Header, addresses []common.Address, topics [][]common.Hash, remove bool) []Log {
|
func (es *EventSystem) lightFilterLogs(header *types.Header, addresses []common.Address, topics [][]common.Hash, remove bool) []*vm.Log {
|
||||||
//fmt.Println("lightFilterLogs", header.Number.Uint64(), remove)
|
|
||||||
if bloomFilter(header.Bloom, addresses, topics) {
|
if bloomFilter(header.Bloom, addresses, topics) {
|
||||||
//fmt.Println("bloom match")
|
|
||||||
// Get the logs of the block
|
// Get the logs of the block
|
||||||
ctx, _ := context.WithTimeout(context.Background(), time.Second*5)
|
ctx, _ := context.WithTimeout(context.Background(), time.Second*5)
|
||||||
receipts, err := es.backend.GetReceipts(ctx, header.Hash())
|
receipts, err := es.backend.GetReceipts(ctx, header.Hash())
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
var unfiltered []Log
|
var unfiltered []*vm.Log
|
||||||
for _, receipt := range receipts {
|
for _, receipt := range receipts {
|
||||||
rl := make([]Log, len(receipt.Logs))
|
for _, log := range receipt.Logs {
|
||||||
for i, l := range receipt.Logs {
|
logcopy := *log
|
||||||
rl[i] = Log{l, remove}
|
logcopy.Removed = remove
|
||||||
|
unfiltered = append(unfiltered, &logcopy)
|
||||||
}
|
}
|
||||||
unfiltered = append(unfiltered, rl...)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
logs := filterLogs(unfiltered, nil, nil, addresses, topics)
|
logs := filterLogs(unfiltered, nil, nil, addresses, topics)
|
||||||
return logs
|
return logs
|
||||||
}
|
}
|
||||||
@ -465,13 +431,3 @@ func (es *EventSystem) eventLoop() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// convertLogs is a helper utility that converts vm.Logs to []filter.Log.
|
|
||||||
func convertLogs(in vm.Logs, removed bool) []Log {
|
|
||||||
|
|
||||||
logs := make([]Log, len(in))
|
|
||||||
for i, l := range in {
|
|
||||||
logs[i] = Log{l, removed}
|
|
||||||
}
|
|
||||||
return logs
|
|
||||||
}
|
|
||||||
|
@ -321,14 +321,14 @@ func TestLogFilter(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
for i, tt := range testCases {
|
for i, tt := range testCases {
|
||||||
var fetched []Log
|
var fetched []*vm.Log
|
||||||
for { // fetch all expected logs
|
for { // fetch all expected logs
|
||||||
results, err := api.GetFilterChanges(tt.id)
|
results, err := api.GetFilterChanges(tt.id)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("Unable to fetch logs: %v", err)
|
t.Fatalf("Unable to fetch logs: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
fetched = append(fetched, results.([]Log)...)
|
fetched = append(fetched, results.([]*vm.Log)...)
|
||||||
if len(fetched) >= len(tt.expected) {
|
if len(fetched) >= len(tt.expected) {
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
@ -345,7 +345,7 @@ func TestLogFilter(t *testing.T) {
|
|||||||
if fetched[l].Removed {
|
if fetched[l].Removed {
|
||||||
t.Errorf("expected log not to be removed for log %d in case %d", l, i)
|
t.Errorf("expected log not to be removed for log %d in case %d", l, i)
|
||||||
}
|
}
|
||||||
if !reflect.DeepEqual(fetched[l].Log, tt.expected[l]) {
|
if !reflect.DeepEqual(fetched[l], tt.expected[l]) {
|
||||||
t.Errorf("invalid log on index %d for case %d", l, i)
|
t.Errorf("invalid log on index %d for case %d", l, i)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -397,7 +397,7 @@ func TestPendingLogsSubscription(t *testing.T) {
|
|||||||
testCases = []struct {
|
testCases = []struct {
|
||||||
crit FilterCriteria
|
crit FilterCriteria
|
||||||
expected vm.Logs
|
expected vm.Logs
|
||||||
c chan []Log
|
c chan []*vm.Log
|
||||||
sub *Subscription
|
sub *Subscription
|
||||||
}{
|
}{
|
||||||
// match all
|
// match all
|
||||||
@ -423,7 +423,7 @@ func TestPendingLogsSubscription(t *testing.T) {
|
|||||||
// on slow machines this could otherwise lead to missing events when the subscription is created after
|
// on slow machines this could otherwise lead to missing events when the subscription is created after
|
||||||
// (some) events are posted.
|
// (some) events are posted.
|
||||||
for i := range testCases {
|
for i := range testCases {
|
||||||
testCases[i].c = make(chan []Log)
|
testCases[i].c = make(chan []*vm.Log)
|
||||||
testCases[i].sub, _ = api.events.SubscribeLogs(testCases[i].crit, testCases[i].c)
|
testCases[i].sub, _ = api.events.SubscribeLogs(testCases[i].crit, testCases[i].c)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -431,7 +431,7 @@ func TestPendingLogsSubscription(t *testing.T) {
|
|||||||
i := n
|
i := n
|
||||||
tt := test
|
tt := test
|
||||||
go func() {
|
go func() {
|
||||||
var fetched []Log
|
var fetched []*vm.Log
|
||||||
fetchLoop:
|
fetchLoop:
|
||||||
for {
|
for {
|
||||||
logs := <-tt.c
|
logs := <-tt.c
|
||||||
@ -449,7 +449,7 @@ func TestPendingLogsSubscription(t *testing.T) {
|
|||||||
if fetched[l].Removed {
|
if fetched[l].Removed {
|
||||||
t.Errorf("expected log not to be removed for log %d in case %d", l, i)
|
t.Errorf("expected log not to be removed for log %d in case %d", l, i)
|
||||||
}
|
}
|
||||||
if !reflect.DeepEqual(fetched[l].Log, tt.expected[l]) {
|
if !reflect.DeepEqual(fetched[l], tt.expected[l]) {
|
||||||
t.Errorf("invalid log on index %d for case %d", l, i)
|
t.Errorf("invalid log on index %d for case %d", l, i)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -81,6 +81,8 @@ func (ec *Client) getBlock(ctx context.Context, method string, args ...interface
|
|||||||
err := ec.c.CallContext(ctx, &raw, method, args...)
|
err := ec.c.CallContext(ctx, &raw, method, args...)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
|
} else if len(raw) == 0 {
|
||||||
|
return nil, ethereum.NotFound
|
||||||
}
|
}
|
||||||
// Decode header and transactions.
|
// Decode header and transactions.
|
||||||
var head *types.Header
|
var head *types.Header
|
||||||
@ -112,7 +114,7 @@ func (ec *Client) getBlock(ctx context.Context, method string, args ...interface
|
|||||||
for i := range reqs {
|
for i := range reqs {
|
||||||
reqs[i] = rpc.BatchElem{
|
reqs[i] = rpc.BatchElem{
|
||||||
Method: "eth_getUncleByBlockHashAndIndex",
|
Method: "eth_getUncleByBlockHashAndIndex",
|
||||||
Args: []interface{}{body.Hash, fmt.Sprintf("%#x", i)},
|
Args: []interface{}{body.Hash, hexutil.EncodeUint64(uint64(i))},
|
||||||
Result: &uncles[i],
|
Result: &uncles[i],
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -123,6 +125,9 @@ func (ec *Client) getBlock(ctx context.Context, method string, args ...interface
|
|||||||
if reqs[i].Error != nil {
|
if reqs[i].Error != nil {
|
||||||
return nil, reqs[i].Error
|
return nil, reqs[i].Error
|
||||||
}
|
}
|
||||||
|
if uncles[i] == nil {
|
||||||
|
return nil, fmt.Errorf("got null header for uncle %d of block %x", i, body.Hash[:])
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return types.NewBlockWithHeader(head).WithBody(body.Transactions, uncles), nil
|
return types.NewBlockWithHeader(head).WithBody(body.Transactions, uncles), nil
|
||||||
@ -132,6 +137,9 @@ func (ec *Client) getBlock(ctx context.Context, method string, args ...interface
|
|||||||
func (ec *Client) HeaderByHash(ctx context.Context, hash common.Hash) (*types.Header, error) {
|
func (ec *Client) HeaderByHash(ctx context.Context, hash common.Hash) (*types.Header, error) {
|
||||||
var head *types.Header
|
var head *types.Header
|
||||||
err := ec.c.CallContext(ctx, &head, "eth_getBlockByHash", hash, false)
|
err := ec.c.CallContext(ctx, &head, "eth_getBlockByHash", hash, false)
|
||||||
|
if err == nil && head == nil {
|
||||||
|
err = ethereum.NotFound
|
||||||
|
}
|
||||||
return head, err
|
return head, err
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -140,19 +148,31 @@ func (ec *Client) HeaderByHash(ctx context.Context, hash common.Hash) (*types.He
|
|||||||
func (ec *Client) HeaderByNumber(ctx context.Context, number *big.Int) (*types.Header, error) {
|
func (ec *Client) HeaderByNumber(ctx context.Context, number *big.Int) (*types.Header, error) {
|
||||||
var head *types.Header
|
var head *types.Header
|
||||||
err := ec.c.CallContext(ctx, &head, "eth_getBlockByNumber", toBlockNumArg(number), false)
|
err := ec.c.CallContext(ctx, &head, "eth_getBlockByNumber", toBlockNumArg(number), false)
|
||||||
|
if err == nil && head == nil {
|
||||||
|
err = ethereum.NotFound
|
||||||
|
}
|
||||||
return head, err
|
return head, err
|
||||||
}
|
}
|
||||||
|
|
||||||
// TransactionByHash returns the transaction with the given hash.
|
// TransactionByHash returns the transaction with the given hash.
|
||||||
func (ec *Client) TransactionByHash(ctx context.Context, hash common.Hash) (*types.Transaction, error) {
|
func (ec *Client) TransactionByHash(ctx context.Context, hash common.Hash) (tx *types.Transaction, isPending bool, err error) {
|
||||||
var tx *types.Transaction
|
var raw json.RawMessage
|
||||||
err := ec.c.CallContext(ctx, &tx, "eth_getTransactionByHash", hash)
|
err = ec.c.CallContext(ctx, &raw, "eth_getTransactionByHash", hash)
|
||||||
if err == nil {
|
if err != nil {
|
||||||
if _, r, _ := tx.RawSignatureValues(); r == nil {
|
return nil, false, err
|
||||||
return nil, fmt.Errorf("server returned transaction without signature")
|
} else if len(raw) == 0 {
|
||||||
|
return nil, false, ethereum.NotFound
|
||||||
}
|
}
|
||||||
|
if err := json.Unmarshal(raw, tx); err != nil {
|
||||||
|
return nil, false, err
|
||||||
|
} else if _, r, _ := tx.RawSignatureValues(); r == nil {
|
||||||
|
return nil, false, fmt.Errorf("server returned transaction without signature")
|
||||||
}
|
}
|
||||||
return tx, err
|
var block struct{ BlockHash *common.Hash }
|
||||||
|
if err := json.Unmarshal(raw, &block); err != nil {
|
||||||
|
return nil, false, err
|
||||||
|
}
|
||||||
|
return tx, block.BlockHash == nil, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// TransactionCount returns the total number of transactions in the given block.
|
// TransactionCount returns the total number of transactions in the given block.
|
||||||
@ -167,11 +187,9 @@ func (ec *Client) TransactionInBlock(ctx context.Context, blockHash common.Hash,
|
|||||||
var tx *types.Transaction
|
var tx *types.Transaction
|
||||||
err := ec.c.CallContext(ctx, &tx, "eth_getTransactionByBlockHashAndIndex", blockHash, index)
|
err := ec.c.CallContext(ctx, &tx, "eth_getTransactionByBlockHashAndIndex", blockHash, index)
|
||||||
if err == nil {
|
if err == nil {
|
||||||
var signer types.Signer = types.HomesteadSigner{}
|
if tx == nil {
|
||||||
if tx.Protected() {
|
return nil, ethereum.NotFound
|
||||||
signer = types.NewEIP155Signer(tx.ChainId())
|
} else if _, r, _ := tx.RawSignatureValues(); r == nil {
|
||||||
}
|
|
||||||
if _, r, _ := types.SignatureValues(signer, tx); r == nil {
|
|
||||||
return nil, fmt.Errorf("server returned transaction without signature")
|
return nil, fmt.Errorf("server returned transaction without signature")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -183,9 +201,13 @@ func (ec *Client) TransactionInBlock(ctx context.Context, blockHash common.Hash,
|
|||||||
func (ec *Client) TransactionReceipt(ctx context.Context, txHash common.Hash) (*types.Receipt, error) {
|
func (ec *Client) TransactionReceipt(ctx context.Context, txHash common.Hash) (*types.Receipt, error) {
|
||||||
var r *types.Receipt
|
var r *types.Receipt
|
||||||
err := ec.c.CallContext(ctx, &r, "eth_getTransactionReceipt", txHash)
|
err := ec.c.CallContext(ctx, &r, "eth_getTransactionReceipt", txHash)
|
||||||
if err == nil && r != nil && len(r.PostState) == 0 {
|
if err == nil {
|
||||||
|
if r == nil {
|
||||||
|
return nil, ethereum.NotFound
|
||||||
|
} else if len(r.PostState) == 0 {
|
||||||
return nil, fmt.Errorf("server returned receipt without post state")
|
return nil, fmt.Errorf("server returned receipt without post state")
|
||||||
}
|
}
|
||||||
|
}
|
||||||
return r, err
|
return r, err
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -193,7 +215,7 @@ func toBlockNumArg(number *big.Int) string {
|
|||||||
if number == nil {
|
if number == nil {
|
||||||
return "latest"
|
return "latest"
|
||||||
}
|
}
|
||||||
return fmt.Sprintf("%#x", number)
|
return hexutil.EncodeBig(number)
|
||||||
}
|
}
|
||||||
|
|
||||||
type rpcProgress struct {
|
type rpcProgress struct {
|
||||||
|
@ -21,9 +21,9 @@ import "github.com/ethereum/go-ethereum"
|
|||||||
// Verify that Client implements the ethereum interfaces.
|
// Verify that Client implements the ethereum interfaces.
|
||||||
var (
|
var (
|
||||||
_ = ethereum.ChainReader(&Client{})
|
_ = ethereum.ChainReader(&Client{})
|
||||||
|
_ = ethereum.TransactionReader(&Client{})
|
||||||
_ = ethereum.ChainStateReader(&Client{})
|
_ = ethereum.ChainStateReader(&Client{})
|
||||||
_ = ethereum.ChainSyncReader(&Client{})
|
_ = ethereum.ChainSyncReader(&Client{})
|
||||||
_ = ethereum.ChainHeadEventer(&Client{})
|
|
||||||
_ = ethereum.ContractCaller(&Client{})
|
_ = ethereum.ContractCaller(&Client{})
|
||||||
_ = ethereum.GasEstimator(&Client{})
|
_ = ethereum.GasEstimator(&Client{})
|
||||||
_ = ethereum.GasPricer(&Client{})
|
_ = ethereum.GasPricer(&Client{})
|
||||||
|
@ -18,6 +18,7 @@
|
|||||||
package ethereum
|
package ethereum
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"errors"
|
||||||
"math/big"
|
"math/big"
|
||||||
|
|
||||||
"github.com/ethereum/go-ethereum/common"
|
"github.com/ethereum/go-ethereum/common"
|
||||||
@ -26,6 +27,9 @@ import (
|
|||||||
"golang.org/x/net/context"
|
"golang.org/x/net/context"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// NotFound is returned by API methods if the requested item does not exist.
|
||||||
|
var NotFound = errors.New("not found")
|
||||||
|
|
||||||
// TODO: move subscription to package event
|
// TODO: move subscription to package event
|
||||||
|
|
||||||
// Subscription represents an event subscription where events are
|
// Subscription represents an event subscription where events are
|
||||||
@ -46,6 +50,8 @@ type Subscription interface {
|
|||||||
// blockchain fork that was previously downloaded and processed by the node. The block
|
// blockchain fork that was previously downloaded and processed by the node. The block
|
||||||
// number argument can be nil to select the latest canonical block. Reading block headers
|
// number argument can be nil to select the latest canonical block. Reading block headers
|
||||||
// should be preferred over full blocks whenever possible.
|
// should be preferred over full blocks whenever possible.
|
||||||
|
//
|
||||||
|
// The returned error is NotFound if the requested item does not exist.
|
||||||
type ChainReader interface {
|
type ChainReader interface {
|
||||||
BlockByHash(ctx context.Context, hash common.Hash) (*types.Block, error)
|
BlockByHash(ctx context.Context, hash common.Hash) (*types.Block, error)
|
||||||
BlockByNumber(ctx context.Context, number *big.Int) (*types.Block, error)
|
BlockByNumber(ctx context.Context, number *big.Int) (*types.Block, error)
|
||||||
@ -53,7 +59,30 @@ type ChainReader interface {
|
|||||||
HeaderByNumber(ctx context.Context, number *big.Int) (*types.Header, error)
|
HeaderByNumber(ctx context.Context, number *big.Int) (*types.Header, error)
|
||||||
TransactionCount(ctx context.Context, blockHash common.Hash) (uint, error)
|
TransactionCount(ctx context.Context, blockHash common.Hash) (uint, error)
|
||||||
TransactionInBlock(ctx context.Context, blockHash common.Hash, index uint) (*types.Transaction, error)
|
TransactionInBlock(ctx context.Context, blockHash common.Hash, index uint) (*types.Transaction, error)
|
||||||
TransactionByHash(ctx context.Context, txHash common.Hash) (*types.Transaction, error)
|
|
||||||
|
// This method subscribes to notifications about changes of the head block of
|
||||||
|
// the canonical chain.
|
||||||
|
SubscribeNewHead(ctx context.Context, ch chan<- *types.Header) (Subscription, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
// TransactionReader provides access to past transactions and their receipts.
|
||||||
|
// Implementations may impose arbitrary restrictions on the transactions and receipts that
|
||||||
|
// can be retrieved. Historic transactions may not be available.
|
||||||
|
//
|
||||||
|
// Avoid relying on this interface if possible. Contract logs (through the LogFilterer
|
||||||
|
// interface) are more reliable and usually safer in the presence of chain
|
||||||
|
// reorganisations.
|
||||||
|
//
|
||||||
|
// The returned error is NotFound if the requested item does not exist.
|
||||||
|
type TransactionReader interface {
|
||||||
|
// TransactionByHash checks the pool of pending transactions in addition to the
|
||||||
|
// blockchain. The isPending return value indicates whether the transaction has been
|
||||||
|
// mined yet. Note that the transaction may not be part of the canonical chain even if
|
||||||
|
// it's not pending.
|
||||||
|
TransactionByHash(ctx context.Context, txHash common.Hash) (tx *types.Transaction, isPending bool, err error)
|
||||||
|
// TransactionReceipt returns the receipt of a mined transaction. Note that the
|
||||||
|
// transaction may not be included in the current canonical chain even if a receipt
|
||||||
|
// exists.
|
||||||
TransactionReceipt(ctx context.Context, txHash common.Hash) (*types.Receipt, error)
|
TransactionReceipt(ctx context.Context, txHash common.Hash) (*types.Receipt, error)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -83,11 +112,6 @@ type ChainSyncReader interface {
|
|||||||
SyncProgress(ctx context.Context) (*SyncProgress, error)
|
SyncProgress(ctx context.Context) (*SyncProgress, error)
|
||||||
}
|
}
|
||||||
|
|
||||||
// A ChainHeadEventer returns notifications whenever the canonical head block is updated.
|
|
||||||
type ChainHeadEventer interface {
|
|
||||||
SubscribeNewHead(ctx context.Context, ch chan<- *types.Header) (Subscription, error)
|
|
||||||
}
|
|
||||||
|
|
||||||
// CallMsg contains parameters for contract calls.
|
// CallMsg contains parameters for contract calls.
|
||||||
type CallMsg struct {
|
type CallMsg struct {
|
||||||
From common.Address // the sender of the 'transaction'
|
From common.Address // the sender of the 'transaction'
|
||||||
@ -128,6 +152,9 @@ type FilterQuery struct {
|
|||||||
|
|
||||||
// LogFilterer provides access to contract log events using a one-off query or continuous
|
// LogFilterer provides access to contract log events using a one-off query or continuous
|
||||||
// event subscription.
|
// event subscription.
|
||||||
|
//
|
||||||
|
// Logs received through a streaming query subscription may have Removed set to true,
|
||||||
|
// indicating that the log was reverted due to a chain reorganisation.
|
||||||
type LogFilterer interface {
|
type LogFilterer interface {
|
||||||
FilterLogs(ctx context.Context, q FilterQuery) ([]vm.Log, error)
|
FilterLogs(ctx context.Context, q FilterQuery) ([]vm.Log, error)
|
||||||
SubscribeFilterLogs(ctx context.Context, q FilterQuery, ch chan<- vm.Log) (Subscription, error)
|
SubscribeFilterLogs(ctx context.Context, q FilterQuery, ch chan<- vm.Log) (Subscription, error)
|
||||||
|
@ -73,7 +73,8 @@ func (ec *EthereumClient) GetHeaderByNumber(ctx *Context, number int64) (*Header
|
|||||||
|
|
||||||
// GetTransactionByHash returns the transaction with the given hash.
|
// GetTransactionByHash returns the transaction with the given hash.
|
||||||
func (ec *EthereumClient) GetTransactionByHash(ctx *Context, hash *Hash) (*Transaction, error) {
|
func (ec *EthereumClient) GetTransactionByHash(ctx *Context, hash *Hash) (*Transaction, error) {
|
||||||
tx, err := ec.client.TransactionByHash(ctx.context, hash.hash)
|
// TODO(karalabe): handle isPending
|
||||||
|
tx, _, err := ec.client.TransactionByHash(ctx.context, hash.hash)
|
||||||
return &Transaction{tx}, err
|
return &Transaction{tx}, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user