0f7b7099d8
Handle conflicts (#244) * Handle conflicts * Update go mod file versions * Make lint changes Disassociate block number from the indexer object Update ipld-eth-db ref Refactor builder code to make it reusable Use prefix comparison for account selective statediffing Update builder unit tests Add mode to write to CSV files in statediff file writer (#249) * Change file writing mode to csv files * Implement writer interface for file indexer * Implement option for csv or sql in file mode * Close files in CSV writer * Add tests for CSV file mode * Implement CSV file for watched addresses * Separate test configs for CSV and SQL * Refactor common code for file indexer tests Update indexer to include block hash in receipts and logs (#256) * Update indexer to include block hash in receipts and logs * Upgrade ipld-eth-db image in docker-compose to run tests Use watched addresses from direct indexing params by default while serving statediff APIs (#262) * Use watched addresses from direct indexing params in statediff APIs by default * Avoid using indexer object when direct indexing is off * Add nil check before accessing watched addresses from direct indexing params
176 lines
4.1 KiB
Go
176 lines
4.1 KiB
Go
// VulcanizeDB
|
|
// Copyright © 2019 Vulcanize
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU Affero General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// (at your option) any later version.
|
|
|
|
// This program 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 Affero General Public License for more details.
|
|
|
|
// You should have received a copy of the GNU Affero General Public License
|
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
package ipld
|
|
|
|
import (
|
|
"fmt"
|
|
|
|
"github.com/ipfs/go-cid"
|
|
node "github.com/ipfs/go-ipld-format"
|
|
"github.com/multiformats/go-multihash"
|
|
|
|
"github.com/ethereum/go-ethereum/core/types"
|
|
)
|
|
|
|
// EthRctTrie (eth-tx-trie codec 0x92) represents
|
|
// a node from the transaction trie in ethereum.
|
|
type EthRctTrie struct {
|
|
*TrieNode
|
|
}
|
|
|
|
// Static (compile time) check that EthRctTrie satisfies the node.Node interface.
|
|
var _ node.Node = (*EthRctTrie)(nil)
|
|
|
|
/*
|
|
INPUT
|
|
*/
|
|
|
|
// To create a proper trie of the eth-tx-trie objects, it is required
|
|
// to input all transactions belonging to a forest in a single step.
|
|
// We are adding the transactions, and creating its trie on
|
|
// block body parsing time.
|
|
|
|
/*
|
|
OUTPUT
|
|
*/
|
|
|
|
// DecodeEthRctTrie returns an EthRctTrie object from its cid and rawdata.
|
|
func DecodeEthRctTrie(c cid.Cid, b []byte) (*EthRctTrie, error) {
|
|
tn, err := decodeTrieNode(c, b, decodeEthRctTrieLeaf)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &EthRctTrie{TrieNode: tn}, nil
|
|
}
|
|
|
|
// decodeEthRctTrieLeaf parses a eth-rct-trie leaf
|
|
//from decoded RLP elements
|
|
func decodeEthRctTrieLeaf(i []interface{}) ([]interface{}, error) {
|
|
r := new(types.Receipt)
|
|
if err := r.UnmarshalBinary(i[1].([]byte)); err != nil {
|
|
return nil, err
|
|
}
|
|
c, err := RawdataToCid(MEthTxReceipt, i[1].([]byte), multihash.KECCAK_256)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return []interface{}{
|
|
i[0].([]byte),
|
|
&EthReceipt{
|
|
Receipt: r,
|
|
cid: c,
|
|
rawdata: i[1].([]byte),
|
|
},
|
|
}, nil
|
|
}
|
|
|
|
/*
|
|
Block INTERFACE
|
|
*/
|
|
|
|
// RawData returns the binary of the RLP encode of the transaction.
|
|
func (t *EthRctTrie) RawData() []byte {
|
|
return t.rawdata
|
|
}
|
|
|
|
// Cid returns the cid of the transaction.
|
|
func (t *EthRctTrie) Cid() cid.Cid {
|
|
return t.cid
|
|
}
|
|
|
|
// String is a helper for output
|
|
func (t *EthRctTrie) String() string {
|
|
return fmt.Sprintf("<EthereumRctTrie %s>", t.cid)
|
|
}
|
|
|
|
// Loggable returns in a map the type of IPLD Link.
|
|
func (t *EthRctTrie) Loggable() map[string]interface{} {
|
|
return map[string]interface{}{
|
|
"type": "eth-rct-trie",
|
|
}
|
|
}
|
|
|
|
/*
|
|
EthRctTrie functions
|
|
*/
|
|
|
|
// rctTrie wraps a localTrie for use on the receipt trie.
|
|
type rctTrie struct {
|
|
*localTrie
|
|
}
|
|
|
|
// NewRctTrie initializes and returns a rctTrie.
|
|
func NewRctTrie() *rctTrie {
|
|
return &rctTrie{
|
|
localTrie: newLocalTrie(),
|
|
}
|
|
}
|
|
|
|
// GetNodes invokes the localTrie, which computes the root hash of the
|
|
// transaction trie and returns its sql keys, to return a slice
|
|
// of EthRctTrie nodes.
|
|
func (rt *rctTrie) GetNodes() ([]*EthRctTrie, error) {
|
|
keys, err := rt.getKeys()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
var out []*EthRctTrie
|
|
|
|
for _, k := range keys {
|
|
n, err := rt.getNodeFromDB(k)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
out = append(out, n)
|
|
}
|
|
|
|
return out, nil
|
|
}
|
|
|
|
// GetLeafNodes invokes the localTrie, which returns a slice
|
|
// of EthRctTrie leaf nodes.
|
|
func (rt *rctTrie) GetLeafNodes() ([]*EthRctTrie, []*nodeKey, error) {
|
|
keys, err := rt.getLeafKeys()
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
|
|
out := make([]*EthRctTrie, 0, len(keys))
|
|
for _, k := range keys {
|
|
n, err := rt.getNodeFromDB(k.dbKey)
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
out = append(out, n)
|
|
}
|
|
|
|
return out, keys, nil
|
|
}
|
|
|
|
func (rt *rctTrie) getNodeFromDB(key []byte) (*EthRctTrie, error) {
|
|
rawdata, err := rt.db.Get(key)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
tn := &TrieNode{
|
|
cid: keccak256ToCid(MEthTxReceiptTrie, key),
|
|
rawdata: rawdata,
|
|
}
|
|
|
|
return &EthRctTrie{TrieNode: tn}, nil
|
|
}
|