2020-01-30 22:46:08 +00:00
|
|
|
// VulcanizeDB
|
|
|
|
// Copyright © 2019 Vulcanize
|
|
|
|
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU Affero General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU Affero General Public License for more details.
|
|
|
|
|
|
|
|
// You should have received a copy of the GNU Affero General Public License
|
|
|
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
package btc
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"strconv"
|
|
|
|
|
2020-06-22 18:12:32 +00:00
|
|
|
"github.com/vulcanize/ipfs-blockchain-watcher/pkg/ipfs/ipld"
|
2020-08-10 17:46:03 +00:00
|
|
|
"github.com/vulcanize/ipfs-blockchain-watcher/pkg/postgres"
|
2020-06-22 18:12:32 +00:00
|
|
|
"github.com/vulcanize/ipfs-blockchain-watcher/pkg/shared"
|
2020-01-30 22:46:08 +00:00
|
|
|
)
|
|
|
|
|
2020-08-10 18:04:07 +00:00
|
|
|
// IPLDPublisher satisfies the IPLDPublisher interface for bitcoin
|
2020-08-10 17:46:03 +00:00
|
|
|
// 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
|
2020-08-10 18:04:07 +00:00
|
|
|
type IPLDPublisher struct {
|
2020-08-10 17:46:03 +00:00
|
|
|
indexer *CIDIndexer
|
2020-01-30 22:46:08 +00:00
|
|
|
}
|
|
|
|
|
2020-08-10 18:04:07 +00:00
|
|
|
// NewIPLDPublisher creates a pointer to a new eth IPLDPublisher which satisfies the IPLDPublisher interface
|
|
|
|
func NewIPLDPublisher(db *postgres.DB) *IPLDPublisher {
|
|
|
|
return &IPLDPublisher{
|
2020-08-10 17:46:03 +00:00
|
|
|
indexer: NewCIDIndexer(db),
|
2020-01-30 22:46:08 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Publish publishes an IPLDPayload to IPFS and returns the corresponding CIDPayload
|
2020-08-10 18:04:07 +00:00
|
|
|
func (pub *IPLDPublisher) Publish(payload shared.ConvertedData) error {
|
2020-02-20 22:12:52 +00:00
|
|
|
ipldPayload, ok := payload.(ConvertedPayload)
|
2020-01-30 22:46:08 +00:00
|
|
|
if !ok {
|
2020-08-10 18:04:07 +00:00
|
|
|
return fmt.Errorf("btc publisher expected payload type %T got %T", ConvertedPayload{}, payload)
|
2020-01-30 22:46:08 +00:00
|
|
|
}
|
2020-08-10 17:46:03 +00:00
|
|
|
// Generate the iplds
|
2020-03-17 18:05:19 +00:00
|
|
|
headerNode, txNodes, txTrieNodes, err := ipld.FromHeaderAndTxs(ipldPayload.Header, ipldPayload.Txs)
|
|
|
|
if err != nil {
|
2020-08-10 18:04:07 +00:00
|
|
|
return err
|
2020-03-17 18:05:19 +00:00
|
|
|
}
|
2020-08-10 17:46:03 +00:00
|
|
|
|
|
|
|
// Begin new db tx
|
|
|
|
tx, err := pub.indexer.db.Beginx()
|
2020-01-30 22:46:08 +00:00
|
|
|
if err != nil {
|
2020-08-10 18:04:07 +00:00
|
|
|
return err
|
2020-01-30 22:46:08 +00:00
|
|
|
}
|
2020-08-10 17:46:03 +00:00
|
|
|
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 {
|
2020-08-10 18:04:07 +00:00
|
|
|
return err
|
2020-08-10 17:46:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Publish and index header
|
|
|
|
if err := shared.PublishIPLD(tx, headerNode); err != nil {
|
2020-08-10 18:04:07 +00:00
|
|
|
return err
|
2020-08-10 17:46:03 +00:00
|
|
|
}
|
2020-01-30 22:46:08 +00:00
|
|
|
header := HeaderModel{
|
2020-08-10 17:46:03 +00:00
|
|
|
CID: headerNode.Cid().String(),
|
|
|
|
MhKey: shared.MultihashKeyFromCID(headerNode.Cid()),
|
2020-01-30 22:46:08 +00:00
|
|
|
ParentHash: ipldPayload.Header.PrevBlock.String(),
|
2020-02-19 22:09:33 +00:00
|
|
|
BlockNumber: strconv.Itoa(int(ipldPayload.BlockPayload.BlockHeight)),
|
2020-01-30 22:46:08 +00:00
|
|
|
BlockHash: ipldPayload.Header.BlockHash().String(),
|
2020-02-02 21:58:07 +00:00
|
|
|
Timestamp: ipldPayload.Header.Timestamp.UnixNano(),
|
|
|
|
Bits: ipldPayload.Header.Bits,
|
2020-01-30 22:46:08 +00:00
|
|
|
}
|
2020-08-10 17:46:03 +00:00
|
|
|
headerID, err := pub.indexer.indexHeaderCID(tx, header)
|
2020-01-30 22:46:08 +00:00
|
|
|
if err != nil {
|
2020-08-10 18:04:07 +00:00
|
|
|
return err
|
2020-01-30 22:46:08 +00:00
|
|
|
}
|
|
|
|
|
2020-08-10 17:46:03 +00:00
|
|
|
// Publish and index txs
|
|
|
|
for i, txNode := range txNodes {
|
|
|
|
if err := shared.PublishIPLD(tx, txNode); err != nil {
|
2020-08-10 18:04:07 +00:00
|
|
|
return err
|
2020-08-10 17:46:03 +00:00
|
|
|
}
|
|
|
|
txModel := ipldPayload.TxMetaData[i]
|
|
|
|
txModel.CID = txNode.Cid().String()
|
|
|
|
txModel.MhKey = shared.MultihashKeyFromCID(txNode.Cid())
|
|
|
|
txID, err := pub.indexer.indexTransactionCID(tx, txModel, headerID)
|
2020-03-17 18:05:19 +00:00
|
|
|
if err != nil {
|
2020-08-10 18:04:07 +00:00
|
|
|
return err
|
2020-03-17 18:05:19 +00:00
|
|
|
}
|
2020-08-10 17:46:03 +00:00
|
|
|
for _, input := range txModel.TxInputs {
|
|
|
|
if err := pub.indexer.indexTxInput(tx, input, txID); err != nil {
|
2020-08-10 18:04:07 +00:00
|
|
|
return err
|
2020-08-10 17:46:03 +00:00
|
|
|
}
|
2020-01-30 22:46:08 +00:00
|
|
|
}
|
2020-08-10 17:46:03 +00:00
|
|
|
for _, output := range txModel.TxOutputs {
|
|
|
|
if err := pub.indexer.indexTxOutput(tx, output, txID); err != nil {
|
2020-08-10 18:04:07 +00:00
|
|
|
return err
|
2020-08-10 17:46:03 +00:00
|
|
|
}
|
2020-03-17 18:05:19 +00:00
|
|
|
}
|
|
|
|
}
|
2020-08-10 17:46:03 +00:00
|
|
|
|
2020-08-10 18:04:07 +00:00
|
|
|
return err
|
2020-01-30 22:46:08 +00:00
|
|
|
}
|