diff --git a/README.md b/README.md
index 21eba062..306ed2a1 100644
--- a/README.md
+++ b/README.md
@@ -161,10 +161,6 @@ This set of parameters needs to be set no matter the chain type.
user = "vdbm" # $DATABASE_USER
password = "" # $DATABASE_PASSWORD
-[ipfs]
- path = "~/.ipfs" # $IPFS_PATH
- mode = "postgres" # $IPFS_MODE
-
[watcher]
chain = "bitcoin" # $SUPERNODE_CHAIN
server = true # $SUPERNODE_SERVER
@@ -207,6 +203,7 @@ For Ethereum:
clientName = "Geth" # $ETH_CLIENT_NAME
genesisBlock = "0xd4e56740f876aef8c010b86a40d5f56745a118d0906a34e69aec8c0db1cb8fa3" # $ETH_GENESIS_BLOCK
networkID = "1" # $ETH_NETWORK_ID
+ chainID = "1" # $ETH_CHAIN_ID
```
### Exposing the data
diff --git a/cmd/resync.go b/cmd/resync.go
index 9ffde23e..0a04f3d5 100644
--- a/cmd/resync.go
+++ b/cmd/resync.go
@@ -19,9 +19,6 @@ import (
log "github.com/sirupsen/logrus"
"github.com/spf13/cobra"
"github.com/spf13/viper"
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/shared"
-
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/ipfs"
"github.com/vulcanize/ipfs-blockchain-watcher/pkg/resync"
v "github.com/vulcanize/ipfs-blockchain-watcher/version"
)
@@ -46,11 +43,6 @@ func rsyncCmdCommand() {
logWithCommand.Fatal(err)
}
logWithCommand.Infof("resync config: %+v", rConfig)
- if rConfig.IPFSMode == shared.LocalInterface {
- if err := ipfs.InitIPFSPlugins(); err != nil {
- logWithCommand.Fatal(err)
- }
- }
logWithCommand.Debug("initializing new resync service")
rService, err := resync.NewResyncService(rConfig)
if err != nil {
diff --git a/cmd/watch.go b/cmd/watch.go
index de4acc92..f0512557 100644
--- a/cmd/watch.go
+++ b/cmd/watch.go
@@ -26,7 +26,6 @@ import (
"github.com/spf13/viper"
h "github.com/vulcanize/ipfs-blockchain-watcher/pkg/historical"
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/ipfs"
"github.com/vulcanize/ipfs-blockchain-watcher/pkg/shared"
w "github.com/vulcanize/ipfs-blockchain-watcher/pkg/watch"
v "github.com/vulcanize/ipfs-blockchain-watcher/version"
@@ -65,11 +64,6 @@ func watch() {
logWithCommand.Fatal(err)
}
logWithCommand.Infof("watcher config: %+v", watcherConfig)
- if watcherConfig.IPFSMode == shared.LocalInterface {
- if err := ipfs.InitIPFSPlugins(); err != nil {
- logWithCommand.Fatal(err)
- }
- }
logWithCommand.Debug("initializing new watcher service")
watcher, err := w.NewWatcher(watcherConfig)
if err != nil {
diff --git a/db/migrations/00006_create_eth_transaction_cids_table.sql b/db/migrations/00006_create_eth_transaction_cids_table.sql
index cbbef782..f41412c5 100644
--- a/db/migrations/00006_create_eth_transaction_cids_table.sql
+++ b/db/migrations/00006_create_eth_transaction_cids_table.sql
@@ -8,6 +8,8 @@ CREATE TABLE eth.transaction_cids (
mh_key TEXT NOT NULL REFERENCES public.blocks (key) ON DELETE CASCADE DEFERRABLE INITIALLY DEFERRED,
dst VARCHAR(66) NOT NULL,
src VARCHAR(66) NOT NULL,
+ deployment BOOL NOT NULL,
+ data BYTEA,
UNIQUE (header_id, tx_hash)
);
diff --git a/db/schema.sql b/db/schema.sql
index caa1bb67..05c36096 100644
--- a/db/schema.sql
+++ b/db/schema.sql
@@ -415,7 +415,9 @@ CREATE TABLE eth.transaction_cids (
cid text NOT NULL,
mh_key text NOT NULL,
dst character varying(66) NOT NULL,
- src character varying(66) NOT NULL
+ src character varying(66) NOT NULL,
+ deployment boolean NOT NULL,
+ data bytea
);
diff --git a/dockerfiles/super_node/entrypoint.sh b/dockerfiles/super_node/entrypoint.sh
index 7df52bc5..ff93bc60 100755
--- a/dockerfiles/super_node/entrypoint.sh
+++ b/dockerfiles/super_node/entrypoint.sh
@@ -13,7 +13,6 @@ set +x
#test $DATABASE_USER
#test $DATABASE_PASSWORD
#test $IPFS_INIT
-#test $IPFS_PATH
VDB_COMMAND=${VDB_COMMAND:-watch}
set +e
diff --git a/dockerfiles/super_node/startup_script.sh b/dockerfiles/super_node/startup_script.sh
index 89039372..3ff9ca02 100755
--- a/dockerfiles/super_node/startup_script.sh
+++ b/dockerfiles/super_node/startup_script.sh
@@ -12,7 +12,6 @@ test $DATABASE_PORT
test $DATABASE_USER
test $DATABASE_PASSWORD
test $IPFS_INIT
-test $IPFS_PATH
test $VDB_COMMAND
set +e
diff --git a/documentation/architecture.md b/documentation/architecture.md
index 8b617c10..dd18e100 100644
--- a/documentation/architecture.md
+++ b/documentation/architecture.md
@@ -52,10 +52,6 @@ This set of parameters needs to be set no matter the chain type.
user = "vdbm" # $DATABASE_USER
password = "" # $DATABASE_PASSWORD
-[ipfs]
- path = "~/.ipfs" # $IPFS_PATH
- mode = "direct" # $IPFS_MODE
-
[watcher]
chain = "bitcoin" # $SUPERNODE_CHAIN
server = true # $SUPERNODE_SERVER
diff --git a/documentation/resync.md b/documentation/resync.md
index 44e5332d..b0de3c2e 100644
--- a/documentation/resync.md
+++ b/documentation/resync.md
@@ -30,9 +30,6 @@ This set of parameters needs to be set no matter the chain type.
port = 5432 # $DATABASE_PORT
user = "vdbm" # $DATABASE_USER
password = "" # $DATABASE_PASSWORD
-
-[ipfs]
- path = "~/.ipfs" # $IPFS_PATH
[resync]
chain = "ethereum" # $RESYNC_CHAIN
diff --git a/environments/superNodeETH.toml b/environments/superNodeETH.toml
index 11bcc709..02cd5d22 100644
--- a/environments/superNodeETH.toml
+++ b/environments/superNodeETH.toml
@@ -43,3 +43,4 @@
clientName = "Geth" # $ETH_CLIENT_NAME
genesisBlock = "0xd4e56740f876aef8c010b86a40d5f56745a118d0906a34e69aec8c0db1cb8fa3" # $ETH_GENESIS_BLOCK
networkID = "1" # $ETH_NETWORK_ID
+ chainID = "1" # $ETH_CHAIN_ID
diff --git a/pkg/btc/ipld_fetcher.go b/pkg/btc/ipld_fetcher.go
index 787fad4c..0e8f1213 100644
--- a/pkg/btc/ipld_fetcher.go
+++ b/pkg/btc/ipld_fetcher.go
@@ -17,38 +17,27 @@
package btc
import (
- "context"
- "errors"
"fmt"
- "github.com/ipfs/go-block-format"
- "github.com/ipfs/go-blockservice"
- "github.com/ipfs/go-cid"
+ "github.com/jmoiron/sqlx"
log "github.com/sirupsen/logrus"
"github.com/vulcanize/ipfs-blockchain-watcher/pkg/ipfs"
+ "github.com/vulcanize/ipfs-blockchain-watcher/pkg/postgres"
"github.com/vulcanize/ipfs-blockchain-watcher/pkg/shared"
)
-var (
- errUnexpectedNumberOfIPLDs = errors.New("ipfs batch fetch returned unexpected number of IPLDs")
-)
-
// IPLDFetcher satisfies the IPLDFetcher interface for ethereum
+// it interfaces directly with PG-IPFS instead of going through a node-interface or remote node
type IPLDFetcher struct {
- BlockService blockservice.BlockService
+ db *postgres.DB
}
// NewIPLDFetcher creates a pointer to a new IPLDFetcher
-// It interfaces with PG-IPFS through an internalized IPFS node interface
-func NewIPLDFetcher(ipfsPath string) (*IPLDFetcher, error) {
- blockService, err := ipfs.InitIPFSBlockService(ipfsPath)
- if err != nil {
- return nil, err
- }
+func NewIPLDFetcher(db *postgres.DB) *IPLDFetcher {
return &IPLDFetcher{
- BlockService: blockService,
- }, nil
+ db: db,
+ }
}
// Fetch is the exported method for fetching and returning all the IPLDS specified in the CIDWrapper
@@ -60,76 +49,59 @@ func (f *IPLDFetcher) Fetch(cids shared.CIDsForFetching) (shared.IPLDs, error) {
log.Debug("fetching iplds")
iplds := IPLDs{}
iplds.BlockNumber = cidWrapper.BlockNumber
- var err error
- iplds.Header, err = f.FetchHeader(cidWrapper.Header)
+
+ tx, err := f.db.Beginx()
if err != nil {
return nil, err
}
- iplds.Transactions, err = f.FetchTrxs(cidWrapper.Transactions)
+ defer func() {
+ if p := recover(); p != nil {
+ shared.Rollback(tx)
+ panic(p)
+ } else if err != nil {
+ shared.Rollback(tx)
+ } else {
+ err = tx.Commit()
+ }
+ }()
+
+ iplds.Header, err = f.FetchHeader(tx, cidWrapper.Header)
if err != nil {
- return nil, err
+ return nil, fmt.Errorf("btc pg fetcher: header fetching error: %s", err.Error())
}
- return iplds, nil
+ iplds.Transactions, err = f.FetchTrxs(tx, cidWrapper.Transactions)
+ if err != nil {
+ return nil, fmt.Errorf("btc pg fetcher: transaction fetching error: %s", err.Error())
+ }
+ return iplds, err
}
// FetchHeaders fetches headers
-// It uses the f.fetch method
-func (f *IPLDFetcher) FetchHeader(c HeaderModel) (ipfs.BlockModel, error) {
+func (f *IPLDFetcher) FetchHeader(tx *sqlx.Tx, c HeaderModel) (ipfs.BlockModel, error) {
log.Debug("fetching header ipld")
- dc, err := cid.Decode(c.CID)
- if err != nil {
- return ipfs.BlockModel{}, err
- }
- header, err := f.fetch(dc)
+ headerBytes, err := shared.FetchIPLDByMhKey(tx, c.MhKey)
if err != nil {
return ipfs.BlockModel{}, err
}
return ipfs.BlockModel{
- Data: header.RawData(),
- CID: header.Cid().String(),
+ Data: headerBytes,
+ CID: c.CID,
}, nil
}
// FetchTrxs fetches transactions
-// It uses the f.fetchBatch method
-func (f *IPLDFetcher) FetchTrxs(cids []TxModel) ([]ipfs.BlockModel, error) {
+func (f *IPLDFetcher) FetchTrxs(tx *sqlx.Tx, cids []TxModel) ([]ipfs.BlockModel, error) {
log.Debug("fetching transaction iplds")
- trxCids := make([]cid.Cid, len(cids))
+ trxIPLDs := make([]ipfs.BlockModel, len(cids))
for i, c := range cids {
- dc, err := cid.Decode(c.CID)
+ trxBytes, err := shared.FetchIPLDByMhKey(tx, c.MhKey)
if err != nil {
return nil, err
}
- trxCids[i] = dc
- }
- trxs := f.fetchBatch(trxCids)
- trxIPLDs := make([]ipfs.BlockModel, len(trxs))
- for i, trx := range trxs {
trxIPLDs[i] = ipfs.BlockModel{
- Data: trx.RawData(),
- CID: trx.Cid().String(),
+ Data: trxBytes,
+ CID: c.CID,
}
}
- if len(trxIPLDs) != len(trxCids) {
- log.Errorf("ipfs fetcher: number of transaction blocks returned (%d) does not match number expected (%d)", len(trxs), len(trxCids))
- return trxIPLDs, errUnexpectedNumberOfIPLDs
- }
return trxIPLDs, nil
}
-
-// fetch is used to fetch a single cid
-func (f *IPLDFetcher) fetch(cid cid.Cid) (blocks.Block, error) {
- return f.BlockService.GetBlock(context.Background(), cid)
-}
-
-// fetchBatch is used to fetch a batch of IPFS data blocks by cid
-// There is no guarantee all are fetched, and no error in such a case, so
-// downstream we will need to confirm which CIDs were fetched in the result set
-func (f *IPLDFetcher) fetchBatch(cids []cid.Cid) []blocks.Block {
- fetchedBlocks := make([]blocks.Block, 0, len(cids))
- blockChan := f.BlockService.GetBlocks(context.Background(), cids)
- for block := range blockChan {
- fetchedBlocks = append(fetchedBlocks, block)
- }
- return fetchedBlocks
-}
diff --git a/pkg/btc/ipld_pg_fetcher.go b/pkg/btc/ipld_pg_fetcher.go
deleted file mode 100644
index cd673c75..00000000
--- a/pkg/btc/ipld_pg_fetcher.go
+++ /dev/null
@@ -1,107 +0,0 @@
-// 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 .
-
-package btc
-
-import (
- "fmt"
-
- "github.com/jmoiron/sqlx"
- log "github.com/sirupsen/logrus"
-
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/ipfs"
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/postgres"
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/shared"
-)
-
-// IPLDPGFetcher satisfies the IPLDFetcher interface for ethereum
-// it interfaces directly with PG-IPFS instead of going through a node-interface or remote node
-type IPLDPGFetcher struct {
- db *postgres.DB
-}
-
-// NewIPLDPGFetcher creates a pointer to a new IPLDPGFetcher
-func NewIPLDPGFetcher(db *postgres.DB) *IPLDPGFetcher {
- return &IPLDPGFetcher{
- db: db,
- }
-}
-
-// Fetch is the exported method for fetching and returning all the IPLDS specified in the CIDWrapper
-func (f *IPLDPGFetcher) Fetch(cids shared.CIDsForFetching) (shared.IPLDs, error) {
- cidWrapper, ok := cids.(*CIDWrapper)
- if !ok {
- return nil, fmt.Errorf("btc fetcher: expected cids type %T got %T", &CIDWrapper{}, cids)
- }
- log.Debug("fetching iplds")
- iplds := IPLDs{}
- iplds.BlockNumber = cidWrapper.BlockNumber
-
- tx, err := f.db.Beginx()
- if err != nil {
- return nil, err
- }
- defer func() {
- if p := recover(); p != nil {
- shared.Rollback(tx)
- panic(p)
- } else if err != nil {
- shared.Rollback(tx)
- } else {
- err = tx.Commit()
- }
- }()
-
- iplds.Header, err = f.FetchHeader(tx, cidWrapper.Header)
- if err != nil {
- return nil, fmt.Errorf("btc pg fetcher: header fetching error: %s", err.Error())
- }
- iplds.Transactions, err = f.FetchTrxs(tx, cidWrapper.Transactions)
- if err != nil {
- return nil, fmt.Errorf("btc pg fetcher: transaction fetching error: %s", err.Error())
- }
- return iplds, err
-}
-
-// FetchHeaders fetches headers
-func (f *IPLDPGFetcher) FetchHeader(tx *sqlx.Tx, c HeaderModel) (ipfs.BlockModel, error) {
- log.Debug("fetching header ipld")
- headerBytes, err := shared.FetchIPLDByMhKey(tx, c.MhKey)
- if err != nil {
- return ipfs.BlockModel{}, err
- }
- return ipfs.BlockModel{
- Data: headerBytes,
- CID: c.CID,
- }, nil
-}
-
-// FetchTrxs fetches transactions
-func (f *IPLDPGFetcher) FetchTrxs(tx *sqlx.Tx, cids []TxModel) ([]ipfs.BlockModel, error) {
- log.Debug("fetching transaction iplds")
- trxIPLDs := make([]ipfs.BlockModel, len(cids))
- for i, c := range cids {
- trxBytes, err := shared.FetchIPLDByMhKey(tx, c.MhKey)
- if err != nil {
- return nil, err
- }
- trxIPLDs[i] = ipfs.BlockModel{
- Data: trxBytes,
- CID: c.CID,
- }
- }
- return trxIPLDs, nil
-}
diff --git a/pkg/btc/publish_and_indexer.go b/pkg/btc/publish_and_indexer.go
deleted file mode 100644
index dd5a72cb..00000000
--- a/pkg/btc/publish_and_indexer.go
+++ /dev/null
@@ -1,126 +0,0 @@
-// 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 .
-
-package btc
-
-import (
- "fmt"
- "strconv"
-
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/ipfs/ipld"
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/postgres"
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/shared"
-)
-
-// IPLDPublisherAndIndexer satisfies the IPLDPublisher interface for bitcoin
-// It interfaces directly with the public.blocks table of PG-IPFS rather than going through an ipfs intermediary
-// It publishes and indexes IPLDs together in a single sqlx.Tx
-type IPLDPublisherAndIndexer struct {
- indexer *CIDIndexer
-}
-
-// NewIPLDPublisherAndIndexer creates a pointer to a new IPLDPublisherAndIndexer which satisfies the IPLDPublisher interface
-func NewIPLDPublisherAndIndexer(db *postgres.DB) *IPLDPublisherAndIndexer {
- return &IPLDPublisherAndIndexer{
- indexer: NewCIDIndexer(db),
- }
-}
-
-// Publish publishes an IPLDPayload to IPFS and returns the corresponding CIDPayload
-func (pub *IPLDPublisherAndIndexer) Publish(payload shared.ConvertedData) (shared.CIDsForIndexing, error) {
- ipldPayload, ok := payload.(ConvertedPayload)
- if !ok {
- return nil, fmt.Errorf("btc publisher expected payload type %T got %T", ConvertedPayload{}, payload)
- }
- // Generate the iplds
- headerNode, txNodes, txTrieNodes, err := ipld.FromHeaderAndTxs(ipldPayload.Header, ipldPayload.Txs)
- if err != nil {
- return nil, err
- }
-
- // Begin new db tx
- tx, err := pub.indexer.db.Beginx()
- if err != nil {
- return nil, err
- }
- defer func() {
- if p := recover(); p != nil {
- shared.Rollback(tx)
- panic(p)
- } else if err != nil {
- shared.Rollback(tx)
- } else {
- err = tx.Commit()
- }
- }()
-
- // Publish trie nodes
- for _, node := range txTrieNodes {
- if err := shared.PublishIPLD(tx, node); err != nil {
- return nil, err
- }
- }
-
- // Publish and index header
- if err := shared.PublishIPLD(tx, headerNode); err != nil {
- return nil, err
- }
- header := HeaderModel{
- CID: headerNode.Cid().String(),
- MhKey: shared.MultihashKeyFromCID(headerNode.Cid()),
- ParentHash: ipldPayload.Header.PrevBlock.String(),
- BlockNumber: strconv.Itoa(int(ipldPayload.BlockPayload.BlockHeight)),
- BlockHash: ipldPayload.Header.BlockHash().String(),
- Timestamp: ipldPayload.Header.Timestamp.UnixNano(),
- Bits: ipldPayload.Header.Bits,
- }
- headerID, err := pub.indexer.indexHeaderCID(tx, header)
- if err != nil {
- return nil, err
- }
-
- // Publish and index txs
- for i, txNode := range txNodes {
- if err := shared.PublishIPLD(tx, txNode); err != nil {
- return nil, err
- }
- txModel := ipldPayload.TxMetaData[i]
- txModel.CID = txNode.Cid().String()
- txModel.MhKey = shared.MultihashKeyFromCID(txNode.Cid())
- txID, err := pub.indexer.indexTransactionCID(tx, txModel, headerID)
- if err != nil {
- return nil, err
- }
- for _, input := range txModel.TxInputs {
- if err := pub.indexer.indexTxInput(tx, input, txID); err != nil {
- return nil, err
- }
- }
- for _, output := range txModel.TxOutputs {
- if err := pub.indexer.indexTxOutput(tx, output, txID); err != nil {
- return nil, err
- }
- }
- }
-
- // This IPLDPublisher does both publishing and indexing, we do not need to pass anything forward to the indexer
- return nil, err
-}
-
-// Index satisfies the shared.CIDIndexer interface
-func (pub *IPLDPublisherAndIndexer) Index(cids shared.CIDsForIndexing) error {
- return nil
-}
diff --git a/pkg/btc/publish_and_indexer_test.go b/pkg/btc/publish_and_indexer_test.go
deleted file mode 100644
index 3271db9a..00000000
--- a/pkg/btc/publish_and_indexer_test.go
+++ /dev/null
@@ -1,121 +0,0 @@
-// 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 .
-
-package btc_test
-
-import (
- "bytes"
-
- "github.com/ipfs/go-cid"
- "github.com/ipfs/go-ipfs-blockstore"
- "github.com/ipfs/go-ipfs-ds-help"
- "github.com/multiformats/go-multihash"
- . "github.com/onsi/ginkgo"
- . "github.com/onsi/gomega"
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/ipfs/ipld"
-
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/btc"
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/btc/mocks"
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/postgres"
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/shared"
-)
-
-var _ = Describe("PublishAndIndexer", func() {
- var (
- db *postgres.DB
- err error
- repo *btc.IPLDPublisherAndIndexer
- ipfsPgGet = `SELECT data FROM public.blocks
- WHERE key = $1`
- )
- BeforeEach(func() {
- db, err = shared.SetupDB()
- Expect(err).ToNot(HaveOccurred())
- repo = btc.NewIPLDPublisherAndIndexer(db)
- })
- AfterEach(func() {
- btc.TearDownDB(db)
- })
-
- Describe("Publish", func() {
- It("Published and indexes header and transaction IPLDs in a single tx", func() {
- emptyReturn, err := repo.Publish(mocks.MockConvertedPayload)
- Expect(emptyReturn).To(BeNil())
- Expect(err).ToNot(HaveOccurred())
- pgStr := `SELECT * FROM btc.header_cids
- WHERE block_number = $1`
- // check header was properly indexed
- buf := bytes.NewBuffer(make([]byte, 0, 80))
- err = mocks.MockBlock.Header.Serialize(buf)
- Expect(err).ToNot(HaveOccurred())
- headerBytes := buf.Bytes()
- c, _ := ipld.RawdataToCid(ipld.MBitcoinHeader, headerBytes, multihash.DBL_SHA2_256)
- header := new(btc.HeaderModel)
- err = db.Get(header, pgStr, mocks.MockHeaderMetaData.BlockNumber)
- Expect(err).ToNot(HaveOccurred())
- Expect(header.CID).To(Equal(c.String()))
- Expect(header.BlockNumber).To(Equal(mocks.MockHeaderMetaData.BlockNumber))
- Expect(header.Bits).To(Equal(mocks.MockHeaderMetaData.Bits))
- Expect(header.Timestamp).To(Equal(mocks.MockHeaderMetaData.Timestamp))
- Expect(header.BlockHash).To(Equal(mocks.MockHeaderMetaData.BlockHash))
- Expect(header.ParentHash).To(Equal(mocks.MockHeaderMetaData.ParentHash))
- dc, err := cid.Decode(header.CID)
- Expect(err).ToNot(HaveOccurred())
- mhKey := dshelp.MultihashToDsKey(dc.Hash())
- prefixedKey := blockstore.BlockPrefix.String() + mhKey.String()
- var data []byte
- err = db.Get(&data, ipfsPgGet, prefixedKey)
- Expect(err).ToNot(HaveOccurred())
- Expect(data).To(Equal(headerBytes))
-
- // check that txs were properly indexed
- trxs := make([]btc.TxModel, 0)
- pgStr = `SELECT transaction_cids.id, transaction_cids.header_id, transaction_cids.index,
- transaction_cids.tx_hash, transaction_cids.cid, transaction_cids.segwit, transaction_cids.witness_hash
- FROM btc.transaction_cids INNER JOIN btc.header_cids ON (transaction_cids.header_id = header_cids.id)
- WHERE header_cids.block_number = $1`
- err = db.Select(&trxs, pgStr, mocks.MockHeaderMetaData.BlockNumber)
- Expect(err).ToNot(HaveOccurred())
- Expect(len(trxs)).To(Equal(3))
- txData := make([][]byte, len(mocks.MockTransactions))
- txCIDs := make([]string, len(mocks.MockTransactions))
- for i, m := range mocks.MockTransactions {
- buf := bytes.NewBuffer(make([]byte, 0))
- err = m.MsgTx().Serialize(buf)
- Expect(err).ToNot(HaveOccurred())
- tx := buf.Bytes()
- txData[i] = tx
- c, _ := ipld.RawdataToCid(ipld.MBitcoinTx, tx, multihash.DBL_SHA2_256)
- txCIDs[i] = c.String()
- }
- for _, tx := range trxs {
- Expect(tx.SegWit).To(Equal(false))
- Expect(tx.HeaderID).To(Equal(header.ID))
- Expect(tx.WitnessHash).To(Equal(""))
- Expect(tx.CID).To(Equal(txCIDs[tx.Index]))
- Expect(tx.TxHash).To(Equal(mocks.MockBlock.Transactions[tx.Index].TxHash().String()))
- dc, err := cid.Decode(tx.CID)
- Expect(err).ToNot(HaveOccurred())
- mhKey := dshelp.MultihashToDsKey(dc.Hash())
- prefixedKey := blockstore.BlockPrefix.String() + mhKey.String()
- var data []byte
- err = db.Get(&data, ipfsPgGet, prefixedKey)
- Expect(err).ToNot(HaveOccurred())
- Expect(data).To(Equal(txData[tx.Index]))
- }
- })
- })
-})
diff --git a/pkg/btc/publisher.go b/pkg/btc/publisher.go
index 5c6c4d07..74a3ba56 100644
--- a/pkg/btc/publisher.go
+++ b/pkg/btc/publisher.go
@@ -20,102 +20,101 @@ import (
"fmt"
"strconv"
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/ipfs"
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/ipfs/dag_putters"
"github.com/vulcanize/ipfs-blockchain-watcher/pkg/ipfs/ipld"
+ "github.com/vulcanize/ipfs-blockchain-watcher/pkg/postgres"
"github.com/vulcanize/ipfs-blockchain-watcher/pkg/shared"
)
-// IPLDPublisher satisfies the IPLDPublisher for ethereum
+// IPLDPublisher satisfies the IPLDPublisher interface for bitcoin
+// It interfaces directly with the public.blocks table of PG-IPFS rather than going through an ipfs intermediary
+// It publishes and indexes IPLDs together in a single sqlx.Tx
type IPLDPublisher struct {
- HeaderPutter ipfs.DagPutter
- TransactionPutter ipfs.DagPutter
- TransactionTriePutter ipfs.DagPutter
+ indexer *CIDIndexer
}
-// NewIPLDPublisher creates a pointer to a new Publisher which satisfies the IPLDPublisher interface
-func NewIPLDPublisher(ipfsPath string) (*IPLDPublisher, error) {
- node, err := ipfs.InitIPFSNode(ipfsPath)
- if err != nil {
- return nil, err
- }
+// NewIPLDPublisher creates a pointer to a new eth IPLDPublisher which satisfies the IPLDPublisher interface
+func NewIPLDPublisher(db *postgres.DB) *IPLDPublisher {
return &IPLDPublisher{
- HeaderPutter: dag_putters.NewBtcHeaderDagPutter(node),
- TransactionPutter: dag_putters.NewBtcTxDagPutter(node),
- TransactionTriePutter: dag_putters.NewBtcTxTrieDagPutter(node),
- }, nil
+ indexer: NewCIDIndexer(db),
+ }
}
// Publish publishes an IPLDPayload to IPFS and returns the corresponding CIDPayload
-func (pub *IPLDPublisher) Publish(payload shared.ConvertedData) (shared.CIDsForIndexing, error) {
+func (pub *IPLDPublisher) Publish(payload shared.ConvertedData) error {
ipldPayload, ok := payload.(ConvertedPayload)
if !ok {
- return nil, fmt.Errorf("eth publisher expected payload type %T got %T", &ConvertedPayload{}, payload)
+ return fmt.Errorf("btc publisher expected payload type %T got %T", ConvertedPayload{}, payload)
}
- // Generate nodes
+ // Generate the iplds
headerNode, txNodes, txTrieNodes, err := ipld.FromHeaderAndTxs(ipldPayload.Header, ipldPayload.Txs)
if err != nil {
- return nil, err
+ return err
}
- // Process and publish headers
- headerCid, err := pub.publishHeader(headerNode)
+
+ // Begin new db tx
+ tx, err := pub.indexer.db.Beginx()
if err != nil {
- return nil, err
+ return err
+ }
+ defer func() {
+ if p := recover(); p != nil {
+ shared.Rollback(tx)
+ panic(p)
+ } else if err != nil {
+ shared.Rollback(tx)
+ } else {
+ err = tx.Commit()
+ }
+ }()
+
+ // Publish trie nodes
+ for _, node := range txTrieNodes {
+ if err := shared.PublishIPLD(tx, node); err != nil {
+ return err
+ }
+ }
+
+ // Publish and index header
+ if err := shared.PublishIPLD(tx, headerNode); err != nil {
+ return err
}
- mhKey, _ := shared.MultihashKeyFromCIDString(headerCid)
header := HeaderModel{
- CID: headerCid,
- MhKey: mhKey,
+ CID: headerNode.Cid().String(),
+ MhKey: shared.MultihashKeyFromCID(headerNode.Cid()),
ParentHash: ipldPayload.Header.PrevBlock.String(),
BlockNumber: strconv.Itoa(int(ipldPayload.BlockPayload.BlockHeight)),
BlockHash: ipldPayload.Header.BlockHash().String(),
Timestamp: ipldPayload.Header.Timestamp.UnixNano(),
Bits: ipldPayload.Header.Bits,
}
- // Process and publish transactions
- transactionCids, err := pub.publishTransactions(txNodes, txTrieNodes, ipldPayload.TxMetaData)
+ headerID, err := pub.indexer.indexHeaderCID(tx, header)
if err != nil {
- return nil, err
+ return err
}
- // Package CIDs and their metadata into a single struct
- return &CIDPayload{
- HeaderCID: header,
- TransactionCIDs: transactionCids,
- }, nil
-}
-func (pub *IPLDPublisher) publishHeader(header *ipld.BtcHeader) (string, error) {
- cid, err := pub.HeaderPutter.DagPut(header)
- if err != nil {
- return "", err
- }
- return cid, nil
-}
-
-func (pub *IPLDPublisher) publishTransactions(transactions []*ipld.BtcTx, txTrie []*ipld.BtcTxTrie, trxMeta []TxModelWithInsAndOuts) ([]TxModelWithInsAndOuts, error) {
- txCids := make([]TxModelWithInsAndOuts, len(transactions))
- for i, tx := range transactions {
- cid, err := pub.TransactionPutter.DagPut(tx)
+ // Publish and index txs
+ for i, txNode := range txNodes {
+ if err := shared.PublishIPLD(tx, txNode); err != nil {
+ return err
+ }
+ txModel := ipldPayload.TxMetaData[i]
+ txModel.CID = txNode.Cid().String()
+ txModel.MhKey = shared.MultihashKeyFromCID(txNode.Cid())
+ txID, err := pub.indexer.indexTransactionCID(tx, txModel, headerID)
if err != nil {
- return nil, err
+ return err
}
- mhKey, _ := shared.MultihashKeyFromCIDString(cid)
- txCids[i] = TxModelWithInsAndOuts{
- CID: cid,
- MhKey: mhKey,
- Index: trxMeta[i].Index,
- TxHash: trxMeta[i].TxHash,
- SegWit: trxMeta[i].SegWit,
- WitnessHash: trxMeta[i].WitnessHash,
- TxInputs: trxMeta[i].TxInputs,
- TxOutputs: trxMeta[i].TxOutputs,
+ for _, input := range txModel.TxInputs {
+ if err := pub.indexer.indexTxInput(tx, input, txID); err != nil {
+ return err
+ }
+ }
+ for _, output := range txModel.TxOutputs {
+ if err := pub.indexer.indexTxOutput(tx, output, txID); err != nil {
+ return err
+ }
}
}
- for _, txNode := range txTrie {
- // We don't do anything with the tx trie cids atm
- if _, err := pub.TransactionTriePutter.DagPut(txNode); err != nil {
- return nil, err
- }
- }
- return txCids, nil
+
+ return err
}
diff --git a/pkg/btc/publisher_test.go b/pkg/btc/publisher_test.go
index ef809f3a..b2277b6d 100644
--- a/pkg/btc/publisher_test.go
+++ b/pkg/btc/publisher_test.go
@@ -19,63 +19,102 @@ package btc_test
import (
"bytes"
- "github.com/ethereum/go-ethereum/common"
+ "github.com/ipfs/go-cid"
+ "github.com/ipfs/go-ipfs-blockstore"
+ "github.com/ipfs/go-ipfs-ds-help"
+ "github.com/multiformats/go-multihash"
. "github.com/onsi/ginkgo"
. "github.com/onsi/gomega"
+ "github.com/vulcanize/ipfs-blockchain-watcher/pkg/ipfs/ipld"
"github.com/vulcanize/ipfs-blockchain-watcher/pkg/btc"
"github.com/vulcanize/ipfs-blockchain-watcher/pkg/btc/mocks"
- mocks2 "github.com/vulcanize/ipfs-blockchain-watcher/pkg/ipfs/mocks"
+ "github.com/vulcanize/ipfs-blockchain-watcher/pkg/postgres"
+ "github.com/vulcanize/ipfs-blockchain-watcher/pkg/shared"
)
-var (
- mockHeaderDagPutter *mocks2.MappedDagPutter
- mockTrxDagPutter *mocks2.MappedDagPutter
- mockTrxTrieDagPutter *mocks2.DagPutter
-)
-
-var _ = Describe("Publisher", func() {
+var _ = Describe("PublishAndIndexer", func() {
+ var (
+ db *postgres.DB
+ err error
+ repo *btc.IPLDPublisher
+ ipfsPgGet = `SELECT data FROM public.blocks
+ WHERE key = $1`
+ )
BeforeEach(func() {
- mockHeaderDagPutter = new(mocks2.MappedDagPutter)
- mockTrxDagPutter = new(mocks2.MappedDagPutter)
- mockTrxTrieDagPutter = new(mocks2.DagPutter)
+ db, err = shared.SetupDB()
+ Expect(err).ToNot(HaveOccurred())
+ repo = btc.NewIPLDPublisher(db)
+ })
+ AfterEach(func() {
+ btc.TearDownDB(db)
})
Describe("Publish", func() {
- It("Publishes the passed IPLDPayload objects to IPFS and returns a CIDPayload for indexing", func() {
- by := new(bytes.Buffer)
- err := mocks.MockConvertedPayload.BlockPayload.Header.Serialize(by)
+ It("Published and indexes header and transaction IPLDs in a single tx", func() {
+ err = repo.Publish(mocks.MockConvertedPayload)
Expect(err).ToNot(HaveOccurred())
- headerBytes := by.Bytes()
- err = mocks.MockTransactions[0].MsgTx().Serialize(by)
+ pgStr := `SELECT * FROM btc.header_cids
+ WHERE block_number = $1`
+ // check header was properly indexed
+ buf := bytes.NewBuffer(make([]byte, 0, 80))
+ err = mocks.MockBlock.Header.Serialize(buf)
Expect(err).ToNot(HaveOccurred())
- tx1Bytes := by.Bytes()
- err = mocks.MockTransactions[1].MsgTx().Serialize(by)
+ headerBytes := buf.Bytes()
+ c, _ := ipld.RawdataToCid(ipld.MBitcoinHeader, headerBytes, multihash.DBL_SHA2_256)
+ header := new(btc.HeaderModel)
+ err = db.Get(header, pgStr, mocks.MockHeaderMetaData.BlockNumber)
Expect(err).ToNot(HaveOccurred())
- tx2Bytes := by.Bytes()
- err = mocks.MockTransactions[2].MsgTx().Serialize(by)
+ Expect(header.CID).To(Equal(c.String()))
+ Expect(header.BlockNumber).To(Equal(mocks.MockHeaderMetaData.BlockNumber))
+ Expect(header.Bits).To(Equal(mocks.MockHeaderMetaData.Bits))
+ Expect(header.Timestamp).To(Equal(mocks.MockHeaderMetaData.Timestamp))
+ Expect(header.BlockHash).To(Equal(mocks.MockHeaderMetaData.BlockHash))
+ Expect(header.ParentHash).To(Equal(mocks.MockHeaderMetaData.ParentHash))
+ dc, err := cid.Decode(header.CID)
Expect(err).ToNot(HaveOccurred())
- tx3Bytes := by.Bytes()
- mockHeaderDagPutter.CIDsToReturn = map[common.Hash]string{
- common.BytesToHash(headerBytes): mocks.MockHeaderCID.String(),
+ mhKey := dshelp.MultihashToDsKey(dc.Hash())
+ prefixedKey := blockstore.BlockPrefix.String() + mhKey.String()
+ var data []byte
+ err = db.Get(&data, ipfsPgGet, prefixedKey)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(data).To(Equal(headerBytes))
+
+ // check that txs were properly indexed
+ trxs := make([]btc.TxModel, 0)
+ pgStr = `SELECT transaction_cids.id, transaction_cids.header_id, transaction_cids.index,
+ transaction_cids.tx_hash, transaction_cids.cid, transaction_cids.segwit, transaction_cids.witness_hash
+ FROM btc.transaction_cids INNER JOIN btc.header_cids ON (transaction_cids.header_id = header_cids.id)
+ WHERE header_cids.block_number = $1`
+ err = db.Select(&trxs, pgStr, mocks.MockHeaderMetaData.BlockNumber)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(len(trxs)).To(Equal(3))
+ txData := make([][]byte, len(mocks.MockTransactions))
+ txCIDs := make([]string, len(mocks.MockTransactions))
+ for i, m := range mocks.MockTransactions {
+ buf := bytes.NewBuffer(make([]byte, 0))
+ err = m.MsgTx().Serialize(buf)
+ Expect(err).ToNot(HaveOccurred())
+ tx := buf.Bytes()
+ txData[i] = tx
+ c, _ := ipld.RawdataToCid(ipld.MBitcoinTx, tx, multihash.DBL_SHA2_256)
+ txCIDs[i] = c.String()
}
- mockTrxDagPutter.CIDsToReturn = map[common.Hash]string{
- common.BytesToHash(tx1Bytes): mocks.MockTrxCID1.String(),
- common.BytesToHash(tx2Bytes): mocks.MockTrxCID2.String(),
- common.BytesToHash(tx3Bytes): mocks.MockTrxCID3.String(),
+ for _, tx := range trxs {
+ Expect(tx.SegWit).To(Equal(false))
+ Expect(tx.HeaderID).To(Equal(header.ID))
+ Expect(tx.WitnessHash).To(Equal(""))
+ Expect(tx.CID).To(Equal(txCIDs[tx.Index]))
+ Expect(tx.TxHash).To(Equal(mocks.MockBlock.Transactions[tx.Index].TxHash().String()))
+ dc, err := cid.Decode(tx.CID)
+ Expect(err).ToNot(HaveOccurred())
+ mhKey := dshelp.MultihashToDsKey(dc.Hash())
+ prefixedKey := blockstore.BlockPrefix.String() + mhKey.String()
+ var data []byte
+ err = db.Get(&data, ipfsPgGet, prefixedKey)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(data).To(Equal(txData[tx.Index]))
}
- publisher := btc.IPLDPublisher{
- HeaderPutter: mockHeaderDagPutter,
- TransactionPutter: mockTrxDagPutter,
- TransactionTriePutter: mockTrxTrieDagPutter,
- }
- payload, err := publisher.Publish(mocks.MockConvertedPayload)
- Expect(err).ToNot(HaveOccurred())
- cidPayload, ok := payload.(*btc.CIDPayload)
- Expect(ok).To(BeTrue())
- Expect(cidPayload).To(Equal(&mocks.MockCIDPayload))
- Expect(cidPayload.HeaderCID).To(Equal(mocks.MockHeaderMetaData))
- Expect(cidPayload.TransactionCIDs).To(Equal(mocks.MockTxsMetaDataPostPublish))
})
})
})
diff --git a/pkg/builders/builders.go b/pkg/builders/builders.go
index 51aa3432..226dc2ef 100644
--- a/pkg/builders/builders.go
+++ b/pkg/builders/builders.go
@@ -22,7 +22,6 @@ import (
"github.com/btcsuite/btcd/chaincfg"
"github.com/btcsuite/btcd/rpcclient"
- "github.com/ethereum/go-ethereum/params"
"github.com/ethereum/go-ethereum/rpc"
"github.com/vulcanize/ipfs-blockchain-watcher/pkg/btc"
@@ -43,32 +42,6 @@ func NewResponseFilterer(chain shared.ChainType) (shared.ResponseFilterer, error
}
}
-// NewCIDIndexer constructs a CIDIndexer for the provided chain type
-func NewCIDIndexer(chain shared.ChainType, db *postgres.DB, ipfsMode shared.IPFSMode) (shared.CIDIndexer, error) {
- switch chain {
- case shared.Ethereum:
- switch ipfsMode {
- case shared.LocalInterface, shared.RemoteClient:
- return eth.NewCIDIndexer(db), nil
- case shared.DirectPostgres:
- return eth.NewIPLDPublisherAndIndexer(db), nil
- default:
- return nil, fmt.Errorf("ethereum CIDIndexer unexpected ipfs mode %s", ipfsMode.String())
- }
- case shared.Bitcoin:
- switch ipfsMode {
- case shared.LocalInterface, shared.RemoteClient:
- return btc.NewCIDIndexer(db), nil
- case shared.DirectPostgres:
- return eth.NewIPLDPublisherAndIndexer(db), nil
- default:
- return nil, fmt.Errorf("bitcoin CIDIndexer unexpected ipfs mode %s", ipfsMode.String())
- }
- default:
- return nil, fmt.Errorf("invalid chain %s for indexer constructor", chain.String())
- }
-}
-
// NewCIDRetriever constructs a CIDRetriever for the provided chain type
func NewCIDRetriever(chain shared.ChainType, db *postgres.DB) (shared.CIDRetriever, error) {
switch chain {
@@ -124,71 +97,47 @@ func NewPaylaodFetcher(chain shared.ChainType, client interface{}, timeout time.
}
// NewPayloadConverter constructs a PayloadConverter for the provided chain type
-func NewPayloadConverter(chain shared.ChainType) (shared.PayloadConverter, error) {
- switch chain {
+func NewPayloadConverter(chainType shared.ChainType, chainID uint64) (shared.PayloadConverter, error) {
+ switch chainType {
case shared.Ethereum:
- return eth.NewPayloadConverter(params.MainnetChainConfig), nil
+ chainConfig, err := eth.ChainConfig(chainID)
+ if err != nil {
+ return nil, err
+ }
+ return eth.NewPayloadConverter(chainConfig), nil
case shared.Bitcoin:
return btc.NewPayloadConverter(&chaincfg.MainNetParams), nil
default:
- return nil, fmt.Errorf("invalid chain %s for converter constructor", chain.String())
+ return nil, fmt.Errorf("invalid chain %s for converter constructor", chainType.String())
}
}
// NewIPLDFetcher constructs an IPLDFetcher for the provided chain type
-func NewIPLDFetcher(chain shared.ChainType, ipfsPath string, db *postgres.DB, ipfsMode shared.IPFSMode) (shared.IPLDFetcher, error) {
+func NewIPLDFetcher(chain shared.ChainType, db *postgres.DB) (shared.IPLDFetcher, error) {
switch chain {
case shared.Ethereum:
- switch ipfsMode {
- case shared.LocalInterface, shared.RemoteClient:
- return eth.NewIPLDFetcher(ipfsPath)
- case shared.DirectPostgres:
- return eth.NewIPLDPGFetcher(db), nil
- default:
- return nil, fmt.Errorf("ethereum IPLDFetcher unexpected ipfs mode %s", ipfsMode.String())
- }
+ return eth.NewIPLDFetcher(db), nil
case shared.Bitcoin:
- switch ipfsMode {
- case shared.LocalInterface, shared.RemoteClient:
- return btc.NewIPLDFetcher(ipfsPath)
- case shared.DirectPostgres:
- return btc.NewIPLDPGFetcher(db), nil
- default:
- return nil, fmt.Errorf("bitcoin IPLDFetcher unexpected ipfs mode %s", ipfsMode.String())
- }
+ return btc.NewIPLDFetcher(db), nil
default:
return nil, fmt.Errorf("invalid chain %s for IPLD fetcher constructor", chain.String())
}
}
// NewIPLDPublisher constructs an IPLDPublisher for the provided chain type
-func NewIPLDPublisher(chain shared.ChainType, ipfsPath string, db *postgres.DB, ipfsMode shared.IPFSMode) (shared.IPLDPublisher, error) {
+func NewIPLDPublisher(chain shared.ChainType, db *postgres.DB) (shared.IPLDPublisher, error) {
switch chain {
case shared.Ethereum:
- switch ipfsMode {
- case shared.LocalInterface, shared.RemoteClient:
- return eth.NewIPLDPublisher(ipfsPath)
- case shared.DirectPostgres:
- return eth.NewIPLDPublisherAndIndexer(db), nil
- default:
- return nil, fmt.Errorf("ethereum IPLDPublisher unexpected ipfs mode %s", ipfsMode.String())
- }
+ return eth.NewIPLDPublisher(db), nil
case shared.Bitcoin:
- switch ipfsMode {
- case shared.LocalInterface, shared.RemoteClient:
- return btc.NewIPLDPublisher(ipfsPath)
- case shared.DirectPostgres:
- return btc.NewIPLDPublisherAndIndexer(db), nil
- default:
- return nil, fmt.Errorf("bitcoin IPLDPublisher unexpected ipfs mode %s", ipfsMode.String())
- }
+ return btc.NewIPLDPublisher(db), nil
default:
return nil, fmt.Errorf("invalid chain %s for publisher constructor", chain.String())
}
}
// NewPublicAPI constructs a PublicAPI for the provided chain type
-func NewPublicAPI(chain shared.ChainType, db *postgres.DB, ipfsPath string) (rpc.API, error) {
+func NewPublicAPI(chain shared.ChainType, db *postgres.DB) (rpc.API, error) {
switch chain {
case shared.Ethereum:
backend, err := eth.NewEthBackend(db)
diff --git a/pkg/eth/api_test.go b/pkg/eth/api_test.go
index edf4ec40..f41b3b44 100644
--- a/pkg/eth/api_test.go
+++ b/pkg/eth/api_test.go
@@ -84,8 +84,8 @@ var _ = Describe("API", func() {
var (
db *postgres.DB
retriever *eth.CIDRetriever
- fetcher *eth.IPLDPGFetcher
- indexAndPublisher *eth.IPLDPublisherAndIndexer
+ fetcher *eth.IPLDFetcher
+ indexAndPublisher *eth.IPLDPublisher
backend *eth.Backend
api *eth.PublicEthAPI
)
@@ -94,15 +94,15 @@ var _ = Describe("API", func() {
db, err = shared.SetupDB()
Expect(err).ToNot(HaveOccurred())
retriever = eth.NewCIDRetriever(db)
- fetcher = eth.NewIPLDPGFetcher(db)
- indexAndPublisher = eth.NewIPLDPublisherAndIndexer(db)
+ fetcher = eth.NewIPLDFetcher(db)
+ indexAndPublisher = eth.NewIPLDPublisher(db)
backend = ð.Backend{
Retriever: retriever,
Fetcher: fetcher,
DB: db,
}
api = eth.NewPublicEthAPI(backend)
- _, err = indexAndPublisher.Publish(mocks.MockConvertedPayload)
+ err = indexAndPublisher.Publish(mocks.MockConvertedPayload)
Expect(err).ToNot(HaveOccurred())
uncles := mocks.MockBlock.Uncles()
uncleHashes := make([]common.Hash, len(uncles))
diff --git a/pkg/eth/backend.go b/pkg/eth/backend.go
index 81ad8382..49c20b06 100644
--- a/pkg/eth/backend.go
+++ b/pkg/eth/backend.go
@@ -39,7 +39,7 @@ var (
type Backend struct {
Retriever *CIDRetriever
- Fetcher *IPLDPGFetcher
+ Fetcher *IPLDFetcher
DB *postgres.DB
}
diff --git a/pkg/eth/cid_retriever_test.go b/pkg/eth/cid_retriever_test.go
index 668d763d..20e2abc0 100644
--- a/pkg/eth/cid_retriever_test.go
+++ b/pkg/eth/cid_retriever_test.go
@@ -211,14 +211,14 @@ var (
var _ = Describe("Retriever", func() {
var (
db *postgres.DB
- repo *eth2.IPLDPublisherAndIndexer
+ repo *eth2.IPLDPublisher
retriever *eth2.CIDRetriever
)
BeforeEach(func() {
var err error
db, err = shared.SetupDB()
Expect(err).ToNot(HaveOccurred())
- repo = eth2.NewIPLDPublisherAndIndexer(db)
+ repo = eth2.NewIPLDPublisher(db)
retriever = eth2.NewCIDRetriever(db)
})
AfterEach(func() {
@@ -227,7 +227,7 @@ var _ = Describe("Retriever", func() {
Describe("Retrieve", func() {
BeforeEach(func() {
- _, err := repo.Publish(mocks.MockConvertedPayload)
+ err := repo.Publish(mocks.MockConvertedPayload)
Expect(err).ToNot(HaveOccurred())
})
It("Retrieves all CIDs for the given blocknumber when provided an open filter", func() {
@@ -413,7 +413,7 @@ var _ = Describe("Retriever", func() {
Expect(err).To(HaveOccurred())
})
It("Gets the number of the first block that has data in the database", func() {
- _, err := repo.Publish(mocks.MockConvertedPayload)
+ err := repo.Publish(mocks.MockConvertedPayload)
Expect(err).ToNot(HaveOccurred())
num, err := retriever.RetrieveFirstBlockNumber()
Expect(err).ToNot(HaveOccurred())
@@ -423,7 +423,7 @@ var _ = Describe("Retriever", func() {
It("Gets the number of the first block that has data in the database", func() {
payload := mocks.MockConvertedPayload
payload.Block = newMockBlock(1010101)
- _, err := repo.Publish(payload)
+ err := repo.Publish(payload)
Expect(err).ToNot(HaveOccurred())
num, err := retriever.RetrieveFirstBlockNumber()
Expect(err).ToNot(HaveOccurred())
@@ -435,9 +435,9 @@ var _ = Describe("Retriever", func() {
payload1.Block = newMockBlock(1010101)
payload2 := payload1
payload2.Block = newMockBlock(5)
- _, err := repo.Publish(payload1)
+ err := repo.Publish(payload1)
Expect(err).ToNot(HaveOccurred())
- _, err = repo.Publish(payload2)
+ err = repo.Publish(payload2)
Expect(err).ToNot(HaveOccurred())
num, err := retriever.RetrieveFirstBlockNumber()
Expect(err).ToNot(HaveOccurred())
@@ -451,7 +451,7 @@ var _ = Describe("Retriever", func() {
Expect(err).To(HaveOccurred())
})
It("Gets the number of the latest block that has data in the database", func() {
- _, err := repo.Publish(mocks.MockConvertedPayload)
+ err := repo.Publish(mocks.MockConvertedPayload)
Expect(err).ToNot(HaveOccurred())
num, err := retriever.RetrieveLastBlockNumber()
Expect(err).ToNot(HaveOccurred())
@@ -461,7 +461,7 @@ var _ = Describe("Retriever", func() {
It("Gets the number of the latest block that has data in the database", func() {
payload := mocks.MockConvertedPayload
payload.Block = newMockBlock(1010101)
- _, err := repo.Publish(payload)
+ err := repo.Publish(payload)
Expect(err).ToNot(HaveOccurred())
num, err := retriever.RetrieveLastBlockNumber()
Expect(err).ToNot(HaveOccurred())
@@ -473,9 +473,9 @@ var _ = Describe("Retriever", func() {
payload1.Block = newMockBlock(1010101)
payload2 := payload1
payload2.Block = newMockBlock(5)
- _, err := repo.Publish(payload1)
+ err := repo.Publish(payload1)
Expect(err).ToNot(HaveOccurred())
- _, err = repo.Publish(payload2)
+ err = repo.Publish(payload2)
Expect(err).ToNot(HaveOccurred())
num, err := retriever.RetrieveLastBlockNumber()
Expect(err).ToNot(HaveOccurred())
@@ -492,13 +492,13 @@ var _ = Describe("Retriever", func() {
payload2.Block = newMockBlock(2)
payload3 := payload2
payload3.Block = newMockBlock(3)
- _, err := repo.Publish(payload0)
+ err := repo.Publish(payload0)
Expect(err).ToNot(HaveOccurred())
- _, err = repo.Publish(payload1)
+ err = repo.Publish(payload1)
Expect(err).ToNot(HaveOccurred())
- _, err = repo.Publish(payload2)
+ err = repo.Publish(payload2)
Expect(err).ToNot(HaveOccurred())
- _, err = repo.Publish(payload3)
+ err = repo.Publish(payload3)
Expect(err).ToNot(HaveOccurred())
gaps, err := retriever.RetrieveGapsInData(1)
Expect(err).ToNot(HaveOccurred())
@@ -508,7 +508,7 @@ var _ = Describe("Retriever", func() {
It("Returns the gap from 0 to the earliest block", func() {
payload := mocks.MockConvertedPayload
payload.Block = newMockBlock(5)
- _, err := repo.Publish(payload)
+ err := repo.Publish(payload)
Expect(err).ToNot(HaveOccurred())
gaps, err := retriever.RetrieveGapsInData(1)
Expect(err).ToNot(HaveOccurred())
@@ -523,11 +523,11 @@ var _ = Describe("Retriever", func() {
payload1 := mocks.MockConvertedPayload
payload3 := payload1
payload3.Block = newMockBlock(3)
- _, err := repo.Publish(payload0)
+ err := repo.Publish(payload0)
Expect(err).ToNot(HaveOccurred())
- _, err = repo.Publish(payload1)
+ err = repo.Publish(payload1)
Expect(err).ToNot(HaveOccurred())
- _, err = repo.Publish(payload3)
+ err = repo.Publish(payload3)
Expect(err).ToNot(HaveOccurred())
gaps, err := retriever.RetrieveGapsInData(1)
Expect(err).ToNot(HaveOccurred())
@@ -541,9 +541,9 @@ var _ = Describe("Retriever", func() {
payload1.Block = newMockBlock(1010101)
payload2 := payload1
payload2.Block = newMockBlock(0)
- _, err := repo.Publish(payload1)
+ err := repo.Publish(payload1)
Expect(err).ToNot(HaveOccurred())
- _, err = repo.Publish(payload2)
+ err = repo.Publish(payload2)
Expect(err).ToNot(HaveOccurred())
gaps, err := retriever.RetrieveGapsInData(1)
Expect(err).ToNot(HaveOccurred())
@@ -576,27 +576,27 @@ var _ = Describe("Retriever", func() {
payload11 := mocks.MockConvertedPayload
payload11.Block = newMockBlock(1000)
- _, err := repo.Publish(payload1)
+ err := repo.Publish(payload1)
Expect(err).ToNot(HaveOccurred())
- _, err = repo.Publish(payload2)
+ err = repo.Publish(payload2)
Expect(err).ToNot(HaveOccurred())
- _, err = repo.Publish(payload3)
+ err = repo.Publish(payload3)
Expect(err).ToNot(HaveOccurred())
- _, err = repo.Publish(payload4)
+ err = repo.Publish(payload4)
Expect(err).ToNot(HaveOccurred())
- _, err = repo.Publish(payload5)
+ err = repo.Publish(payload5)
Expect(err).ToNot(HaveOccurred())
- _, err = repo.Publish(payload6)
+ err = repo.Publish(payload6)
Expect(err).ToNot(HaveOccurred())
- _, err = repo.Publish(payload7)
+ err = repo.Publish(payload7)
Expect(err).ToNot(HaveOccurred())
- _, err = repo.Publish(payload8)
+ err = repo.Publish(payload8)
Expect(err).ToNot(HaveOccurred())
- _, err = repo.Publish(payload9)
+ err = repo.Publish(payload9)
Expect(err).ToNot(HaveOccurred())
- _, err = repo.Publish(payload10)
+ err = repo.Publish(payload10)
Expect(err).ToNot(HaveOccurred())
- _, err = repo.Publish(payload11)
+ err = repo.Publish(payload11)
Expect(err).ToNot(HaveOccurred())
gaps, err := retriever.RetrieveGapsInData(1)
@@ -640,33 +640,33 @@ var _ = Describe("Retriever", func() {
payload14 := mocks.MockConvertedPayload
payload14.Block = newMockBlock(1000)
- _, err := repo.Publish(payload1)
+ err := repo.Publish(payload1)
Expect(err).ToNot(HaveOccurred())
- _, err = repo.Publish(payload2)
+ err = repo.Publish(payload2)
Expect(err).ToNot(HaveOccurred())
- _, err = repo.Publish(payload3)
+ err = repo.Publish(payload3)
Expect(err).ToNot(HaveOccurred())
- _, err = repo.Publish(payload4)
+ err = repo.Publish(payload4)
Expect(err).ToNot(HaveOccurred())
- _, err = repo.Publish(payload5)
+ err = repo.Publish(payload5)
Expect(err).ToNot(HaveOccurred())
- _, err = repo.Publish(payload6)
+ err = repo.Publish(payload6)
Expect(err).ToNot(HaveOccurred())
- _, err = repo.Publish(payload7)
+ err = repo.Publish(payload7)
Expect(err).ToNot(HaveOccurred())
- _, err = repo.Publish(payload8)
+ err = repo.Publish(payload8)
Expect(err).ToNot(HaveOccurred())
- _, err = repo.Publish(payload9)
+ err = repo.Publish(payload9)
Expect(err).ToNot(HaveOccurred())
- _, err = repo.Publish(payload10)
+ err = repo.Publish(payload10)
Expect(err).ToNot(HaveOccurred())
- _, err = repo.Publish(payload11)
+ err = repo.Publish(payload11)
Expect(err).ToNot(HaveOccurred())
- _, err = repo.Publish(payload12)
+ err = repo.Publish(payload12)
Expect(err).ToNot(HaveOccurred())
- _, err = repo.Publish(payload13)
+ err = repo.Publish(payload13)
Expect(err).ToNot(HaveOccurred())
- _, err = repo.Publish(payload14)
+ err = repo.Publish(payload14)
Expect(err).ToNot(HaveOccurred())
cleaner := eth.NewCleaner(db)
diff --git a/pkg/eth/converter.go b/pkg/eth/converter.go
index 1d6d2d31..eb534773 100644
--- a/pkg/eth/converter.go
+++ b/pkg/eth/converter.go
@@ -71,11 +71,13 @@ func (pc *PayloadConverter) Convert(payload shared.RawChainData) (shared.Convert
if err != nil {
return nil, err
}
+
txMeta := TxModel{
Dst: shared.HandleZeroAddrPointer(trx.To()),
Src: shared.HandleZeroAddr(from),
TxHash: trx.Hash().String(),
Index: int64(i),
+ Data: trx.Data(),
}
// txMeta will have same index as its corresponding trx in the convertedPayload.BlockBody
convertedPayload.TxMetaData = append(convertedPayload.TxMetaData, txMeta)
@@ -90,7 +92,7 @@ func (pc *PayloadConverter) Convert(payload shared.RawChainData) (shared.Convert
if err := receipts.DeriveFields(pc.chainConfig, block.Hash(), block.NumberU64(), block.Transactions()); err != nil {
return nil, err
}
- for _, receipt := range receipts {
+ for i, receipt := range receipts {
// Extract topic and contract data from the receipt for indexing
topicSets := make([][]string, 4)
mappedContracts := make(map[string]bool) // use map to avoid duplicate addresses
@@ -109,6 +111,7 @@ func (pc *PayloadConverter) Convert(payload shared.RawChainData) (shared.Convert
contract := shared.HandleZeroAddr(receipt.ContractAddress)
var contractHash string
if contract != "" {
+ convertedPayload.TxMetaData[i].Deployment = true
contractHash = crypto.Keccak256Hash(common.HexToAddress(contract).Bytes()).String()
}
rctMeta := ReceiptModel{
diff --git a/pkg/eth/helpers.go b/pkg/eth/helpers.go
index 7d9021ba..73b1f0ca 100644
--- a/pkg/eth/helpers.go
+++ b/pkg/eth/helpers.go
@@ -16,7 +16,12 @@
package eth
-import "github.com/ethereum/go-ethereum/statediff"
+import (
+ "fmt"
+
+ "github.com/ethereum/go-ethereum/params"
+ "github.com/ethereum/go-ethereum/statediff"
+)
func ResolveFromNodeType(nodeType statediff.NodeType) int {
switch nodeType {
@@ -47,3 +52,19 @@ func ResolveToNodeType(nodeType int) statediff.NodeType {
return statediff.Unknown
}
}
+
+// ChainConfig returns the appropriate ethereum chain config for the provided chain id
+func ChainConfig(chainID uint64) (*params.ChainConfig, error) {
+ switch chainID {
+ case 1:
+ return params.MainnetChainConfig, nil
+ case 3:
+ return params.TestnetChainConfig, nil // Ropsten
+ case 4:
+ return params.RinkebyChainConfig, nil
+ case 5:
+ return params.GoerliChainConfig, nil
+ default:
+ return nil, fmt.Errorf("chain config for chainid %d not available", chainID)
+ }
+}
diff --git a/pkg/eth/indexer.go b/pkg/eth/indexer.go
index ff57d2d4..b48f9a3b 100644
--- a/pkg/eth/indexer.go
+++ b/pkg/eth/indexer.go
@@ -109,10 +109,10 @@ func (in *CIDIndexer) indexUncleCID(tx *sqlx.Tx, uncle UncleModel, headerID int6
func (in *CIDIndexer) indexTransactionAndReceiptCIDs(tx *sqlx.Tx, payload *CIDPayload, headerID int64) error {
for _, trxCidMeta := range payload.TransactionCIDs {
var txID int64
- err := tx.QueryRowx(`INSERT INTO eth.transaction_cids (header_id, tx_hash, cid, dst, src, index, mh_key) VALUES ($1, $2, $3, $4, $5, $6, $7)
- ON CONFLICT (header_id, tx_hash) DO UPDATE SET (cid, dst, src, index, mh_key) = ($3, $4, $5, $6, $7)
+ err := tx.QueryRowx(`INSERT INTO eth.transaction_cids (header_id, tx_hash, cid, dst, src, index, mh_key, data, deployment) VALUES ($1, $2, $3, $4, $5, $6, $7, $8, $9)
+ ON CONFLICT (header_id, tx_hash) DO UPDATE SET (cid, dst, src, index, mh_key, data, deployment) = ($3, $4, $5, $6, $7, $8, $9)
RETURNING id`,
- headerID, trxCidMeta.TxHash, trxCidMeta.CID, trxCidMeta.Dst, trxCidMeta.Src, trxCidMeta.Index, trxCidMeta.MhKey).Scan(&txID)
+ headerID, trxCidMeta.TxHash, trxCidMeta.CID, trxCidMeta.Dst, trxCidMeta.Src, trxCidMeta.Index, trxCidMeta.MhKey, trxCidMeta.Data, trxCidMeta.Deployment).Scan(&txID)
if err != nil {
return err
}
@@ -128,10 +128,10 @@ func (in *CIDIndexer) indexTransactionAndReceiptCIDs(tx *sqlx.Tx, payload *CIDPa
func (in *CIDIndexer) indexTransactionCID(tx *sqlx.Tx, transaction TxModel, headerID int64) (int64, error) {
var txID int64
- err := tx.QueryRowx(`INSERT INTO eth.transaction_cids (header_id, tx_hash, cid, dst, src, index, mh_key) VALUES ($1, $2, $3, $4, $5, $6, $7)
- ON CONFLICT (header_id, tx_hash) DO UPDATE SET (cid, dst, src, index, mh_key) = ($3, $4, $5, $6, $7)
+ err := tx.QueryRowx(`INSERT INTO eth.transaction_cids (header_id, tx_hash, cid, dst, src, index, mh_key, data, deployment) VALUES ($1, $2, $3, $4, $5, $6, $7, $8, $9)
+ ON CONFLICT (header_id, tx_hash) DO UPDATE SET (cid, dst, src, index, mh_key, data, deployment) = ($3, $4, $5, $6, $7, $8, $9)
RETURNING id`,
- headerID, transaction.TxHash, transaction.CID, transaction.Dst, transaction.Src, transaction.Index, transaction.MhKey).Scan(&txID)
+ headerID, transaction.TxHash, transaction.CID, transaction.Dst, transaction.Src, transaction.Index, transaction.MhKey, transaction.Data, transaction.Deployment).Scan(&txID)
return txID, err
}
diff --git a/pkg/eth/ipld_fetcher.go b/pkg/eth/ipld_fetcher.go
index e17e85b5..acea6197 100644
--- a/pkg/eth/ipld_fetcher.go
+++ b/pkg/eth/ipld_fetcher.go
@@ -17,39 +17,30 @@
package eth
import (
- "context"
"errors"
"fmt"
"math/big"
"github.com/ethereum/go-ethereum/common"
- "github.com/ipfs/go-block-format"
- "github.com/ipfs/go-blockservice"
- "github.com/ipfs/go-cid"
+ "github.com/jmoiron/sqlx"
log "github.com/sirupsen/logrus"
"github.com/vulcanize/ipfs-blockchain-watcher/pkg/ipfs"
+ "github.com/vulcanize/ipfs-blockchain-watcher/pkg/postgres"
"github.com/vulcanize/ipfs-blockchain-watcher/pkg/shared"
)
-var (
- errUnexpectedNumberOfIPLDs = errors.New("ipfs batch fetch returned unexpected number of IPLDs")
-)
-
// IPLDFetcher satisfies the IPLDFetcher interface for ethereum
+// It interfaces directly with PG-IPFS
type IPLDFetcher struct {
- BlockService blockservice.BlockService
+ db *postgres.DB
}
// NewIPLDFetcher creates a pointer to a new IPLDFetcher
-func NewIPLDFetcher(ipfsPath string) (*IPLDFetcher, error) {
- blockService, err := ipfs.InitIPFSBlockService(ipfsPath)
- if err != nil {
- return nil, err
- }
+func NewIPLDFetcher(db *postgres.DB) *IPLDFetcher {
return &IPLDFetcher{
- BlockService: blockService,
- }, nil
+ db: db,
+ }
}
// Fetch is the exported method for fetching and returning all the IPLDS specified in the CIDWrapper
@@ -59,161 +50,135 @@ func (f *IPLDFetcher) Fetch(cids shared.CIDsForFetching) (shared.IPLDs, error) {
return nil, fmt.Errorf("eth fetcher: expected cids type %T got %T", &CIDWrapper{}, cids)
}
log.Debug("fetching iplds")
- var err error
iplds := IPLDs{}
iplds.TotalDifficulty, ok = new(big.Int).SetString(cidWrapper.Header.TotalDifficulty, 10)
if !ok {
return nil, errors.New("eth fetcher: unable to set total difficulty")
}
iplds.BlockNumber = cidWrapper.BlockNumber
- iplds.Header, err = f.FetchHeader(cidWrapper.Header)
+
+ tx, err := f.db.Beginx()
if err != nil {
return nil, err
}
- iplds.Uncles, err = f.FetchUncles(cidWrapper.Uncles)
+ defer func() {
+ if p := recover(); p != nil {
+ shared.Rollback(tx)
+ panic(p)
+ } else if err != nil {
+ shared.Rollback(tx)
+ } else {
+ err = tx.Commit()
+ }
+ }()
+
+ iplds.Header, err = f.FetchHeader(tx, cidWrapper.Header)
if err != nil {
- return nil, err
+ return nil, fmt.Errorf("eth pg fetcher: header fetching error: %s", err.Error())
}
- iplds.Transactions, err = f.FetchTrxs(cidWrapper.Transactions)
+ iplds.Uncles, err = f.FetchUncles(tx, cidWrapper.Uncles)
if err != nil {
- return nil, err
+ return nil, fmt.Errorf("eth pg fetcher: uncle fetching error: %s", err.Error())
}
- iplds.Receipts, err = f.FetchRcts(cidWrapper.Receipts)
+ iplds.Transactions, err = f.FetchTrxs(tx, cidWrapper.Transactions)
if err != nil {
- return nil, err
+ return nil, fmt.Errorf("eth pg fetcher: transaction fetching error: %s", err.Error())
}
- iplds.StateNodes, err = f.FetchState(cidWrapper.StateNodes)
+ iplds.Receipts, err = f.FetchRcts(tx, cidWrapper.Receipts)
if err != nil {
- return nil, err
+ return nil, fmt.Errorf("eth pg fetcher: receipt fetching error: %s", err.Error())
}
- iplds.StorageNodes, err = f.FetchStorage(cidWrapper.StorageNodes)
+ iplds.StateNodes, err = f.FetchState(tx, cidWrapper.StateNodes)
if err != nil {
- return nil, err
+ return nil, fmt.Errorf("eth pg fetcher: state fetching error: %s", err.Error())
}
- return iplds, nil
+ iplds.StorageNodes, err = f.FetchStorage(tx, cidWrapper.StorageNodes)
+ if err != nil {
+ return nil, fmt.Errorf("eth pg fetcher: storage fetching error: %s", err.Error())
+ }
+ return iplds, err
}
// FetchHeaders fetches headers
-// It uses the f.fetch method
-func (f *IPLDFetcher) FetchHeader(c HeaderModel) (ipfs.BlockModel, error) {
+func (f *IPLDFetcher) FetchHeader(tx *sqlx.Tx, c HeaderModel) (ipfs.BlockModel, error) {
log.Debug("fetching header ipld")
- dc, err := cid.Decode(c.CID)
- if err != nil {
- return ipfs.BlockModel{}, err
- }
- header, err := f.fetch(dc)
+ headerBytes, err := shared.FetchIPLDByMhKey(tx, c.MhKey)
if err != nil {
return ipfs.BlockModel{}, err
}
return ipfs.BlockModel{
- Data: header.RawData(),
- CID: header.Cid().String(),
+ Data: headerBytes,
+ CID: c.CID,
}, nil
}
// FetchUncles fetches uncles
-// It uses the f.fetchBatch method
-func (f *IPLDFetcher) FetchUncles(cids []UncleModel) ([]ipfs.BlockModel, error) {
+func (f *IPLDFetcher) FetchUncles(tx *sqlx.Tx, cids []UncleModel) ([]ipfs.BlockModel, error) {
log.Debug("fetching uncle iplds")
- uncleCids := make([]cid.Cid, len(cids))
+ uncleIPLDs := make([]ipfs.BlockModel, len(cids))
for i, c := range cids {
- dc, err := cid.Decode(c.CID)
+ uncleBytes, err := shared.FetchIPLDByMhKey(tx, c.MhKey)
if err != nil {
return nil, err
}
- uncleCids[i] = dc
- }
- uncles := f.fetchBatch(uncleCids)
- uncleIPLDs := make([]ipfs.BlockModel, len(uncles))
- for i, uncle := range uncles {
uncleIPLDs[i] = ipfs.BlockModel{
- Data: uncle.RawData(),
- CID: uncle.Cid().String(),
+ Data: uncleBytes,
+ CID: c.CID,
}
}
- if len(uncleIPLDs) != len(uncleCids) {
- log.Errorf("ipfs fetcher: number of uncle blocks returned (%d) does not match number expected (%d)", len(uncles), len(uncleCids))
- return uncleIPLDs, errUnexpectedNumberOfIPLDs
- }
return uncleIPLDs, nil
}
// FetchTrxs fetches transactions
-// It uses the f.fetchBatch method
-func (f *IPLDFetcher) FetchTrxs(cids []TxModel) ([]ipfs.BlockModel, error) {
+func (f *IPLDFetcher) FetchTrxs(tx *sqlx.Tx, cids []TxModel) ([]ipfs.BlockModel, error) {
log.Debug("fetching transaction iplds")
- trxCids := make([]cid.Cid, len(cids))
+ trxIPLDs := make([]ipfs.BlockModel, len(cids))
for i, c := range cids {
- dc, err := cid.Decode(c.CID)
+ txBytes, err := shared.FetchIPLDByMhKey(tx, c.MhKey)
if err != nil {
return nil, err
}
- trxCids[i] = dc
- }
- trxs := f.fetchBatch(trxCids)
- trxIPLDs := make([]ipfs.BlockModel, len(trxs))
- for i, trx := range trxs {
trxIPLDs[i] = ipfs.BlockModel{
- Data: trx.RawData(),
- CID: trx.Cid().String(),
+ Data: txBytes,
+ CID: c.CID,
}
}
- if len(trxIPLDs) != len(trxCids) {
- log.Errorf("ipfs fetcher: number of transaction blocks returned (%d) does not match number expected (%d)", len(trxs), len(trxCids))
- return trxIPLDs, errUnexpectedNumberOfIPLDs
- }
return trxIPLDs, nil
}
// FetchRcts fetches receipts
-// It uses the f.fetchBatch method
-func (f *IPLDFetcher) FetchRcts(cids []ReceiptModel) ([]ipfs.BlockModel, error) {
+func (f *IPLDFetcher) FetchRcts(tx *sqlx.Tx, cids []ReceiptModel) ([]ipfs.BlockModel, error) {
log.Debug("fetching receipt iplds")
- rctCids := make([]cid.Cid, len(cids))
+ rctIPLDs := make([]ipfs.BlockModel, len(cids))
for i, c := range cids {
- dc, err := cid.Decode(c.CID)
+ rctBytes, err := shared.FetchIPLDByMhKey(tx, c.MhKey)
if err != nil {
return nil, err
}
- rctCids[i] = dc
- }
- rcts := f.fetchBatch(rctCids)
- rctIPLDs := make([]ipfs.BlockModel, len(rcts))
- for i, rct := range rcts {
rctIPLDs[i] = ipfs.BlockModel{
- Data: rct.RawData(),
- CID: rct.Cid().String(),
+ Data: rctBytes,
+ CID: c.CID,
}
}
- if len(rctIPLDs) != len(rctCids) {
- log.Errorf("ipfs fetcher: number of receipt blocks returned (%d) does not match number expected (%d)", len(rcts), len(rctCids))
- return rctIPLDs, errUnexpectedNumberOfIPLDs
- }
return rctIPLDs, nil
}
// FetchState fetches state nodes
-// It uses the single f.fetch method instead of the batch fetch, because it
-// needs to maintain the data's relation to state keys
-func (f *IPLDFetcher) FetchState(cids []StateNodeModel) ([]StateNode, error) {
+func (f *IPLDFetcher) FetchState(tx *sqlx.Tx, cids []StateNodeModel) ([]StateNode, error) {
log.Debug("fetching state iplds")
stateNodes := make([]StateNode, 0, len(cids))
for _, stateNode := range cids {
if stateNode.CID == "" {
continue
}
- dc, err := cid.Decode(stateNode.CID)
- if err != nil {
- return nil, err
- }
- state, err := f.fetch(dc)
+ stateBytes, err := shared.FetchIPLDByMhKey(tx, stateNode.MhKey)
if err != nil {
return nil, err
}
stateNodes = append(stateNodes, StateNode{
IPLD: ipfs.BlockModel{
- Data: state.RawData(),
- CID: state.Cid().String(),
+ Data: stateBytes,
+ CID: stateNode.CID,
},
StateLeafKey: common.HexToHash(stateNode.StateKey),
Type: ResolveToNodeType(stateNode.NodeType),
@@ -224,27 +189,21 @@ func (f *IPLDFetcher) FetchState(cids []StateNodeModel) ([]StateNode, error) {
}
// FetchStorage fetches storage nodes
-// It uses the single f.fetch method instead of the batch fetch, because it
-// needs to maintain the data's relation to state and storage keys
-func (f *IPLDFetcher) FetchStorage(cids []StorageNodeWithStateKeyModel) ([]StorageNode, error) {
+func (f *IPLDFetcher) FetchStorage(tx *sqlx.Tx, cids []StorageNodeWithStateKeyModel) ([]StorageNode, error) {
log.Debug("fetching storage iplds")
storageNodes := make([]StorageNode, 0, len(cids))
for _, storageNode := range cids {
if storageNode.CID == "" || storageNode.StateKey == "" {
continue
}
- dc, err := cid.Decode(storageNode.CID)
- if err != nil {
- return nil, err
- }
- storage, err := f.fetch(dc)
+ storageBytes, err := shared.FetchIPLDByMhKey(tx, storageNode.MhKey)
if err != nil {
return nil, err
}
storageNodes = append(storageNodes, StorageNode{
IPLD: ipfs.BlockModel{
- Data: storage.RawData(),
- CID: storage.Cid().String(),
+ Data: storageBytes,
+ CID: storageNode.CID,
},
StateLeafKey: common.HexToHash(storageNode.StateKey),
StorageLeafKey: common.HexToHash(storageNode.StorageKey),
@@ -254,20 +213,3 @@ func (f *IPLDFetcher) FetchStorage(cids []StorageNodeWithStateKeyModel) ([]Stora
}
return storageNodes, nil
}
-
-// fetch is used to fetch a single cid
-func (f *IPLDFetcher) fetch(cid cid.Cid) (blocks.Block, error) {
- return f.BlockService.GetBlock(context.Background(), cid)
-}
-
-// fetchBatch is used to fetch a batch of IPFS data blocks by cid
-// There is no guarantee all are fetched, and no error in such a case, so
-// downstream we will need to confirm which CIDs were fetched in the result set
-func (f *IPLDFetcher) fetchBatch(cids []cid.Cid) []blocks.Block {
- fetchedBlocks := make([]blocks.Block, 0, len(cids))
- blockChan := f.BlockService.GetBlocks(context.Background(), cids)
- for block := range blockChan {
- fetchedBlocks = append(fetchedBlocks, block)
- }
- return fetchedBlocks
-}
diff --git a/pkg/eth/ipld_fetcher_test.go b/pkg/eth/ipld_fetcher_test.go
index 3ee52a65..c1165d8c 100644
--- a/pkg/eth/ipld_fetcher_test.go
+++ b/pkg/eth/ipld_fetcher_test.go
@@ -17,139 +17,49 @@
package eth_test
import (
- "bytes"
- "math/big"
-
- "github.com/ethereum/go-ethereum/common"
- "github.com/ethereum/go-ethereum/statediff"
- "github.com/ipfs/go-block-format"
. "github.com/onsi/ginkgo"
. "github.com/onsi/gomega"
"github.com/vulcanize/ipfs-blockchain-watcher/pkg/eth"
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/ipfs"
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/ipfs/mocks"
+ "github.com/vulcanize/ipfs-blockchain-watcher/pkg/eth/mocks"
+ "github.com/vulcanize/ipfs-blockchain-watcher/pkg/postgres"
+ "github.com/vulcanize/ipfs-blockchain-watcher/pkg/shared"
)
var (
- mockHeaderData = []byte{0, 1, 2, 3, 4}
- mockUncleData = []byte{1, 2, 3, 4, 5}
- mockTrxData = []byte{2, 3, 4, 5, 6}
- mockReceiptData = []byte{3, 4, 5, 6, 7}
- mockStateData = []byte{4, 5, 6, 7, 8}
- mockStorageData = []byte{5, 6, 7, 8, 9}
- mockStorageData2 = []byte{6, 7, 8, 9, 1}
- mockHeaderBlock = blocks.NewBlock(mockHeaderData)
- mockUncleBlock = blocks.NewBlock(mockUncleData)
- mockTrxBlock = blocks.NewBlock(mockTrxData)
- mockReceiptBlock = blocks.NewBlock(mockReceiptData)
- mockStateBlock = blocks.NewBlock(mockStateData)
- mockStorageBlock1 = blocks.NewBlock(mockStorageData)
- mockStorageBlock2 = blocks.NewBlock(mockStorageData2)
- mockBlocks = []blocks.Block{mockHeaderBlock, mockUncleBlock, mockTrxBlock, mockReceiptBlock, mockStateBlock, mockStorageBlock1, mockStorageBlock2}
- mockBlockService *mocks.MockIPFSBlockService
- mockCIDWrapper = ð.CIDWrapper{
- BlockNumber: big.NewInt(9000),
- Header: eth.HeaderModel{
- TotalDifficulty: "1337",
- CID: mockHeaderBlock.Cid().String(),
- },
- Uncles: []eth.UncleModel{
- {
- CID: mockUncleBlock.Cid().String(),
- },
- },
- Transactions: []eth.TxModel{
- {
- CID: mockTrxBlock.Cid().String(),
- },
- },
- Receipts: []eth.ReceiptModel{
- {
- CID: mockReceiptBlock.Cid().String(),
- },
- },
- StateNodes: []eth.StateNodeModel{{
- CID: mockStateBlock.Cid().String(),
- NodeType: 2,
- StateKey: "0xc5d2460186f7233c927e7db2dcc703c0e500b653ca82273b7bfad8045d85a470",
- }},
- StorageNodes: []eth.StorageNodeWithStateKeyModel{{
- CID: mockStorageBlock1.Cid().String(),
- NodeType: 2,
- StateKey: "0xc5d2460186f7233c927e7db2dcc703c0e500b653ca82273b7bfad8045d85a470",
- StorageKey: "0000000000000000000000000000000000000000000000000000000000000001",
- },
- {
- CID: mockStorageBlock2.Cid().String(),
- NodeType: 2,
- StateKey: "0xc5d2460186f7233c927e7db2dcc703c0e500b653ca82273b7bfad8045d85a470",
- StorageKey: "0000000000000000000000000000000000000000000000000000000000000002",
- }},
- }
+ db *postgres.DB
+ pubAndIndexer *eth.IPLDPublisher
+ fetcher *eth.IPLDFetcher
)
var _ = Describe("IPLDFetcher", func() {
Describe("Fetch", func() {
BeforeEach(func() {
- mockBlockService = new(mocks.MockIPFSBlockService)
- err := mockBlockService.AddBlocks(mockBlocks)
+ var err error
+ db, err = shared.SetupDB()
Expect(err).ToNot(HaveOccurred())
- Expect(len(mockBlockService.Blocks)).To(Equal(7))
+ pubAndIndexer = eth.NewIPLDPublisher(db)
+ err = pubAndIndexer.Publish(mocks.MockConvertedPayload)
+ Expect(err).ToNot(HaveOccurred())
+ fetcher = eth.NewIPLDFetcher(db)
+ })
+ AfterEach(func() {
+ eth.TearDownDB(db)
})
It("Fetches and returns IPLDs for the CIDs provided in the CIDWrapper", func() {
- fetcher := new(eth.IPLDFetcher)
- fetcher.BlockService = mockBlockService
- i, err := fetcher.Fetch(mockCIDWrapper)
+ i, err := fetcher.Fetch(mocks.MockCIDWrapper)
Expect(err).ToNot(HaveOccurred())
iplds, ok := i.(eth.IPLDs)
Expect(ok).To(BeTrue())
- Expect(iplds.TotalDifficulty).To(Equal(big.NewInt(1337)))
- Expect(iplds.BlockNumber).To(Equal(mockCIDWrapper.BlockNumber))
- Expect(iplds.Header).To(Equal(ipfs.BlockModel{
- Data: mockHeaderBlock.RawData(),
- CID: mockHeaderBlock.Cid().String(),
- }))
- Expect(len(iplds.Uncles)).To(Equal(1))
- Expect(iplds.Uncles[0]).To(Equal(ipfs.BlockModel{
- Data: mockUncleBlock.RawData(),
- CID: mockUncleBlock.Cid().String(),
- }))
- Expect(len(iplds.Transactions)).To(Equal(1))
- Expect(iplds.Transactions[0]).To(Equal(ipfs.BlockModel{
- Data: mockTrxBlock.RawData(),
- CID: mockTrxBlock.Cid().String(),
- }))
- Expect(len(iplds.Receipts)).To(Equal(1))
- Expect(iplds.Receipts[0]).To(Equal(ipfs.BlockModel{
- Data: mockReceiptBlock.RawData(),
- CID: mockReceiptBlock.Cid().String(),
- }))
- Expect(len(iplds.StateNodes)).To(Equal(1))
- Expect(iplds.StateNodes[0].StateLeafKey).To(Equal(common.HexToHash("0xc5d2460186f7233c927e7db2dcc703c0e500b653ca82273b7bfad8045d85a470")))
- Expect(iplds.StateNodes[0].Type).To(Equal(statediff.Leaf))
- Expect(iplds.StateNodes[0].IPLD).To(Equal(ipfs.BlockModel{
- Data: mockStateBlock.RawData(),
- CID: mockStateBlock.Cid().String(),
- }))
- Expect(len(iplds.StorageNodes)).To(Equal(2))
- for _, storage := range iplds.StorageNodes {
- Expect(storage.Type).To(Equal(statediff.Leaf))
- Expect(storage.StateLeafKey).To(Equal(common.HexToHash("0xc5d2460186f7233c927e7db2dcc703c0e500b653ca82273b7bfad8045d85a470")))
- if bytes.Equal(storage.StorageLeafKey.Bytes(), common.HexToHash("0000000000000000000000000000000000000000000000000000000000000001").Bytes()) {
- Expect(storage.IPLD).To(Equal(ipfs.BlockModel{
- Data: mockStorageBlock1.RawData(),
- CID: mockStorageBlock1.Cid().String(),
- }))
- }
- if bytes.Equal(storage.StorageLeafKey.Bytes(), common.HexToHash("0000000000000000000000000000000000000000000000000000000000000002").Bytes()) {
- Expect(storage.IPLD).To(Equal(ipfs.BlockModel{
- Data: mockStorageBlock2.RawData(),
- CID: mockStorageBlock2.Cid().String(),
- }))
- }
- }
+ Expect(iplds.TotalDifficulty).To(Equal(mocks.MockConvertedPayload.TotalDifficulty))
+ Expect(iplds.BlockNumber).To(Equal(mocks.MockConvertedPayload.Block.Number()))
+ Expect(iplds.Header).To(Equal(mocks.MockIPLDs.Header))
+ Expect(len(iplds.Uncles)).To(Equal(0))
+ Expect(iplds.Transactions).To(Equal(mocks.MockIPLDs.Transactions))
+ Expect(iplds.Receipts).To(Equal(mocks.MockIPLDs.Receipts))
+ Expect(iplds.StateNodes).To(Equal(mocks.MockIPLDs.StateNodes))
+ Expect(iplds.StorageNodes).To(Equal(mocks.MockIPLDs.StorageNodes))
})
})
})
diff --git a/pkg/eth/ipld_pg_fetcher.go b/pkg/eth/ipld_pg_fetcher.go
deleted file mode 100644
index bbfe6b3d..00000000
--- a/pkg/eth/ipld_pg_fetcher.go
+++ /dev/null
@@ -1,215 +0,0 @@
-// 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 .
-
-package eth
-
-import (
- "errors"
- "fmt"
- "math/big"
-
- "github.com/ethereum/go-ethereum/common"
- "github.com/jmoiron/sqlx"
- log "github.com/sirupsen/logrus"
-
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/ipfs"
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/postgres"
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/shared"
-)
-
-// IPLDPGFetcher satisfies the IPLDFetcher interface for ethereum
-// It interfaces directly with PG-IPFS
-type IPLDPGFetcher struct {
- db *postgres.DB
-}
-
-// NewIPLDPGFetcher creates a pointer to a new IPLDPGFetcher
-func NewIPLDPGFetcher(db *postgres.DB) *IPLDPGFetcher {
- return &IPLDPGFetcher{
- db: db,
- }
-}
-
-// Fetch is the exported method for fetching and returning all the IPLDS specified in the CIDWrapper
-func (f *IPLDPGFetcher) Fetch(cids shared.CIDsForFetching) (shared.IPLDs, error) {
- cidWrapper, ok := cids.(*CIDWrapper)
- if !ok {
- return nil, fmt.Errorf("eth fetcher: expected cids type %T got %T", &CIDWrapper{}, cids)
- }
- log.Debug("fetching iplds")
- iplds := IPLDs{}
- iplds.TotalDifficulty, ok = new(big.Int).SetString(cidWrapper.Header.TotalDifficulty, 10)
- if !ok {
- return nil, errors.New("eth fetcher: unable to set total difficulty")
- }
- iplds.BlockNumber = cidWrapper.BlockNumber
-
- tx, err := f.db.Beginx()
- if err != nil {
- return nil, err
- }
- defer func() {
- if p := recover(); p != nil {
- shared.Rollback(tx)
- panic(p)
- } else if err != nil {
- shared.Rollback(tx)
- } else {
- err = tx.Commit()
- }
- }()
-
- iplds.Header, err = f.FetchHeader(tx, cidWrapper.Header)
- if err != nil {
- return nil, fmt.Errorf("eth pg fetcher: header fetching error: %s", err.Error())
- }
- iplds.Uncles, err = f.FetchUncles(tx, cidWrapper.Uncles)
- if err != nil {
- return nil, fmt.Errorf("eth pg fetcher: uncle fetching error: %s", err.Error())
- }
- iplds.Transactions, err = f.FetchTrxs(tx, cidWrapper.Transactions)
- if err != nil {
- return nil, fmt.Errorf("eth pg fetcher: transaction fetching error: %s", err.Error())
- }
- iplds.Receipts, err = f.FetchRcts(tx, cidWrapper.Receipts)
- if err != nil {
- return nil, fmt.Errorf("eth pg fetcher: receipt fetching error: %s", err.Error())
- }
- iplds.StateNodes, err = f.FetchState(tx, cidWrapper.StateNodes)
- if err != nil {
- return nil, fmt.Errorf("eth pg fetcher: state fetching error: %s", err.Error())
- }
- iplds.StorageNodes, err = f.FetchStorage(tx, cidWrapper.StorageNodes)
- if err != nil {
- return nil, fmt.Errorf("eth pg fetcher: storage fetching error: %s", err.Error())
- }
- return iplds, err
-}
-
-// FetchHeaders fetches headers
-func (f *IPLDPGFetcher) FetchHeader(tx *sqlx.Tx, c HeaderModel) (ipfs.BlockModel, error) {
- log.Debug("fetching header ipld")
- headerBytes, err := shared.FetchIPLDByMhKey(tx, c.MhKey)
- if err != nil {
- return ipfs.BlockModel{}, err
- }
- return ipfs.BlockModel{
- Data: headerBytes,
- CID: c.CID,
- }, nil
-}
-
-// FetchUncles fetches uncles
-func (f *IPLDPGFetcher) FetchUncles(tx *sqlx.Tx, cids []UncleModel) ([]ipfs.BlockModel, error) {
- log.Debug("fetching uncle iplds")
- uncleIPLDs := make([]ipfs.BlockModel, len(cids))
- for i, c := range cids {
- uncleBytes, err := shared.FetchIPLDByMhKey(tx, c.MhKey)
- if err != nil {
- return nil, err
- }
- uncleIPLDs[i] = ipfs.BlockModel{
- Data: uncleBytes,
- CID: c.CID,
- }
- }
- return uncleIPLDs, nil
-}
-
-// FetchTrxs fetches transactions
-func (f *IPLDPGFetcher) FetchTrxs(tx *sqlx.Tx, cids []TxModel) ([]ipfs.BlockModel, error) {
- log.Debug("fetching transaction iplds")
- trxIPLDs := make([]ipfs.BlockModel, len(cids))
- for i, c := range cids {
- txBytes, err := shared.FetchIPLDByMhKey(tx, c.MhKey)
- if err != nil {
- return nil, err
- }
- trxIPLDs[i] = ipfs.BlockModel{
- Data: txBytes,
- CID: c.CID,
- }
- }
- return trxIPLDs, nil
-}
-
-// FetchRcts fetches receipts
-func (f *IPLDPGFetcher) FetchRcts(tx *sqlx.Tx, cids []ReceiptModel) ([]ipfs.BlockModel, error) {
- log.Debug("fetching receipt iplds")
- rctIPLDs := make([]ipfs.BlockModel, len(cids))
- for i, c := range cids {
- rctBytes, err := shared.FetchIPLDByMhKey(tx, c.MhKey)
- if err != nil {
- return nil, err
- }
- rctIPLDs[i] = ipfs.BlockModel{
- Data: rctBytes,
- CID: c.CID,
- }
- }
- return rctIPLDs, nil
-}
-
-// FetchState fetches state nodes
-func (f *IPLDPGFetcher) FetchState(tx *sqlx.Tx, cids []StateNodeModel) ([]StateNode, error) {
- log.Debug("fetching state iplds")
- stateNodes := make([]StateNode, 0, len(cids))
- for _, stateNode := range cids {
- if stateNode.CID == "" {
- continue
- }
- stateBytes, err := shared.FetchIPLDByMhKey(tx, stateNode.MhKey)
- if err != nil {
- return nil, err
- }
- stateNodes = append(stateNodes, StateNode{
- IPLD: ipfs.BlockModel{
- Data: stateBytes,
- CID: stateNode.CID,
- },
- StateLeafKey: common.HexToHash(stateNode.StateKey),
- Type: ResolveToNodeType(stateNode.NodeType),
- Path: stateNode.Path,
- })
- }
- return stateNodes, nil
-}
-
-// FetchStorage fetches storage nodes
-func (f *IPLDPGFetcher) FetchStorage(tx *sqlx.Tx, cids []StorageNodeWithStateKeyModel) ([]StorageNode, error) {
- log.Debug("fetching storage iplds")
- storageNodes := make([]StorageNode, 0, len(cids))
- for _, storageNode := range cids {
- if storageNode.CID == "" || storageNode.StateKey == "" {
- continue
- }
- storageBytes, err := shared.FetchIPLDByMhKey(tx, storageNode.MhKey)
- if err != nil {
- return nil, err
- }
- storageNodes = append(storageNodes, StorageNode{
- IPLD: ipfs.BlockModel{
- Data: storageBytes,
- CID: storageNode.CID,
- },
- StateLeafKey: common.HexToHash(storageNode.StateKey),
- StorageLeafKey: common.HexToHash(storageNode.StorageKey),
- Type: ResolveToNodeType(storageNode.NodeType),
- Path: storageNode.Path,
- })
- }
- return storageNodes, nil
-}
diff --git a/pkg/eth/ipld_pg_fetcher_test.go b/pkg/eth/ipld_pg_fetcher_test.go
deleted file mode 100644
index a3ee7c78..00000000
--- a/pkg/eth/ipld_pg_fetcher_test.go
+++ /dev/null
@@ -1,65 +0,0 @@
-// 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 .
-
-package eth_test
-
-import (
- . "github.com/onsi/ginkgo"
- . "github.com/onsi/gomega"
-
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/eth"
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/eth/mocks"
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/postgres"
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/shared"
-)
-
-var (
- db *postgres.DB
- pubAndIndexer *eth.IPLDPublisherAndIndexer
- fetcher *eth.IPLDPGFetcher
-)
-
-var _ = Describe("IPLDPGFetcher", func() {
- Describe("Fetch", func() {
- BeforeEach(func() {
- var err error
- db, err = shared.SetupDB()
- Expect(err).ToNot(HaveOccurred())
- pubAndIndexer = eth.NewIPLDPublisherAndIndexer(db)
- _, err = pubAndIndexer.Publish(mocks.MockConvertedPayload)
- Expect(err).ToNot(HaveOccurred())
- fetcher = eth.NewIPLDPGFetcher(db)
- })
- AfterEach(func() {
- eth.TearDownDB(db)
- })
-
- It("Fetches and returns IPLDs for the CIDs provided in the CIDWrapper", func() {
- i, err := fetcher.Fetch(mocks.MockCIDWrapper)
- Expect(err).ToNot(HaveOccurred())
- iplds, ok := i.(eth.IPLDs)
- Expect(ok).To(BeTrue())
- Expect(iplds.TotalDifficulty).To(Equal(mocks.MockConvertedPayload.TotalDifficulty))
- Expect(iplds.BlockNumber).To(Equal(mocks.MockConvertedPayload.Block.Number()))
- Expect(iplds.Header).To(Equal(mocks.MockIPLDs.Header))
- Expect(len(iplds.Uncles)).To(Equal(0))
- Expect(iplds.Transactions).To(Equal(mocks.MockIPLDs.Transactions))
- Expect(iplds.Receipts).To(Equal(mocks.MockIPLDs.Receipts))
- Expect(iplds.StateNodes).To(Equal(mocks.MockIPLDs.StateNodes))
- Expect(iplds.StorageNodes).To(Equal(mocks.MockIPLDs.StorageNodes))
- })
- })
-})
diff --git a/pkg/eth/mocks/publisher.go b/pkg/eth/mocks/publisher.go
index a86f303e..c3e9a26a 100644
--- a/pkg/eth/mocks/publisher.go
+++ b/pkg/eth/mocks/publisher.go
@@ -32,13 +32,13 @@ type IPLDPublisher struct {
}
// Publish publishes an IPLDPayload to IPFS and returns the corresponding CIDPayload
-func (pub *IPLDPublisher) Publish(payload shared.ConvertedData) (shared.CIDsForIndexing, error) {
+func (pub *IPLDPublisher) Publish(payload shared.ConvertedData) error {
ipldPayload, ok := payload.(eth.ConvertedPayload)
if !ok {
- return nil, fmt.Errorf("publish expected payload type %T got %T", ð.ConvertedPayload{}, payload)
+ return fmt.Errorf("publish expected payload type %T got %T", ð.ConvertedPayload{}, payload)
}
pub.PassedIPLDPayload = ipldPayload
- return pub.ReturnCIDPayload, pub.ReturnErr
+ return pub.ReturnErr
}
// IterativeIPLDPublisher is the underlying struct for the Publisher interface; used in testing
@@ -50,16 +50,12 @@ type IterativeIPLDPublisher struct {
}
// Publish publishes an IPLDPayload to IPFS and returns the corresponding CIDPayload
-func (pub *IterativeIPLDPublisher) Publish(payload shared.ConvertedData) (shared.CIDsForIndexing, error) {
+func (pub *IterativeIPLDPublisher) Publish(payload shared.ConvertedData) error {
ipldPayload, ok := payload.(eth.ConvertedPayload)
if !ok {
- return nil, fmt.Errorf("publish expected payload type %T got %T", ð.ConvertedPayload{}, payload)
+ return fmt.Errorf("publish expected payload type %T got %T", ð.ConvertedPayload{}, payload)
}
pub.PassedIPLDPayload = append(pub.PassedIPLDPayload, ipldPayload)
- if len(pub.ReturnCIDPayload) < pub.iteration+1 {
- return nil, fmt.Errorf("IterativeIPLDPublisher does not have a payload to return at iteration %d", pub.iteration)
- }
- returnPayload := pub.ReturnCIDPayload[pub.iteration]
pub.iteration++
- return returnPayload, pub.ReturnErr
+ return pub.ReturnErr
}
diff --git a/pkg/eth/mocks/test_data.go b/pkg/eth/mocks/test_data.go
index e4cf42bb..ecd62b49 100644
--- a/pkg/eth/mocks/test_data.go
+++ b/pkg/eth/mocks/test_data.go
@@ -63,6 +63,7 @@ var (
AnotherAddress = common.HexToAddress("0xaE9BEa628c4Ce503DcFD7E305CaB4e29E7476593")
ContractAddress = crypto.CreateAddress(SenderAddr, MockTransactions[2].Nonce())
ContractHash = crypto.Keccak256Hash(ContractAddress.Bytes()).String()
+ MockContractByteCode = []byte{0, 1, 2, 3, 4, 5}
mockTopic11 = common.HexToHash("0x04")
mockTopic12 = common.HexToHash("0x06")
mockTopic21 = common.HexToHash("0x05")
@@ -99,28 +100,34 @@ var (
StorageMhKey = shared.MultihashKeyFromCID(StorageCID)
MockTrxMeta = []eth.TxModel{
{
- CID: "", // This is empty until we go to publish to ipfs
- MhKey: "",
- Src: SenderAddr.Hex(),
- Dst: Address.String(),
- Index: 0,
- TxHash: MockTransactions[0].Hash().String(),
+ CID: "", // This is empty until we go to publish to ipfs
+ MhKey: "",
+ Src: SenderAddr.Hex(),
+ Dst: Address.String(),
+ Index: 0,
+ TxHash: MockTransactions[0].Hash().String(),
+ Data: []byte{},
+ Deployment: false,
},
{
- CID: "",
- MhKey: "",
- Src: SenderAddr.Hex(),
- Dst: AnotherAddress.String(),
- Index: 1,
- TxHash: MockTransactions[1].Hash().String(),
+ CID: "",
+ MhKey: "",
+ Src: SenderAddr.Hex(),
+ Dst: AnotherAddress.String(),
+ Index: 1,
+ TxHash: MockTransactions[1].Hash().String(),
+ Data: []byte{},
+ Deployment: false,
},
{
- CID: "",
- MhKey: "",
- Src: SenderAddr.Hex(),
- Dst: "",
- Index: 2,
- TxHash: MockTransactions[2].Hash().String(),
+ CID: "",
+ MhKey: "",
+ Src: SenderAddr.Hex(),
+ Dst: "",
+ Index: 2,
+ TxHash: MockTransactions[2].Hash().String(),
+ Data: MockContractByteCode,
+ Deployment: true,
},
}
MockTrxMetaPostPublsh = []eth.TxModel{
@@ -532,7 +539,7 @@ func createTransactionsAndReceipts() (types.Transactions, types.Receipts, common
// make transactions
trx1 := types.NewTransaction(0, Address, big.NewInt(1000), 50, big.NewInt(100), []byte{})
trx2 := types.NewTransaction(1, AnotherAddress, big.NewInt(2000), 100, big.NewInt(200), []byte{})
- trx3 := types.NewContractCreation(2, big.NewInt(1500), 75, big.NewInt(150), []byte{0, 1, 2, 3, 4, 5})
+ trx3 := types.NewContractCreation(2, big.NewInt(1500), 75, big.NewInt(150), MockContractByteCode)
transactionSigner := types.MakeSigner(params.MainnetChainConfig, new(big.Int).Set(BlockNumber))
mockCurve := elliptic.P256()
mockPrvKey, err := ecdsa.GenerateKey(mockCurve, rand.Reader)
diff --git a/pkg/eth/models.go b/pkg/eth/models.go
index 6c44a801..e5375d6c 100644
--- a/pkg/eth/models.go
+++ b/pkg/eth/models.go
@@ -51,14 +51,16 @@ type UncleModel struct {
// TxModel is the db model for eth.transaction_cids
type TxModel struct {
- ID int64 `db:"id"`
- HeaderID int64 `db:"header_id"`
- Index int64 `db:"index"`
- TxHash string `db:"tx_hash"`
- CID string `db:"cid"`
- MhKey string `db:"mh_key"`
- Dst string `db:"dst"`
- Src string `db:"src"`
+ ID int64 `db:"id"`
+ HeaderID int64 `db:"header_id"`
+ Index int64 `db:"index"`
+ TxHash string `db:"tx_hash"`
+ CID string `db:"cid"`
+ MhKey string `db:"mh_key"`
+ Dst string `db:"dst"`
+ Src string `db:"src"`
+ Data []byte `db:"data"`
+ Deployment bool `db:"deployment"`
}
// ReceiptModel is the db model for eth.receipt_cids
diff --git a/pkg/eth/publish_and_indexer.go b/pkg/eth/publish_and_indexer.go
deleted file mode 100644
index eceaf59d..00000000
--- a/pkg/eth/publish_and_indexer.go
+++ /dev/null
@@ -1,228 +0,0 @@
-// 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 .
-
-package eth
-
-import (
- "fmt"
-
- "github.com/ethereum/go-ethereum/common"
- "github.com/ethereum/go-ethereum/core/state"
- "github.com/ethereum/go-ethereum/rlp"
- "github.com/ethereum/go-ethereum/statediff"
- "github.com/jmoiron/sqlx"
- "github.com/multiformats/go-multihash"
-
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/ipfs/ipld"
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/postgres"
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/shared"
-)
-
-// IPLDPublisherAndIndexer satisfies the IPLDPublisher interface for ethereum
-// It interfaces directly with the public.blocks table of PG-IPFS rather than going through an ipfs intermediary
-// It publishes and indexes IPLDs together in a single sqlx.Tx
-type IPLDPublisherAndIndexer struct {
- indexer *CIDIndexer
-}
-
-// NewIPLDPublisherAndIndexer creates a pointer to a new IPLDPublisherAndIndexer which satisfies the IPLDPublisher interface
-func NewIPLDPublisherAndIndexer(db *postgres.DB) *IPLDPublisherAndIndexer {
- return &IPLDPublisherAndIndexer{
- indexer: NewCIDIndexer(db),
- }
-}
-
-// Publish publishes an IPLDPayload to IPFS and returns the corresponding CIDPayload
-func (pub *IPLDPublisherAndIndexer) Publish(payload shared.ConvertedData) (shared.CIDsForIndexing, error) {
- ipldPayload, ok := payload.(ConvertedPayload)
- if !ok {
- return nil, fmt.Errorf("eth IPLDPublisherAndIndexer expected payload type %T got %T", ConvertedPayload{}, payload)
- }
- // Generate the iplds
- headerNode, uncleNodes, txNodes, txTrieNodes, rctNodes, rctTrieNodes, err := ipld.FromBlockAndReceipts(ipldPayload.Block, ipldPayload.Receipts)
- if err != nil {
- return nil, err
- }
-
- // Begin new db tx
- tx, err := pub.indexer.db.Beginx()
- if err != nil {
- return nil, err
- }
- defer func() {
- if p := recover(); p != nil {
- shared.Rollback(tx)
- panic(p)
- } else if err != nil {
- shared.Rollback(tx)
- } else {
- err = tx.Commit()
- }
- }()
-
- // Publish trie nodes
- for _, node := range txTrieNodes {
- if err := shared.PublishIPLD(tx, node); err != nil {
- return nil, err
- }
- }
- for _, node := range rctTrieNodes {
- if err := shared.PublishIPLD(tx, node); err != nil {
- return nil, err
- }
- }
-
- // Publish and index header
- if err := shared.PublishIPLD(tx, headerNode); err != nil {
- return nil, err
- }
- reward := CalcEthBlockReward(ipldPayload.Block.Header(), ipldPayload.Block.Uncles(), ipldPayload.Block.Transactions(), ipldPayload.Receipts)
- header := HeaderModel{
- CID: headerNode.Cid().String(),
- MhKey: shared.MultihashKeyFromCID(headerNode.Cid()),
- ParentHash: ipldPayload.Block.ParentHash().String(),
- BlockNumber: ipldPayload.Block.Number().String(),
- BlockHash: ipldPayload.Block.Hash().String(),
- TotalDifficulty: ipldPayload.TotalDifficulty.String(),
- Reward: reward.String(),
- Bloom: ipldPayload.Block.Bloom().Bytes(),
- StateRoot: ipldPayload.Block.Root().String(),
- RctRoot: ipldPayload.Block.ReceiptHash().String(),
- TxRoot: ipldPayload.Block.TxHash().String(),
- UncleRoot: ipldPayload.Block.UncleHash().String(),
- Timestamp: ipldPayload.Block.Time(),
- }
- headerID, err := pub.indexer.indexHeaderCID(tx, header)
- if err != nil {
- return nil, err
- }
-
- // Publish and index uncles
- for _, uncleNode := range uncleNodes {
- if err := shared.PublishIPLD(tx, uncleNode); err != nil {
- return nil, err
- }
- uncleReward := CalcUncleMinerReward(ipldPayload.Block.Number().Int64(), uncleNode.Number.Int64())
- uncle := UncleModel{
- CID: uncleNode.Cid().String(),
- MhKey: shared.MultihashKeyFromCID(uncleNode.Cid()),
- ParentHash: uncleNode.ParentHash.String(),
- BlockHash: uncleNode.Hash().String(),
- Reward: uncleReward.String(),
- }
- if err := pub.indexer.indexUncleCID(tx, uncle, headerID); err != nil {
- return nil, err
- }
- }
-
- // Publish and index txs and receipts
- for i, txNode := range txNodes {
- if err := shared.PublishIPLD(tx, txNode); err != nil {
- return nil, err
- }
- rctNode := rctNodes[i]
- if err := shared.PublishIPLD(tx, rctNode); err != nil {
- return nil, err
- }
- txModel := ipldPayload.TxMetaData[i]
- txModel.CID = txNode.Cid().String()
- txModel.MhKey = shared.MultihashKeyFromCID(txNode.Cid())
- txID, err := pub.indexer.indexTransactionCID(tx, txModel, headerID)
- if err != nil {
- return nil, err
- }
- rctModel := ipldPayload.ReceiptMetaData[i]
- rctModel.CID = rctNode.Cid().String()
- rctModel.MhKey = shared.MultihashKeyFromCID(rctNode.Cid())
- if err := pub.indexer.indexReceiptCID(tx, rctModel, txID); err != nil {
- return nil, err
- }
- }
-
- // Publish and index state and storage
- err = pub.publishAndIndexStateAndStorage(tx, ipldPayload, headerID)
-
- // This IPLDPublisher does both publishing and indexing, we do not need to pass anything forward to the indexer
- return nil, err // return err variable explicitly so that we return the err = tx.Commit() assignment in the defer
-}
-
-func (pub *IPLDPublisherAndIndexer) publishAndIndexStateAndStorage(tx *sqlx.Tx, ipldPayload ConvertedPayload, headerID int64) error {
- // Publish and index state and storage
- for _, stateNode := range ipldPayload.StateNodes {
- stateCIDStr, err := shared.PublishRaw(tx, ipld.MEthStateTrie, multihash.KECCAK_256, stateNode.Value)
- if err != nil {
- return err
- }
- mhKey, _ := shared.MultihashKeyFromCIDString(stateCIDStr)
- stateModel := StateNodeModel{
- Path: stateNode.Path,
- StateKey: stateNode.LeafKey.String(),
- CID: stateCIDStr,
- MhKey: mhKey,
- NodeType: ResolveFromNodeType(stateNode.Type),
- }
- stateID, err := pub.indexer.indexStateCID(tx, stateModel, headerID)
- if err != nil {
- return err
- }
- // If we have a leaf, decode and index the account data and any associated storage diffs
- if stateNode.Type == statediff.Leaf {
- var i []interface{}
- if err := rlp.DecodeBytes(stateNode.Value, &i); err != nil {
- return err
- }
- if len(i) != 2 {
- return fmt.Errorf("eth IPLDPublisherAndIndexer expected state leaf node rlp to decode into two elements")
- }
- var account state.Account
- if err := rlp.DecodeBytes(i[1].([]byte), &account); err != nil {
- return err
- }
- accountModel := StateAccountModel{
- Balance: account.Balance.String(),
- Nonce: account.Nonce,
- CodeHash: account.CodeHash,
- StorageRoot: account.Root.String(),
- }
- if err := pub.indexer.indexStateAccount(tx, accountModel, stateID); err != nil {
- return err
- }
- for _, storageNode := range ipldPayload.StorageNodes[common.Bytes2Hex(stateNode.Path)] {
- storageCIDStr, err := shared.PublishRaw(tx, ipld.MEthStorageTrie, multihash.KECCAK_256, storageNode.Value)
- if err != nil {
- return err
- }
- mhKey, _ := shared.MultihashKeyFromCIDString(storageCIDStr)
- storageModel := StorageNodeModel{
- Path: storageNode.Path,
- StorageKey: storageNode.LeafKey.Hex(),
- CID: storageCIDStr,
- MhKey: mhKey,
- NodeType: ResolveFromNodeType(storageNode.Type),
- }
- if err := pub.indexer.indexStorageCID(tx, storageModel, stateID); err != nil {
- return err
- }
- }
- }
- }
- return nil
-}
-
-// Index satisfies the shared.CIDIndexer interface
-func (pub *IPLDPublisherAndIndexer) Index(cids shared.CIDsForIndexing) error {
- return nil
-}
diff --git a/pkg/eth/publish_and_indexer_test.go b/pkg/eth/publish_and_indexer_test.go
deleted file mode 100644
index c90d10ae..00000000
--- a/pkg/eth/publish_and_indexer_test.go
+++ /dev/null
@@ -1,238 +0,0 @@
-// 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 .
-
-package eth_test
-
-import (
- "github.com/ethereum/go-ethereum/common"
- "github.com/ipfs/go-cid"
- "github.com/ipfs/go-ipfs-blockstore"
- "github.com/ipfs/go-ipfs-ds-help"
- . "github.com/onsi/ginkgo"
- . "github.com/onsi/gomega"
-
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/eth"
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/eth/mocks"
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/postgres"
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/shared"
-)
-
-var _ = Describe("PublishAndIndexer", func() {
- var (
- db *postgres.DB
- err error
- repo *eth.IPLDPublisherAndIndexer
- ipfsPgGet = `SELECT data FROM public.blocks
- WHERE key = $1`
- )
- BeforeEach(func() {
- db, err = shared.SetupDB()
- Expect(err).ToNot(HaveOccurred())
- repo = eth.NewIPLDPublisherAndIndexer(db)
- })
- AfterEach(func() {
- eth.TearDownDB(db)
- })
-
- Describe("Publish", func() {
- It("Published and indexes header IPLDs in a single tx", func() {
- emptyReturn, err := repo.Publish(mocks.MockConvertedPayload)
- Expect(emptyReturn).To(BeNil())
- Expect(err).ToNot(HaveOccurred())
- pgStr := `SELECT cid, td, reward, id
- FROM eth.header_cids
- WHERE block_number = $1`
- // check header was properly indexed
- type res struct {
- CID string
- TD string
- Reward string
- ID int
- }
- header := new(res)
- err = db.QueryRowx(pgStr, 1).StructScan(header)
- Expect(err).ToNot(HaveOccurred())
- Expect(header.CID).To(Equal(mocks.HeaderCID.String()))
- Expect(header.TD).To(Equal(mocks.MockBlock.Difficulty().String()))
- Expect(header.Reward).To(Equal("5000000000000000000"))
- dc, err := cid.Decode(header.CID)
- Expect(err).ToNot(HaveOccurred())
- mhKey := dshelp.MultihashToDsKey(dc.Hash())
- prefixedKey := blockstore.BlockPrefix.String() + mhKey.String()
- var data []byte
- err = db.Get(&data, ipfsPgGet, prefixedKey)
- Expect(err).ToNot(HaveOccurred())
- Expect(data).To(Equal(mocks.MockHeaderRlp))
- })
-
- It("Publishes and indexes transaction IPLDs in a single tx", func() {
- emptyReturn, err := repo.Publish(mocks.MockConvertedPayload)
- Expect(emptyReturn).To(BeNil())
- Expect(err).ToNot(HaveOccurred())
- // check that txs were properly indexed
- trxs := make([]string, 0)
- pgStr := `SELECT transaction_cids.cid FROM eth.transaction_cids INNER JOIN eth.header_cids ON (transaction_cids.header_id = header_cids.id)
- WHERE header_cids.block_number = $1`
- err = db.Select(&trxs, pgStr, 1)
- Expect(err).ToNot(HaveOccurred())
- Expect(len(trxs)).To(Equal(3))
- Expect(shared.ListContainsString(trxs, mocks.Trx1CID.String())).To(BeTrue())
- Expect(shared.ListContainsString(trxs, mocks.Trx2CID.String())).To(BeTrue())
- Expect(shared.ListContainsString(trxs, mocks.Trx3CID.String())).To(BeTrue())
- // and published
- for _, c := range trxs {
- dc, err := cid.Decode(c)
- Expect(err).ToNot(HaveOccurred())
- mhKey := dshelp.MultihashToDsKey(dc.Hash())
- prefixedKey := blockstore.BlockPrefix.String() + mhKey.String()
- var data []byte
- err = db.Get(&data, ipfsPgGet, prefixedKey)
- Expect(err).ToNot(HaveOccurred())
- switch c {
- case mocks.Trx1CID.String():
- Expect(data).To(Equal(mocks.MockTransactions.GetRlp(0)))
- case mocks.Trx2CID.String():
- Expect(data).To(Equal(mocks.MockTransactions.GetRlp(1)))
- case mocks.Trx3CID.String():
- Expect(data).To(Equal(mocks.MockTransactions.GetRlp(2)))
- }
- }
- })
-
- It("Publishes and indexes receipt IPLDs in a single tx", func() {
- emptyReturn, err := repo.Publish(mocks.MockConvertedPayload)
- Expect(emptyReturn).To(BeNil())
- Expect(err).ToNot(HaveOccurred())
- // check receipts were properly indexed
- rcts := make([]string, 0)
- pgStr := `SELECT receipt_cids.cid FROM eth.receipt_cids, eth.transaction_cids, eth.header_cids
- WHERE receipt_cids.tx_id = transaction_cids.id
- AND transaction_cids.header_id = header_cids.id
- AND header_cids.block_number = $1`
- err = db.Select(&rcts, pgStr, 1)
- Expect(err).ToNot(HaveOccurred())
- Expect(len(rcts)).To(Equal(3))
- Expect(shared.ListContainsString(rcts, mocks.Rct1CID.String())).To(BeTrue())
- Expect(shared.ListContainsString(rcts, mocks.Rct2CID.String())).To(BeTrue())
- Expect(shared.ListContainsString(rcts, mocks.Rct3CID.String())).To(BeTrue())
- // and published
- for _, c := range rcts {
- dc, err := cid.Decode(c)
- Expect(err).ToNot(HaveOccurred())
- mhKey := dshelp.MultihashToDsKey(dc.Hash())
- prefixedKey := blockstore.BlockPrefix.String() + mhKey.String()
- var data []byte
- err = db.Get(&data, ipfsPgGet, prefixedKey)
- Expect(err).ToNot(HaveOccurred())
- switch c {
- case mocks.Rct1CID.String():
- Expect(data).To(Equal(mocks.MockReceipts.GetRlp(0)))
- case mocks.Rct2CID.String():
- Expect(data).To(Equal(mocks.MockReceipts.GetRlp(1)))
- case mocks.Rct3CID.String():
- Expect(data).To(Equal(mocks.MockReceipts.GetRlp(2)))
- }
- }
- })
-
- It("Publishes and indexes state IPLDs in a single tx", func() {
- emptyReturn, err := repo.Publish(mocks.MockConvertedPayload)
- Expect(emptyReturn).To(BeNil())
- Expect(err).ToNot(HaveOccurred())
- // check that state nodes were properly indexed and published
- stateNodes := make([]eth.StateNodeModel, 0)
- pgStr := `SELECT state_cids.id, state_cids.cid, state_cids.state_leaf_key, state_cids.node_type, state_cids.state_path, state_cids.header_id
- FROM eth.state_cids INNER JOIN eth.header_cids ON (state_cids.header_id = header_cids.id)
- WHERE header_cids.block_number = $1`
- err = db.Select(&stateNodes, pgStr, 1)
- Expect(err).ToNot(HaveOccurred())
- Expect(len(stateNodes)).To(Equal(2))
- for _, stateNode := range stateNodes {
- var data []byte
- dc, err := cid.Decode(stateNode.CID)
- Expect(err).ToNot(HaveOccurred())
- mhKey := dshelp.MultihashToDsKey(dc.Hash())
- prefixedKey := blockstore.BlockPrefix.String() + mhKey.String()
- err = db.Get(&data, ipfsPgGet, prefixedKey)
- Expect(err).ToNot(HaveOccurred())
- pgStr = `SELECT * from eth.state_accounts WHERE state_id = $1`
- var account eth.StateAccountModel
- err = db.Get(&account, pgStr, stateNode.ID)
- Expect(err).ToNot(HaveOccurred())
- if stateNode.CID == mocks.State1CID.String() {
- Expect(stateNode.NodeType).To(Equal(2))
- Expect(stateNode.StateKey).To(Equal(common.BytesToHash(mocks.ContractLeafKey).Hex()))
- Expect(stateNode.Path).To(Equal([]byte{'\x06'}))
- Expect(data).To(Equal(mocks.ContractLeafNode))
- Expect(account).To(Equal(eth.StateAccountModel{
- ID: account.ID,
- StateID: stateNode.ID,
- Balance: "0",
- CodeHash: mocks.ContractCodeHash.Bytes(),
- StorageRoot: mocks.ContractRoot,
- Nonce: 1,
- }))
- }
- if stateNode.CID == mocks.State2CID.String() {
- Expect(stateNode.NodeType).To(Equal(2))
- Expect(stateNode.StateKey).To(Equal(common.BytesToHash(mocks.AccountLeafKey).Hex()))
- Expect(stateNode.Path).To(Equal([]byte{'\x0c'}))
- Expect(data).To(Equal(mocks.AccountLeafNode))
- Expect(account).To(Equal(eth.StateAccountModel{
- ID: account.ID,
- StateID: stateNode.ID,
- Balance: "1000",
- CodeHash: mocks.AccountCodeHash.Bytes(),
- StorageRoot: mocks.AccountRoot,
- Nonce: 0,
- }))
- }
- }
- pgStr = `SELECT * from eth.state_accounts WHERE state_id = $1`
- })
-
- It("Publishes and indexes storage IPLDs in a single tx", func() {
- emptyReturn, err := repo.Publish(mocks.MockConvertedPayload)
- Expect(emptyReturn).To(BeNil())
- Expect(err).ToNot(HaveOccurred())
- // check that storage nodes were properly indexed
- storageNodes := make([]eth.StorageNodeWithStateKeyModel, 0)
- pgStr := `SELECT storage_cids.cid, state_cids.state_leaf_key, storage_cids.storage_leaf_key, storage_cids.node_type, storage_cids.storage_path
- FROM eth.storage_cids, eth.state_cids, eth.header_cids
- WHERE storage_cids.state_id = state_cids.id
- AND state_cids.header_id = header_cids.id
- AND header_cids.block_number = $1`
- err = db.Select(&storageNodes, pgStr, 1)
- Expect(err).ToNot(HaveOccurred())
- Expect(len(storageNodes)).To(Equal(1))
- Expect(storageNodes[0]).To(Equal(eth.StorageNodeWithStateKeyModel{
- CID: mocks.StorageCID.String(),
- NodeType: 2,
- StorageKey: common.BytesToHash(mocks.StorageLeafKey).Hex(),
- StateKey: common.BytesToHash(mocks.ContractLeafKey).Hex(),
- Path: []byte{},
- }))
- var data []byte
- dc, err := cid.Decode(storageNodes[0].CID)
- Expect(err).ToNot(HaveOccurred())
- mhKey := dshelp.MultihashToDsKey(dc.Hash())
- prefixedKey := blockstore.BlockPrefix.String() + mhKey.String()
- err = db.Get(&data, ipfsPgGet, prefixedKey)
- Expect(err).ToNot(HaveOccurred())
- Expect(data).To(Equal(mocks.StorageLeafNode))
- })
- })
-})
diff --git a/pkg/eth/publisher.go b/pkg/eth/publisher.go
index 89169270..de8bdcc0 100644
--- a/pkg/eth/publisher.go
+++ b/pkg/eth/publisher.go
@@ -21,64 +21,77 @@ import (
"github.com/ethereum/go-ethereum/common"
"github.com/ethereum/go-ethereum/core/state"
- "github.com/ethereum/go-ethereum/core/types"
"github.com/ethereum/go-ethereum/rlp"
"github.com/ethereum/go-ethereum/statediff"
+ "github.com/jmoiron/sqlx"
+ "github.com/multiformats/go-multihash"
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/ipfs"
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/ipfs/dag_putters"
"github.com/vulcanize/ipfs-blockchain-watcher/pkg/ipfs/ipld"
+ "github.com/vulcanize/ipfs-blockchain-watcher/pkg/postgres"
"github.com/vulcanize/ipfs-blockchain-watcher/pkg/shared"
)
-// IPLDPublisher satisfies the IPLDPublisher for ethereum
+// IPLDPublisher satisfies the IPLDPublisher interface for ethereum
+// It interfaces directly with the public.blocks table of PG-IPFS rather than going through an ipfs intermediary
+// It publishes and indexes IPLDs together in a single sqlx.Tx
type IPLDPublisher struct {
- HeaderPutter ipfs.DagPutter
- TransactionPutter ipfs.DagPutter
- TransactionTriePutter ipfs.DagPutter
- ReceiptPutter ipfs.DagPutter
- ReceiptTriePutter ipfs.DagPutter
- StatePutter ipfs.DagPutter
- StoragePutter ipfs.DagPutter
+ indexer *CIDIndexer
}
// NewIPLDPublisher creates a pointer to a new IPLDPublisher which satisfies the IPLDPublisher interface
-func NewIPLDPublisher(ipfsPath string) (*IPLDPublisher, error) {
- node, err := ipfs.InitIPFSNode(ipfsPath)
- if err != nil {
- return nil, err
- }
+func NewIPLDPublisher(db *postgres.DB) *IPLDPublisher {
return &IPLDPublisher{
- HeaderPutter: dag_putters.NewEthBlockHeaderDagPutter(node),
- TransactionPutter: dag_putters.NewEthTxsDagPutter(node),
- TransactionTriePutter: dag_putters.NewEthTxTrieDagPutter(node),
- ReceiptPutter: dag_putters.NewEthReceiptDagPutter(node),
- ReceiptTriePutter: dag_putters.NewEthRctTrieDagPutter(node),
- StatePutter: dag_putters.NewEthStateDagPutter(node),
- StoragePutter: dag_putters.NewEthStorageDagPutter(node),
- }, nil
+ indexer: NewCIDIndexer(db),
+ }
}
// Publish publishes an IPLDPayload to IPFS and returns the corresponding CIDPayload
-func (pub *IPLDPublisher) Publish(payload shared.ConvertedData) (shared.CIDsForIndexing, error) {
+func (pub *IPLDPublisher) Publish(payload shared.ConvertedData) error {
ipldPayload, ok := payload.(ConvertedPayload)
if !ok {
- return nil, fmt.Errorf("eth publisher expected payload type %T got %T", ConvertedPayload{}, payload)
+ return fmt.Errorf("eth IPLDPublisher expected payload type %T got %T", ConvertedPayload{}, payload)
}
- // Generate the nodes for publishing
+ // Generate the iplds
headerNode, uncleNodes, txNodes, txTrieNodes, rctNodes, rctTrieNodes, err := ipld.FromBlockAndReceipts(ipldPayload.Block, ipldPayload.Receipts)
if err != nil {
- return nil, err
+ return err
}
- // Process and publish headers
- headerCid, err := pub.publishHeader(headerNode)
+ // Begin new db tx
+ tx, err := pub.indexer.db.Beginx()
if err != nil {
- return nil, err
+ return err
+ }
+ defer func() {
+ if p := recover(); p != nil {
+ shared.Rollback(tx)
+ panic(p)
+ } else if err != nil {
+ shared.Rollback(tx)
+ } else {
+ err = tx.Commit()
+ }
+ }()
+
+ // Publish trie nodes
+ for _, node := range txTrieNodes {
+ if err := shared.PublishIPLD(tx, node); err != nil {
+ return err
+ }
+ }
+ for _, node := range rctTrieNodes {
+ if err := shared.PublishIPLD(tx, node); err != nil {
+ return err
+ }
+ }
+
+ // Publish and index header
+ if err := shared.PublishIPLD(tx, headerNode); err != nil {
+ return err
}
reward := CalcEthBlockReward(ipldPayload.Block.Header(), ipldPayload.Block.Uncles(), ipldPayload.Block.Transactions(), ipldPayload.Receipts)
header := HeaderModel{
- CID: headerCid,
+ CID: headerNode.Cid().String(),
MhKey: shared.MultihashKeyFromCID(headerNode.Cid()),
ParentHash: ipldPayload.Block.ParentHash().String(),
BlockNumber: ipldPayload.Block.Number().String(),
@@ -92,189 +105,124 @@ func (pub *IPLDPublisher) Publish(payload shared.ConvertedData) (shared.CIDsForI
UncleRoot: ipldPayload.Block.UncleHash().String(),
Timestamp: ipldPayload.Block.Time(),
}
+ headerID, err := pub.indexer.indexHeaderCID(tx, header)
+ if err != nil {
+ return err
+ }
- // Process and publish uncles
- uncleCids := make([]UncleModel, len(uncleNodes))
- for i, uncle := range uncleNodes {
- uncleCid, err := pub.publishHeader(uncle)
- if err != nil {
- return nil, err
+ // Publish and index uncles
+ for _, uncleNode := range uncleNodes {
+ if err := shared.PublishIPLD(tx, uncleNode); err != nil {
+ return err
}
- uncleReward := CalcUncleMinerReward(ipldPayload.Block.Number().Int64(), uncle.Number.Int64())
- uncleCids[i] = UncleModel{
- CID: uncleCid,
- MhKey: shared.MultihashKeyFromCID(uncle.Cid()),
- ParentHash: uncle.ParentHash.String(),
- BlockHash: uncle.Hash().String(),
+ uncleReward := CalcUncleMinerReward(ipldPayload.Block.Number().Int64(), uncleNode.Number.Int64())
+ uncle := UncleModel{
+ CID: uncleNode.Cid().String(),
+ MhKey: shared.MultihashKeyFromCID(uncleNode.Cid()),
+ ParentHash: uncleNode.ParentHash.String(),
+ BlockHash: uncleNode.Hash().String(),
Reward: uncleReward.String(),
}
+ if err := pub.indexer.indexUncleCID(tx, uncle, headerID); err != nil {
+ return err
+ }
}
- // Process and publish transactions
- transactionCids, err := pub.publishTransactions(txNodes, txTrieNodes, ipldPayload.TxMetaData)
- if err != nil {
- return nil, err
- }
-
- // Process and publish receipts
- receiptsCids, err := pub.publishReceipts(rctNodes, rctTrieNodes, ipldPayload.ReceiptMetaData)
- if err != nil {
- return nil, err
- }
-
- // Process and publish state leafs
- stateNodeCids, stateAccounts, err := pub.publishStateNodes(ipldPayload.StateNodes)
- if err != nil {
- return nil, err
- }
-
- // Process and publish storage leafs
- storageNodeCids, err := pub.publishStorageNodes(ipldPayload.StorageNodes)
- if err != nil {
- return nil, err
- }
-
- // Package CIDs and their metadata into a single struct
- return &CIDPayload{
- HeaderCID: header,
- UncleCIDs: uncleCids,
- TransactionCIDs: transactionCids,
- ReceiptCIDs: receiptsCids,
- StateNodeCIDs: stateNodeCids,
- StorageNodeCIDs: storageNodeCids,
- StateAccounts: stateAccounts,
- }, nil
-}
-
-func (pub *IPLDPublisher) generateBlockNodes(body *types.Block, receipts types.Receipts) (*ipld.EthHeader,
- []*ipld.EthHeader, []*ipld.EthTx, []*ipld.EthTxTrie, []*ipld.EthReceipt, []*ipld.EthRctTrie, error) {
- return ipld.FromBlockAndReceipts(body, receipts)
-}
-
-func (pub *IPLDPublisher) publishHeader(header *ipld.EthHeader) (string, error) {
- return pub.HeaderPutter.DagPut(header)
-}
-
-func (pub *IPLDPublisher) publishTransactions(transactions []*ipld.EthTx, txTrie []*ipld.EthTxTrie, trxMeta []TxModel) ([]TxModel, error) {
- trxCids := make([]TxModel, len(transactions))
- for i, tx := range transactions {
- cid, err := pub.TransactionPutter.DagPut(tx)
+ // Publish and index txs and receipts
+ for i, txNode := range txNodes {
+ if err := shared.PublishIPLD(tx, txNode); err != nil {
+ return err
+ }
+ rctNode := rctNodes[i]
+ if err := shared.PublishIPLD(tx, rctNode); err != nil {
+ return err
+ }
+ txModel := ipldPayload.TxMetaData[i]
+ txModel.CID = txNode.Cid().String()
+ txModel.MhKey = shared.MultihashKeyFromCID(txNode.Cid())
+ txID, err := pub.indexer.indexTransactionCID(tx, txModel, headerID)
if err != nil {
- return nil, err
+ return err
}
- trxCids[i] = TxModel{
- CID: cid,
- MhKey: shared.MultihashKeyFromCID(tx.Cid()),
- Index: trxMeta[i].Index,
- TxHash: trxMeta[i].TxHash,
- Src: trxMeta[i].Src,
- Dst: trxMeta[i].Dst,
+ // If tx is a contract deployment, publish the data (code)
+ if txModel.Deployment {
+ if _, err = shared.PublishRaw(tx, ipld.MEthStorageTrie, multihash.KECCAK_256, txModel.Data); err != nil {
+ return err
+ }
+ }
+ rctModel := ipldPayload.ReceiptMetaData[i]
+ rctModel.CID = rctNode.Cid().String()
+ rctModel.MhKey = shared.MultihashKeyFromCID(rctNode.Cid())
+ if err := pub.indexer.indexReceiptCID(tx, rctModel, txID); err != nil {
+ return err
}
}
- for _, txNode := range txTrie {
- // We don't do anything with the tx trie cids atm
- if _, err := pub.TransactionTriePutter.DagPut(txNode); err != nil {
- return nil, err
- }
- }
- return trxCids, nil
+
+ // Publish and index state and storage
+ err = pub.publishAndIndexStateAndStorage(tx, ipldPayload, headerID)
+
+ return err // return err variable explicitly so that we return the err = tx.Commit() assignment in the defer
}
-func (pub *IPLDPublisher) publishReceipts(receipts []*ipld.EthReceipt, receiptTrie []*ipld.EthRctTrie, receiptMeta []ReceiptModel) (map[common.Hash]ReceiptModel, error) {
- rctCids := make(map[common.Hash]ReceiptModel)
- for i, rct := range receipts {
- cid, err := pub.ReceiptPutter.DagPut(rct)
+func (pub *IPLDPublisher) publishAndIndexStateAndStorage(tx *sqlx.Tx, ipldPayload ConvertedPayload, headerID int64) error {
+ // Publish and index state and storage
+ for _, stateNode := range ipldPayload.StateNodes {
+ stateCIDStr, err := shared.PublishRaw(tx, ipld.MEthStateTrie, multihash.KECCAK_256, stateNode.Value)
if err != nil {
- return nil, err
+ return err
}
- rctCids[rct.TxHash] = ReceiptModel{
- CID: cid,
- MhKey: shared.MultihashKeyFromCID(rct.Cid()),
- Contract: receiptMeta[i].Contract,
- ContractHash: receiptMeta[i].ContractHash,
- Topic0s: receiptMeta[i].Topic0s,
- Topic1s: receiptMeta[i].Topic1s,
- Topic2s: receiptMeta[i].Topic2s,
- Topic3s: receiptMeta[i].Topic3s,
- LogContracts: receiptMeta[i].LogContracts,
- }
- }
- for _, rctNode := range receiptTrie {
- // We don't do anything with the rct trie cids atm
- if _, err := pub.ReceiptTriePutter.DagPut(rctNode); err != nil {
- return nil, err
- }
- }
- return rctCids, nil
-}
-
-func (pub *IPLDPublisher) publishStateNodes(stateNodes []TrieNode) ([]StateNodeModel, map[string]StateAccountModel, error) {
- stateNodeCids := make([]StateNodeModel, 0, len(stateNodes))
- stateAccounts := make(map[string]StateAccountModel)
- for _, stateNode := range stateNodes {
- node, err := ipld.FromStateTrieRLP(stateNode.Value)
- if err != nil {
- return nil, nil, err
- }
- cid, err := pub.StatePutter.DagPut(node)
- if err != nil {
- return nil, nil, err
- }
- stateNodeCids = append(stateNodeCids, StateNodeModel{
+ mhKey, _ := shared.MultihashKeyFromCIDString(stateCIDStr)
+ stateModel := StateNodeModel{
Path: stateNode.Path,
StateKey: stateNode.LeafKey.String(),
- CID: cid,
- MhKey: shared.MultihashKeyFromCID(node.Cid()),
+ CID: stateCIDStr,
+ MhKey: mhKey,
NodeType: ResolveFromNodeType(stateNode.Type),
- })
- // If we have a leaf, decode the account to extract additional metadata for indexing
+ }
+ stateID, err := pub.indexer.indexStateCID(tx, stateModel, headerID)
+ if err != nil {
+ return err
+ }
+ // If we have a leaf, decode and index the account data and any associated storage diffs
if stateNode.Type == statediff.Leaf {
var i []interface{}
if err := rlp.DecodeBytes(stateNode.Value, &i); err != nil {
- return nil, nil, err
+ return err
}
if len(i) != 2 {
- return nil, nil, fmt.Errorf("IPLDPublisher expected state leaf node rlp to decode into two elements")
+ return fmt.Errorf("eth IPLDPublisher expected state leaf node rlp to decode into two elements")
}
var account state.Account
if err := rlp.DecodeBytes(i[1].([]byte), &account); err != nil {
- return nil, nil, err
+ return err
}
- // Map state account to the state path hash
- statePath := common.Bytes2Hex(stateNode.Path)
- stateAccounts[statePath] = StateAccountModel{
+ accountModel := StateAccountModel{
Balance: account.Balance.String(),
Nonce: account.Nonce,
CodeHash: account.CodeHash,
StorageRoot: account.Root.String(),
}
+ if err := pub.indexer.indexStateAccount(tx, accountModel, stateID); err != nil {
+ return err
+ }
+ for _, storageNode := range ipldPayload.StorageNodes[common.Bytes2Hex(stateNode.Path)] {
+ storageCIDStr, err := shared.PublishRaw(tx, ipld.MEthStorageTrie, multihash.KECCAK_256, storageNode.Value)
+ if err != nil {
+ return err
+ }
+ mhKey, _ := shared.MultihashKeyFromCIDString(storageCIDStr)
+ storageModel := StorageNodeModel{
+ Path: storageNode.Path,
+ StorageKey: storageNode.LeafKey.Hex(),
+ CID: storageCIDStr,
+ MhKey: mhKey,
+ NodeType: ResolveFromNodeType(storageNode.Type),
+ }
+ if err := pub.indexer.indexStorageCID(tx, storageModel, stateID); err != nil {
+ return err
+ }
+ }
}
}
- return stateNodeCids, stateAccounts, nil
-}
-
-func (pub *IPLDPublisher) publishStorageNodes(storageNodes map[string][]TrieNode) (map[string][]StorageNodeModel, error) {
- storageLeafCids := make(map[string][]StorageNodeModel)
- for path, storageTrie := range storageNodes {
- storageLeafCids[path] = make([]StorageNodeModel, 0, len(storageTrie))
- for _, storageNode := range storageTrie {
- node, err := ipld.FromStorageTrieRLP(storageNode.Value)
- if err != nil {
- return nil, err
- }
- cid, err := pub.StoragePutter.DagPut(node)
- if err != nil {
- return nil, err
- }
- // Map storage node cids to the state path hash
- storageLeafCids[path] = append(storageLeafCids[path], StorageNodeModel{
- Path: storageNode.Path,
- StorageKey: storageNode.LeafKey.Hex(),
- CID: cid,
- MhKey: shared.MultihashKeyFromCID(node.Cid()),
- NodeType: ResolveFromNodeType(storageNode.Type),
- })
- }
- }
- return storageLeafCids, nil
+ return nil
}
diff --git a/pkg/eth/publisher_test.go b/pkg/eth/publisher_test.go
index cd98954f..559e9b5a 100644
--- a/pkg/eth/publisher_test.go
+++ b/pkg/eth/publisher_test.go
@@ -18,88 +18,216 @@ package eth_test
import (
"github.com/ethereum/go-ethereum/common"
+ "github.com/ipfs/go-cid"
+ "github.com/ipfs/go-ipfs-blockstore"
+ "github.com/ipfs/go-ipfs-ds-help"
. "github.com/onsi/ginkgo"
. "github.com/onsi/gomega"
"github.com/vulcanize/ipfs-blockchain-watcher/pkg/eth"
"github.com/vulcanize/ipfs-blockchain-watcher/pkg/eth/mocks"
- mocks2 "github.com/vulcanize/ipfs-blockchain-watcher/pkg/ipfs/mocks"
+ "github.com/vulcanize/ipfs-blockchain-watcher/pkg/postgres"
+ "github.com/vulcanize/ipfs-blockchain-watcher/pkg/shared"
)
-var (
- mockHeaderDagPutter *mocks2.MappedDagPutter
- mockTrxDagPutter *mocks2.MappedDagPutter
- mockTrxTrieDagPutter *mocks2.DagPutter
- mockRctDagPutter *mocks2.MappedDagPutter
- mockRctTrieDagPutter *mocks2.DagPutter
- mockStateDagPutter *mocks2.MappedDagPutter
- mockStorageDagPutter *mocks2.MappedDagPutter
-)
-
-var _ = Describe("Publisher", func() {
+var _ = Describe("PublishAndIndexer", func() {
+ var (
+ db *postgres.DB
+ err error
+ repo *eth.IPLDPublisher
+ ipfsPgGet = `SELECT data FROM public.blocks
+ WHERE key = $1`
+ )
BeforeEach(func() {
- mockHeaderDagPutter = new(mocks2.MappedDagPutter)
- mockTrxDagPutter = new(mocks2.MappedDagPutter)
- mockTrxTrieDagPutter = new(mocks2.DagPutter)
- mockRctDagPutter = new(mocks2.MappedDagPutter)
- mockRctTrieDagPutter = new(mocks2.DagPutter)
- mockStateDagPutter = new(mocks2.MappedDagPutter)
- mockStorageDagPutter = new(mocks2.MappedDagPutter)
+ db, err = shared.SetupDB()
+ Expect(err).ToNot(HaveOccurred())
+ repo = eth.NewIPLDPublisher(db)
+ })
+ AfterEach(func() {
+ eth.TearDownDB(db)
})
Describe("Publish", func() {
- It("Publishes the passed IPLDPayload objects to IPFS and returns a CIDPayload for indexing", func() {
- mockHeaderDagPutter.CIDsToReturn = map[common.Hash]string{
- common.BytesToHash(mocks.HeaderIPLD.RawData()): mocks.HeaderCID.String(),
- }
- mockTrxDagPutter.CIDsToReturn = map[common.Hash]string{
- common.BytesToHash(mocks.Trx1IPLD.RawData()): mocks.Trx1CID.String(),
- common.BytesToHash(mocks.Trx2IPLD.RawData()): mocks.Trx2CID.String(),
- common.BytesToHash(mocks.Trx3IPLD.RawData()): mocks.Trx3CID.String(),
- }
- mockRctDagPutter.CIDsToReturn = map[common.Hash]string{
- common.BytesToHash(mocks.Rct1IPLD.RawData()): mocks.Rct1CID.String(),
- common.BytesToHash(mocks.Rct2IPLD.RawData()): mocks.Rct2CID.String(),
- common.BytesToHash(mocks.Rct3IPLD.RawData()): mocks.Rct3CID.String(),
- }
- mockStateDagPutter.CIDsToReturn = map[common.Hash]string{
- common.BytesToHash(mocks.State1IPLD.RawData()): mocks.State1CID.String(),
- common.BytesToHash(mocks.State2IPLD.RawData()): mocks.State2CID.String(),
- }
- mockStorageDagPutter.CIDsToReturn = map[common.Hash]string{
- common.BytesToHash(mocks.StorageIPLD.RawData()): mocks.StorageCID.String(),
- }
- publisher := eth.IPLDPublisher{
- HeaderPutter: mockHeaderDagPutter,
- TransactionPutter: mockTrxDagPutter,
- TransactionTriePutter: mockTrxTrieDagPutter,
- ReceiptPutter: mockRctDagPutter,
- ReceiptTriePutter: mockRctTrieDagPutter,
- StatePutter: mockStateDagPutter,
- StoragePutter: mockStorageDagPutter,
- }
- payload, err := publisher.Publish(mocks.MockConvertedPayload)
+ It("Published and indexes header IPLDs in a single tx", func() {
+ err = repo.Publish(mocks.MockConvertedPayload)
Expect(err).ToNot(HaveOccurred())
- cidPayload, ok := payload.(*eth.CIDPayload)
- Expect(ok).To(BeTrue())
- Expect(cidPayload.HeaderCID.TotalDifficulty).To(Equal(mocks.MockConvertedPayload.TotalDifficulty.String()))
- Expect(cidPayload.HeaderCID.BlockNumber).To(Equal(mocks.MockCIDPayload.HeaderCID.BlockNumber))
- Expect(cidPayload.HeaderCID.BlockHash).To(Equal(mocks.MockCIDPayload.HeaderCID.BlockHash))
- Expect(cidPayload.HeaderCID.Reward).To(Equal(mocks.MockCIDPayload.HeaderCID.Reward))
- Expect(cidPayload.UncleCIDs).To(Equal(mocks.MockCIDPayload.UncleCIDs))
- Expect(cidPayload.HeaderCID).To(Equal(mocks.MockCIDPayload.HeaderCID))
- Expect(len(cidPayload.TransactionCIDs)).To(Equal(3))
- Expect(cidPayload.TransactionCIDs[0]).To(Equal(mocks.MockCIDPayload.TransactionCIDs[0]))
- Expect(cidPayload.TransactionCIDs[1]).To(Equal(mocks.MockCIDPayload.TransactionCIDs[1]))
- Expect(cidPayload.TransactionCIDs[2]).To(Equal(mocks.MockCIDPayload.TransactionCIDs[2]))
- Expect(len(cidPayload.ReceiptCIDs)).To(Equal(3))
- Expect(cidPayload.ReceiptCIDs[mocks.MockTransactions[0].Hash()]).To(Equal(mocks.MockCIDPayload.ReceiptCIDs[mocks.MockTransactions[0].Hash()]))
- Expect(cidPayload.ReceiptCIDs[mocks.MockTransactions[1].Hash()]).To(Equal(mocks.MockCIDPayload.ReceiptCIDs[mocks.MockTransactions[1].Hash()]))
- Expect(cidPayload.ReceiptCIDs[mocks.MockTransactions[2].Hash()]).To(Equal(mocks.MockCIDPayload.ReceiptCIDs[mocks.MockTransactions[2].Hash()]))
- Expect(len(cidPayload.StateNodeCIDs)).To(Equal(2))
- Expect(cidPayload.StateNodeCIDs[0]).To(Equal(mocks.MockCIDPayload.StateNodeCIDs[0]))
- Expect(cidPayload.StateNodeCIDs[1]).To(Equal(mocks.MockCIDPayload.StateNodeCIDs[1]))
- Expect(cidPayload.StorageNodeCIDs).To(Equal(mocks.MockCIDPayload.StorageNodeCIDs))
+ pgStr := `SELECT cid, td, reward, id
+ FROM eth.header_cids
+ WHERE block_number = $1`
+ // check header was properly indexed
+ type res struct {
+ CID string
+ TD string
+ Reward string
+ ID int
+ }
+ header := new(res)
+ err = db.QueryRowx(pgStr, 1).StructScan(header)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(header.CID).To(Equal(mocks.HeaderCID.String()))
+ Expect(header.TD).To(Equal(mocks.MockBlock.Difficulty().String()))
+ Expect(header.Reward).To(Equal("5000000000000000000"))
+ dc, err := cid.Decode(header.CID)
+ Expect(err).ToNot(HaveOccurred())
+ mhKey := dshelp.MultihashToDsKey(dc.Hash())
+ prefixedKey := blockstore.BlockPrefix.String() + mhKey.String()
+ var data []byte
+ err = db.Get(&data, ipfsPgGet, prefixedKey)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(data).To(Equal(mocks.MockHeaderRlp))
+ })
+
+ It("Publishes and indexes transaction IPLDs in a single tx", func() {
+ err = repo.Publish(mocks.MockConvertedPayload)
+ Expect(err).ToNot(HaveOccurred())
+ // check that txs were properly indexed
+ trxs := make([]string, 0)
+ pgStr := `SELECT transaction_cids.cid FROM eth.transaction_cids INNER JOIN eth.header_cids ON (transaction_cids.header_id = header_cids.id)
+ WHERE header_cids.block_number = $1`
+ err = db.Select(&trxs, pgStr, 1)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(len(trxs)).To(Equal(3))
+ Expect(shared.ListContainsString(trxs, mocks.Trx1CID.String())).To(BeTrue())
+ Expect(shared.ListContainsString(trxs, mocks.Trx2CID.String())).To(BeTrue())
+ Expect(shared.ListContainsString(trxs, mocks.Trx3CID.String())).To(BeTrue())
+ // and published
+ for _, c := range trxs {
+ dc, err := cid.Decode(c)
+ Expect(err).ToNot(HaveOccurred())
+ mhKey := dshelp.MultihashToDsKey(dc.Hash())
+ prefixedKey := blockstore.BlockPrefix.String() + mhKey.String()
+ var data []byte
+ err = db.Get(&data, ipfsPgGet, prefixedKey)
+ Expect(err).ToNot(HaveOccurred())
+ switch c {
+ case mocks.Trx1CID.String():
+ Expect(data).To(Equal(mocks.MockTransactions.GetRlp(0)))
+ case mocks.Trx2CID.String():
+ Expect(data).To(Equal(mocks.MockTransactions.GetRlp(1)))
+ case mocks.Trx3CID.String():
+ Expect(data).To(Equal(mocks.MockTransactions.GetRlp(2)))
+ }
+ }
+ })
+
+ It("Publishes and indexes receipt IPLDs in a single tx", func() {
+ err = repo.Publish(mocks.MockConvertedPayload)
+ Expect(err).ToNot(HaveOccurred())
+ // check receipts were properly indexed
+ rcts := make([]string, 0)
+ pgStr := `SELECT receipt_cids.cid FROM eth.receipt_cids, eth.transaction_cids, eth.header_cids
+ WHERE receipt_cids.tx_id = transaction_cids.id
+ AND transaction_cids.header_id = header_cids.id
+ AND header_cids.block_number = $1`
+ err = db.Select(&rcts, pgStr, 1)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(len(rcts)).To(Equal(3))
+ Expect(shared.ListContainsString(rcts, mocks.Rct1CID.String())).To(BeTrue())
+ Expect(shared.ListContainsString(rcts, mocks.Rct2CID.String())).To(BeTrue())
+ Expect(shared.ListContainsString(rcts, mocks.Rct3CID.String())).To(BeTrue())
+ // and published
+ for _, c := range rcts {
+ dc, err := cid.Decode(c)
+ Expect(err).ToNot(HaveOccurred())
+ mhKey := dshelp.MultihashToDsKey(dc.Hash())
+ prefixedKey := blockstore.BlockPrefix.String() + mhKey.String()
+ var data []byte
+ err = db.Get(&data, ipfsPgGet, prefixedKey)
+ Expect(err).ToNot(HaveOccurred())
+ switch c {
+ case mocks.Rct1CID.String():
+ Expect(data).To(Equal(mocks.MockReceipts.GetRlp(0)))
+ case mocks.Rct2CID.String():
+ Expect(data).To(Equal(mocks.MockReceipts.GetRlp(1)))
+ case mocks.Rct3CID.String():
+ Expect(data).To(Equal(mocks.MockReceipts.GetRlp(2)))
+ }
+ }
+ })
+
+ It("Publishes and indexes state IPLDs in a single tx", func() {
+ err = repo.Publish(mocks.MockConvertedPayload)
+ Expect(err).ToNot(HaveOccurred())
+ // check that state nodes were properly indexed and published
+ stateNodes := make([]eth.StateNodeModel, 0)
+ pgStr := `SELECT state_cids.id, state_cids.cid, state_cids.state_leaf_key, state_cids.node_type, state_cids.state_path, state_cids.header_id
+ FROM eth.state_cids INNER JOIN eth.header_cids ON (state_cids.header_id = header_cids.id)
+ WHERE header_cids.block_number = $1`
+ err = db.Select(&stateNodes, pgStr, 1)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(len(stateNodes)).To(Equal(2))
+ for _, stateNode := range stateNodes {
+ var data []byte
+ dc, err := cid.Decode(stateNode.CID)
+ Expect(err).ToNot(HaveOccurred())
+ mhKey := dshelp.MultihashToDsKey(dc.Hash())
+ prefixedKey := blockstore.BlockPrefix.String() + mhKey.String()
+ err = db.Get(&data, ipfsPgGet, prefixedKey)
+ Expect(err).ToNot(HaveOccurred())
+ pgStr = `SELECT * from eth.state_accounts WHERE state_id = $1`
+ var account eth.StateAccountModel
+ err = db.Get(&account, pgStr, stateNode.ID)
+ Expect(err).ToNot(HaveOccurred())
+ if stateNode.CID == mocks.State1CID.String() {
+ Expect(stateNode.NodeType).To(Equal(2))
+ Expect(stateNode.StateKey).To(Equal(common.BytesToHash(mocks.ContractLeafKey).Hex()))
+ Expect(stateNode.Path).To(Equal([]byte{'\x06'}))
+ Expect(data).To(Equal(mocks.ContractLeafNode))
+ Expect(account).To(Equal(eth.StateAccountModel{
+ ID: account.ID,
+ StateID: stateNode.ID,
+ Balance: "0",
+ CodeHash: mocks.ContractCodeHash.Bytes(),
+ StorageRoot: mocks.ContractRoot,
+ Nonce: 1,
+ }))
+ }
+ if stateNode.CID == mocks.State2CID.String() {
+ Expect(stateNode.NodeType).To(Equal(2))
+ Expect(stateNode.StateKey).To(Equal(common.BytesToHash(mocks.AccountLeafKey).Hex()))
+ Expect(stateNode.Path).To(Equal([]byte{'\x0c'}))
+ Expect(data).To(Equal(mocks.AccountLeafNode))
+ Expect(account).To(Equal(eth.StateAccountModel{
+ ID: account.ID,
+ StateID: stateNode.ID,
+ Balance: "1000",
+ CodeHash: mocks.AccountCodeHash.Bytes(),
+ StorageRoot: mocks.AccountRoot,
+ Nonce: 0,
+ }))
+ }
+ }
+ pgStr = `SELECT * from eth.state_accounts WHERE state_id = $1`
+ })
+
+ It("Publishes and indexes storage IPLDs in a single tx", func() {
+ err = repo.Publish(mocks.MockConvertedPayload)
+ Expect(err).ToNot(HaveOccurred())
+ // check that storage nodes were properly indexed
+ storageNodes := make([]eth.StorageNodeWithStateKeyModel, 0)
+ pgStr := `SELECT storage_cids.cid, state_cids.state_leaf_key, storage_cids.storage_leaf_key, storage_cids.node_type, storage_cids.storage_path
+ FROM eth.storage_cids, eth.state_cids, eth.header_cids
+ WHERE storage_cids.state_id = state_cids.id
+ AND state_cids.header_id = header_cids.id
+ AND header_cids.block_number = $1`
+ err = db.Select(&storageNodes, pgStr, 1)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(len(storageNodes)).To(Equal(1))
+ Expect(storageNodes[0]).To(Equal(eth.StorageNodeWithStateKeyModel{
+ CID: mocks.StorageCID.String(),
+ NodeType: 2,
+ StorageKey: common.BytesToHash(mocks.StorageLeafKey).Hex(),
+ StateKey: common.BytesToHash(mocks.ContractLeafKey).Hex(),
+ Path: []byte{},
+ }))
+ var data []byte
+ dc, err := cid.Decode(storageNodes[0].CID)
+ Expect(err).ToNot(HaveOccurred())
+ mhKey := dshelp.MultihashToDsKey(dc.Hash())
+ prefixedKey := blockstore.BlockPrefix.String() + mhKey.String()
+ err = db.Get(&data, ipfsPgGet, prefixedKey)
+ Expect(err).ToNot(HaveOccurred())
+ Expect(data).To(Equal(mocks.StorageLeafNode))
})
})
})
diff --git a/pkg/historical/config.go b/pkg/historical/config.go
index bdb88481..c435f187 100644
--- a/pkg/historical/config.go
+++ b/pkg/historical/config.go
@@ -45,8 +45,6 @@ const (
// Config struct
type Config struct {
Chain shared.ChainType
- IPFSPath string
- IPFSMode shared.IPFSMode
DBConfig config.Database
DB *postgres.DB
@@ -71,19 +69,7 @@ func NewConfig() (*Config, error) {
return nil, err
}
- c.IPFSMode, err = shared.GetIPFSMode()
- if err != nil {
- return nil, err
- }
- if c.IPFSMode == shared.LocalInterface || c.IPFSMode == shared.RemoteClient {
- c.IPFSPath, err = shared.GetIPFSPath()
- if err != nil {
- return nil, err
- }
- }
-
c.DBConfig.Init()
-
if err := c.init(); err != nil {
return nil, err
}
diff --git a/pkg/historical/service.go b/pkg/historical/service.go
index f942c1dc..95173435 100644
--- a/pkg/historical/service.go
+++ b/pkg/historical/service.go
@@ -40,8 +40,6 @@ type BackFillService struct {
Converter shared.PayloadConverter
// Interface for publishing the IPLD payloads to IPFS
Publisher shared.IPLDPublisher
- // Interface for indexing the CIDs of the published IPLDs in Postgres
- Indexer shared.CIDIndexer
// Interface for searching and retrieving CIDs from Postgres index
Retriever shared.CIDRetriever
// Interface for fetching payloads over at historical blocks; over http
@@ -64,15 +62,11 @@ type BackFillService struct {
// NewBackFillService returns a new BackFillInterface
func NewBackFillService(settings *Config, screenAndServeChan chan shared.ConvertedData) (BackFillInterface, error) {
- publisher, err := builders.NewIPLDPublisher(settings.Chain, settings.IPFSPath, settings.DB, settings.IPFSMode)
+ publisher, err := builders.NewIPLDPublisher(settings.Chain, settings.DB)
if err != nil {
return nil, err
}
- indexer, err := builders.NewCIDIndexer(settings.Chain, settings.DB, settings.IPFSMode)
- if err != nil {
- return nil, err
- }
- converter, err := builders.NewPayloadConverter(settings.Chain)
+ converter, err := builders.NewPayloadConverter(settings.Chain, settings.NodeInfo.ChainID)
if err != nil {
return nil, err
}
@@ -93,7 +87,6 @@ func NewBackFillService(settings *Config, screenAndServeChan chan shared.Convert
batchNumber = shared.DefaultMaxBatchNumber
}
return &BackFillService{
- Indexer: indexer,
Converter: converter,
Publisher: publisher,
Retriever: retriever,
@@ -183,14 +176,10 @@ func (bfs *BackFillService) backFill(wg *sync.WaitGroup, id int, heightChan chan
default:
log.Debugf("%s backFill worker %d unable to forward converted payload to server; no channel ready to receive", bfs.chain.String(), id)
}
- cidPayload, err := bfs.Publisher.Publish(ipldPayload)
- if err != nil {
+ if err := bfs.Publisher.Publish(ipldPayload); err != nil {
log.Errorf("%s backFill worker %d publisher error: %s", bfs.chain.String(), id, err.Error())
continue
}
- if err := bfs.Indexer.Index(cidPayload); err != nil {
- log.Errorf("%s backFill worker %d indexer error: %s", bfs.chain.String(), id, err.Error())
- }
}
log.Infof("%s backFill worker %d finished section from %d to %d", bfs.chain.String(), id, heights[0], heights[len(heights)-1])
case <-bfs.QuitChan:
diff --git a/pkg/historical/service_test.go b/pkg/historical/service_test.go
index 2e3ef81b..51cf3e23 100644
--- a/pkg/historical/service_test.go
+++ b/pkg/historical/service_test.go
@@ -33,9 +33,6 @@ import (
var _ = Describe("BackFiller", func() {
Describe("FillGaps", func() {
It("Periodically checks for and fills in gaps in the watcher's data", func() {
- mockCidRepo := &mocks.CIDIndexer{
- ReturnErr: nil,
- }
mockPublisher := &mocks.IterativeIPLDPublisher{
ReturnCIDPayload: []*eth.CIDPayload{mocks.MockCIDPayload, mocks.MockCIDPayload},
ReturnErr: nil,
@@ -60,7 +57,6 @@ var _ = Describe("BackFiller", func() {
}
quitChan := make(chan bool, 1)
backfiller := &historical.BackFillService{
- Indexer: mockCidRepo,
Publisher: mockPublisher,
Converter: mockConverter,
Fetcher: mockFetcher,
@@ -74,9 +70,6 @@ var _ = Describe("BackFiller", func() {
backfiller.BackFill(wg)
time.Sleep(time.Second * 3)
quitChan <- true
- Expect(len(mockCidRepo.PassedCIDPayload)).To(Equal(2))
- Expect(mockCidRepo.PassedCIDPayload[0]).To(Equal(mocks.MockCIDPayload))
- Expect(mockCidRepo.PassedCIDPayload[1]).To(Equal(mocks.MockCIDPayload))
Expect(len(mockPublisher.PassedIPLDPayload)).To(Equal(2))
Expect(mockPublisher.PassedIPLDPayload[0]).To(Equal(mocks.MockConvertedPayload))
Expect(mockPublisher.PassedIPLDPayload[1]).To(Equal(mocks.MockConvertedPayload))
@@ -89,9 +82,6 @@ var _ = Describe("BackFiller", func() {
})
It("Works for single block `ranges`", func() {
- mockCidRepo := &mocks.CIDIndexer{
- ReturnErr: nil,
- }
mockPublisher := &mocks.IterativeIPLDPublisher{
ReturnCIDPayload: []*eth.CIDPayload{mocks.MockCIDPayload},
ReturnErr: nil,
@@ -115,7 +105,6 @@ var _ = Describe("BackFiller", func() {
}
quitChan := make(chan bool, 1)
backfiller := &historical.BackFillService{
- Indexer: mockCidRepo,
Publisher: mockPublisher,
Converter: mockConverter,
Fetcher: mockFetcher,
@@ -129,8 +118,6 @@ var _ = Describe("BackFiller", func() {
backfiller.BackFill(wg)
time.Sleep(time.Second * 3)
quitChan <- true
- Expect(len(mockCidRepo.PassedCIDPayload)).To(Equal(1))
- Expect(mockCidRepo.PassedCIDPayload[0]).To(Equal(mocks.MockCIDPayload))
Expect(len(mockPublisher.PassedIPLDPayload)).To(Equal(1))
Expect(mockPublisher.PassedIPLDPayload[0]).To(Equal(mocks.MockConvertedPayload))
Expect(len(mockConverter.PassedStatediffPayload)).To(Equal(1))
@@ -141,9 +128,6 @@ var _ = Describe("BackFiller", func() {
})
It("Finds beginning gap", func() {
- mockCidRepo := &mocks.CIDIndexer{
- ReturnErr: nil,
- }
mockPublisher := &mocks.IterativeIPLDPublisher{
ReturnCIDPayload: []*eth.CIDPayload{mocks.MockCIDPayload, mocks.MockCIDPayload},
ReturnErr: nil,
@@ -169,7 +153,6 @@ var _ = Describe("BackFiller", func() {
}
quitChan := make(chan bool, 1)
backfiller := &historical.BackFillService{
- Indexer: mockCidRepo,
Publisher: mockPublisher,
Converter: mockConverter,
Fetcher: mockFetcher,
@@ -183,9 +166,6 @@ var _ = Describe("BackFiller", func() {
backfiller.BackFill(wg)
time.Sleep(time.Second * 3)
quitChan <- true
- Expect(len(mockCidRepo.PassedCIDPayload)).To(Equal(2))
- Expect(mockCidRepo.PassedCIDPayload[0]).To(Equal(mocks.MockCIDPayload))
- Expect(mockCidRepo.PassedCIDPayload[1]).To(Equal(mocks.MockCIDPayload))
Expect(len(mockPublisher.PassedIPLDPayload)).To(Equal(2))
Expect(mockPublisher.PassedIPLDPayload[0]).To(Equal(mocks.MockConvertedPayload))
Expect(mockPublisher.PassedIPLDPayload[1]).To(Equal(mocks.MockConvertedPayload))
diff --git a/pkg/ipfs/builders.go b/pkg/ipfs/builders.go
deleted file mode 100644
index 317a9454..00000000
--- a/pkg/ipfs/builders.go
+++ /dev/null
@@ -1,90 +0,0 @@
-// 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 .
-
-package ipfs
-
-import (
- "context"
-
- "github.com/sirupsen/logrus"
-
- "github.com/ipfs/go-blockservice"
- "github.com/ipfs/go-ipfs/core"
- "github.com/ipfs/go-ipfs/plugin/loader"
- "github.com/ipfs/go-ipfs/repo/fsrepo"
- ipld "github.com/ipfs/go-ipld-format"
-)
-
-// InitIPFSPlugins is used to initialized IPFS plugins before creating a new IPFS node
-// This should only be called once
-func InitIPFSPlugins() error {
- logrus.Debug("initializing IPFS plugins")
- l, err := loader.NewPluginLoader("")
- if err != nil {
- return err
- }
- err = l.Initialize()
- if err != nil {
- return err
- }
- return l.Inject()
-}
-
-// InitIPFSBlockService is used to configure and return a BlockService using an ipfs repo path (e.g. ~/.ipfs)
-func InitIPFSBlockService(ipfsPath string) (blockservice.BlockService, error) {
- logrus.Debug("initializing IPFS block service interface")
- r, openErr := fsrepo.Open(ipfsPath)
- if openErr != nil {
- return nil, openErr
- }
- ctx := context.Background()
- cfg := &core.BuildCfg{
- Online: false,
- Repo: r,
- }
- ipfsNode, newNodeErr := core.NewNode(ctx, cfg)
- if newNodeErr != nil {
- return nil, newNodeErr
- }
- return ipfsNode.Blocks, nil
-}
-
-type IPFS struct {
- n *core.IpfsNode
- ctx context.Context
-}
-
-func (ipfs IPFS) Add(node ipld.Node) error {
- return ipfs.n.DAG.Add(ipfs.n.Context(), node)
-}
-
-func InitIPFSNode(repoPath string) (*IPFS, error) {
- logrus.Debug("initializing IPFS node interface")
- r, err := fsrepo.Open(repoPath)
- if err != nil {
- return nil, err
- }
- ctx := context.Background()
- cfg := &core.BuildCfg{
- Online: false,
- Repo: r,
- }
- ipfsNode, err := core.NewNode(ctx, cfg)
- if err != nil {
- return nil, err
- }
- return &IPFS{n: ipfsNode, ctx: ctx}, nil
-}
diff --git a/pkg/ipfs/dag_putters/btc_header.go b/pkg/ipfs/dag_putters/btc_header.go
deleted file mode 100644
index e1ff52a1..00000000
--- a/pkg/ipfs/dag_putters/btc_header.go
+++ /dev/null
@@ -1,50 +0,0 @@
-// 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 .
-
-package dag_putters
-
-import (
- "fmt"
- "strings"
-
- node "github.com/ipfs/go-ipld-format"
-
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/ipfs"
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/ipfs/ipld"
-)
-
-var (
- duplicateKeyErrorString = "pq: duplicate key value violates unique constraint"
-)
-
-type BtcHeaderDagPutter struct {
- adder *ipfs.IPFS
-}
-
-func NewBtcHeaderDagPutter(adder *ipfs.IPFS) *BtcHeaderDagPutter {
- return &BtcHeaderDagPutter{adder: adder}
-}
-
-func (bhdp *BtcHeaderDagPutter) DagPut(n node.Node) (string, error) {
- header, ok := n.(*ipld.BtcHeader)
- if !ok {
- return "", fmt.Errorf("BtcHeaderDagPutter expected input type %T got %T", &ipld.BtcHeader{}, n)
- }
- if err := bhdp.adder.Add(header); err != nil && !strings.Contains(err.Error(), duplicateKeyErrorString) {
- return "", err
- }
- return header.Cid().String(), nil
-}
diff --git a/pkg/ipfs/dag_putters/btc_tx.go b/pkg/ipfs/dag_putters/btc_tx.go
deleted file mode 100644
index 97d23fca..00000000
--- a/pkg/ipfs/dag_putters/btc_tx.go
+++ /dev/null
@@ -1,46 +0,0 @@
-// 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 .
-
-package dag_putters
-
-import (
- "fmt"
- "strings"
-
- node "github.com/ipfs/go-ipld-format"
-
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/ipfs"
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/ipfs/ipld"
-)
-
-type BtcTxDagPutter struct {
- adder *ipfs.IPFS
-}
-
-func NewBtcTxDagPutter(adder *ipfs.IPFS) *BtcTxDagPutter {
- return &BtcTxDagPutter{adder: adder}
-}
-
-func (etdp *BtcTxDagPutter) DagPut(n node.Node) (string, error) {
- transaction, ok := n.(*ipld.BtcTx)
- if !ok {
- return "", fmt.Errorf("BtcTxDagPutter expected input type %T got %T", &ipld.BtcTx{}, n)
- }
- if err := etdp.adder.Add(transaction); err != nil && !strings.Contains(err.Error(), duplicateKeyErrorString) {
- return "", err
- }
- return transaction.Cid().String(), nil
-}
diff --git a/pkg/ipfs/dag_putters/btc_tx_trie.go b/pkg/ipfs/dag_putters/btc_tx_trie.go
deleted file mode 100644
index 7d4623d5..00000000
--- a/pkg/ipfs/dag_putters/btc_tx_trie.go
+++ /dev/null
@@ -1,46 +0,0 @@
-// 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 .
-
-package dag_putters
-
-import (
- "fmt"
- "strings"
-
- node "github.com/ipfs/go-ipld-format"
-
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/ipfs"
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/ipfs/ipld"
-)
-
-type BtcTxTrieDagPutter struct {
- adder *ipfs.IPFS
-}
-
-func NewBtcTxTrieDagPutter(adder *ipfs.IPFS) *BtcTxTrieDagPutter {
- return &BtcTxTrieDagPutter{adder: adder}
-}
-
-func (etdp *BtcTxTrieDagPutter) DagPut(n node.Node) (string, error) {
- txTrieNode, ok := n.(*ipld.BtcTxTrie)
- if !ok {
- return "", fmt.Errorf("BtcTxTrieDagPutter expected input type %T got %T", &ipld.BtcTxTrie{}, n)
- }
- if err := etdp.adder.Add(txTrieNode); err != nil && !strings.Contains(err.Error(), duplicateKeyErrorString) {
- return "", err
- }
- return txTrieNode.Cid().String(), nil
-}
diff --git a/pkg/ipfs/dag_putters/eth_header.go b/pkg/ipfs/dag_putters/eth_header.go
deleted file mode 100644
index fd368f7c..00000000
--- a/pkg/ipfs/dag_putters/eth_header.go
+++ /dev/null
@@ -1,46 +0,0 @@
-// 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 .
-
-package dag_putters
-
-import (
- "fmt"
- "strings"
-
- node "github.com/ipfs/go-ipld-format"
-
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/ipfs"
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/ipfs/ipld"
-)
-
-type EthHeaderDagPutter struct {
- adder *ipfs.IPFS
-}
-
-func NewEthBlockHeaderDagPutter(adder *ipfs.IPFS) *EthHeaderDagPutter {
- return &EthHeaderDagPutter{adder: adder}
-}
-
-func (bhdp *EthHeaderDagPutter) DagPut(n node.Node) (string, error) {
- header, ok := n.(*ipld.EthHeader)
- if !ok {
- return "", fmt.Errorf("EthHeaderDagPutter expected input type %T got %T", &ipld.EthHeader{}, n)
- }
- if err := bhdp.adder.Add(header); err != nil && !strings.Contains(err.Error(), duplicateKeyErrorString) {
- return "", err
- }
- return header.Cid().String(), nil
-}
diff --git a/pkg/ipfs/dag_putters/eth_receipt.go b/pkg/ipfs/dag_putters/eth_receipt.go
deleted file mode 100644
index 35ecb279..00000000
--- a/pkg/ipfs/dag_putters/eth_receipt.go
+++ /dev/null
@@ -1,46 +0,0 @@
-// 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 .
-
-package dag_putters
-
-import (
- "fmt"
- "strings"
-
- node "github.com/ipfs/go-ipld-format"
-
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/ipfs"
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/ipfs/ipld"
-)
-
-type EthReceiptDagPutter struct {
- adder *ipfs.IPFS
-}
-
-func NewEthReceiptDagPutter(adder *ipfs.IPFS) *EthReceiptDagPutter {
- return &EthReceiptDagPutter{adder: adder}
-}
-
-func (erdp *EthReceiptDagPutter) DagPut(n node.Node) (string, error) {
- receipt, ok := n.(*ipld.EthReceipt)
- if !ok {
- return "", fmt.Errorf("EthReceiptDagPutter expected input type %T got type %T", &ipld.EthReceipt{}, n)
- }
- if err := erdp.adder.Add(receipt); err != nil && !strings.Contains(err.Error(), duplicateKeyErrorString) {
- return "", err
- }
- return receipt.Cid().String(), nil
-}
diff --git a/pkg/ipfs/dag_putters/eth_receipt_trie.go b/pkg/ipfs/dag_putters/eth_receipt_trie.go
deleted file mode 100644
index 531f5303..00000000
--- a/pkg/ipfs/dag_putters/eth_receipt_trie.go
+++ /dev/null
@@ -1,46 +0,0 @@
-// 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 .
-
-package dag_putters
-
-import (
- "fmt"
- "strings"
-
- node "github.com/ipfs/go-ipld-format"
-
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/ipfs"
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/ipfs/ipld"
-)
-
-type EthRctTrieDagPutter struct {
- adder *ipfs.IPFS
-}
-
-func NewEthRctTrieDagPutter(adder *ipfs.IPFS) *EthRctTrieDagPutter {
- return &EthRctTrieDagPutter{adder: adder}
-}
-
-func (etdp *EthRctTrieDagPutter) DagPut(n node.Node) (string, error) {
- rctTrieNode, ok := n.(*ipld.EthRctTrie)
- if !ok {
- return "", fmt.Errorf("EthRctTrieDagPutter expected input type %T got %T", &ipld.EthRctTrie{}, n)
- }
- if err := etdp.adder.Add(rctTrieNode); err != nil && !strings.Contains(err.Error(), duplicateKeyErrorString) {
- return "", err
- }
- return rctTrieNode.Cid().String(), nil
-}
diff --git a/pkg/ipfs/dag_putters/eth_state.go b/pkg/ipfs/dag_putters/eth_state.go
deleted file mode 100644
index 50d39092..00000000
--- a/pkg/ipfs/dag_putters/eth_state.go
+++ /dev/null
@@ -1,46 +0,0 @@
-// 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 .
-
-package dag_putters
-
-import (
- "fmt"
- "strings"
-
- node "github.com/ipfs/go-ipld-format"
-
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/ipfs"
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/ipfs/ipld"
-)
-
-type EthStateDagPutter struct {
- adder *ipfs.IPFS
-}
-
-func NewEthStateDagPutter(adder *ipfs.IPFS) *EthStateDagPutter {
- return &EthStateDagPutter{adder: adder}
-}
-
-func (erdp *EthStateDagPutter) DagPut(n node.Node) (string, error) {
- stateNode, ok := n.(*ipld.EthStateTrie)
- if !ok {
- return "", fmt.Errorf("EthStateDagPutter expected input type %T got %T", &ipld.EthStateTrie{}, n)
- }
- if err := erdp.adder.Add(stateNode); err != nil && !strings.Contains(err.Error(), duplicateKeyErrorString) {
- return "", err
- }
- return stateNode.Cid().String(), nil
-}
diff --git a/pkg/ipfs/dag_putters/eth_storage.go b/pkg/ipfs/dag_putters/eth_storage.go
deleted file mode 100644
index 1f6b8acd..00000000
--- a/pkg/ipfs/dag_putters/eth_storage.go
+++ /dev/null
@@ -1,46 +0,0 @@
-// 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 .
-
-package dag_putters
-
-import (
- "fmt"
- "strings"
-
- node "github.com/ipfs/go-ipld-format"
-
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/ipfs"
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/ipfs/ipld"
-)
-
-type EthStorageDagPutter struct {
- adder *ipfs.IPFS
-}
-
-func NewEthStorageDagPutter(adder *ipfs.IPFS) *EthStorageDagPutter {
- return &EthStorageDagPutter{adder: adder}
-}
-
-func (erdp *EthStorageDagPutter) DagPut(n node.Node) (string, error) {
- storageNode, ok := n.(*ipld.EthStorageTrie)
- if !ok {
- return "", fmt.Errorf("EthStorageDagPutter expected input type %T got %T", &ipld.EthStorageTrie{}, n)
- }
- if err := erdp.adder.Add(storageNode); err != nil && !strings.Contains(err.Error(), duplicateKeyErrorString) {
- return "", err
- }
- return storageNode.Cid().String(), nil
-}
diff --git a/pkg/ipfs/dag_putters/eth_tx.go b/pkg/ipfs/dag_putters/eth_tx.go
deleted file mode 100644
index d4ef103f..00000000
--- a/pkg/ipfs/dag_putters/eth_tx.go
+++ /dev/null
@@ -1,46 +0,0 @@
-// 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 .
-
-package dag_putters
-
-import (
- "fmt"
- "strings"
-
- node "github.com/ipfs/go-ipld-format"
-
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/ipfs"
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/ipfs/ipld"
-)
-
-type EthTxsDagPutter struct {
- adder *ipfs.IPFS
-}
-
-func NewEthTxsDagPutter(adder *ipfs.IPFS) *EthTxsDagPutter {
- return &EthTxsDagPutter{adder: adder}
-}
-
-func (etdp *EthTxsDagPutter) DagPut(n node.Node) (string, error) {
- transaction, ok := n.(*ipld.EthTx)
- if !ok {
- return "", fmt.Errorf("EthTxsDagPutter expected input type %T got %T", &ipld.EthTx{}, n)
- }
- if err := etdp.adder.Add(transaction); err != nil && !strings.Contains(err.Error(), duplicateKeyErrorString) {
- return "", err
- }
- return transaction.Cid().String(), nil
-}
diff --git a/pkg/ipfs/dag_putters/eth_tx_trie.go b/pkg/ipfs/dag_putters/eth_tx_trie.go
deleted file mode 100644
index e802d7f3..00000000
--- a/pkg/ipfs/dag_putters/eth_tx_trie.go
+++ /dev/null
@@ -1,46 +0,0 @@
-// 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 .
-
-package dag_putters
-
-import (
- "fmt"
- "strings"
-
- node "github.com/ipfs/go-ipld-format"
-
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/ipfs"
- "github.com/vulcanize/ipfs-blockchain-watcher/pkg/ipfs/ipld"
-)
-
-type EthTxTrieDagPutter struct {
- adder *ipfs.IPFS
-}
-
-func NewEthTxTrieDagPutter(adder *ipfs.IPFS) *EthTxTrieDagPutter {
- return &EthTxTrieDagPutter{adder: adder}
-}
-
-func (etdp *EthTxTrieDagPutter) DagPut(n node.Node) (string, error) {
- txTrieNode, ok := n.(*ipld.EthTxTrie)
- if !ok {
- return "", fmt.Errorf("EthTxTrieDagPutter expected input type %T got %T", &ipld.EthTxTrie{}, n)
- }
- if err := etdp.adder.Add(txTrieNode); err != nil && !strings.Contains(err.Error(), duplicateKeyErrorString) {
- return "", err
- }
- return txTrieNode.Cid().String(), nil
-}
diff --git a/pkg/ipfs/interfaces.go b/pkg/ipfs/interfaces.go
deleted file mode 100644
index cb0f25d9..00000000
--- a/pkg/ipfs/interfaces.go
+++ /dev/null
@@ -1,26 +0,0 @@
-// 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 .
-
-package ipfs
-
-import (
- ipld "github.com/ipfs/go-ipld-format"
-)
-
-// DagPutter is a general interface for a dag putter
-type DagPutter interface {
- DagPut(n ipld.Node) (string, error)
-}
diff --git a/pkg/ipfs/mocks/blockservice.go b/pkg/ipfs/mocks/blockservice.go
deleted file mode 100644
index fdab2fd9..00000000
--- a/pkg/ipfs/mocks/blockservice.go
+++ /dev/null
@@ -1,86 +0,0 @@
-package mocks
-
-import (
- "context"
- "errors"
-
- "github.com/ipfs/go-block-format"
- "github.com/ipfs/go-cid"
- "github.com/ipfs/go-ipfs-blockstore"
- "github.com/ipfs/go-ipfs-exchange-interface"
-)
-
-// MockIPFSBlockService is a mock for testing the ipfs fetcher
-type MockIPFSBlockService struct {
- Blocks map[cid.Cid]blocks.Block
-}
-
-// GetBlock is used to retrieve a block from the mock BlockService
-func (bs *MockIPFSBlockService) GetBlock(ctx context.Context, c cid.Cid) (blocks.Block, error) {
- if bs.Blocks == nil {
- return nil, errors.New("BlockService has not been initialized")
- }
- blk, ok := bs.Blocks[c]
- if ok {
- return blk, nil
- }
- return nil, nil
-}
-
-// GetBlocks is used to retrieve a set of blocks from the mock BlockService
-func (bs *MockIPFSBlockService) GetBlocks(ctx context.Context, cs []cid.Cid) <-chan blocks.Block {
- if bs.Blocks == nil {
- panic("BlockService has not been initialized")
- }
- blkChan := make(chan blocks.Block)
- go func() {
- for _, c := range cs {
- blk, ok := bs.Blocks[c]
- if ok {
- blkChan <- blk
- }
- }
- close(blkChan)
- }()
- return blkChan
-}
-
-// AddBlock adds a block to the mock BlockService
-func (bs *MockIPFSBlockService) AddBlock(blk blocks.Block) error {
- if bs.Blocks == nil {
- bs.Blocks = make(map[cid.Cid]blocks.Block)
- }
- bs.Blocks[blk.Cid()] = blk
- return nil
-}
-
-// AddBlocks adds a set of blocks to the mock BlockService
-func (bs *MockIPFSBlockService) AddBlocks(blks []blocks.Block) error {
- if bs.Blocks == nil {
- bs.Blocks = make(map[cid.Cid]blocks.Block)
- }
- for _, block := range blks {
- bs.Blocks[block.Cid()] = block
- }
- return nil
-}
-
-// Close is here to satisfy the interface
-func (*MockIPFSBlockService) Close() error {
- panic("implement me")
-}
-
-// Blockstore is here to satisfy the interface
-func (*MockIPFSBlockService) Blockstore() blockstore.Blockstore {
- panic("implement me")
-}
-
-// DeleteBlock is here to satisfy the interface
-func (*MockIPFSBlockService) DeleteBlock(c cid.Cid) error {
- panic("implement me")
-}
-
-// Exchange is here to satisfy the interface
-func (*MockIPFSBlockService) Exchange() exchange.Interface {
- panic("implement me")
-}
diff --git a/pkg/ipfs/mocks/dag_putters.go b/pkg/ipfs/mocks/dag_putters.go
deleted file mode 100644
index dd7a9380..00000000
--- a/pkg/ipfs/mocks/dag_putters.go
+++ /dev/null
@@ -1,53 +0,0 @@
-// 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 .
-
-package mocks
-
-import (
- "errors"
-
- node "github.com/ipfs/go-ipld-format"
-
- "github.com/ethereum/go-ethereum/common"
-)
-
-// DagPutter is a mock for testing the ipfs publisher
-type DagPutter struct {
- PassedNode node.Node
- ErrToReturn error
-}
-
-// DagPut returns the pre-loaded CIDs or error
-func (dp *DagPutter) DagPut(n node.Node) (string, error) {
- dp.PassedNode = n
- return n.Cid().String(), dp.ErrToReturn
-}
-
-// MappedDagPutter is a mock for testing the ipfs publisher
-type MappedDagPutter struct {
- CIDsToReturn map[common.Hash]string
- PassedNode node.Node
- ErrToReturn error
-}
-
-// DagPut returns the pre-loaded CIDs or error
-func (mdp *MappedDagPutter) DagPut(n node.Node) (string, error) {
- if mdp.CIDsToReturn == nil {
- return "", errors.New("mapped dag putter needs to be initialized with a map of cids to return")
- }
- hash := common.BytesToHash(n.RawData())
- return mdp.CIDsToReturn[hash], nil
-}
diff --git a/pkg/node/node.go b/pkg/node/node.go
index 7faa881f..66f91d6d 100644
--- a/pkg/node/node.go
+++ b/pkg/node/node.go
@@ -19,6 +19,7 @@ package node
type Node struct {
GenesisBlock string
NetworkID string
+ ChainID uint64
ID string
ClientName string
}
diff --git a/pkg/resync/config.go b/pkg/resync/config.go
index 1451b2a6..c68ad7f6 100644
--- a/pkg/resync/config.go
+++ b/pkg/resync/config.go
@@ -51,8 +51,6 @@ type Config struct {
// DB info
DB *postgres.DB
DBConfig config.Database
- IPFSPath string
- IPFSMode shared.IPFSMode
HTTPClient interface{} // Note this client is expected to support the retrieval of the specified data type(s)
NodeInfo node.Node // Info for the associated node
@@ -91,16 +89,6 @@ func NewConfig() (*Config, error) {
c.ClearOldCache = viper.GetBool("resync.clearOldCache")
c.ResetValidation = viper.GetBool("resync.resetValidation")
- c.IPFSMode, err = shared.GetIPFSMode()
- if err != nil {
- return nil, err
- }
- if c.IPFSMode == shared.LocalInterface || c.IPFSMode == shared.RemoteClient {
- c.IPFSPath, err = shared.GetIPFSPath()
- if err != nil {
- return nil, err
- }
- }
resyncType := viper.GetString("resync.type")
c.ResyncType, err = shared.GenerateDataTypeFromString(resyncType)
if err != nil {
diff --git a/pkg/resync/service.go b/pkg/resync/service.go
index 34bd1bf0..1dc39753 100644
--- a/pkg/resync/service.go
+++ b/pkg/resync/service.go
@@ -35,8 +35,6 @@ type Service struct {
Converter shared.PayloadConverter
// Interface for publishing the IPLD payloads to IPFS
Publisher shared.IPLDPublisher
- // Interface for indexing the CIDs of the published IPLDs in Postgres
- Indexer shared.CIDIndexer
// Interface for searching and retrieving CIDs from Postgres index
Retriever shared.CIDRetriever
// Interface for fetching payloads over at historical blocks; over http
@@ -63,15 +61,11 @@ type Service struct {
// NewResyncService creates and returns a resync service from the provided settings
func NewResyncService(settings *Config) (Resync, error) {
- publisher, err := builders.NewIPLDPublisher(settings.Chain, settings.IPFSPath, settings.DB, settings.IPFSMode)
+ publisher, err := builders.NewIPLDPublisher(settings.Chain, settings.DB)
if err != nil {
return nil, err
}
- indexer, err := builders.NewCIDIndexer(settings.Chain, settings.DB, settings.IPFSMode)
- if err != nil {
- return nil, err
- }
- converter, err := builders.NewPayloadConverter(settings.Chain)
+ converter, err := builders.NewPayloadConverter(settings.Chain, settings.NodeInfo.ChainID)
if err != nil {
return nil, err
}
@@ -96,7 +90,6 @@ func NewResyncService(settings *Config) (Resync, error) {
batchNumber = shared.DefaultMaxBatchNumber
}
return &Service{
- Indexer: indexer,
Converter: converter,
Publisher: publisher,
Retriever: retriever,
@@ -168,13 +161,9 @@ func (rs *Service) resync(id int, heightChan chan []uint64) {
if err != nil {
logrus.Errorf("%s resync worker %d converter error: %s", rs.chain.String(), id, err.Error())
}
- cidPayload, err := rs.Publisher.Publish(ipldPayload)
- if err != nil {
+ if err := rs.Publisher.Publish(ipldPayload); err != nil {
logrus.Errorf("%s resync worker %d publisher error: %s", rs.chain.String(), id, err.Error())
}
- if err := rs.Indexer.Index(cidPayload); err != nil {
- logrus.Errorf("%s resync worker %d indexer error: %s", rs.chain.String(), id, err.Error())
- }
}
logrus.Infof("%s resync worker %d finished section from %d to %d", rs.chain.String(), id, heights[0], heights[len(heights)-1])
case <-rs.quitChan:
diff --git a/pkg/shared/chain_type.go b/pkg/shared/chain_type.go
index 83445192..c3dedfe3 100644
--- a/pkg/shared/chain_type.go
+++ b/pkg/shared/chain_type.go
@@ -29,6 +29,7 @@ const (
Ethereum
Bitcoin
Omni
+ EthereumClassic
)
func (c ChainType) String() string {
@@ -39,6 +40,8 @@ func (c ChainType) String() string {
return "Bitcoin"
case Omni:
return "Omni"
+ case EthereumClassic:
+ return "EthereumClassic"
default:
return ""
}
@@ -52,6 +55,8 @@ func (c ChainType) API() string {
return "btc"
case Omni:
return "omni"
+ case EthereumClassic:
+ return "etc"
default:
return ""
}
@@ -65,6 +70,8 @@ func NewChainType(name string) (ChainType, error) {
return Bitcoin, nil
case "omni":
return Omni, nil
+ case "classic", "etc":
+ return EthereumClassic, nil
default:
return UnknownChain, errors.New("invalid name for chain")
}
diff --git a/pkg/shared/env.go b/pkg/shared/env.go
index 58191613..b11c726e 100644
--- a/pkg/shared/env.go
+++ b/pkg/shared/env.go
@@ -17,9 +17,6 @@
package shared
import (
- "os"
- "path/filepath"
-
"github.com/ethereum/go-ethereum/rpc"
"github.com/btcsuite/btcd/rpcclient"
@@ -29,8 +26,6 @@ import (
// Env variables
const (
- IPFS_PATH = "IPFS_PATH"
- IPFS_MODE = "IPFS_MODE"
HTTP_TIMEOUT = "HTTP_TIMEOUT"
ETH_WS_PATH = "ETH_WS_PATH"
@@ -39,6 +34,7 @@ const (
ETH_CLIENT_NAME = "ETH_CLIENT_NAME"
ETH_GENESIS_BLOCK = "ETH_GENESIS_BLOCK"
ETH_NETWORK_ID = "ETH_NETWORK_ID"
+ ETH_CHAIN_ID = "ETH_CHAIN_ID"
BTC_WS_PATH = "BTC_WS_PATH"
BTC_HTTP_PATH = "BTC_HTTP_PATH"
@@ -48,6 +44,7 @@ const (
BTC_CLIENT_NAME = "BTC_CLIENT_NAME"
BTC_GENESIS_BLOCK = "BTC_GENESIS_BLOCK"
BTC_NETWORK_ID = "BTC_NETWORK_ID"
+ BTC_CHAIN_ID = "BTC_CHAIN_ID"
)
// GetEthNodeAndClient returns eth node info and client from path url
@@ -56,6 +53,7 @@ func GetEthNodeAndClient(path string) (node.Node, *rpc.Client, error) {
viper.BindEnv("ethereum.clientName", ETH_CLIENT_NAME)
viper.BindEnv("ethereum.genesisBlock", ETH_GENESIS_BLOCK)
viper.BindEnv("ethereum.networkID", ETH_NETWORK_ID)
+ viper.BindEnv("ethereum.chainID", ETH_CHAIN_ID)
rpcClient, err := rpc.Dial(path)
if err != nil {
@@ -66,33 +64,10 @@ func GetEthNodeAndClient(path string) (node.Node, *rpc.Client, error) {
ClientName: viper.GetString("ethereum.clientName"),
GenesisBlock: viper.GetString("ethereum.genesisBlock"),
NetworkID: viper.GetString("ethereum.networkID"),
+ ChainID: viper.GetUint64("ethereum.chainID"),
}, rpcClient, nil
}
-// GetIPFSPath returns the ipfs path from the config or env variable
-func GetIPFSPath() (string, error) {
- viper.BindEnv("ipfs.path", IPFS_PATH)
- ipfsPath := viper.GetString("ipfs.path")
- if ipfsPath == "" {
- home, err := os.UserHomeDir()
- if err != nil {
- return "", err
- }
- ipfsPath = filepath.Join(home, ".ipfs")
- }
- return ipfsPath, nil
-}
-
-// GetIPFSMode returns the ipfs mode of operation from the config or env variable
-func GetIPFSMode() (IPFSMode, error) {
- viper.BindEnv("ipfs.mode", IPFS_MODE)
- ipfsMode := viper.GetString("ipfs.mode")
- if ipfsMode == "" {
- return DirectPostgres, nil
- }
- return NewIPFSMode(ipfsMode)
-}
-
// GetBtcNodeAndClient returns btc node info from path url
func GetBtcNodeAndClient(path string) (node.Node, *rpcclient.ConnConfig) {
viper.BindEnv("bitcoin.nodeID", BTC_NODE_ID)
@@ -101,6 +76,7 @@ func GetBtcNodeAndClient(path string) (node.Node, *rpcclient.ConnConfig) {
viper.BindEnv("bitcoin.networkID", BTC_NETWORK_ID)
viper.BindEnv("bitcoin.pass", BTC_NODE_PASSWORD)
viper.BindEnv("bitcoin.user", BTC_NODE_USER)
+ viper.BindEnv("bitcoin.chainID", BTC_CHAIN_ID)
// For bitcoin we load in node info from the config because there is no RPC endpoint to retrieve this from the node
return node.Node{
@@ -108,6 +84,7 @@ func GetBtcNodeAndClient(path string) (node.Node, *rpcclient.ConnConfig) {
ClientName: viper.GetString("bitcoin.clientName"),
GenesisBlock: viper.GetString("bitcoin.genesisBlock"),
NetworkID: viper.GetString("bitcoin.networkID"),
+ ChainID: viper.GetUint64("bitcoin.chainID"),
}, &rpcclient.ConnConfig{
Host: path,
HTTPPostMode: true, // Bitcoin core only supports HTTP POST mode
diff --git a/pkg/shared/intefaces.go b/pkg/shared/intefaces.go
index 3c2e0110..03839b4d 100644
--- a/pkg/shared/intefaces.go
+++ b/pkg/shared/intefaces.go
@@ -37,12 +37,7 @@ type PayloadConverter interface {
// IPLDPublisher publishes IPLD payloads and returns a CID payload for indexing
type IPLDPublisher interface {
- Publish(payload ConvertedData) (CIDsForIndexing, error)
-}
-
-// CIDIndexer indexes a CID payload in Postgres
-type CIDIndexer interface {
- Index(cids CIDsForIndexing) error
+ Publish(payload ConvertedData) error
}
// ResponseFilterer applies a filter to an IPLD payload to return a subscription response packet
diff --git a/pkg/shared/ipfs_mode.go b/pkg/shared/ipfs_mode.go
deleted file mode 100644
index 14d7d856..00000000
--- a/pkg/shared/ipfs_mode.go
+++ /dev/null
@@ -1,58 +0,0 @@
-// 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 .
-
-package shared
-
-import (
- "errors"
- "strings"
-)
-
-// IPFSMode enum for specifying how we want to interface and publish objects to IPFS
-type IPFSMode int
-
-const (
- Unknown IPFSMode = iota
- LocalInterface
- RemoteClient
- DirectPostgres
-)
-
-func (c IPFSMode) String() string {
- switch c {
- case LocalInterface:
- return "Local"
- case RemoteClient:
- return "Remote"
- case DirectPostgres:
- return "Postgres"
- default:
- return ""
- }
-}
-
-func NewIPFSMode(name string) (IPFSMode, error) {
- switch strings.ToLower(name) {
- case "local", "interface":
- return LocalInterface, nil
- case "remote", "client":
- return RemoteClient, errors.New("remote IPFS client mode is not currently supported")
- case "postgres", "direct":
- return DirectPostgres, nil
- default:
- return Unknown, errors.New("unrecognized name for ipfs mode")
- }
-}
diff --git a/pkg/shared/types.go b/pkg/shared/types.go
index e213f7f4..7dd52f30 100644
--- a/pkg/shared/types.go
+++ b/pkg/shared/types.go
@@ -16,6 +16,9 @@
package shared
+// Very loose interface types for generic processing of different blockchains
+// TODO: split different blockchain support into separate repos
+
// These types serve as very loose wrappers around a generic underlying interface{}
type RawChainData interface{}
diff --git a/pkg/validate/service.go b/pkg/validate/service.go
deleted file mode 100644
index 77fb2737..00000000
--- a/pkg/validate/service.go
+++ /dev/null
@@ -1 +0,0 @@
-package validate
diff --git a/pkg/watch/config.go b/pkg/watch/config.go
index 267c3887..2f8bac20 100644
--- a/pkg/watch/config.go
+++ b/pkg/watch/config.go
@@ -53,8 +53,6 @@ const (
// Config struct
type Config struct {
Chain shared.ChainType
- IPFSPath string
- IPFSMode shared.IPFSMode
DBConfig config.Database
// Server fields
Serve bool
@@ -96,19 +94,7 @@ func NewConfig() (*Config, error) {
return nil, err
}
- c.IPFSMode, err = shared.GetIPFSMode()
- if err != nil {
- return nil, err
- }
- if c.IPFSMode == shared.LocalInterface || c.IPFSMode == shared.RemoteClient {
- c.IPFSPath, err = shared.GetIPFSPath()
- if err != nil {
- return nil, err
- }
- }
-
c.DBConfig.Init()
-
c.Sync = viper.GetBool("watcher.sync")
if c.Sync {
workers := viper.GetInt("watcher.workers")
diff --git a/pkg/watch/service.go b/pkg/watch/service.go
index 922ff6fe..74cb7422 100644
--- a/pkg/watch/service.go
+++ b/pkg/watch/service.go
@@ -66,10 +66,8 @@ type Service struct {
Streamer shared.PayloadStreamer
// Interface for converting raw payloads into IPLD object payloads
Converter shared.PayloadConverter
- // Interface for publishing the IPLD payloads to IPFS
+ // Interface for publishing and indexing the PG-IPLD payloads
Publisher shared.IPLDPublisher
- // Interface for indexing the CIDs of the published IPLDs in Postgres
- Indexer shared.CIDIndexer
// Interface for filtering and serving data according to subscribed clients according to their specification
Filterer shared.ResponseFilterer
// Interface for fetching IPLD objects from IPFS
@@ -86,12 +84,10 @@ type Service struct {
SubscriptionTypes map[common.Hash]shared.SubscriptionSettings
// Info for the Geth node that this watcher is working with
NodeInfo *node.Node
- // Number of publishAndIndex workers
+ // Number of publish workers
WorkerPoolSize int
// chain type for this service
chain shared.ChainType
- // Path to ipfs data dir
- ipfsPath string
// Underlying db
db *postgres.DB
// wg for syncing serve processes
@@ -108,15 +104,11 @@ func NewWatcher(settings *Config) (Watcher, error) {
if err != nil {
return nil, err
}
- sn.Converter, err = builders.NewPayloadConverter(settings.Chain)
+ sn.Converter, err = builders.NewPayloadConverter(settings.Chain, settings.NodeInfo.ChainID)
if err != nil {
return nil, err
}
- sn.Publisher, err = builders.NewIPLDPublisher(settings.Chain, settings.IPFSPath, settings.SyncDBConn, settings.IPFSMode)
- if err != nil {
- return nil, err
- }
- sn.Indexer, err = builders.NewCIDIndexer(settings.Chain, settings.SyncDBConn, settings.IPFSMode)
+ sn.Publisher, err = builders.NewIPLDPublisher(settings.Chain, settings.SyncDBConn)
if err != nil {
return nil, err
}
@@ -131,7 +123,7 @@ func NewWatcher(settings *Config) (Watcher, error) {
if err != nil {
return nil, err
}
- sn.IPLDFetcher, err = builders.NewIPLDFetcher(settings.Chain, settings.IPFSPath, settings.ServeDBConn, settings.IPFSMode)
+ sn.IPLDFetcher, err = builders.NewIPLDFetcher(settings.Chain, settings.ServeDBConn)
if err != nil {
return nil, err
}
@@ -142,7 +134,6 @@ func NewWatcher(settings *Config) (Watcher, error) {
sn.SubscriptionTypes = make(map[common.Hash]shared.SubscriptionSettings)
sn.WorkerPoolSize = settings.Workers
sn.NodeInfo = &settings.NodeInfo
- sn.ipfsPath = settings.IPFSPath
sn.chain = settings.Chain
return sn, nil
}
@@ -154,7 +145,7 @@ func (sap *Service) Protocols() []p2p.Protocol {
// APIs returns the RPC descriptors the watcher service offers
func (sap *Service) APIs() []rpc.API {
- ifnoAPI := NewInfoAPI()
+ infoAPI := NewInfoAPI()
apis := []rpc.API{
{
Namespace: APIName,
@@ -165,23 +156,23 @@ func (sap *Service) APIs() []rpc.API {
{
Namespace: "rpc",
Version: APIVersion,
- Service: ifnoAPI,
+ Service: infoAPI,
Public: true,
},
{
Namespace: "net",
Version: APIVersion,
- Service: ifnoAPI,
+ Service: infoAPI,
Public: true,
},
{
Namespace: "admin",
Version: APIVersion,
- Service: ifnoAPI,
+ Service: infoAPI,
Public: true,
},
}
- chainAPI, err := builders.NewPublicAPI(sap.chain, sap.db, sap.ipfsPath)
+ chainAPI, err := builders.NewPublicAPI(sap.chain, sap.db)
if err != nil {
log.Error(err)
return apis
@@ -190,7 +181,7 @@ func (sap *Service) APIs() []rpc.API {
}
// Sync streams incoming raw chain data and converts it for further processing
-// It forwards the converted data to the publishAndIndex process(es) it spins up
+// It forwards the converted data to the publish process(es) it spins up
// If forwards the converted data to a ScreenAndServe process if it there is one listening on the passed screenAndServePayload channel
// This continues on no matter if or how many subscribers there are
func (sap *Service) Sync(wg *sync.WaitGroup, screenAndServePayload chan<- shared.ConvertedData) error {
@@ -198,11 +189,11 @@ func (sap *Service) Sync(wg *sync.WaitGroup, screenAndServePayload chan<- shared
if err != nil {
return err
}
- // spin up publishAndIndex worker goroutines
- publishAndIndexPayload := make(chan shared.ConvertedData, PayloadChanBufferSize)
+ // spin up publish worker goroutines
+ publishPayload := make(chan shared.ConvertedData, PayloadChanBufferSize)
for i := 1; i <= sap.WorkerPoolSize; i++ {
- go sap.publishAndIndex(wg, i, publishAndIndexPayload)
- log.Debugf("%s publishAndIndex worker %d successfully spun up", sap.chain.String(), i)
+ go sap.publish(wg, i, publishPayload)
+ log.Debugf("%s publish worker %d successfully spun up", sap.chain.String(), i)
}
go func() {
wg.Add(1)
@@ -221,13 +212,13 @@ func (sap *Service) Sync(wg *sync.WaitGroup, screenAndServePayload chan<- shared
case screenAndServePayload <- ipldPayload:
default:
}
- // Forward the payload to the publishAndIndex workers
+ // Forward the payload to the publish workers
// this channel acts as a ring buffer
select {
- case publishAndIndexPayload <- ipldPayload:
+ case publishPayload <- ipldPayload:
default:
- <-publishAndIndexPayload
- publishAndIndexPayload <- ipldPayload
+ <-publishPayload
+ publishPayload <- ipldPayload
}
case err := <-sub.Err():
log.Errorf("watcher subscription error for chain %s: %v", sap.chain.String(), err)
@@ -241,26 +232,21 @@ func (sap *Service) Sync(wg *sync.WaitGroup, screenAndServePayload chan<- shared
return nil
}
-// publishAndIndex is spun up by SyncAndConvert and receives converted chain data from that process
+// publish is spun up by SyncAndConvert and receives converted chain data from that process
// it publishes this data to IPFS and indexes their CIDs with useful metadata in Postgres
-func (sap *Service) publishAndIndex(wg *sync.WaitGroup, id int, publishAndIndexPayload <-chan shared.ConvertedData) {
+func (sap *Service) publish(wg *sync.WaitGroup, id int, publishPayload <-chan shared.ConvertedData) {
wg.Add(1)
defer wg.Done()
for {
select {
- case payload := <-publishAndIndexPayload:
- log.Debugf("%s watcher publishAndIndex worker %d publishing data streamed at head height %d", sap.chain.String(), id, payload.Height())
- cidPayload, err := sap.Publisher.Publish(payload)
- if err != nil {
- log.Errorf("%s watcher publishAndIndex worker %d publishing error: %v", sap.chain.String(), id, err)
+ case payload := <-publishPayload:
+ log.Debugf("%s watcher sync worker %d publishing and indexing data streamed at head height %d", sap.chain.String(), id, payload.Height())
+ if err := sap.Publisher.Publish(payload); err != nil {
+ log.Errorf("%s watcher publish worker %d publishing error: %v", sap.chain.String(), id, err)
continue
}
- log.Debugf("%s watcher publishAndIndex worker %d indexing data streamed at head height %d", sap.chain.String(), id, payload.Height())
- if err := sap.Indexer.Index(cidPayload); err != nil {
- log.Errorf("%s watcher publishAndIndex worker %d indexing error: %v", sap.chain.String(), id, err)
- }
case <-sap.QuitChan:
- log.Infof("%s watcher publishAndIndex worker %d shutting down", sap.chain.String(), id)
+ log.Infof("%s watcher publish worker %d shutting down", sap.chain.String(), id)
return
}
}
diff --git a/pkg/watch/service_test.go b/pkg/watch/service_test.go
index b94b5a8a..d2902c88 100644
--- a/pkg/watch/service_test.go
+++ b/pkg/watch/service_test.go
@@ -36,9 +36,6 @@ var _ = Describe("Service", func() {
wg := new(sync.WaitGroup)
payloadChan := make(chan shared.RawChainData, 1)
quitChan := make(chan bool, 1)
- mockCidIndexer := &mocks.CIDIndexer{
- ReturnErr: nil,
- }
mockPublisher := &mocks.IPLDPublisher{
ReturnCIDPayload: mocks.MockCIDPayload,
ReturnErr: nil,
@@ -55,7 +52,6 @@ var _ = Describe("Service", func() {
ReturnErr: nil,
}
processor := &watch.Service{
- Indexer: mockCidIndexer,
Publisher: mockPublisher,
Streamer: mockStreamer,
Converter: mockConverter,
@@ -69,8 +65,6 @@ var _ = Describe("Service", func() {
close(quitChan)
wg.Wait()
Expect(mockConverter.PassedStatediffPayload).To(Equal(mocks.MockStateDiffPayload))
- Expect(len(mockCidIndexer.PassedCIDPayload)).To(Equal(1))
- Expect(mockCidIndexer.PassedCIDPayload[0]).To(Equal(mocks.MockCIDPayload))
Expect(mockPublisher.PassedIPLDPayload).To(Equal(mocks.MockConvertedPayload))
Expect(mockStreamer.PassedPayloadChan).To(Equal(payloadChan))
})