log.go 4.49 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"
obscuren's avatar
obscuren committed
26
	"github.com/ethereum/go-ethereum/rlp"
obscuren's avatar
obscuren committed
27
)
obscuren's avatar
obscuren committed
28

29 30 31 32
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.
33
type Log struct {
34 35 36 37 38 39 40 41 42 43 44 45
	// 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
46

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

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

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

66 67 68 69 70 71 72 73 74 75 76 77 78 79 80
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
81 82
}

83
// MarshalJSON implements json.Marshaler.
84
func (r *Log) MarshalJSON() ([]byte, error) {
85 86 87 88 89 90 91 92 93 94 95
	return json.Marshal(&jsonLog{
		Address:     &r.Address,
		Topics:      &r.Topics,
		Data:        fmt.Sprintf("0x%x", r.Data),
		BlockNumber: fmt.Sprintf("0x%x", r.BlockNumber),
		TxIndex:     fmt.Sprintf("0x%x", r.TxIndex),
		TxHash:      &r.TxHash,
		BlockHash:   &r.BlockHash,
		Index:       fmt.Sprintf("0x%x", r.Index),
	})
}
96

97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126
// UnmarshalJSON implements json.Umarshaler.
func (r *Log) UnmarshalJSON(input []byte) error {
	var dec jsonLog
	if err := json.Unmarshal(input, &dec); err != nil {
		return err
	}
	if dec.Address == nil || dec.Topics == nil || dec.Data == "" || dec.BlockNumber == "" ||
		dec.TxIndex == "" || dec.TxHash == nil || dec.BlockHash == nil || dec.Index == "" {
		return errMissingLogFields
	}
	declog := Log{
		Address:   *dec.Address,
		Topics:    *dec.Topics,
		TxHash:    *dec.TxHash,
		BlockHash: *dec.BlockHash,
	}
	if _, err := fmt.Sscanf(dec.Data, "0x%x", &declog.Data); err != nil {
		return fmt.Errorf("invalid hex log data")
	}
	if _, err := fmt.Sscanf(dec.BlockNumber, "0x%x", &declog.BlockNumber); err != nil {
		return fmt.Errorf("invalid hex log block number")
	}
	if _, err := fmt.Sscanf(dec.TxIndex, "0x%x", &declog.TxIndex); err != nil {
		return fmt.Errorf("invalid hex log tx index")
	}
	if _, err := fmt.Sscanf(dec.Index, "0x%x", &declog.Index); err != nil {
		return fmt.Errorf("invalid hex log index")
	}
	*r = declog
	return nil
127 128
}

129
type Logs []*Log
obscuren's avatar
obscuren committed
130

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