channeldb: differentiate force vs coop close

This commit is contained in:
Conner Fromknecht 2019-12-04 13:29:30 -08:00
parent 59c4db7f33
commit 1c0dc98a7c
No known key found for this signature in database
GPG Key ID: E7D737B67FA592C7
2 changed files with 82 additions and 14 deletions

@ -59,9 +59,13 @@ var (
// remote peer during a channel sync in case we have lost channel state. // remote peer during a channel sync in case we have lost channel state.
dataLossCommitPointKey = []byte("data-loss-commit-point-key") dataLossCommitPointKey = []byte("data-loss-commit-point-key")
// closingTxKey points to a the closing tx that we broadcasted when // forceCloseTxKey points to a the unilateral closing tx that we
// moving the channel to state CommitBroadcasted. // broadcasted when moving the channel to state CommitBroadcasted.
closingTxKey = []byte("closing-tx-key") forceCloseTxKey = []byte("closing-tx-key")
// coopCloseTxKey points to a the cooperative closing tx that we
// broadcasted when moving the channel to state CoopBroadcasted.
coopCloseTxKey = []byte("coop-closing-tx-key")
// commitDiffKey stores the current pending commitment state we've // commitDiffKey stores the current pending commitment state we've
// extended to the remote party (if any). Each time we propose a new // extended to the remote party (if any). Each time we propose a new
@ -361,6 +365,10 @@ var (
// has been restored, and doesn't have all the fields a typical channel // has been restored, and doesn't have all the fields a typical channel
// will have. // will have.
ChanStatusRestored ChannelStatus = 1 << 3 ChanStatusRestored ChannelStatus = 1 << 3
// ChanStatusCoopBroadcasted indicates that a cooperative close for this
// channel has been broadcasted.
ChanStatusCoopBroadcasted ChannelStatus = 1 << 4
) )
// chanStatusStrings maps a ChannelStatus to a human friendly string that // chanStatusStrings maps a ChannelStatus to a human friendly string that
@ -371,6 +379,7 @@ var chanStatusStrings = map[ChannelStatus]string{
ChanStatusCommitBroadcasted: "ChanStatusCommitBroadcasted", ChanStatusCommitBroadcasted: "ChanStatusCommitBroadcasted",
ChanStatusLocalDataLoss: "ChanStatusLocalDataLoss", ChanStatusLocalDataLoss: "ChanStatusLocalDataLoss",
ChanStatusRestored: "ChanStatusRestored", ChanStatusRestored: "ChanStatusRestored",
ChanStatusCoopBroadcasted: "ChanStatusCoopBroadcasted",
} }
// orderedChanStatusFlags is an in-order list of all that channel status flags. // orderedChanStatusFlags is an in-order list of all that channel status flags.
@ -380,6 +389,7 @@ var orderedChanStatusFlags = []ChannelStatus{
ChanStatusCommitBroadcasted, ChanStatusCommitBroadcasted,
ChanStatusLocalDataLoss, ChanStatusLocalDataLoss,
ChanStatusRestored, ChanStatusRestored,
ChanStatusCoopBroadcasted,
} }
// String returns a human-readable representation of the ChannelStatus. // String returns a human-readable representation of the ChannelStatus.
@ -919,6 +929,30 @@ func (c *OpenChannel) isBorked(chanBucket *bbolt.Bucket) (bool, error) {
// republish this tx at startup to ensure propagation, and we should still // republish this tx at startup to ensure propagation, and we should still
// handle the case where a different tx actually hits the chain. // handle the case where a different tx actually hits the chain.
func (c *OpenChannel) MarkCommitmentBroadcasted(closeTx *wire.MsgTx) error { func (c *OpenChannel) MarkCommitmentBroadcasted(closeTx *wire.MsgTx) error {
return c.markBroadcasted(
ChanStatusCommitBroadcasted, forceCloseTxKey, closeTx,
)
}
// MarkCoopBroadcasted marks the channel to indicate that a cooperative close
// transaction has been broadcast, either our own or the remote, and that we
// should wach the chain for it to confirm before taking further action. It
// takes as argument a cooperative close tx that could appear on chain, and
// should be rebroadcast upon startup. This is only used to republish and ensure
// propagation, and we should still handle the case where a different tx
// actually hits the chain.
func (c *OpenChannel) MarkCoopBroadcasted(closeTx *wire.MsgTx) error {
return c.markBroadcasted(
ChanStatusCoopBroadcasted, coopCloseTxKey, closeTx,
)
}
// markBroadcasted is a helper function which modifies the channel status of the
// receiving channel and inserts a close transaction under the requested key,
// which should specify either a coop or force close.
func (c *OpenChannel) markBroadcasted(status ChannelStatus, key []byte,
closeTx *wire.MsgTx) error {
c.Lock() c.Lock()
defer c.Unlock() defer c.Unlock()
@ -928,15 +962,27 @@ func (c *OpenChannel) MarkCommitmentBroadcasted(closeTx *wire.MsgTx) error {
} }
putClosingTx := func(chanBucket *bbolt.Bucket) error { putClosingTx := func(chanBucket *bbolt.Bucket) error {
return chanBucket.Put(closingTxKey, b.Bytes()) return chanBucket.Put(key, b.Bytes())
} }
return c.putChanStatus(ChanStatusCommitBroadcasted, putClosingTx) return c.putChanStatus(status, putClosingTx)
} }
// BroadcastedCommitment retrieves the stored closing tx set during // BroadcastedCommitment retrieves the stored unilateral closing tx set during
// MarkCommitmentBroadcasted. If not found ErrNoCloseTx is returned. // MarkCommitmentBroadcasted. If not found ErrNoCloseTx is returned.
func (c *OpenChannel) BroadcastedCommitment() (*wire.MsgTx, error) { func (c *OpenChannel) BroadcastedCommitment() (*wire.MsgTx, error) {
return c.getClosingTx(forceCloseTxKey)
}
// BroadcastedCooperative retrieves the stored cooperative closing tx set during
// MarkCoopBroadcasted. If not found ErrNoCloseTx is returned.
func (c *OpenChannel) BroadcastedCooperative() (*wire.MsgTx, error) {
return c.getClosingTx(coopCloseTxKey)
}
// getClosingTx is a helper method which returns the stored closing transaction
// for key. The caller should use either the force or coop closing keys.
func (c *OpenChannel) getClosingTx(key []byte) (*wire.MsgTx, error) {
var closeTx *wire.MsgTx var closeTx *wire.MsgTx
err := c.Db.View(func(tx *bbolt.Tx) error { err := c.Db.View(func(tx *bbolt.Tx) error {
@ -951,7 +997,7 @@ func (c *OpenChannel) BroadcastedCommitment() (*wire.MsgTx, error) {
return err return err
} }
bs := chanBucket.Get(closingTxKey) bs := chanBucket.Get(key)
if bs == nil { if bs == nil {
return ErrNoCloseTx return ErrNoCloseTx
} }

@ -900,6 +900,14 @@ func TestFetchWaitingCloseChannels(t *testing.T) {
if err := channel.MarkCommitmentBroadcasted(closeTx); err != nil { if err := channel.MarkCommitmentBroadcasted(closeTx); err != nil {
t.Fatalf("unable to mark commitment broadcast: %v", err) t.Fatalf("unable to mark commitment broadcast: %v", err)
} }
// Modify the close tx deterministically and also mark it as
// coop closed. Later we will test that distinct transactions
// are returned for both coop and force closes.
closeTx.TxIn[0].PreviousOutPoint.Index ^= 1
if err := channel.MarkCoopBroadcasted(closeTx); err != nil {
t.Fatalf("unable to mark coop broadcast: %v", err)
}
} }
// Now, we'll fetch all the channels waiting to be closed from the // Now, we'll fetch all the channels waiting to be closed from the
@ -909,7 +917,7 @@ func TestFetchWaitingCloseChannels(t *testing.T) {
if err != nil { if err != nil {
t.Fatalf("unable to fetch all waiting close channels: %v", err) t.Fatalf("unable to fetch all waiting close channels: %v", err)
} }
if len(waitingCloseChannels) != 2 { if len(waitingCloseChannels) != numChannels {
t.Fatalf("expected %d channels waiting to be closed, got %d", 2, t.Fatalf("expected %d channels waiting to be closed, got %d", 2,
len(waitingCloseChannels)) len(waitingCloseChannels))
} }
@ -923,17 +931,31 @@ func TestFetchWaitingCloseChannels(t *testing.T) {
channel.FundingOutpoint) channel.FundingOutpoint)
} }
// Finally, make sure we can retrieve the closing tx for the chanPoint := channel.FundingOutpoint
// channel.
closeTx, err := channel.BroadcastedCommitment() // Assert that the force close transaction is retrievable.
forceCloseTx, err := channel.BroadcastedCommitment()
if err != nil { if err != nil {
t.Fatalf("Unable to retrieve commitment: %v", err) t.Fatalf("Unable to retrieve commitment: %v", err)
} }
if closeTx.TxIn[0].PreviousOutPoint != channel.FundingOutpoint { if forceCloseTx.TxIn[0].PreviousOutPoint != chanPoint {
t.Fatalf("expected outpoint %v, got %v", t.Fatalf("expected outpoint %v, got %v",
channel.FundingOutpoint, chanPoint,
closeTx.TxIn[0].PreviousOutPoint) forceCloseTx.TxIn[0].PreviousOutPoint)
}
// Assert that the coop close transaction is retrievable.
coopCloseTx, err := channel.BroadcastedCooperative()
if err != nil {
t.Fatalf("unable to retrieve coop close: %v", err)
}
chanPoint.Index ^= 1
if coopCloseTx.TxIn[0].PreviousOutPoint != chanPoint {
t.Fatalf("expected outpoint %v, got %v",
chanPoint,
coopCloseTx.TxIn[0].PreviousOutPoint)
} }
} }
} }