From 6418f0257cec277e9fe100c27305de31a4f10be3 Mon Sep 17 00:00:00 2001 From: alainjr10 Date: Tue, 23 Jul 2024 16:38:10 +0100 Subject: [PATCH 01/28] add function to create headers for utreexo cf --- btcutil/gcs/builder/builder.go | 16 ++++++++++++++++ 1 file changed, 16 insertions(+) diff --git a/btcutil/gcs/builder/builder.go b/btcutil/gcs/builder/builder.go index fe11b80d..33bbd3e8 100644 --- a/btcutil/gcs/builder/builder.go +++ b/btcutil/gcs/builder/builder.go @@ -369,3 +369,19 @@ func MakeHeaderForFilter(filter *gcs.Filter, prevHeader chainhash.Hash) (chainha // above. return chainhash.DoubleHashH(filterTip), nil } + +// MakeHeaderForUtreexoCFilter makes a filter chain header for a utreexoc filter, given the +// filter data and the previous filter chain header. +func MakeHeaderForUtreexoCFilter(filterData []byte, prevHeader chainhash.Hash) (chainhash.Hash, error) { + filterTip := make([]byte, 2*chainhash.HashSize) + filterHash := chainhash.DoubleHashH(filterData) + + // In the buffer we created above we'll compute hash || prevHash as an + // intermediate value. + copy(filterTip, filterHash[:]) + copy(filterTip[chainhash.HashSize:], prevHeader[:]) + + // The final filter hash is the double-sha256 of the hash computed + // above. + return chainhash.DoubleHashH(filterTip), nil +} From cab02ccbda89f9dbf3502b955c7330f43f4f20ed Mon Sep 17 00:00:00 2001 From: alainjr10 Date: Tue, 23 Jul 2024 16:39:03 +0100 Subject: [PATCH 02/28] add new filter type --- wire/msgcfilter.go | 1 + 1 file changed, 1 insertion(+) diff --git a/wire/msgcfilter.go b/wire/msgcfilter.go index 682e9fd2..1e33d6e2 100644 --- a/wire/msgcfilter.go +++ b/wire/msgcfilter.go @@ -17,6 +17,7 @@ type FilterType uint8 const ( // GCSFilterRegular is the regular filter type. GCSFilterRegular FilterType = iota + UtreexoCFilter ) const ( From 1f938ffe352e23b171561ad2c5e4f9a0f7c11977 Mon Sep 17 00:00:00 2001 From: alainjr10 Date: Tue, 23 Jul 2024 16:41:20 +0100 Subject: [PATCH 03/28] add indexer for utreexoc filter --- blockchain/indexers/utreexocfindex.go | 301 ++++++++++++++++++++++++++ 1 file changed, 301 insertions(+) create mode 100644 blockchain/indexers/utreexocfindex.go diff --git a/blockchain/indexers/utreexocfindex.go b/blockchain/indexers/utreexocfindex.go new file mode 100644 index 00000000..c0ac2910 --- /dev/null +++ b/blockchain/indexers/utreexocfindex.go @@ -0,0 +1,301 @@ +package indexers + +import ( + "errors" + + "github.com/utreexo/utreexod/blockchain" + "github.com/utreexo/utreexod/btcutil" + "github.com/utreexo/utreexod/btcutil/gcs/builder" + "github.com/utreexo/utreexod/chaincfg" + "github.com/utreexo/utreexod/chaincfg/chainhash" + "github.com/utreexo/utreexod/database" + "github.com/utreexo/utreexod/wire" +) + +// utreexoProofIndexName is the human-readable name for the index. +const ( + utreexoCFIndexName = "utreexo custom cfilter index" +) + +// utreexocfilter is a custom commited filter which serves utreexo roots +// these roots are already present, so they need not be created/stored, their +// headers could be stored though +var ( + // utreexoCFIndexParentBucketKey is the name of the parent bucket used to + // house the index. The rest of the buckets live below this bucket. + utreexoCFIndexParentBucketKey = []byte("utreexocfindexparentbucket") + + // utreexoCfHeaderKeys is an array of db bucket names used to house indexes of + // block hashes to cf headers. + utreexoCfHeaderKeys = [][]byte{ + []byte("utreexocfheaderbyhashidx"), + } +) + +// dbFetchFilterIdxEntry retrieves a data blob from the filter index database. +// An entry's absence is not considered an error. +func dbFetchUtreexoCFilterIdxEntry(dbTx database.Tx, key []byte, h *chainhash.Hash) ([]byte, error) { + idx := dbTx.Metadata().Bucket(utreexoCFIndexParentBucketKey).Bucket(key) + return idx.Get(h[:]), nil +} + +// dbStoreFilterIdxEntry stores a data blob in the filter index database. +func dbStoreUtreexoCFilterIdxEntry(dbTx database.Tx, key []byte, h *chainhash.Hash, f []byte) error { + idx := dbTx.Metadata().Bucket(utreexoCFIndexParentBucketKey).Bucket(key) + return idx.Put(h[:], f) +} + +// dbDeleteFilterIdxEntry deletes a data blob from the filter index database. +func dbDeleteUtreexoCFilterIdxEntry(dbTx database.Tx, key []byte, h *chainhash.Hash) error { + idx := dbTx.Metadata().Bucket(utreexoCFIndexParentBucketKey).Bucket(key) + return idx.Delete(h[:]) +} + +var _ Indexer = (*UtreexoCFIndex)(nil) + +var _ NeedsInputser = (*UtreexoCFIndex)(nil) + +type UtreexoCFIndex struct { + db database.DB + chainParams *chaincfg.Params + + chain *blockchain.BlockChain + + utreexoProofIndex *UtreexoProofIndex + + flatUtreexoProofIndex *FlatUtreexoProofIndex +} + +func (idx *UtreexoCFIndex) NeedsInputs() bool { + return true +} + +// Init initializes the utreexo cf index. This is part of the Indexer +// interface. +func (idx *UtreexoCFIndex) Init(_ *blockchain.BlockChain) error { + return nil // Nothing to do. +} + +// Key returns the database key to use for the index as a byte slice. This is +// part of the Indexer interface. +func (idx *UtreexoCFIndex) Key() []byte { + return utreexoCFIndexParentBucketKey +} + +// Name returns the human-readable name of the index. This is part of the +// Indexer interface. +func (idx *UtreexoCFIndex) Name() string { + return utreexoCFIndexName +} + +// Create is invoked when the index manager determines the index needs to +// be created for the first time. It creates buckets for the custom utreexo +// filter index. +func (idx *UtreexoCFIndex) Create(dbTx database.Tx) error { + meta := dbTx.Metadata() + + utreexoCfIndexParentBucket, err := meta.CreateBucket(utreexoCFIndexParentBucketKey) + if err != nil { + return err + } + + for _, bucketName := range utreexoCfHeaderKeys { + _, err = utreexoCfIndexParentBucket.CreateBucket(bucketName) + if err != nil { + return err + } + } + + return nil +} + +// storeUtreexoCFilter stores a given utreexocfilter header +func storeUtreexoCFHeader(dbTx database.Tx, block *btcutil.Block, filterData []byte, + filterType wire.FilterType) error { + if filterType != wire.UtreexoCFilter { + return errors.New("invalid filter type") + } + + // Figure out which header bucket to use. + hkey := utreexoCfHeaderKeys[0] + h := block.Hash() + + // fetch the previous block's filter header. + var prevHeader *chainhash.Hash + ph := &block.MsgBlock().Header.PrevBlock + if ph.IsEqual(&zeroHash) { + prevHeader = &zeroHash + } else { + pfh, err := dbFetchUtreexoCFilterIdxEntry(dbTx, hkey, ph) + if err != nil { + return err + } + + // Construct the new block's filter header, and store it. + prevHeader, err = chainhash.NewHash(pfh) + if err != nil { + return err + } + } + + fh, err := builder.MakeHeaderForUtreexoCFilter(filterData, *prevHeader) + if err != nil { + return err + } + return dbStoreUtreexoCFilterIdxEntry(dbTx, hkey, h, fh[:]) +} + +// ConnectBlock is invoked by the index manager when a new block has been +// connected to the main chain. +// This is part of the Indexer interface. +func (idx *UtreexoCFIndex) ConnectBlock(dbTx database.Tx, block *btcutil.Block, + stxos []blockchain.SpentTxOut) error { + + blockHash := block.Hash() + roots, leaves, err := idx.fetchUtreexoRoots(dbTx, blockHash) + + if err != nil { + return err + } + + // serialize the hashes of the utreexo roots hash + serializedUtreexo, err := blockchain.SerializeUtreexoRootsHash(leaves, roots) + if err != nil { + return err + } + + return storeUtreexoCFHeader(dbTx, block, serializedUtreexo, wire.UtreexoCFilter) +} + +// fetches the utreexo roots for a given block hash +func (idx *UtreexoCFIndex) fetchUtreexoRoots(dbTx database.Tx, + blockHash *chainhash.Hash) ([]*chainhash.Hash, uint64, error) { + + var leaves uint64 + var roots []*chainhash.Hash + + // For compact state nodes + if idx.chain.IsUtreexoViewActive() && idx.chain.IsAssumeUtreexo() { + viewPoint, err := idx.chain.FetchUtreexoViewpoint(blockHash) + if err != nil { + return nil, 0, err + } + roots = viewPoint.GetRoots() + leaves = viewPoint.NumLeaves() + } + // for bridge nodes + if idx.utreexoProofIndex != nil { + roots, leaves, err := idx.utreexoProofIndex.FetchUtreexoState(dbTx, blockHash) + if err != nil { + return nil, 0, err + } + return roots, leaves, nil + } else if idx.flatUtreexoProofIndex != nil { + height, err := idx.chain.BlockHeightByHash(blockHash) + if err != nil { + return nil, 0, err + } + roots, leaves, err := idx.flatUtreexoProofIndex.FetchUtreexoState(height) + if err != nil { + return nil, 0, err + } + return roots, leaves, nil + } + + return roots, leaves, nil +} + +// DisconnectBlock is invoked by the index manager when a block has been +// disconnected from the main chain. This indexer removes the hash-to-cf +// mapping for every passed block. This is part of the Indexer interface. +func (idx *UtreexoCFIndex) DisconnectBlock(dbTx database.Tx, block *btcutil.Block, + _ []blockchain.SpentTxOut) error { + + for _, key := range utreexoCfHeaderKeys { + err := dbDeleteUtreexoCFilterIdxEntry(dbTx, key, block.Hash()) + if err != nil { + return err + } + } + + return nil +} + +// PruneBlock is invoked when an older block is deleted after it's been +// processed. +// TODO (kcalvinalvin): Consider keeping the filters at a later date to help with +// reindexing as a pruned node. +// +// This is part of the Indexer interface. +func (idx *UtreexoCFIndex) PruneBlock(dbTx database.Tx, blockHash *chainhash.Hash) error { + + for _, key := range utreexoCfHeaderKeys { + err := dbDeleteUtreexoCFilterIdxEntry(dbTx, key, blockHash) + if err != nil { + return err + } + } + + return nil +} + +// entryByBlockHash fetches a filter index entry of a particular type +// (eg. filter, filter header, etc) for a filter type and block hash. +func (idx *UtreexoCFIndex) entryByBlockHash(dbTx database.Tx, + filterType wire.FilterType, h *chainhash.Hash) ([]byte, error) { + + if uint8(filterType) != uint8(wire.UtreexoCFilter) { + return nil, errors.New("unsupported filter type") + } + + roots, leaves, err := idx.fetchUtreexoRoots(dbTx, h) + + if err != nil { + return nil, err + } + + // serialize the hashes of the utreexo roots hash + serializedUtreexo, err := blockchain.SerializeUtreexoRootsHash(leaves, roots) + if err != nil { + return nil, err + } + + return serializedUtreexo, err +} + +// FilterByBlockHash returns the serialized contents of a block's utreexo +// cfilter. +func (idx *UtreexoCFIndex) FilterByBlockHash(dbTx database.Tx, h *chainhash.Hash, + filterType wire.FilterType) ([]byte, error) { + return idx.entryByBlockHash(dbTx, filterType, h) +} + +// NewCfIndex returns a new instance of an indexer that is used to create a +// mapping of the hashes of all blocks in the blockchain to their respective +// committed filters. +// +// It implements the Indexer interface which plugs into the IndexManager that +// in turn is used by the blockchain package. This allows the index to be +// seamlessly maintained along with the chain. +func NewUtreexoCfIndex(db database.DB, chainParams *chaincfg.Params, utreexoProofIndex *UtreexoProofIndex, + flatUtreexoProofIndex *FlatUtreexoProofIndex) *UtreexoCFIndex { + return &UtreexoCFIndex{db: db, chainParams: chainParams, utreexoProofIndex: utreexoProofIndex, + flatUtreexoProofIndex: flatUtreexoProofIndex} +} + +// DropCfIndex drops the CF index from the provided database if exists. +func DropUtreexoCfIndex(db database.DB, interrupt <-chan struct{}) error { + return dropIndex(db, utreexoCFIndexParentBucketKey, utreexoCFIndexName, interrupt) +} + +// CfIndexInitialized returns true if the cfindex has been created previously. +func UtreexoCfIndexInitialized(db database.DB) bool { + var exists bool + db.View(func(dbTx database.Tx) error { + bucket := dbTx.Metadata().Bucket(utreexoCFIndexParentBucketKey) + exists = bucket != nil + return nil + }) + + return exists +} From a3e6852612c97bc4028037ef5d6c9e17857f7201 Mon Sep 17 00:00:00 2001 From: alainjr10 Date: Tue, 23 Jul 2024 16:42:27 +0100 Subject: [PATCH 04/28] add func to serialize utreexo roots hash --- blockchain/chainio.go | 26 ++++++++++++++++++++++++++ 1 file changed, 26 insertions(+) diff --git a/blockchain/chainio.go b/blockchain/chainio.go index fdcbb4c3..3a86fcd6 100644 --- a/blockchain/chainio.go +++ b/blockchain/chainio.go @@ -1080,6 +1080,32 @@ func SerializeUtreexoRoots(numLeaves uint64, roots []utreexo.Hash) ([]byte, erro return w.Bytes(), nil } +// SerializeUtreexoRootsHash serializes the numLeaves and the roots into a byte slice. +// it takes in a slice of chainhash.Hash instead of utreexo.Hash. chainhash.Hash is the hashed +// value of the utreexo.Hash. +func SerializeUtreexoRootsHash(numLeaves uint64, roots []*chainhash.Hash) ([]byte, error) { + // 8 byte NumLeaves + (32 byte roots * len(roots)) + w := bytes.NewBuffer(make([]byte, 0, 8+(len(roots)*chainhash.HashSize))) + + // Write the NumLeaves first. + var buf [8]byte + byteOrder.PutUint64(buf[:], numLeaves) + _, err := w.Write(buf[:]) + if err != nil { + return nil, err + } + + // Then write the roots. + for _, root := range roots { + _, err = w.Write(root[:]) + if err != nil { + return nil, err + } + } + + return w.Bytes(), nil +} + // DeserializeUtreexoRoots deserializes the provided byte slice into numLeaves and roots. func DeserializeUtreexoRoots(serializedUView []byte) (uint64, []utreexo.Hash, error) { totalLen := len(serializedUView) From 001749f8844da233c4dab0dea8211e0df56c497d Mon Sep 17 00:00:00 2001 From: alainjr10 Date: Tue, 23 Jul 2024 16:44:35 +0100 Subject: [PATCH 05/28] handle new getcfilters and getcfheaders message type --- server.go | 416 +++++++++++++++++++++++++++++++++++++++--------------- 1 file changed, 299 insertions(+), 117 deletions(-) diff --git a/server.go b/server.go index 74fcc96f..aeaf88bc 100644 --- a/server.go +++ b/server.go @@ -252,6 +252,7 @@ type server struct { txIndex *indexers.TxIndex addrIndex *indexers.AddrIndex cfIndex *indexers.CfIndex + utreexoCfIndex *indexers.UtreexoCFIndex ttlIndex *indexers.TTLIndex utreexoProofIndex *indexers.UtreexoProofIndex flatUtreexoProofIndex *indexers.FlatUtreexoProofIndex @@ -863,170 +864,345 @@ func (sp *serverPeer) OnGetCFilters(_ *peer.Peer, msg *wire.MsgGetCFilters) { return } + var hashes []chainhash.Hash + var hashPtrs []*chainhash.Hash + // if the filter type is supported, we initialize variables to avoid duplicate code + if msg.FilterType == wire.GCSFilterRegular || msg.FilterType == wire.UtreexoCFilter { + var err error + // get the block hashes included in the getcfilters message + hashes, err = sp.server.chain.HeightToHashRange( + int32(msg.StartHeight), &msg.StopHash, wire.MaxGetCFiltersReqRange, + ) + if err != nil { + peerLog.Debugf("Invalid getcfilters request: %v", err) + return + } + + // Create []*chainhash.Hash from []chainhash.Hash to pass to + // FiltersByBlockHashes. + hashPtrs = make([]*chainhash.Hash, len(hashes)) + for i := range hashes { + hashPtrs[i] = &hashes[i] + } + } + // We'll also ensure that the remote party is requesting a set of // filters that we actually currently maintain. switch msg.FilterType { case wire.GCSFilterRegular: - break + filters, err := sp.server.cfIndex.FiltersByBlockHashes( + hashPtrs, msg.FilterType, + ) + if err != nil { + peerLog.Errorf("Error retrieving cfilters: %v", err) + return + } + + for i, filterBytes := range filters { + if len(filterBytes) == 0 { + peerLog.Warnf("Could not obtain cfilter for %v", + hashes[i]) + return + } + + filterMsg := wire.NewMsgCFilter( + msg.FilterType, &hashes[i], filterBytes, + ) + sp.QueueMessage(filterMsg, nil) + } + + case wire.UtreexoCFilter: + for i, blockHash := range hashPtrs { + var serializedUtreexo []byte + + leaves, roots, err := sp.getUtreexoRoots(blockHash) + if err != nil { + return + } + + // serialize the hashes of the utreexo roots hash + serializedUtreexo, err = blockchain.SerializeUtreexoRootsHash(leaves, roots) + if err != nil { + peerLog.Errorf("error serializing utreexoc filter: %v", err) + return + } + + if len(serializedUtreexo) == 0 { + peerLog.Warnf("Could not obtain utreexocfilter for %v", + hashes[i]) + return + } + + filterMsg := wire.NewMsgCFilter( + msg.FilterType, &hashes[i], serializedUtreexo, + ) + sp.QueueMessage(filterMsg, nil) + } default: peerLog.Debug("Filter request for unknown filter: %v", msg.FilterType) return } +} - hashes, err := sp.server.chain.HeightToHashRange( - int32(msg.StartHeight), &msg.StopHash, wire.MaxGetCFiltersReqRange, - ) - if err != nil { - peerLog.Debugf("Invalid getcfilters request: %v", err) +// OnGetCFHeaders is invoked when a peer receives a getcfheader bitcoin message. +func (sp *serverPeer) OnGetCFHeaders(_ *peer.Peer, msg *wire.MsgGetCFHeaders) { + // Ignore getcfilterheader requests if not in sync. + if !sp.server.syncManager.IsCurrent() { return } - // Create []*chainhash.Hash from []chainhash.Hash to pass to - // FiltersByBlockHashes. - hashPtrs := make([]*chainhash.Hash, len(hashes)) - for i := range hashes { - hashPtrs[i] = &hashes[i] - } + var startHeight int32 + var maxResults int + var hashList []chainhash.Hash + var hashPtrs []*chainhash.Hash + // if the filter type is supported, we initialize variables to avoid duplicate code + if msg.FilterType == wire.GCSFilterRegular || msg.FilterType == wire.UtreexoCFilter { - filters, err := sp.server.cfIndex.FiltersByBlockHashes( - hashPtrs, msg.FilterType, - ) - if err != nil { - peerLog.Errorf("Error retrieving cfilters: %v", err) - return - } + startHeight = int32(msg.StartHeight) + maxResults = wire.MaxCFHeadersPerMsg - for i, filterBytes := range filters { - if len(filterBytes) == 0 { - peerLog.Warnf("Could not obtain cfilter for %v", - hashes[i]) - return + // If StartHeight is positive, fetch the predecessor block hash so we + // can populate the PrevFilterHeader field. + if msg.StartHeight > 0 { + startHeight-- + maxResults++ } - filterMsg := wire.NewMsgCFilter( - msg.FilterType, &hashes[i], filterBytes, + // Fetch the hashes from the block index. + var err error + hashList, err = sp.server.chain.HeightToHashRange( + startHeight, &msg.StopHash, maxResults, ) - sp.QueueMessage(filterMsg, nil) - } -} + if err != nil { + peerLog.Debugf("Invalid getcfheaders request: %v", err) + } -// OnGetCFHeaders is invoked when a peer receives a getcfheader bitcoin message. -func (sp *serverPeer) OnGetCFHeaders(_ *peer.Peer, msg *wire.MsgGetCFHeaders) { - // Ignore getcfilterheader requests if not in sync. - if !sp.server.syncManager.IsCurrent() { - return + // This is possible if StartHeight is one greater that the height of + // StopHash, and we pull a valid range of hashes including the previous + // filter header. + if len(hashList) == 0 || (msg.StartHeight > 0 && len(hashList) == 1) { + peerLog.Debug("No results for getcfheaders request") + return + } + + // Create []*chainhash.Hash from []chainhash.Hash to pass to + // FilterHeadersByBlockHashes. + hashPtrs = make([]*chainhash.Hash, len(hashList)) + for i := range hashList { + hashPtrs[i] = &hashList[i] + } } // We'll also ensure that the remote party is requesting a set of // headers for filters that we actually currently maintain. switch msg.FilterType { case wire.GCSFilterRegular: - break - default: - peerLog.Debug("Filter request for unknown headers for "+ - "filter: %v", msg.FilterType) - return - } + // Fetch the raw filter hash bytes from the database for all blocks. + filterHashes, err := sp.server.cfIndex.FilterHashesByBlockHashes( + hashPtrs, msg.FilterType, + ) + if err != nil { + peerLog.Errorf("Error retrieving cfilter hashes: %v", err) + return + } - startHeight := int32(msg.StartHeight) - maxResults := wire.MaxCFHeadersPerMsg + // Generate cfheaders message and send it. + headersMsg := wire.NewMsgCFHeaders() - // If StartHeight is positive, fetch the predecessor block hash so we - // can populate the PrevFilterHeader field. - if msg.StartHeight > 0 { - startHeight-- - maxResults++ - } + // Populate the PrevFilterHeader field. + if msg.StartHeight > 0 { + prevBlockHash := &hashList[0] - // Fetch the hashes from the block index. - hashList, err := sp.server.chain.HeightToHashRange( - startHeight, &msg.StopHash, maxResults, - ) - if err != nil { - peerLog.Debugf("Invalid getcfheaders request: %v", err) - } - - // This is possible if StartHeight is one greater that the height of - // StopHash, and we pull a valid range of hashes including the previous - // filter header. - if len(hashList) == 0 || (msg.StartHeight > 0 && len(hashList) == 1) { - peerLog.Debug("No results for getcfheaders request") - return - } + // Fetch the raw committed filter header bytes from the + // database. + headerBytes, err := sp.server.cfIndex.FilterHeaderByBlockHash( + prevBlockHash, msg.FilterType) + if err != nil { + peerLog.Errorf("Error retrieving CF header: %v", err) + return + } + if len(headerBytes) == 0 { + peerLog.Warnf("Could not obtain CF header for %v", prevBlockHash) + return + } - // Create []*chainhash.Hash from []chainhash.Hash to pass to - // FilterHeadersByBlockHashes. - hashPtrs := make([]*chainhash.Hash, len(hashList)) - for i := range hashList { - hashPtrs[i] = &hashList[i] - } + // Deserialize the hash into PrevFilterHeader. + err = headersMsg.PrevFilterHeader.SetBytes(headerBytes) + if err != nil { + peerLog.Warnf("Committed filter header deserialize "+ + "failed: %v", err) + return + } - // Fetch the raw filter hash bytes from the database for all blocks. - filterHashes, err := sp.server.cfIndex.FilterHashesByBlockHashes( - hashPtrs, msg.FilterType, - ) - if err != nil { - peerLog.Errorf("Error retrieving cfilter hashes: %v", err) - return - } + hashList = hashList[1:] + filterHashes = filterHashes[1:] + } - // Generate cfheaders message and send it. - headersMsg := wire.NewMsgCFHeaders() + // Populate HeaderHashes. + for i, hashBytes := range filterHashes { + if len(hashBytes) == 0 { + peerLog.Warnf("Could not obtain CF hash for %v", hashList[i]) + return + } - // Populate the PrevFilterHeader field. - if msg.StartHeight > 0 { - prevBlockHash := &hashList[0] + // Deserialize the hash. + filterHash, err := chainhash.NewHash(hashBytes) + if err != nil { + peerLog.Warnf("Committed filter hash deserialize "+ + "failed: %v", err) + return + } - // Fetch the raw committed filter header bytes from the - // database. - headerBytes, err := sp.server.cfIndex.FilterHeaderByBlockHash( - prevBlockHash, msg.FilterType) - if err != nil { - peerLog.Errorf("Error retrieving CF header: %v", err) - return + headersMsg.AddCFHash(filterHash) } - if len(headerBytes) == 0 { - peerLog.Warnf("Could not obtain CF header for %v", prevBlockHash) - return + + headersMsg.FilterType = msg.FilterType + headersMsg.StopHash = msg.StopHash + + sp.QueueMessage(headersMsg, nil) + + // handle custom utreexocfilter message + case wire.UtreexoCFilter: + + // Generate cfheaders message and send it. + headersMsg := wire.NewMsgCFHeaders() + + // Populate the PrevFilterHeader field. + if msg.StartHeight > 0 { + prevBlockHash := &hashList[0] + + // Fetch the raw committed filter header bytes from the + // database. + headerBytes, err := sp.server.cfIndex.FilterHeaderByBlockHash( + prevBlockHash, msg.FilterType) + if err != nil { + peerLog.Errorf("Error retrieving CF header: %v", err) + return + } + if len(headerBytes) == 0 { + peerLog.Warnf("Could not obtain CF header for %v", prevBlockHash) + return + } + + // Deserialize the hash into PrevFilterHeader. + err = headersMsg.PrevFilterHeader.SetBytes(headerBytes) + if err != nil { + peerLog.Warnf("Committed filter header deserialize "+ + "failed: %v", err) + return + } } - // Deserialize the hash into PrevFilterHeader. - err = headersMsg.PrevFilterHeader.SetBytes(headerBytes) - if err != nil { - peerLog.Warnf("Committed filter header deserialize "+ - "failed: %v", err) - return + // fetch filter hashes and add to cf hashes field + for i, blockHash := range hashPtrs { + var serializedUtreexo []byte + // skip the first index as this index was added so as to enable us + // to get the previous filter's header + if i == 0 { + continue + } + + leaves, roots, err := sp.getUtreexoRoots(blockHash) + if err != nil { + return + } + + // serialize the hashes of the utreexo roots hash + serializedUtreexo, err = blockchain.SerializeUtreexoRootsHash(leaves, roots) + if err != nil { + peerLog.Errorf("error serializing utreexoc filter: %v", err) + return + } + + if len(serializedUtreexo) == 0 { + peerLog.Warnf("Could not obtain utreexocfilter for %v", + hashList[i]) + return + } + hashBytes := chainhash.DoubleHashB(serializedUtreexo) + + if len(hashBytes) == 0 { + peerLog.Warnf("Could not obtain CF hash for %v", hashList[i]) + return + } + + // Deserialize the hash. + filterHash, err := chainhash.NewHash(hashBytes) + if err != nil { + peerLog.Warnf("Committed filter hash deserialize "+ + "failed: %v", err) + return + } + + headersMsg.AddCFHash(filterHash) } + headersMsg.FilterType = msg.FilterType + headersMsg.StopHash = msg.StopHash - hashList = hashList[1:] - filterHashes = filterHashes[1:] + sp.QueueMessage(headersMsg, nil) + + default: + peerLog.Debug("Filter request for unknown headers for "+ + "filter: %v", msg.FilterType) + return } +} - // Populate HeaderHashes. - for i, hashBytes := range filterHashes { - if len(hashBytes) == 0 { - peerLog.Warnf("Could not obtain CF hash for %v", hashList[i]) - return - } +// getUtreexoRoots fetches utreexo roots from the appropriate locations, i.e fetches +// roots for CSN from a different location from utreexoviewpoint and flatfile +func (sp *serverPeer) getUtreexoRoots(blockHash *chainhash.Hash) (uint64, []*chainhash.Hash, error) { + + var leaves uint64 + var roots []*chainhash.Hash - // Deserialize the hash. - filterHash, err := chainhash.NewHash(hashBytes) + // For compact state nodes + if !cfg.NoUtreexo { + viewPoint, err := sp.server.chain.FetchUtreexoViewpoint(blockHash) if err != nil { - peerLog.Warnf("Committed filter hash deserialize "+ - "failed: %v", err) - return + peerLog.Errorf("could not obtain utreexo view: %v", err) + return 0, nil, err } - - headersMsg.AddCFHash(filterHash) + roots = viewPoint.GetRoots() + leaves = viewPoint.NumLeaves() } + // for bridge nodes + if sp.server.utreexoProofIndex != nil { + var uleaves uint64 + var uroots []*chainhash.Hash + var err error + err = sp.server.db.View(func(dbTx database.Tx) error { + uroots, uleaves, err = sp.server.utreexoProofIndex.FetchUtreexoState(dbTx, blockHash) + if err != nil { + return err + } - headersMsg.FilterType = msg.FilterType - headersMsg.StopHash = msg.StopHash - - sp.QueueMessage(headersMsg, nil) + return nil + }) + if err != nil { + peerLog.Errorf("error fetching utreexo view for blockhash %s: error: %v", blockHash, err) + return 0, nil, err + } + roots = uroots + leaves = uleaves + } else if sp.server.flatUtreexoProofIndex != nil { + height, err := sp.server.chain.BlockHeightByHash(blockHash) + if err != nil { + peerLog.Errorf("couldn't fetch the block height for blockhash %s from "+ + "the blockindex. Error: %v", blockHash, err) + return 0, nil, err + } + uroots, uleaves, err := sp.server.flatUtreexoProofIndex.FetchUtreexoState(height) + if err != nil { + peerLog.Errorf("error fetching utreexo view for blockhash: %s: error: %v", err) + return 0, nil, err + } + roots = uroots + leaves = uleaves + } + return leaves, roots, nil } // OnGetCFCheckpt is invoked when a peer receives a getcfcheckpt bitcoin message. @@ -3228,6 +3404,12 @@ func newServer(listenAddrs, agentBlacklist, agentWhitelist []string, s.cfIndex = indexers.NewCfIndex(db, chainParams) indexes = append(indexes, s.cfIndex) } + if cfg.UtreexoCFilters { + indxLog.Info("Utreexo C filter index enabled") + s.utreexoCfIndex = indexers.NewUtreexoCfIndex(db, chainParams, + s.utreexoProofIndex, s.flatUtreexoProofIndex) + indexes = append(indexes, s.utreexoCfIndex) + } if cfg.TTLIndex { indxLog.Info("TTL index is enabled") s.ttlIndex = indexers.NewTTLIndex(db, chainParams) From f509bc8f46a271c0d2e2065de2310d396b2268aa Mon Sep 17 00:00:00 2001 From: alainjr10 Date: Tue, 23 Jul 2024 16:45:26 +0100 Subject: [PATCH 06/28] add utreexocf command line params to config struct --- config.go | 2 ++ 1 file changed, 2 insertions(+) diff --git a/config.go b/config.go index 59f21754..0a54de97 100644 --- a/config.go +++ b/config.go @@ -204,9 +204,11 @@ type config struct { FlatUtreexoProofIndex bool `long:"flatutreexoproofindex" description:"Maintain a utreexo proof for all blocks in flat files"` UtreexoProofIndexMaxMemory int64 `long:"utreexoproofindexmaxmemory" description:"The maxmimum memory in mebibytes (MiB) that the utreexo proof indexes will use up. Passing in 0 will make the entire proof index stay on disk. Passing in a negative value will make the entire proof index stay in memory. Default of 250MiB."` CFilters bool `long:"cfilters" description:"Enable committed filtering (CF) support"` + UtreexoCFilters bool `long:"utreexocfilters" description:"Enable committed filtering (CF) support serving utreexo roots."` NoPeerBloomFilters bool `long:"nopeerbloomfilters" description:"Disable bloom filtering support"` DropAddrIndex bool `long:"dropaddrindex" description:"Deletes the address-based transaction index from the database on start up and then exits."` DropCfIndex bool `long:"dropcfindex" description:"Deletes the index used for committed filtering (CF) support from the database on start up and then exits."` + DropUtreexoCfIndex bool `long:"droputreexocfindex" description:"Deletes the index used for custom utreexo commited filter indexing support serving utreexo roots from the database on start up and then exits."` DropTxIndex bool `long:"droptxindex" description:"Deletes the hash-based transaction index from the database on start up and then exits."` DropTTLIndex bool `long:"dropttlindex" description:"Deletes the time to live index from the database on start up and then exits."` DropUtreexoProofIndex bool `long:"droputreexoproofindex" description:"Deletes the utreexo proof index from the database on start up and then exits."` From afae8ca8039e33ab220c03342be546d482dae56b Mon Sep 17 00:00:00 2001 From: alainjr10 Date: Tue, 23 Jul 2024 16:47:08 +0100 Subject: [PATCH 07/28] add checks and error messages on trying to disable utreexocf index --- utreexod.go | 33 +++++++++++++++++++++++++++++++++ 1 file changed, 33 insertions(+) diff --git a/utreexod.go b/utreexod.go index 427e408a..0da65281 100644 --- a/utreexod.go +++ b/utreexod.go @@ -94,6 +94,14 @@ func pruneChecks(db database.DB) error { "and sync from the beginning to enable the desired index. You may "+ "start the node up without the --cfilters flag", cfg.DataDir) } + // If we've previously been pruned and the utreexocfindex isn't present, it means that the + // user wants to enable the utreexocfindex after the node has already synced up while being pruned. + if beenPruned && !indexers.UtreexoCfIndexInitialized(db) && cfg.UtreexoCFilters { + return fmt.Errorf("utreeco cfilters cannot be enabled as the node has been "+ + "previously pruned. You must delete the files in the datadir: \"%s\" "+ + "and sync from the beginning to enable the desired index. You may "+ + "start the node up without the --utreexocfilters flag", cfg.DataDir) + } // If the user wants to disable the cfindex and is pruned or has enabled pruning, force // the user to either drop the cfindex manually or restart the node without the --cfilters @@ -114,6 +122,25 @@ func pruneChecks(db database.DB) error { "To keep the compact filters, restart the node with the --cfilters "+ "flag", prunedStr) } + // If the user wants to disable the utreexocfindex and is pruned or has enabled pruning, force + // the user to either drop the utreexocfindex manually or restart the node without the + // --utreexocfilters flag. + if (beenPruned || cfg.Prune != 0) && indexers.UtreexoCfIndexInitialized(db) && !cfg.UtreexoCFilters { + var prunedStr string + if beenPruned { + prunedStr = "has been previously pruned" + } else { + prunedStr = fmt.Sprintf("was started with prune flag (--prune=%d)", cfg.Prune) + } + return fmt.Errorf("--utreexocfilters flag was not given but the utreexo cfilters have "+ + "previously been enabled on this node and the index data currently "+ + "exists in the database. The node %s and "+ + "the database would be left in an inconsistent state if the utreexo c "+ + "filters don't get indexed now. To disable utreeco cfilters, please drop the "+ + "index completely with the --droputreexocfindex flag and restart the node. "+ + "To keep the compact filters, restart the node with the --utreexocfilters "+ + "flag", prunedStr) + } // If the user wants to disable the utreexo proof index and is pruned or has enabled pruning, // force the user to either drop the utreexo proof index manually or restart the node without // the --utreexoproofindex flag. @@ -287,6 +314,12 @@ func btcdMain(serverChan chan<- *server) error { return nil } + if cfg.DropUtreexoCfIndex { + if err := indexers.DropUtreexoCfIndex(db, interrupt); err != nil { + btcdLog.Errorf("%v", err) + return err + } + } if cfg.DropTTLIndex { if err := indexers.DropTTLIndex(db, interrupt); err != nil { btcdLog.Errorf("%v", err) From 525420803f3113dfd91f53c78f886f27fdb5ff2a Mon Sep 17 00:00:00 2001 From: alainjr10 Date: Tue, 23 Jul 2024 16:48:23 +0100 Subject: [PATCH 08/28] update implementeation of getcfilter handler --- rpcserver.go | 18 ++++++++++++++++-- 1 file changed, 16 insertions(+), 2 deletions(-) diff --git a/rpcserver.go b/rpcserver.go index 14b0cea3..a3bc1163 100644 --- a/rpcserver.go +++ b/rpcserver.go @@ -2381,7 +2381,7 @@ func handleGetChainTips(s *rpcServer, cmd interface{}, closeChan <-chan struct{} // handleGetCFilter implements the getcfilter command. func handleGetCFilter(s *rpcServer, cmd interface{}, closeChan <-chan struct{}) (interface{}, error) { - if s.cfg.CfIndex == nil { + if s.cfg.CfIndex == nil || s.cfg.UtreexoCfIndex != nil { return nil, &btcjson.RPCError{ Code: btcjson.ErrRPCNoCFIndex, Message: "The CF index must be enabled for this command", @@ -2394,7 +2394,20 @@ func handleGetCFilter(s *rpcServer, cmd interface{}, closeChan <-chan struct{}) return nil, rpcDecodeHexError(c.Hash) } - filterBytes, err := s.cfg.CfIndex.FilterByBlockHash(hash, c.FilterType) + var filterBytes []byte + if c.FilterType == wire.UtreexoCFilter { + err = s.cfg.DB.View(func(dbTx database.Tx) error { + var err error + filterBytes, err = s.cfg.UtreexoCfIndex.FilterByBlockHash(dbTx, hash, c.FilterType) + return err + }) + if err != nil { + return nil, rpcNoTxInfoError(hash) + } + } else { + filterBytes, err = s.cfg.CfIndex.FilterByBlockHash(hash, c.FilterType) + } + if err != nil { rpcsLog.Debugf("Could not find committed filter for %v: %v", hash, err) @@ -5649,6 +5662,7 @@ type rpcserverConfig struct { TxIndex *indexers.TxIndex AddrIndex *indexers.AddrIndex CfIndex *indexers.CfIndex + UtreexoCfIndex *indexers.UtreexoCFIndex TTLIndex *indexers.TTLIndex UtreexoProofIndex *indexers.UtreexoProofIndex FlatUtreexoProofIndex *indexers.FlatUtreexoProofIndex From f70880611aab3672ae2844bf35fac280cb208519 Mon Sep 17 00:00:00 2001 From: alainjr10 Date: Thu, 25 Jul 2024 10:28:55 +0100 Subject: [PATCH 09/28] fix: fix node panic on start --- blockchain/indexers/utreexocfindex.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/blockchain/indexers/utreexocfindex.go b/blockchain/indexers/utreexocfindex.go index c0ac2910..2de723b4 100644 --- a/blockchain/indexers/utreexocfindex.go +++ b/blockchain/indexers/utreexocfindex.go @@ -175,7 +175,7 @@ func (idx *UtreexoCFIndex) fetchUtreexoRoots(dbTx database.Tx, var roots []*chainhash.Hash // For compact state nodes - if idx.chain.IsUtreexoViewActive() && idx.chain.IsAssumeUtreexo() { + if idx.chain.IsUtreexoViewActive() { viewPoint, err := idx.chain.FetchUtreexoViewpoint(blockHash) if err != nil { return nil, 0, err From e7b1d55a538f994efda63a66f5ab7676cedfb4c0 Mon Sep 17 00:00:00 2001 From: alainjr10 Date: Sun, 4 Aug 2024 16:09:03 +0100 Subject: [PATCH 10/28] add function to filter headers by blockhashes --- blockchain/indexers/utreexocfindex.go | 38 +++++++++++++++++++++++++++ 1 file changed, 38 insertions(+) diff --git a/blockchain/indexers/utreexocfindex.go b/blockchain/indexers/utreexocfindex.go index 2de723b4..4f9ee652 100644 --- a/blockchain/indexers/utreexocfindex.go +++ b/blockchain/indexers/utreexocfindex.go @@ -34,6 +34,9 @@ var ( // dbFetchFilterIdxEntry retrieves a data blob from the filter index database. // An entry's absence is not considered an error. +// Right now, the value of 'key' will always be the key to the utreexocfheader key +// as we don't need to fetch filters or filter hashes from the main bucket, as those +// are not stored func dbFetchUtreexoCFilterIdxEntry(dbTx database.Tx, key []byte, h *chainhash.Hash) ([]byte, error) { idx := dbTx.Metadata().Bucket(utreexoCFIndexParentBucketKey).Bucket(key) return idx.Get(h[:]), nil @@ -263,6 +266,34 @@ func (idx *UtreexoCFIndex) entryByBlockHash(dbTx database.Tx, return serializedUtreexo, err } +// entriesByBlockHashes batch fetches a filter index entry of a particular type +// (eg. filter, filter header, etc) for a filter type and slice of block hashes. +func (idx *UtreexoCFIndex) entriesByBlockHashes(filterTypeKeys [][]byte, + filterType wire.FilterType, blockHashes []*chainhash.Hash) ([][]byte, error) { + + if uint8(filterType) != uint8(wire.UtreexoCFilter) { + return nil, errors.New("unsupported filter type") + } + + // we use filterTypeKeys[0] as the key for the utreexo cfilter since for now, + // there is only one type of utreexo cfilter and the filterTypeKeys always + // returns the filterheaderkeys, which has just the one value + key := filterTypeKeys[0] + + entries := make([][]byte, 0, len(blockHashes)) + err := idx.db.View(func(dbTx database.Tx) error { + for _, blockHash := range blockHashes { + entry, err := dbFetchUtreexoCFilterIdxEntry(dbTx, key, blockHash) + if err != nil { + return err + } + entries = append(entries, entry) + } + return nil + }) + return entries, err +} + // FilterByBlockHash returns the serialized contents of a block's utreexo // cfilter. func (idx *UtreexoCFIndex) FilterByBlockHash(dbTx database.Tx, h *chainhash.Hash, @@ -270,6 +301,13 @@ func (idx *UtreexoCFIndex) FilterByBlockHash(dbTx database.Tx, h *chainhash.Hash return idx.entryByBlockHash(dbTx, filterType, h) } +// FilterHeadersByBlockHashes returns the serialized contents of a block's +// utreexo commited filter header for a set of blocks by hash. +func (idx *UtreexoCFIndex) FilterHeadersByBlockHashes(blockHashes []*chainhash.Hash, + filterType wire.FilterType) ([][]byte, error) { + return idx.entriesByBlockHashes(utreexoCfHeaderKeys, filterType, blockHashes) +} + // NewCfIndex returns a new instance of an indexer that is used to create a // mapping of the hashes of all blocks in the blockchain to their respective // committed filters. From 04b6da9e7ceb33b9571acc72fa90187f537a89c5 Mon Sep 17 00:00:00 2001 From: alainjr10 Date: Sun, 4 Aug 2024 16:09:34 +0100 Subject: [PATCH 11/28] handle on getcfcheckpt in server.go --- server.go | 21 ++++++++++++++++----- 1 file changed, 16 insertions(+), 5 deletions(-) diff --git a/server.go b/server.go index aeaf88bc..39af6574 100644 --- a/server.go +++ b/server.go @@ -1216,6 +1216,7 @@ func (sp *serverPeer) OnGetCFCheckpt(_ *peer.Peer, msg *wire.MsgGetCFCheckpt) { // checkpoints for filters that we actually currently maintain. switch msg.FilterType { case wire.GCSFilterRegular: + case wire.UtreexoCFilter: break default: @@ -1311,11 +1312,21 @@ func (sp *serverPeer) OnGetCFCheckpt(_ *peer.Peer, msg *wire.MsgGetCFCheckpt) { for i := forkIdx; i < len(blockHashes); i++ { blockHashPtrs = append(blockHashPtrs, &blockHashes[i]) } - filterHeaders, err := sp.server.cfIndex.FilterHeadersByBlockHashes( - blockHashPtrs, msg.FilterType, - ) - if err != nil { - peerLog.Errorf("Error retrieving cfilter headers: %v", err) + var filterHeaders [][]byte + var filterHeaderErr error + + if msg.FilterType == wire.GCSFilterRegular { + filterHeaders, filterHeaderErr = sp.server.cfIndex.FilterHeadersByBlockHashes( + blockHashPtrs, msg.FilterType, + ) + } else if msg.FilterType == wire.UtreexoCFilter { + filterHeaders, filterHeaderErr = sp.server.utreexoCfIndex.FilterHeadersByBlockHashes( + blockHashPtrs, msg.FilterType, + ) + } + + if filterHeaderErr != nil { + peerLog.Errorf("Error retrieving cfilter headers: %v", filterHeaderErr) return } From 5573d3434902dc5bbf965849dd2968e59ce2dbac Mon Sep 17 00:00:00 2001 From: alainjr10 Date: Sun, 4 Aug 2024 22:36:31 +0100 Subject: [PATCH 12/28] add filterheaderby block hash and improve other util funcs --- blockchain/indexers/utreexocfindex.go | 58 +++++++++++++++++++++------ 1 file changed, 45 insertions(+), 13 deletions(-) diff --git a/blockchain/indexers/utreexocfindex.go b/blockchain/indexers/utreexocfindex.go index 4f9ee652..e1a0bdbe 100644 --- a/blockchain/indexers/utreexocfindex.go +++ b/blockchain/indexers/utreexocfindex.go @@ -244,26 +244,48 @@ func (idx *UtreexoCFIndex) PruneBlock(dbTx database.Tx, blockHash *chainhash.Has // entryByBlockHash fetches a filter index entry of a particular type // (eg. filter, filter header, etc) for a filter type and block hash. -func (idx *UtreexoCFIndex) entryByBlockHash(dbTx database.Tx, +func (idx *UtreexoCFIndex) entryByBlockHash(filterTypeKeys [][]byte, filterType wire.FilterType, h *chainhash.Hash) ([]byte, error) { if uint8(filterType) != uint8(wire.UtreexoCFilter) { return nil, errors.New("unsupported filter type") } - roots, leaves, err := idx.fetchUtreexoRoots(dbTx, h) + // if the filtertype keys is empty, then we are fetching the filter data itself + // if not, we are fetching the filter header + if len(filterTypeKeys) == 0 { + var serializedUtreexo []byte + err := idx.db.View(func(dbTx database.Tx) error { + var err error + roots, leaves, err := idx.fetchUtreexoRoots(dbTx, h) - if err != nil { - return nil, err - } + if err != nil { + return err + } + // serialize the hashes of the utreexo roots hash + serializedUtreexo, err = blockchain.SerializeUtreexoRootsHash(leaves, roots) + return err + }) - // serialize the hashes of the utreexo roots hash - serializedUtreexo, err := blockchain.SerializeUtreexoRootsHash(leaves, roots) - if err != nil { - return nil, err - } + // serialize the hashes of the utreexo roots hash + // serializedUtreexo, err := blockchain.SerializeUtreexoRootsHash(leaves, roots) + if err != nil { + return nil, err + } - return serializedUtreexo, err + return serializedUtreexo, err + } else { + // using filtertypekeys[0] as there is only one entry in the filterTypeKeys (utreexoCfHeaderKeys) + key := filterTypeKeys[0] + + var entry []byte + err := idx.db.View(func(dbTx database.Tx) error { + var err error + entry, err = dbFetchUtreexoCFilterIdxEntry(dbTx, key, h) + return err + }) + return entry, err + } } // entriesByBlockHashes batch fetches a filter index entry of a particular type @@ -296,9 +318,19 @@ func (idx *UtreexoCFIndex) entriesByBlockHashes(filterTypeKeys [][]byte, // FilterByBlockHash returns the serialized contents of a block's utreexo // cfilter. -func (idx *UtreexoCFIndex) FilterByBlockHash(dbTx database.Tx, h *chainhash.Hash, +func (idx *UtreexoCFIndex) FilterByBlockHash(h *chainhash.Hash, + filterType wire.FilterType) ([]byte, error) { + // we create an ampty variable of type [][]byte to pass to the entryByBlockHash + // in order to fetch the filter data itself + utreexoCfIndexKeys := [][]byte{} + return idx.entryByBlockHash(utreexoCfIndexKeys, filterType, h) +} + +// FilterHeaderByBlockHash returns the serialized contents of a block's utreexo +// committed filter header. +func (idx *UtreexoCFIndex) FilterHeaderByBlockHash(h *chainhash.Hash, filterType wire.FilterType) ([]byte, error) { - return idx.entryByBlockHash(dbTx, filterType, h) + return idx.entryByBlockHash(utreexoCfHeaderKeys, filterType, h) } // FilterHeadersByBlockHashes returns the serialized contents of a block's From 5383c216f774a423210641b3477d2ae56e25596d Mon Sep 17 00:00:00 2001 From: alainjr10 Date: Sun, 4 Aug 2024 22:37:29 +0100 Subject: [PATCH 13/28] fetch hilter headers from proper location ongetcfheaders based on filter type --- server.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/server.go b/server.go index 39af6574..7b46ad68 100644 --- a/server.go +++ b/server.go @@ -1076,7 +1076,7 @@ func (sp *serverPeer) OnGetCFHeaders(_ *peer.Peer, msg *wire.MsgGetCFHeaders) { // Fetch the raw committed filter header bytes from the // database. - headerBytes, err := sp.server.cfIndex.FilterHeaderByBlockHash( + headerBytes, err := sp.server.utreexoCfIndex.FilterHeaderByBlockHash( prevBlockHash, msg.FilterType) if err != nil { peerLog.Errorf("Error retrieving CF header: %v", err) From 593908650d300492d5b75a42db7059b2794be2db Mon Sep 17 00:00:00 2001 From: alainjr10 Date: Sun, 4 Aug 2024 22:38:30 +0100 Subject: [PATCH 14/28] update handlegetcfilterheader to handle both supported cfilters --- rpcserver.go | 21 ++++++++++----------- 1 file changed, 10 insertions(+), 11 deletions(-) diff --git a/rpcserver.go b/rpcserver.go index a3bc1163..c6c7a877 100644 --- a/rpcserver.go +++ b/rpcserver.go @@ -2396,14 +2396,7 @@ func handleGetCFilter(s *rpcServer, cmd interface{}, closeChan <-chan struct{}) var filterBytes []byte if c.FilterType == wire.UtreexoCFilter { - err = s.cfg.DB.View(func(dbTx database.Tx) error { - var err error - filterBytes, err = s.cfg.UtreexoCfIndex.FilterByBlockHash(dbTx, hash, c.FilterType) - return err - }) - if err != nil { - return nil, rpcNoTxInfoError(hash) - } + filterBytes, err = s.cfg.UtreexoCfIndex.FilterByBlockHash(hash, c.FilterType) } else { filterBytes, err = s.cfg.CfIndex.FilterByBlockHash(hash, c.FilterType) } @@ -2423,10 +2416,10 @@ func handleGetCFilter(s *rpcServer, cmd interface{}, closeChan <-chan struct{}) // handleGetCFilterHeader implements the getcfilterheader command. func handleGetCFilterHeader(s *rpcServer, cmd interface{}, closeChan <-chan struct{}) (interface{}, error) { - if s.cfg.CfIndex == nil { + if s.cfg.CfIndex == nil && s.cfg.UtreexoCfIndex != nil { return nil, &btcjson.RPCError{ Code: btcjson.ErrRPCNoCFIndex, - Message: "The CF index must be enabled for this command", + Message: "One of CF index or Utreexo CF index must be enabled for this command", } } @@ -2436,7 +2429,13 @@ func handleGetCFilterHeader(s *rpcServer, cmd interface{}, closeChan <-chan stru return nil, rpcDecodeHexError(c.Hash) } - headerBytes, err := s.cfg.CfIndex.FilterHeaderByBlockHash(hash, c.FilterType) + var headerBytes []byte + if c.FilterType == wire.GCSFilterRegular { + headerBytes, err = s.cfg.CfIndex.FilterHeaderByBlockHash(hash, c.FilterType) + } else if c.FilterType == wire.UtreexoCFilter { + headerBytes, err = s.cfg.UtreexoCfIndex.FilterHeaderByBlockHash(hash, c.FilterType) + + } if len(headerBytes) > 0 { rpcsLog.Debugf("Found header of committed filter for %v", hash) } else { From 6394bbced82eb895ef736436df23c47b8ba5cd8b Mon Sep 17 00:00:00 2001 From: alainjr10 Date: Mon, 5 Aug 2024 21:08:05 +0100 Subject: [PATCH 15/28] add comment for new utreexocfilter type --- wire/msgcfilter.go | 2 ++ 1 file changed, 2 insertions(+) diff --git a/wire/msgcfilter.go b/wire/msgcfilter.go index 1e33d6e2..43a87b86 100644 --- a/wire/msgcfilter.go +++ b/wire/msgcfilter.go @@ -17,6 +17,8 @@ type FilterType uint8 const ( // GCSFilterRegular is the regular filter type. GCSFilterRegular FilterType = iota + // UtreexoCFilter is a custom filter type for serving Utreexo roots. + // creates a new filter type with FilterType value of 1 UtreexoCFilter ) From e74b647598f1474fed1d75ae7dda8fec6e554a02 Mon Sep 17 00:00:00 2001 From: alainjr10 Date: Mon, 5 Aug 2024 21:13:10 +0100 Subject: [PATCH 16/28] adjust utreexoCFIndexName comment --- blockchain/indexers/utreexocfindex.go | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/blockchain/indexers/utreexocfindex.go b/blockchain/indexers/utreexocfindex.go index e1a0bdbe..1296f25d 100644 --- a/blockchain/indexers/utreexocfindex.go +++ b/blockchain/indexers/utreexocfindex.go @@ -12,9 +12,9 @@ import ( "github.com/utreexo/utreexod/wire" ) -// utreexoProofIndexName is the human-readable name for the index. +// utreexoCFIndexName is the human-readable name for the index. const ( - utreexoCFIndexName = "utreexo custom cfilter index" + utreexoCFIndexName = "utreexo custom commited filter index" ) // utreexocfilter is a custom commited filter which serves utreexo roots From 16de187028500ee98d0340849901f4b04f65fc7a Mon Sep 17 00:00:00 2001 From: alainjr10 Date: Mon, 5 Aug 2024 21:56:09 +0100 Subject: [PATCH 17/28] indexers: make more appropriate comments --- blockchain/indexers/utreexocfindex.go | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/blockchain/indexers/utreexocfindex.go b/blockchain/indexers/utreexocfindex.go index 1296f25d..8eac6a5c 100644 --- a/blockchain/indexers/utreexocfindex.go +++ b/blockchain/indexers/utreexocfindex.go @@ -340,9 +340,9 @@ func (idx *UtreexoCFIndex) FilterHeadersByBlockHashes(blockHashes []*chainhash.H return idx.entriesByBlockHashes(utreexoCfHeaderKeys, filterType, blockHashes) } -// NewCfIndex returns a new instance of an indexer that is used to create a +// NewUtreexoCfIndex returns a new instance of an indexer that is used to create a // mapping of the hashes of all blocks in the blockchain to their respective -// committed filters. +// utreexo committed filters. // // It implements the Indexer interface which plugs into the IndexManager that // in turn is used by the blockchain package. This allows the index to be @@ -353,12 +353,12 @@ func NewUtreexoCfIndex(db database.DB, chainParams *chaincfg.Params, utreexoProo flatUtreexoProofIndex: flatUtreexoProofIndex} } -// DropCfIndex drops the CF index from the provided database if exists. +// DropUtreexoCfIndex drops the utreexo CF index from the provided database if exists. func DropUtreexoCfIndex(db database.DB, interrupt <-chan struct{}) error { return dropIndex(db, utreexoCFIndexParentBucketKey, utreexoCFIndexName, interrupt) } -// CfIndexInitialized returns true if the cfindex has been created previously. +// UtreexoCfIndexInitialized returns true if the utreexocfindex has been created previously. func UtreexoCfIndexInitialized(db database.DB) bool { var exists bool db.View(func(dbTx database.Tx) error { From 068bbea59f439a9d0343f239501f389209585aeb Mon Sep 17 00:00:00 2001 From: alainjr10 Date: Tue, 6 Aug 2024 23:41:48 +0100 Subject: [PATCH 18/28] indexer: init chain in utreexocfindex init --- blockchain/indexers/utreexocfindex.go | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/blockchain/indexers/utreexocfindex.go b/blockchain/indexers/utreexocfindex.go index 8eac6a5c..37dacd52 100644 --- a/blockchain/indexers/utreexocfindex.go +++ b/blockchain/indexers/utreexocfindex.go @@ -75,7 +75,8 @@ func (idx *UtreexoCFIndex) NeedsInputs() bool { // Init initializes the utreexo cf index. This is part of the Indexer // interface. -func (idx *UtreexoCFIndex) Init(_ *blockchain.BlockChain) error { +func (idx *UtreexoCFIndex) Init(chain *blockchain.BlockChain) error { + idx.chain = chain return nil // Nothing to do. } From c6452a431c32740d30882430d2ba7ccd779d0f4f Mon Sep 17 00:00:00 2001 From: alainjr10 Date: Tue, 6 Aug 2024 23:42:41 +0100 Subject: [PATCH 19/28] utreexod: update order of append indexers --- server.go | 13 +++++++------ 1 file changed, 7 insertions(+), 6 deletions(-) diff --git a/server.go b/server.go index 7b46ad68..164652b5 100644 --- a/server.go +++ b/server.go @@ -3415,12 +3415,6 @@ func newServer(listenAddrs, agentBlacklist, agentWhitelist []string, s.cfIndex = indexers.NewCfIndex(db, chainParams) indexes = append(indexes, s.cfIndex) } - if cfg.UtreexoCFilters { - indxLog.Info("Utreexo C filter index enabled") - s.utreexoCfIndex = indexers.NewUtreexoCfIndex(db, chainParams, - s.utreexoProofIndex, s.flatUtreexoProofIndex) - indexes = append(indexes, s.utreexoCfIndex) - } if cfg.TTLIndex { indxLog.Info("TTL index is enabled") s.ttlIndex = indexers.NewTTLIndex(db, chainParams) @@ -3455,6 +3449,12 @@ func newServer(listenAddrs, agentBlacklist, agentWhitelist []string, } indexes = append(indexes, s.flatUtreexoProofIndex) } + if cfg.UtreexoCFilters { + indxLog.Info("Utreexo C filter index enabled") + s.utreexoCfIndex = indexers.NewUtreexoCfIndex(db, chainParams, + s.utreexoProofIndex, s.flatUtreexoProofIndex) + indexes = append(indexes, s.utreexoCfIndex) + } // Create an index manager if any of the optional indexes are enabled. var indexManager blockchain.IndexManager @@ -3801,6 +3801,7 @@ func newServer(listenAddrs, agentBlacklist, agentWhitelist []string, TxIndex: s.txIndex, AddrIndex: s.addrIndex, CfIndex: s.cfIndex, + UtreexoCfIndex: s.utreexoCfIndex, TTLIndex: s.ttlIndex, UtreexoProofIndex: s.utreexoProofIndex, FlatUtreexoProofIndex: s.flatUtreexoProofIndex, From 0e9a41af4a0688776ada4ea0d5ccc4901c0cb98b Mon Sep 17 00:00:00 2001 From: alainjr10 Date: Thu, 8 Aug 2024 18:42:12 +0100 Subject: [PATCH 20/28] indexer: return empty utreexo stump if root is nil --- blockchain/indexers/utreexoproofindex.go | 3 +++ 1 file changed, 3 insertions(+) diff --git a/blockchain/indexers/utreexoproofindex.go b/blockchain/indexers/utreexoproofindex.go index 27224899..7362b41b 100644 --- a/blockchain/indexers/utreexoproofindex.go +++ b/blockchain/indexers/utreexoproofindex.go @@ -648,6 +648,9 @@ func dbStoreUtreexoState(dbTx database.Tx, hash *chainhash.Hash, p utreexo.Utree func dbFetchUtreexoState(dbTx database.Tx, hash *chainhash.Hash) (utreexo.Stump, error) { stateBucket := dbTx.Metadata().Bucket(utreexoParentBucketKey).Bucket(utreexoStateKey) serialized := stateBucket.Get(hash[:]) + if serialized == nil { + return utreexo.Stump{}, nil + } numLeaves, roots, err := blockchain.DeserializeUtreexoRoots(serialized) if err != nil { From e84894da4fa507333b9541f3b1ada1bcb197cf4e Mon Sep 17 00:00:00 2001 From: alainjr10 Date: Mon, 12 Aug 2024 21:29:23 +0100 Subject: [PATCH 21/28] add missing return for absent utreexocfindex for pruned nodes --- utreexod.go | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/utreexod.go b/utreexod.go index 0da65281..8c49d2db 100644 --- a/utreexod.go +++ b/utreexod.go @@ -97,7 +97,7 @@ func pruneChecks(db database.DB) error { // If we've previously been pruned and the utreexocfindex isn't present, it means that the // user wants to enable the utreexocfindex after the node has already synced up while being pruned. if beenPruned && !indexers.UtreexoCfIndexInitialized(db) && cfg.UtreexoCFilters { - return fmt.Errorf("utreeco cfilters cannot be enabled as the node has been "+ + return fmt.Errorf("utreexo cfilters cannot be enabled as the node has been "+ "previously pruned. You must delete the files in the datadir: \"%s\" "+ "and sync from the beginning to enable the desired index. You may "+ "start the node up without the --utreexocfilters flag", cfg.DataDir) @@ -319,6 +319,7 @@ func btcdMain(serverChan chan<- *server) error { btcdLog.Errorf("%v", err) return err } + return nil } if cfg.DropTTLIndex { if err := indexers.DropTTLIndex(db, interrupt); err != nil { From 45853878f01641568f76f4e48cbb26dd74364cb8 Mon Sep 17 00:00:00 2001 From: alainjr10 Date: Mon, 12 Aug 2024 21:30:05 +0100 Subject: [PATCH 22/28] rpc: handle getutreexocfilter in different function --- rpcserver.go | 86 +++++++++++++++++++++++++++++++++++++++++++--------- 1 file changed, 71 insertions(+), 15 deletions(-) diff --git a/rpcserver.go b/rpcserver.go index c6c7a877..c3fa83e0 100644 --- a/rpcserver.go +++ b/rpcserver.go @@ -158,6 +158,8 @@ var rpcHandlersBeforeInit = map[string]commandHandler{ "getchaintips": handleGetChainTips, "getcfilter": handleGetCFilter, "getcfilterheader": handleGetCFilterHeader, + "getutreexocfilter": handleGetUtreexoCFilter, + "getutreexocfilterheader": handleGetUtreexoCFilterHeader, "getconnectioncount": handleGetConnectionCount, "getcurrentnet": handleGetCurrentNet, "getdifficulty": handleGetDifficulty, @@ -294,6 +296,8 @@ var rpcLimited = map[string]struct{}{ "getchaintips": {}, "getcfilter": {}, "getcfilterheader": {}, + "getutreexocfilter": {}, + "getutreexocfilterheader": {}, "getcurrentnet": {}, "getdifficulty": {}, "getheaders": {}, @@ -2381,7 +2385,7 @@ func handleGetChainTips(s *rpcServer, cmd interface{}, closeChan <-chan struct{} // handleGetCFilter implements the getcfilter command. func handleGetCFilter(s *rpcServer, cmd interface{}, closeChan <-chan struct{}) (interface{}, error) { - if s.cfg.CfIndex == nil || s.cfg.UtreexoCfIndex != nil { + if s.cfg.CfIndex == nil { return nil, &btcjson.RPCError{ Code: btcjson.ErrRPCNoCFIndex, Message: "The CF index must be enabled for this command", @@ -2394,12 +2398,7 @@ func handleGetCFilter(s *rpcServer, cmd interface{}, closeChan <-chan struct{}) return nil, rpcDecodeHexError(c.Hash) } - var filterBytes []byte - if c.FilterType == wire.UtreexoCFilter { - filterBytes, err = s.cfg.UtreexoCfIndex.FilterByBlockHash(hash, c.FilterType) - } else { - filterBytes, err = s.cfg.CfIndex.FilterByBlockHash(hash, c.FilterType) - } + filterBytes, err := s.cfg.CfIndex.FilterByBlockHash(hash, c.FilterType) if err != nil { rpcsLog.Debugf("Could not find committed filter for %v: %v", @@ -2416,10 +2415,10 @@ func handleGetCFilter(s *rpcServer, cmd interface{}, closeChan <-chan struct{}) // handleGetCFilterHeader implements the getcfilterheader command. func handleGetCFilterHeader(s *rpcServer, cmd interface{}, closeChan <-chan struct{}) (interface{}, error) { - if s.cfg.CfIndex == nil && s.cfg.UtreexoCfIndex != nil { + if s.cfg.CfIndex == nil { return nil, &btcjson.RPCError{ Code: btcjson.ErrRPCNoCFIndex, - Message: "One of CF index or Utreexo CF index must be enabled for this command", + Message: "The CF index must be enabled for this command", } } @@ -2429,13 +2428,8 @@ func handleGetCFilterHeader(s *rpcServer, cmd interface{}, closeChan <-chan stru return nil, rpcDecodeHexError(c.Hash) } - var headerBytes []byte - if c.FilterType == wire.GCSFilterRegular { - headerBytes, err = s.cfg.CfIndex.FilterHeaderByBlockHash(hash, c.FilterType) - } else if c.FilterType == wire.UtreexoCFilter { - headerBytes, err = s.cfg.UtreexoCfIndex.FilterHeaderByBlockHash(hash, c.FilterType) + headerBytes, err := s.cfg.CfIndex.FilterHeaderByBlockHash(hash, c.FilterType) - } if len(headerBytes) > 0 { rpcsLog.Debugf("Found header of committed filter for %v", hash) } else { @@ -2451,6 +2445,68 @@ func handleGetCFilterHeader(s *rpcServer, cmd interface{}, closeChan <-chan stru return hash.String(), nil } +// handleGetCFilter implements the getcfilter command. +func handleGetUtreexoCFilter(s *rpcServer, cmd interface{}, closeChan <-chan struct{}) (interface{}, error) { + if s.cfg.UtreexoCfIndex == nil { + return nil, &btcjson.RPCError{ + Code: btcjson.ErrRPCNoCFIndex, + Message: "The utreexo CF index must be enabled for this command", + } + } + + c := cmd.(*btcjson.GetCFilterCmd) + hash, err := chainhash.NewHashFromStr(c.Hash) + if err != nil { + return nil, rpcDecodeHexError(c.Hash) + } + + filterBytes, err := s.cfg.UtreexoCfIndex.FilterByBlockHash(hash, c.FilterType) + + if err != nil { + rpcsLog.Debugf("Could not find utreexo committed filter for %v: %v", + hash, err) + return nil, &btcjson.RPCError{ + Code: btcjson.ErrRPCBlockNotFound, + Message: "Block not found", + } + } + + rpcsLog.Debugf("Found utreexo committed filter for %v", hash) + return hex.EncodeToString(filterBytes), nil +} + +// handleGetCFilterHeader implements the getcfilterheader command. +func handleGetUtreexoCFilterHeader(s *rpcServer, cmd interface{}, closeChan <-chan struct{}) (interface{}, error) { + if s.cfg.UtreexoCfIndex == nil { + return nil, &btcjson.RPCError{ + Code: btcjson.ErrRPCNoCFIndex, + Message: "Utreexo CF index must be enabled for this command", + } + } + + c := cmd.(*btcjson.GetCFilterHeaderCmd) + hash, err := chainhash.NewHashFromStr(c.Hash) + if err != nil { + return nil, rpcDecodeHexError(c.Hash) + } + + headerBytes, err := s.cfg.UtreexoCfIndex.FilterHeaderByBlockHash(hash, c.FilterType) + + if len(headerBytes) > 0 { + rpcsLog.Debugf("Found header of utreexo committed filter for %v", hash) + } else { + rpcsLog.Debugf("Could not find header of utreexo committed filter for %v: %v", + hash, err) + return nil, &btcjson.RPCError{ + Code: btcjson.ErrRPCBlockNotFound, + Message: "Block not found", + } + } + + hash.SetBytes(headerBytes) + return hash.String(), nil +} + // handleGetConnectionCount implements the getconnectioncount command. func handleGetConnectionCount(s *rpcServer, cmd interface{}, closeChan <-chan struct{}) (interface{}, error) { return s.cfg.ConnMgr.ConnectedCount(), nil From 271fa628aec65a5ccef379e7f830bcdeadb1a299 Mon Sep 17 00:00:00 2001 From: alainjr10 Date: Tue, 20 Aug 2024 15:17:10 +0100 Subject: [PATCH 23/28] wire: add sfnodeutreexocf flag --- wire/protocol.go | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/wire/protocol.go b/wire/protocol.go index af73714a..9214a71c 100644 --- a/wire/protocol.go +++ b/wire/protocol.go @@ -87,6 +87,10 @@ const ( // filters (CFs). SFNodeCF + // SFNodeUtreexoCF is a flag used to indicate a peer supports utreexo + // committed filters (utreexo CFs). + SFNodeUtreexoCF + // SFNode2X is a flag used to indicate a peer is running the Segwit2X // software. SFNode2X @@ -115,6 +119,7 @@ var sfStrings = map[ServiceFlag]string{ SFNodeCF: "SFNodeCF", SFNode2X: "SFNode2X", SFNodeUtreexo: "SFNodeUtreexo", + SFNodeUtreexoCF: "SFNodeUtreexoCF", } // orderedSFStrings is an ordered list of service flags from highest to @@ -130,6 +135,7 @@ var orderedSFStrings = []ServiceFlag{ SFNodeCF, SFNode2X, SFNodeUtreexo, + SFNodeUtreexoCF, } // String returns the ServiceFlag in human-readable form. From d0a1962910e4ecd459f7f7f72557a274acae6309 Mon Sep 17 00:00:00 2001 From: alainjr10 Date: Tue, 20 Aug 2024 15:17:45 +0100 Subject: [PATCH 24/28] indexers: handle blocking dureing check for CSN --- blockchain/indexers/utreexocfindex.go | 15 +++++++++++---- 1 file changed, 11 insertions(+), 4 deletions(-) diff --git a/blockchain/indexers/utreexocfindex.go b/blockchain/indexers/utreexocfindex.go index 37dacd52..9e55768b 100644 --- a/blockchain/indexers/utreexocfindex.go +++ b/blockchain/indexers/utreexocfindex.go @@ -62,6 +62,8 @@ type UtreexoCFIndex struct { db database.DB chainParams *chaincfg.Params + noUtreexo bool + chain *blockchain.BlockChain utreexoProofIndex *UtreexoProofIndex @@ -179,16 +181,18 @@ func (idx *UtreexoCFIndex) fetchUtreexoRoots(dbTx database.Tx, var roots []*chainhash.Hash // For compact state nodes - if idx.chain.IsUtreexoViewActive() { + if !idx.noUtreexo { viewPoint, err := idx.chain.FetchUtreexoViewpoint(blockHash) if err != nil { return nil, 0, err } roots = viewPoint.GetRoots() leaves = viewPoint.NumLeaves() + return roots, leaves, nil } // for bridge nodes if idx.utreexoProofIndex != nil { + // log.Infof("Is utreexoProofIndex JABURAT %v", blockHash) roots, leaves, err := idx.utreexoProofIndex.FetchUtreexoState(dbTx, blockHash) if err != nil { return nil, 0, err @@ -206,7 +210,10 @@ func (idx *UtreexoCFIndex) fetchUtreexoRoots(dbTx database.Tx, return roots, leaves, nil } - return roots, leaves, nil + leaves = 0 + roots = nil + + return roots, leaves, errors.New("unsupported filter type") } // DisconnectBlock is invoked by the index manager when a block has been @@ -349,9 +356,9 @@ func (idx *UtreexoCFIndex) FilterHeadersByBlockHashes(blockHashes []*chainhash.H // in turn is used by the blockchain package. This allows the index to be // seamlessly maintained along with the chain. func NewUtreexoCfIndex(db database.DB, chainParams *chaincfg.Params, utreexoProofIndex *UtreexoProofIndex, - flatUtreexoProofIndex *FlatUtreexoProofIndex) *UtreexoCFIndex { + flatUtreexoProofIndex *FlatUtreexoProofIndex, noUtreexo bool) *UtreexoCFIndex { return &UtreexoCFIndex{db: db, chainParams: chainParams, utreexoProofIndex: utreexoProofIndex, - flatUtreexoProofIndex: flatUtreexoProofIndex} + flatUtreexoProofIndex: flatUtreexoProofIndex, noUtreexo: noUtreexo} } // DropUtreexoCfIndex drops the utreexo CF index from the provided database if exists. From 146a032117f5dd42874566047dcc8cc6571aba6b Mon Sep 17 00:00:00 2001 From: alainjr10 Date: Tue, 20 Aug 2024 15:18:50 +0100 Subject: [PATCH 25/28] server: add check for sfnodeutreexocf flag --- server.go | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/server.go b/server.go index 164652b5..7edf4d02 100644 --- a/server.go +++ b/server.go @@ -3339,6 +3339,10 @@ func newServer(listenAddrs, agentBlacklist, agentWhitelist []string, services |= wire.SFNodeUtreexo } + if cfg.UtreexoCFilters { + services |= wire.SFNodeUtreexoCF + } + amgr := addrmgr.New(cfg.DataDir, btcdLookup) var listeners []net.Listener @@ -3452,7 +3456,7 @@ func newServer(listenAddrs, agentBlacklist, agentWhitelist []string, if cfg.UtreexoCFilters { indxLog.Info("Utreexo C filter index enabled") s.utreexoCfIndex = indexers.NewUtreexoCfIndex(db, chainParams, - s.utreexoProofIndex, s.flatUtreexoProofIndex) + s.utreexoProofIndex, s.flatUtreexoProofIndex, cfg.NoUtreexo) indexes = append(indexes, s.utreexoCfIndex) } From 51df5baa1ba40221be7f77d8a1467ba2c711ad88 Mon Sep 17 00:00:00 2001 From: alainjr10 Date: Fri, 30 Aug 2024 09:32:54 +0100 Subject: [PATCH 26/28] indexers: fix utreexocfilters not performing ibd --- blockchain/indexers/utreexocfindex.go | 59 ++++++++++++--------------- 1 file changed, 25 insertions(+), 34 deletions(-) diff --git a/blockchain/indexers/utreexocfindex.go b/blockchain/indexers/utreexocfindex.go index 9e55768b..aad85e80 100644 --- a/blockchain/indexers/utreexocfindex.go +++ b/blockchain/indexers/utreexocfindex.go @@ -116,7 +116,7 @@ func (idx *UtreexoCFIndex) Create(dbTx database.Tx) error { } // storeUtreexoCFilter stores a given utreexocfilter header -func storeUtreexoCFHeader(dbTx database.Tx, block *btcutil.Block, filterData []byte, +func (idx *UtreexoCFIndex) storeUtreexoCFHeader(dbTx database.Tx, block *btcutil.Block, filterData []byte, filterType wire.FilterType) error { if filterType != wire.UtreexoCFilter { return errors.New("invalid filter type") @@ -132,7 +132,12 @@ func storeUtreexoCFHeader(dbTx database.Tx, block *btcutil.Block, filterData []b if ph.IsEqual(&zeroHash) { prevHeader = &zeroHash } else { - pfh, err := dbFetchUtreexoCFilterIdxEntry(dbTx, hkey, ph) + var pfh []byte + err := idx.db.View(func(dbTx database.Tx) error { + var err error + pfh, err = dbFetchUtreexoCFilterIdxEntry(dbTx, hkey, ph) + return err + }) if err != nil { return err } @@ -158,7 +163,7 @@ func (idx *UtreexoCFIndex) ConnectBlock(dbTx database.Tx, block *btcutil.Block, stxos []blockchain.SpentTxOut) error { blockHash := block.Hash() - roots, leaves, err := idx.fetchUtreexoRoots(dbTx, blockHash) + roots, leaves, err := idx.fetchUtreexoRoots(blockHash, block) if err != nil { return err @@ -170,15 +175,12 @@ func (idx *UtreexoCFIndex) ConnectBlock(dbTx database.Tx, block *btcutil.Block, return err } - return storeUtreexoCFHeader(dbTx, block, serializedUtreexo, wire.UtreexoCFilter) + return idx.storeUtreexoCFHeader(dbTx, block, serializedUtreexo, wire.UtreexoCFilter) } // fetches the utreexo roots for a given block hash -func (idx *UtreexoCFIndex) fetchUtreexoRoots(dbTx database.Tx, - blockHash *chainhash.Hash) ([]*chainhash.Hash, uint64, error) { - - var leaves uint64 - var roots []*chainhash.Hash +func (idx *UtreexoCFIndex) fetchUtreexoRoots(blockHash *chainhash.Hash, + block *btcutil.Block) ([]*chainhash.Hash, uint64, error) { // For compact state nodes if !idx.noUtreexo { @@ -186,34 +188,19 @@ func (idx *UtreexoCFIndex) fetchUtreexoRoots(dbTx database.Tx, if err != nil { return nil, 0, err } - roots = viewPoint.GetRoots() - leaves = viewPoint.NumLeaves() - return roots, leaves, nil - } - // for bridge nodes - if idx.utreexoProofIndex != nil { - // log.Infof("Is utreexoProofIndex JABURAT %v", blockHash) - roots, leaves, err := idx.utreexoProofIndex.FetchUtreexoState(dbTx, blockHash) - if err != nil { - return nil, 0, err - } + roots := viewPoint.GetRoots() + leaves := viewPoint.NumLeaves() return roots, leaves, nil + } else if idx.utreexoProofIndex != nil { + roots, numLeaves := idx.utreexoProofIndex.FetchCurrentUtreexoState() + return roots, numLeaves, nil + } else if idx.flatUtreexoProofIndex != nil { - height, err := idx.chain.BlockHeightByHash(blockHash) - if err != nil { - return nil, 0, err - } - roots, leaves, err := idx.flatUtreexoProofIndex.FetchUtreexoState(height) - if err != nil { - return nil, 0, err - } - return roots, leaves, nil + roots, numLeaves := idx.flatUtreexoProofIndex.FetchCurrentUtreexoState() + return roots, numLeaves, nil } - leaves = 0 - roots = nil - - return roots, leaves, errors.New("unsupported filter type") + return nil, 0, errors.New("unsupported filter type") } // DisconnectBlock is invoked by the index manager when a block has been @@ -265,7 +252,11 @@ func (idx *UtreexoCFIndex) entryByBlockHash(filterTypeKeys [][]byte, var serializedUtreexo []byte err := idx.db.View(func(dbTx database.Tx) error { var err error - roots, leaves, err := idx.fetchUtreexoRoots(dbTx, h) + block, errheight := idx.chain.BlockByHash(h) + if errheight != nil { + return errheight + } + roots, leaves, err := idx.fetchUtreexoRoots(h, block) if err != nil { return err From 5ba9fbff869834f6cf420404b8f12940a071a5ce Mon Sep 17 00:00:00 2001 From: alainjr10 Date: Sat, 31 Aug 2024 00:22:53 +0100 Subject: [PATCH 27/28] config: throw early error when noutreexo is passed along with utreexocfilter --- config.go | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/config.go b/config.go index 0a54de97..854a1614 100644 --- a/config.go +++ b/config.go @@ -1208,6 +1208,14 @@ func loadConfig() (*config, []string, error) { cfg.NoAssumeUtreexo = true } + if cfg.NoUtreexo && cfg.UtreexoCFilters && (!cfg.UtreexoProofIndex && !cfg.FlatUtreexoProofIndex) { + err := fmt.Errorf("%s: the --utreexocfilters can not be called with --noutreexo option when "+ + "neither of utreexoproofindex or flatutreexoproofindex options are enabled", funcName) + fmt.Fprintln(os.Stderr, err) + fmt.Fprintln(os.Stderr, usageMessage) + return nil, nil, err + } + // Specifying --noonion means the onion address dial function results in // an error. if cfg.NoOnion { From 5cb8c305b41a6b2b3c26e8bc767fc82567fbee7e Mon Sep 17 00:00:00 2001 From: alainjr10 Date: Sat, 31 Aug 2024 00:23:42 +0100 Subject: [PATCH 28/28] indexers: update fetchutreexoroots func to fetch proper roots for all scenarios --- blockchain/indexers/utreexocfindex.go | 34 +++++++++++++++++++++++---- 1 file changed, 29 insertions(+), 5 deletions(-) diff --git a/blockchain/indexers/utreexocfindex.go b/blockchain/indexers/utreexocfindex.go index aad85e80..8a2846d6 100644 --- a/blockchain/indexers/utreexocfindex.go +++ b/blockchain/indexers/utreexocfindex.go @@ -163,7 +163,7 @@ func (idx *UtreexoCFIndex) ConnectBlock(dbTx database.Tx, block *btcutil.Block, stxos []blockchain.SpentTxOut) error { blockHash := block.Hash() - roots, leaves, err := idx.fetchUtreexoRoots(blockHash, block) + roots, leaves, err := idx.fetchUtreexoRoots(dbTx, blockHash, block) if err != nil { return err @@ -179,7 +179,7 @@ func (idx *UtreexoCFIndex) ConnectBlock(dbTx database.Tx, block *btcutil.Block, } // fetches the utreexo roots for a given block hash -func (idx *UtreexoCFIndex) fetchUtreexoRoots(blockHash *chainhash.Hash, +func (idx *UtreexoCFIndex) fetchUtreexoRoots(dbTx database.Tx, blockHash *chainhash.Hash, block *btcutil.Block) ([]*chainhash.Hash, uint64, error) { // For compact state nodes @@ -192,11 +192,35 @@ func (idx *UtreexoCFIndex) fetchUtreexoRoots(blockHash *chainhash.Hash, leaves := viewPoint.NumLeaves() return roots, leaves, nil } else if idx.utreexoProofIndex != nil { - roots, numLeaves := idx.utreexoProofIndex.FetchCurrentUtreexoState() + var roots []*chainhash.Hash + var numLeaves uint64 + var err error + if block.Height() < idx.chain.BestSnapshot().Height { + log.Infof("block height is less than best snapshot") + roots, numLeaves, err = idx.utreexoProofIndex.FetchUtreexoState(dbTx, block.Hash()) + } else { + roots, numLeaves = idx.utreexoProofIndex.FetchCurrentUtreexoState() + } + if err != nil { + log.Errorf("Error fetching utreexo state at block %s: %v", block.Hash(), err) + return nil, 0, err + } return roots, numLeaves, nil } else if idx.flatUtreexoProofIndex != nil { - roots, numLeaves := idx.flatUtreexoProofIndex.FetchCurrentUtreexoState() + var roots []*chainhash.Hash + var numLeaves uint64 + var err error + if block.Height() < idx.chain.BestSnapshot().Height { + log.Infof("block height is less than best snapshot") + roots, numLeaves, err = idx.flatUtreexoProofIndex.FetchUtreexoState(block.Height()) + } else { + roots, numLeaves = idx.flatUtreexoProofIndex.FetchCurrentUtreexoState() + } + if err != nil { + log.Errorf("Error fetching utreexo state at block %s: %v", block.Hash(), err) + return nil, 0, err + } return roots, numLeaves, nil } @@ -256,7 +280,7 @@ func (idx *UtreexoCFIndex) entryByBlockHash(filterTypeKeys [][]byte, if errheight != nil { return errheight } - roots, leaves, err := idx.fetchUtreexoRoots(h, block) + roots, leaves, err := idx.fetchUtreexoRoots(dbTx, h, block) if err != nil { return err