| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // Copyright 2014 The go-ethereum Authors | 
					
						
							| 
									
										
										
										
											2015-07-22 18:48:40 +02:00
										 |  |  | // This file is part of the go-ethereum library. | 
					
						
							| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // | 
					
						
							| 
									
										
										
										
											2015-07-23 18:35:11 +02:00
										 |  |  | // The go-ethereum library is free software: you can redistribute it and/or modify | 
					
						
							| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // 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. | 
					
						
							|  |  |  | // | 
					
						
							| 
									
										
										
										
											2015-07-22 18:48:40 +02:00
										 |  |  | // The go-ethereum library is distributed in the hope that it will be useful, | 
					
						
							| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							| 
									
										
										
										
											2015-07-22 18:48:40 +02:00
										 |  |  | // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | 
					
						
							| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // GNU Lesser General Public License for more details. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // You should have received a copy of the GNU Lesser General Public License | 
					
						
							| 
									
										
										
										
											2015-07-22 18:48:40 +02:00
										 |  |  | // along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>. | 
					
						
							| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-30 10:19:10 +02:00
										 |  |  | package vm | 
					
						
							| 
									
										
										
										
											2014-10-27 11:44:16 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-11 12:16:36 +01:00
										 |  |  | import ( | 
					
						
							| 
									
										
										
										
											2015-10-15 16:07:19 +02:00
										 |  |  | 	"encoding/json" | 
					
						
							| 
									
										
										
										
											2016-08-04 03:55:33 +02:00
										 |  |  | 	"errors" | 
					
						
							| 
									
										
										
										
											2014-11-11 12:16:36 +01:00
										 |  |  | 	"fmt" | 
					
						
							| 
									
										
										
										
											2015-03-16 23:10:26 +01:00
										 |  |  | 	"io" | 
					
						
							| 
									
										
										
										
											2014-11-11 12:16:36 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-16 11:27:38 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/common" | 
					
						
							| 
									
										
										
										
											2016-11-28 00:25:48 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/common/hexutil" | 
					
						
							| 
									
										
										
										
											2015-03-16 23:10:26 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/rlp" | 
					
						
							| 
									
										
										
										
											2014-11-11 12:16:36 +01:00
										 |  |  | ) | 
					
						
							| 
									
										
										
										
											2014-10-27 11:44:16 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-04 03:55:33 +02:00
										 |  |  | var errMissingLogFields = errors.New("missing required JSON log fields") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-04 19:07:24 +01:00
										 |  |  | // Log represents a contract log event. These events are generated by the LOG opcode and | 
					
						
							|  |  |  | // stored/indexed by the node. | 
					
						
							| 
									
										
										
										
											2015-04-08 17:14:58 +02:00
										 |  |  | type Log struct { | 
					
						
							| 
									
										
										
										
											2016-08-04 03:55:33 +02:00
										 |  |  | 	// 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 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-04 19:07:24 +01:00
										 |  |  | 	// Derived fields. These fields are filled in by the node | 
					
						
							|  |  |  | 	// but not secured by consensus. | 
					
						
							| 
									
										
										
										
											2016-08-04 03:55:33 +02:00
										 |  |  | 	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 | 
					
						
							| 
									
										
										
										
											2016-12-04 19:07:24 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// 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 | 
					
						
							| 
									
										
										
										
											2016-08-04 03:55:33 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2015-02-22 13:24:26 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-04 03:55:33 +02:00
										 |  |  | type jsonLog struct { | 
					
						
							|  |  |  | 	Address     *common.Address `json:"address"` | 
					
						
							|  |  |  | 	Topics      *[]common.Hash  `json:"topics"` | 
					
						
							| 
									
										
										
										
											2016-11-28 00:25:48 +01:00
										 |  |  | 	Data        *hexutil.Bytes  `json:"data"` | 
					
						
							|  |  |  | 	BlockNumber *hexutil.Uint64 `json:"blockNumber"` | 
					
						
							|  |  |  | 	TxIndex     *hexutil.Uint   `json:"transactionIndex"` | 
					
						
							| 
									
										
										
										
											2016-08-04 03:55:33 +02:00
										 |  |  | 	TxHash      *common.Hash    `json:"transactionHash"` | 
					
						
							|  |  |  | 	BlockHash   *common.Hash    `json:"blockHash"` | 
					
						
							| 
									
										
										
										
											2016-11-28 00:25:48 +01:00
										 |  |  | 	Index       *hexutil.Uint   `json:"logIndex"` | 
					
						
							| 
									
										
										
										
											2016-12-04 19:07:24 +01:00
										 |  |  | 	Removed     bool            `json:"removed"` | 
					
						
							| 
									
										
										
										
											2014-12-04 12:35:23 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-08 17:14:58 +02:00
										 |  |  | func NewLog(address common.Address, topics []common.Hash, data []byte, number uint64) *Log { | 
					
						
							| 
									
										
										
										
											2015-09-30 19:23:31 +03:00
										 |  |  | 	return &Log{Address: address, Topics: topics, Data: data, BlockNumber: number} | 
					
						
							| 
									
										
										
										
											2014-12-04 12:35:23 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-04 19:07:24 +01:00
										 |  |  | // EncodeRLP implements rlp.Encoder. | 
					
						
							| 
									
										
										
										
											2015-09-29 19:36:16 +03:00
										 |  |  | func (l *Log) EncodeRLP(w io.Writer) error { | 
					
						
							| 
									
										
										
										
											2016-12-04 19:07:24 +01:00
										 |  |  | 	return rlp.Encode(w, rlpLog{Address: l.Address, Topics: l.Topics, Data: l.Data}) | 
					
						
							| 
									
										
										
										
											2014-12-04 12:35:23 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-04 19:07:24 +01:00
										 |  |  | // DecodeRLP implements rlp.Decoder. | 
					
						
							| 
									
										
										
										
											2015-09-29 19:36:16 +03:00
										 |  |  | func (l *Log) DecodeRLP(s *rlp.Stream) error { | 
					
						
							| 
									
										
										
										
											2016-12-04 19:07:24 +01:00
										 |  |  | 	var dec rlpLog | 
					
						
							|  |  |  | 	err := s.Decode(&dec) | 
					
						
							|  |  |  | 	if err == nil { | 
					
						
							|  |  |  | 		l.Address, l.Topics, l.Data = dec.Address, dec.Topics, dec.Data | 
					
						
							| 
									
										
										
										
											2015-09-29 19:36:16 +03:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-12-04 19:07:24 +01:00
										 |  |  | 	return err | 
					
						
							| 
									
										
										
										
											2015-09-29 19:36:16 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 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) | 
					
						
							| 
									
										
										
										
											2014-12-04 12:35:23 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-04 03:55:33 +02:00
										 |  |  | // MarshalJSON implements json.Marshaler. | 
					
						
							| 
									
										
										
										
											2016-12-04 19:07:24 +01:00
										 |  |  | func (l *Log) MarshalJSON() ([]byte, error) { | 
					
						
							|  |  |  | 	jslog := &jsonLog{ | 
					
						
							|  |  |  | 		Address: &l.Address, | 
					
						
							|  |  |  | 		Topics:  &l.Topics, | 
					
						
							|  |  |  | 		Data:    (*hexutil.Bytes)(&l.Data), | 
					
						
							|  |  |  | 		TxIndex: (*hexutil.Uint)(&l.TxIndex), | 
					
						
							|  |  |  | 		TxHash:  &l.TxHash, | 
					
						
							|  |  |  | 		Index:   (*hexutil.Uint)(&l.Index), | 
					
						
							|  |  |  | 		Removed: l.Removed, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// 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) | 
					
						
							| 
									
										
										
										
											2016-08-04 03:55:33 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2015-10-15 16:07:19 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-04 03:55:33 +02:00
										 |  |  | // UnmarshalJSON implements json.Umarshaler. | 
					
						
							| 
									
										
										
										
											2016-12-04 19:07:24 +01:00
										 |  |  | func (l *Log) UnmarshalJSON(input []byte) error { | 
					
						
							| 
									
										
										
										
											2016-08-04 03:55:33 +02:00
										 |  |  | 	var dec jsonLog | 
					
						
							|  |  |  | 	if err := json.Unmarshal(input, &dec); err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-12-04 19:07:24 +01:00
										 |  |  | 	if dec.Address == nil || dec.Topics == nil || dec.Data == nil || | 
					
						
							|  |  |  | 		dec.TxIndex == nil || dec.TxHash == nil || dec.Index == nil { | 
					
						
							| 
									
										
										
										
											2016-08-04 03:55:33 +02:00
										 |  |  | 		return errMissingLogFields | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-12-04 19:07:24 +01:00
										 |  |  | 	declog := Log{ | 
					
						
							|  |  |  | 		Address: *dec.Address, | 
					
						
							|  |  |  | 		Topics:  *dec.Topics, | 
					
						
							|  |  |  | 		Data:    *dec.Data, | 
					
						
							|  |  |  | 		TxHash:  *dec.TxHash, | 
					
						
							|  |  |  | 		TxIndex: uint(*dec.TxIndex), | 
					
						
							|  |  |  | 		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) | 
					
						
							| 
									
										
										
										
											2016-08-04 03:55:33 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-12-04 19:07:24 +01:00
										 |  |  | 	*l = declog | 
					
						
							| 
									
										
										
										
											2016-08-04 03:55:33 +02:00
										 |  |  | 	return nil | 
					
						
							| 
									
										
										
										
											2015-10-15 16:07:19 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-08 17:14:58 +02:00
										 |  |  | type Logs []*Log | 
					
						
							| 
									
										
										
										
											2014-11-11 12:16:36 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-04 19:07:24 +01:00
										 |  |  | // LogForStorage is a wrapper around a Log that flattens and parses the entire content of | 
					
						
							|  |  |  | // a log including non-consensus fields. | 
					
						
							| 
									
										
										
										
											2015-05-22 22:44:51 +02:00
										 |  |  | type LogForStorage Log | 
					
						
							| 
									
										
										
										
											2016-12-04 19:07:24 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | // 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 | 
					
						
							|  |  |  | } |