sweep: add sweeper store
This commit adds a store for the sweeper. The sweeper needs minimal persistent data to be able to recognize its own sweeps.
This commit is contained in:
parent
a2dcca2b08
commit
1f0656559e
247
sweep/store.go
Normal file
247
sweep/store.go
Normal file
@ -0,0 +1,247 @@
|
||||
package sweep
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"encoding/binary"
|
||||
"errors"
|
||||
"fmt"
|
||||
|
||||
"github.com/btcsuite/btcd/chaincfg/chainhash"
|
||||
"github.com/btcsuite/btcd/wire"
|
||||
"github.com/coreos/bbolt"
|
||||
"github.com/lightningnetwork/lnd/channeldb"
|
||||
)
|
||||
|
||||
var (
|
||||
// lastTxBucketKey is the key that points to a bucket containing a
|
||||
// single item storing the last published tx.
|
||||
//
|
||||
// maps: lastTxKey -> serialized_tx
|
||||
lastTxBucketKey = []byte("sweeper-last-tx")
|
||||
|
||||
// lastTxKey is the fixed key under which the serialized tx is stored.
|
||||
lastTxKey = []byte("last-tx")
|
||||
|
||||
// txHashesBucketKey is the key that points to a bucket containing the
|
||||
// hashes of all sweep txes that were published successfully.
|
||||
//
|
||||
// maps: txHash -> empty slice
|
||||
txHashesBucketKey = []byte("sweeper-tx-hashes")
|
||||
|
||||
// utxnChainPrefix is the bucket prefix for nursery buckets.
|
||||
utxnChainPrefix = []byte("utxn")
|
||||
|
||||
// utxnHeightIndexKey is the sub bucket where the nursery stores the
|
||||
// height index.
|
||||
utxnHeightIndexKey = []byte("height-index")
|
||||
|
||||
// utxnFinalizedKndrTxnKey is a static key that can be used to locate
|
||||
// the nursery finalized kindergarten sweep txn.
|
||||
utxnFinalizedKndrTxnKey = []byte("finalized-kndr-txn")
|
||||
|
||||
byteOrder = binary.BigEndian
|
||||
)
|
||||
|
||||
// SweeperStore stores published txes.
|
||||
type SweeperStore interface {
|
||||
// IsOurTx determines whether a tx is published by us, based on its
|
||||
// hash.
|
||||
IsOurTx(hash chainhash.Hash) (bool, error)
|
||||
|
||||
// NotifyPublishTx signals that we are about to publish a tx.
|
||||
NotifyPublishTx(*wire.MsgTx) error
|
||||
|
||||
// GetLastPublishedTx returns the last tx that we called NotifyPublishTx
|
||||
// for.
|
||||
GetLastPublishedTx() (*wire.MsgTx, error)
|
||||
}
|
||||
|
||||
type sweeperStore struct {
|
||||
db *channeldb.DB
|
||||
}
|
||||
|
||||
// NewSweeperStore returns a new store instance.
|
||||
func NewSweeperStore(db *channeldb.DB, chainHash *chainhash.Hash) (
|
||||
SweeperStore, error) {
|
||||
|
||||
err := db.Update(func(tx *bbolt.Tx) error {
|
||||
_, err := tx.CreateBucketIfNotExists(
|
||||
lastTxBucketKey,
|
||||
)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
if tx.Bucket(txHashesBucketKey) != nil {
|
||||
return nil
|
||||
}
|
||||
|
||||
txHashesBucket, err := tx.CreateBucket(txHashesBucketKey)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
// Use non-existence of tx hashes bucket as a signal to migrate
|
||||
// nursery finalized txes.
|
||||
err = migrateTxHashes(tx, txHashesBucket, chainHash)
|
||||
|
||||
return err
|
||||
})
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
return &sweeperStore{
|
||||
db: db,
|
||||
}, nil
|
||||
}
|
||||
|
||||
// migrateTxHashes migrates nursery finalized txes to the tx hashes bucket. This
|
||||
// is not implemented as a database migration, to keep the downgrade path open.
|
||||
func migrateTxHashes(tx *bbolt.Tx, txHashesBucket *bbolt.Bucket,
|
||||
chainHash *chainhash.Hash) error {
|
||||
|
||||
log.Infof("Migrating UTXO nursery finalized TXIDs")
|
||||
|
||||
// Compose chain bucket key.
|
||||
var b bytes.Buffer
|
||||
if _, err := b.Write(utxnChainPrefix); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
if _, err := b.Write(chainHash[:]); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
// Get chain bucket if exists.
|
||||
chainBucket := tx.Bucket(b.Bytes())
|
||||
if chainBucket == nil {
|
||||
return nil
|
||||
}
|
||||
|
||||
// Retrieve the existing height index.
|
||||
hghtIndex := chainBucket.Bucket(utxnHeightIndexKey)
|
||||
if hghtIndex == nil {
|
||||
return nil
|
||||
}
|
||||
|
||||
// Retrieve all heights.
|
||||
err := hghtIndex.ForEach(func(k, v []byte) error {
|
||||
heightBucket := hghtIndex.Bucket(k)
|
||||
if heightBucket == nil {
|
||||
return nil
|
||||
}
|
||||
|
||||
// Get finalized tx for height.
|
||||
txBytes := heightBucket.Get(utxnFinalizedKndrTxnKey)
|
||||
if txBytes == nil {
|
||||
return nil
|
||||
}
|
||||
|
||||
// Deserialize and skip tx if it cannot be deserialized.
|
||||
tx := &wire.MsgTx{}
|
||||
err := tx.Deserialize(bytes.NewReader(txBytes))
|
||||
if err != nil {
|
||||
log.Warnf("Cannot deserialize utxn tx")
|
||||
return nil
|
||||
}
|
||||
|
||||
// Calculate hash.
|
||||
hash := tx.TxHash()
|
||||
|
||||
// Insert utxn tx hash in hashes bucket.
|
||||
log.Debugf("Inserting nursery tx %v in hash list "+
|
||||
"(height=%v)", hash, byteOrder.Uint32(k))
|
||||
|
||||
return txHashesBucket.Put(hash[:], []byte{})
|
||||
})
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
// NotifyPublishTx signals that we are about to publish a tx.
|
||||
func (s *sweeperStore) NotifyPublishTx(sweepTx *wire.MsgTx) error {
|
||||
return s.db.Update(func(tx *bbolt.Tx) error {
|
||||
lastTxBucket := tx.Bucket(lastTxBucketKey)
|
||||
if lastTxBucket == nil {
|
||||
return errors.New("last tx bucket does not exist")
|
||||
}
|
||||
|
||||
txHashesBucket := tx.Bucket(txHashesBucketKey)
|
||||
if txHashesBucket == nil {
|
||||
return errors.New("tx hashes bucket does not exist")
|
||||
}
|
||||
|
||||
var b bytes.Buffer
|
||||
if err := sweepTx.Serialize(&b); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
if err := lastTxBucket.Put(lastTxKey, b.Bytes()); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
hash := sweepTx.TxHash()
|
||||
|
||||
return txHashesBucket.Put(hash[:], []byte{})
|
||||
})
|
||||
}
|
||||
|
||||
// GetLastPublishedTx returns the last tx that we called NotifyPublishTx
|
||||
// for.
|
||||
func (s *sweeperStore) GetLastPublishedTx() (*wire.MsgTx, error) {
|
||||
var sweepTx *wire.MsgTx
|
||||
|
||||
err := s.db.View(func(tx *bbolt.Tx) error {
|
||||
lastTxBucket := tx.Bucket(lastTxBucketKey)
|
||||
if lastTxBucket == nil {
|
||||
return errors.New("last tx bucket does not exist")
|
||||
}
|
||||
|
||||
sweepTxRaw := lastTxBucket.Get(lastTxKey)
|
||||
if sweepTxRaw == nil {
|
||||
return nil
|
||||
}
|
||||
|
||||
sweepTx = &wire.MsgTx{}
|
||||
txReader := bytes.NewReader(sweepTxRaw)
|
||||
if err := sweepTx.Deserialize(txReader); err != nil {
|
||||
return fmt.Errorf("tx deserialize: %v", err)
|
||||
}
|
||||
|
||||
return nil
|
||||
})
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
return sweepTx, nil
|
||||
}
|
||||
|
||||
// IsOurTx determines whether a tx is published by us, based on its
|
||||
// hash.
|
||||
func (s *sweeperStore) IsOurTx(hash chainhash.Hash) (bool, error) {
|
||||
var ours bool
|
||||
|
||||
err := s.db.View(func(tx *bbolt.Tx) error {
|
||||
txHashesBucket := tx.Bucket(txHashesBucketKey)
|
||||
if txHashesBucket == nil {
|
||||
return errors.New("tx hashes bucket does not exist")
|
||||
}
|
||||
|
||||
ours = txHashesBucket.Get(hash[:]) != nil
|
||||
|
||||
return nil
|
||||
})
|
||||
if err != nil {
|
||||
return false, err
|
||||
}
|
||||
|
||||
return ours, nil
|
||||
}
|
||||
|
||||
// Compile-time constraint to ensure sweeperStore implements SweeperStore.
|
||||
var _ SweeperStore = (*sweeperStore)(nil)
|
45
sweep/store_mock.go
Normal file
45
sweep/store_mock.go
Normal file
@ -0,0 +1,45 @@
|
||||
package sweep
|
||||
|
||||
import (
|
||||
"github.com/btcsuite/btcd/chaincfg/chainhash"
|
||||
"github.com/btcsuite/btcd/wire"
|
||||
)
|
||||
|
||||
// MockSweeperStore is a mock implementation of sweeper store. This type is
|
||||
// exported, because it is currently used in nursery tests too.
|
||||
type MockSweeperStore struct {
|
||||
lastTx *wire.MsgTx
|
||||
ourTxes map[chainhash.Hash]struct{}
|
||||
}
|
||||
|
||||
// NewMockSweeperStore returns a new instance.
|
||||
func NewMockSweeperStore() *MockSweeperStore {
|
||||
return &MockSweeperStore{
|
||||
ourTxes: make(map[chainhash.Hash]struct{}),
|
||||
}
|
||||
}
|
||||
|
||||
// IsOurTx determines whether a tx is published by us, based on its
|
||||
// hash.
|
||||
func (s *MockSweeperStore) IsOurTx(hash chainhash.Hash) (bool, error) {
|
||||
_, ok := s.ourTxes[hash]
|
||||
return ok, nil
|
||||
}
|
||||
|
||||
// NotifyPublishTx signals that we are about to publish a tx.
|
||||
func (s *MockSweeperStore) NotifyPublishTx(tx *wire.MsgTx) error {
|
||||
txHash := tx.TxHash()
|
||||
s.ourTxes[txHash] = struct{}{}
|
||||
s.lastTx = tx
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
// GetLastPublishedTx returns the last tx that we called NotifyPublishTx
|
||||
// for.
|
||||
func (s *MockSweeperStore) GetLastPublishedTx() (*wire.MsgTx, error) {
|
||||
return s.lastTx, nil
|
||||
}
|
||||
|
||||
// Compile-time constraint to ensure MockSweeperStore implements SweeperStore.
|
||||
var _ SweeperStore = (*MockSweeperStore)(nil)
|
153
sweep/store_test.go
Normal file
153
sweep/store_test.go
Normal file
@ -0,0 +1,153 @@
|
||||
package sweep
|
||||
|
||||
import (
|
||||
"io/ioutil"
|
||||
"os"
|
||||
"testing"
|
||||
|
||||
"github.com/btcsuite/btcd/chaincfg/chainhash"
|
||||
"github.com/btcsuite/btcd/wire"
|
||||
"github.com/lightningnetwork/lnd/channeldb"
|
||||
)
|
||||
|
||||
// makeTestDB creates a new instance of the ChannelDB for testing purposes. A
|
||||
// callback which cleans up the created temporary directories is also returned
|
||||
// and intended to be executed after the test completes.
|
||||
func makeTestDB() (*channeldb.DB, func(), error) {
|
||||
// First, create a temporary directory to be used for the duration of
|
||||
// this test.
|
||||
tempDirName, err := ioutil.TempDir("", "channeldb")
|
||||
if err != nil {
|
||||
return nil, nil, err
|
||||
}
|
||||
|
||||
// Next, create channeldb for the first time.
|
||||
cdb, err := channeldb.Open(tempDirName)
|
||||
if err != nil {
|
||||
return nil, nil, err
|
||||
}
|
||||
|
||||
cleanUp := func() {
|
||||
cdb.Close()
|
||||
os.RemoveAll(tempDirName)
|
||||
}
|
||||
|
||||
return cdb, cleanUp, nil
|
||||
}
|
||||
|
||||
// TestStore asserts that the store persists the presented data to disk and is
|
||||
// able to retrieve it again.
|
||||
func TestStore(t *testing.T) {
|
||||
t.Run("bolt", func(t *testing.T) {
|
||||
|
||||
// Create new store.
|
||||
cdb, cleanUp, err := makeTestDB()
|
||||
if err != nil {
|
||||
t.Fatalf("unable to open channel db: %v", err)
|
||||
}
|
||||
defer cleanUp()
|
||||
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
|
||||
testStore(t, func() (SweeperStore, error) {
|
||||
var chain chainhash.Hash
|
||||
return NewSweeperStore(cdb, &chain)
|
||||
})
|
||||
})
|
||||
t.Run("mock", func(t *testing.T) {
|
||||
store := NewMockSweeperStore()
|
||||
|
||||
testStore(t, func() (SweeperStore, error) {
|
||||
// Return same store, because the mock has no real
|
||||
// persistence.
|
||||
return store, nil
|
||||
})
|
||||
})
|
||||
}
|
||||
|
||||
func testStore(t *testing.T, createStore func() (SweeperStore, error)) {
|
||||
store, err := createStore()
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
|
||||
// Initially we expect the store not to have a last published tx.
|
||||
retrievedTx, err := store.GetLastPublishedTx()
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
if retrievedTx != nil {
|
||||
t.Fatal("expected no last published tx")
|
||||
}
|
||||
|
||||
// Notify publication of tx1
|
||||
tx1 := wire.MsgTx{}
|
||||
tx1.AddTxIn(&wire.TxIn{
|
||||
PreviousOutPoint: wire.OutPoint{
|
||||
Index: 1,
|
||||
},
|
||||
})
|
||||
|
||||
err = store.NotifyPublishTx(&tx1)
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
|
||||
// Notify publication of tx2
|
||||
tx2 := wire.MsgTx{}
|
||||
tx2.AddTxIn(&wire.TxIn{
|
||||
PreviousOutPoint: wire.OutPoint{
|
||||
Index: 2,
|
||||
},
|
||||
})
|
||||
|
||||
err = store.NotifyPublishTx(&tx2)
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
|
||||
// Recreate the sweeper store
|
||||
store, err = createStore()
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
|
||||
// Assert that last published tx2 is present.
|
||||
retrievedTx, err = store.GetLastPublishedTx()
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
|
||||
if tx2.TxHash() != retrievedTx.TxHash() {
|
||||
t.Fatal("txes do not match")
|
||||
}
|
||||
|
||||
// Assert that both txes are recognized as our own.
|
||||
ours, err := store.IsOurTx(tx1.TxHash())
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
if !ours {
|
||||
t.Fatal("expected tx to be ours")
|
||||
}
|
||||
|
||||
ours, err = store.IsOurTx(tx2.TxHash())
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
if !ours {
|
||||
t.Fatal("expected tx to be ours")
|
||||
}
|
||||
|
||||
// An different hash should be reported on as not being ours.
|
||||
var unknownHash chainhash.Hash
|
||||
ours, err = store.IsOurTx(unknownHash)
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
if ours {
|
||||
t.Fatal("expected tx to be not ours")
|
||||
}
|
||||
}
|
Loading…
Reference in New Issue
Block a user