contractcourt/commit_sweep_resolver: set sweep witness type based on witness script

We use the fact that we can tell whether the commit is local or remote
by inspecting the witness script. We cannot use the maturity delay
anymore, as we can have delayed to_remote outputs also now.

Co-authored-by: Joost Jager <joost.jager@gmail.com>
This commit is contained in:
Johan T. Halseth 2020-03-06 16:11:46 +01:00
parent af68ff1640
commit 1f28bd8086
No known key found for this signature in database
GPG Key ID: 15BAADA29DA20D26
2 changed files with 31 additions and 7 deletions

@ -6,6 +6,7 @@ import (
"io" "io"
"sync" "sync"
"github.com/btcsuite/btcd/txscript"
"github.com/btcsuite/btcd/wire" "github.com/btcsuite/btcd/wire"
"github.com/btcsuite/btcutil" "github.com/btcsuite/btcutil"
"github.com/lightningnetwork/lnd/input" "github.com/lightningnetwork/lnd/input"
@ -172,24 +173,45 @@ func (c *commitSweepResolver) Resolve() (ContractResolver, error) {
} }
} }
// We're dealing with our commitment transaction if the delay on the // The output is on our local commitment if the script starts with
// resolution isn't zero. // OP_IF for the revocation clause. On the remote commitment it will
isLocalCommitTx := c.commitResolution.MaturityDelay != 0 // either be a regular P2WKH or a simple sig spend with a CSV delay.
isLocalCommitTx := c.commitResolution.SelfOutputSignDesc.WitnessScript[0] == txscript.OP_IF
isDelayedOutput := c.commitResolution.MaturityDelay != 0
// There're two types of commitments, those that have tweaks c.log.Debugf("isDelayedOutput=%v, isLocalCommitTx=%v", isDelayedOutput,
// for the remote key (us in this case), and those that don't. isLocalCommitTx)
// We'll rely on the presence of the commitment tweak to to
// discern which type of commitment this is. // There're three types of commitments, those that have tweaks
// for the remote key (us in this case), those that don't, and a third
// where there is no tweak and the output is delayed. On the local
// commitment our output will always be delayed. We'll rely on the
// presence of the commitment tweak to to discern which type of
// commitment this is.
var witnessType input.WitnessType var witnessType input.WitnessType
switch { switch {
// Delayed output to us on our local commitment.
case isLocalCommitTx: case isLocalCommitTx:
witnessType = input.CommitmentTimeLock witnessType = input.CommitmentTimeLock
// A confirmed output to us on the remote commitment.
case isDelayedOutput:
witnessType = input.CommitmentToRemoteConfirmed
// A non-delayed output on the remote commitment where the key is
// tweakless.
case c.commitResolution.SelfOutputSignDesc.SingleTweak == nil: case c.commitResolution.SelfOutputSignDesc.SingleTweak == nil:
witnessType = input.CommitSpendNoDelayTweakless witnessType = input.CommitSpendNoDelayTweakless
// A non-delayed output on the remote commitment where the key is
// tweaked.
default: default:
witnessType = input.CommitmentNoDelay witnessType = input.CommitmentNoDelay
} }
c.log.Infof("Sweeping with witness type: %v", witnessType)
// We'll craft an input with all the information required for // We'll craft an input with all the information required for
// the sweeper to create a fully valid sweeping transaction to // the sweeper to create a fully valid sweeping transaction to
// recover these coins. // recover these coins.

@ -133,6 +133,7 @@ func TestCommitSweepResolverNoDelay(t *testing.T) {
Output: &wire.TxOut{ Output: &wire.TxOut{
Value: 100, Value: 100,
}, },
WitnessScript: []byte{0},
}, },
} }
@ -162,6 +163,7 @@ func TestCommitSweepResolverDelay(t *testing.T) {
Output: &wire.TxOut{ Output: &wire.TxOut{
Value: amt, Value: amt,
}, },
WitnessScript: []byte{0},
}, },
MaturityDelay: 3, MaturityDelay: 3,
SelfOutPoint: outpoint, SelfOutPoint: outpoint,