log.go 4.16 KB
Newer Older
1
// Copyright 2014 The go-ethereum Authors
2
// This file is part of the go-ethereum library.
3
//
4
// The go-ethereum library is free software: you can redistribute it and/or modify
5 6 7 8
// it under the terms of the GNU Lesser General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
9
// The go-ethereum library is distributed in the hope that it will be useful,
10
// but WITHOUT ANY WARRANTY; without even the implied warranty of
11
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 13 14
// GNU Lesser General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public License
15
// along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.
16

17
package vm
obscuren's avatar
obscuren committed
18

obscuren's avatar
obscuren committed
19
import (
20
	"encoding/json"
21
	"errors"
obscuren's avatar
obscuren committed
22
	"fmt"
obscuren's avatar
obscuren committed
23
	"io"
obscuren's avatar
obscuren committed
24

obscuren's avatar
obscuren committed
25
	"github.com/ethereum/go-ethereum/common"
26
	"github.com/ethereum/go-ethereum/common/hexutil"
obscuren's avatar
obscuren committed
27
	"github.com/ethereum/go-ethereum/rlp"
obscuren's avatar
obscuren committed
28
)
obscuren's avatar
obscuren committed
29

30 31 32 33
var errMissingLogFields = errors.New("missing required JSON log fields")

// Log represents a contract log event. These events are generated by the LOG
// opcode and stored/indexed by the node.
34
type Log struct {
35 36 37 38 39 40 41 42 43 44 45 46
	// Consensus fields.
	Address common.Address // address of the contract that generated the event
	Topics  []common.Hash  // list of topics provided by the contract.
	Data    []byte         // supplied by the contract, usually ABI-encoded

	// Derived fields (don't reorder!).
	BlockNumber uint64      // block in which the transaction was included
	TxHash      common.Hash // hash of the transaction
	TxIndex     uint        // index of the transaction in the block
	BlockHash   common.Hash // hash of the block in which the transaction was included
	Index       uint        // index of the log in the receipt
}
obscuren's avatar
obscuren committed
47

48 49 50
type jsonLog struct {
	Address     *common.Address `json:"address"`
	Topics      *[]common.Hash  `json:"topics"`
51 52 53
	Data        *hexutil.Bytes  `json:"data"`
	BlockNumber *hexutil.Uint64 `json:"blockNumber"`
	TxIndex     *hexutil.Uint   `json:"transactionIndex"`
54 55
	TxHash      *common.Hash    `json:"transactionHash"`
	BlockHash   *common.Hash    `json:"blockHash"`
56
	Index       *hexutil.Uint   `json:"logIndex"`
obscuren's avatar
obscuren committed
57 58
}

59
func NewLog(address common.Address, topics []common.Hash, data []byte, number uint64) *Log {
60
	return &Log{Address: address, Topics: topics, Data: data, BlockNumber: number}
obscuren's avatar
obscuren committed
61 62
}

63 64
func (l *Log) EncodeRLP(w io.Writer) error {
	return rlp.Encode(w, []interface{}{l.Address, l.Topics, l.Data})
obscuren's avatar
obscuren committed
65 66
}

67 68 69 70 71 72 73 74 75 76 77 78 79 80 81
func (l *Log) DecodeRLP(s *rlp.Stream) error {
	var log struct {
		Address common.Address
		Topics  []common.Hash
		Data    []byte
	}
	if err := s.Decode(&log); err != nil {
		return err
	}
	l.Address, l.Topics, l.Data = log.Address, log.Topics, log.Data
	return nil
}

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)
obscuren's avatar
obscuren committed
82 83
}

84
// MarshalJSON implements json.Marshaler.
85
func (r *Log) MarshalJSON() ([]byte, error) {
86 87 88
	return json.Marshal(&jsonLog{
		Address:     &r.Address,
		Topics:      &r.Topics,
89 90 91
		Data:        (*hexutil.Bytes)(&r.Data),
		BlockNumber: (*hexutil.Uint64)(&r.BlockNumber),
		TxIndex:     (*hexutil.Uint)(&r.TxIndex),
92 93
		TxHash:      &r.TxHash,
		BlockHash:   &r.BlockHash,
94
		Index:       (*hexutil.Uint)(&r.Index),
95 96
	})
}
97

98 99 100 101 102 103
// UnmarshalJSON implements json.Umarshaler.
func (r *Log) UnmarshalJSON(input []byte) error {
	var dec jsonLog
	if err := json.Unmarshal(input, &dec); err != nil {
		return err
	}
104 105
	if dec.Address == nil || dec.Topics == nil || dec.Data == nil || dec.BlockNumber == nil ||
		dec.TxIndex == nil || dec.TxHash == nil || dec.BlockHash == nil || dec.Index == nil {
106 107
		return errMissingLogFields
	}
108 109 110 111 112 113 114 115 116
	*r = Log{
		Address:     *dec.Address,
		Topics:      *dec.Topics,
		Data:        *dec.Data,
		BlockNumber: uint64(*dec.BlockNumber),
		TxHash:      *dec.TxHash,
		TxIndex:     uint(*dec.TxIndex),
		BlockHash:   *dec.BlockHash,
		Index:       uint(*dec.Index),
117 118
	}
	return nil
119 120
}

121
type Logs []*Log
obscuren's avatar
obscuren committed
122

123
// LogForStorage is a wrapper around a Log that flattens and parses the entire
124
// content of a log, as opposed to only the consensus fields originally (by hiding
125
// the rlp interface methods).
126
type LogForStorage Log