// 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 . // Streamer is used by watchers to stream eth data from a vulcanizedb super node package streamer import ( "encoding/json" "math/big" "github.com/ethereum/go-ethereum/common" "github.com/ethereum/go-ethereum/rpc" "github.com/vulcanize/vulcanizedb/pkg/config" "github.com/vulcanize/vulcanizedb/pkg/core" ) // ISuperNodeStreamer is the interface for streaming SuperNodePayloads from a vulcanizeDB super node type ISuperNodeStreamer interface { Stream(payloadChan chan SuperNodePayload, streamFilters config.Subscription) (*rpc.ClientSubscription, error) } // SuperNodeStreamer is the underlying struct for the ISuperNodeStreamer interface type SuperNodeStreamer struct { Client core.RPCClient } // NewSuperNodeStreamer creates a pointer to a new SuperNodeStreamer which satisfies the ISuperNodeStreamer interface func NewSuperNodeStreamer(client core.RPCClient) *SuperNodeStreamer { return &SuperNodeStreamer{ Client: client, } } // Stream is the main loop for subscribing to data from a vulcanizedb super node func (sds *SuperNodeStreamer) Stream(payloadChan chan SuperNodePayload, streamFilters config.Subscription) (*rpc.ClientSubscription, error) { return sds.Client.Subscribe("vdb", payloadChan, "stream", streamFilters) } // Payload holds the data returned from the super node to the requesting client type SuperNodePayload struct { BlockNumber *big.Int `json:"blockNumber"` HeadersRlp [][]byte `json:"headersRlp"` UnclesRlp [][]byte `json:"unclesRlp"` TransactionsRlp [][]byte `json:"transactionsRlp"` ReceiptsRlp [][]byte `json:"receiptsRlp"` StateNodesRlp map[common.Hash][]byte `json:"stateNodesRlp"` StorageNodesRlp map[common.Hash]map[common.Hash][]byte `json:"storageNodesRlp"` ErrMsg string `json:"errMsg"` encoded []byte err error } func (sd *SuperNodePayload) ensureEncoded() { if sd.encoded == nil && sd.err == nil { sd.encoded, sd.err = json.Marshal(sd) } } // Length to implement Encoder interface for StateDiff func (sd *SuperNodePayload) Length() int { sd.ensureEncoded() return len(sd.encoded) } // Encode to implement Encoder interface for StateDiff func (sd *SuperNodePayload) Encode() ([]byte, error) { sd.ensureEncoded() return sd.encoded, sd.err }