lnwallet: mark channel as frozen based on ShimIntent

In this commit, we make the internal channel funding flow aware of
frozen channels. We also update the testSingleFunderReservationWorkflow
method to ensure that the created channels have the proper type bit set.
This commit is contained in:
Olaoluwa Osuntokun 2020-03-13 16:57:48 -07:00
parent 8bce48d3cc
commit c85f6bb364
No known key found for this signature in database
GPG Key ID: BC13F65E2DC84465
3 changed files with 59 additions and 25 deletions

@ -701,7 +701,7 @@ func testCancelNonExistentReservation(miner *rpctest.Harness,
res, err := lnwallet.NewChannelReservation( res, err := lnwallet.NewChannelReservation(
10000, 10000, feePerKw, alice, 22, 10, &testHdSeed, 10000, 10000, feePerKw, alice, 22, 10, &testHdSeed,
lnwire.FFAnnounceChannel, lnwallet.CommitmentTypeTweakless, lnwire.FFAnnounceChannel, lnwallet.CommitmentTypeTweakless,
nil, [32]byte{}, nil, [32]byte{}, 0,
) )
if err != nil { if err != nil {
t.Fatalf("unable to create res: %v", err) t.Fatalf("unable to create res: %v", err)
@ -796,8 +796,8 @@ func assertContributionInitPopulated(t *testing.T, c *lnwallet.ChannelContributi
func testSingleFunderReservationWorkflow(miner *rpctest.Harness, func testSingleFunderReservationWorkflow(miner *rpctest.Harness,
alice, bob *lnwallet.LightningWallet, t *testing.T, alice, bob *lnwallet.LightningWallet, t *testing.T,
commitType lnwallet.CommitmentType, commitType lnwallet.CommitmentType,
aliceChanFunder chanfunding.Assembler, aliceChanFunder chanfunding.Assembler, fetchFundingTx func() *wire.MsgTx,
fetchFundingTx func() *wire.MsgTx, pendingChanID [32]byte) { pendingChanID [32]byte, thawHeight uint32) {
// For this scenario, Alice will be the channel initiator while bob // For this scenario, Alice will be the channel initiator while bob
// will act as the responder to the workflow. // will act as the responder to the workflow.
@ -1045,6 +1045,24 @@ func testSingleFunderReservationWorkflow(miner *rpctest.Harness,
t.Fatalf("incorrect transaction was mined") t.Fatalf("incorrect transaction was mined")
} }
// If a frozen channel was requested, then we expect that both channel
// types show as being a frozen channel type.
aliceChanFrozen := aliceChannels[0].ChanType.IsFrozen()
bobChanFrozen := bobChannels[0].ChanType.IsFrozen()
if thawHeight != 0 && (!aliceChanFrozen || !bobChanFrozen) {
t.Fatalf("expected both alice and bob to have frozen chans: "+
"alice_frozen=%v, bob_frozen=%v", aliceChanFrozen,
bobChanFrozen)
}
if thawHeight != bobChannels[0].ThawHeight {
t.Fatalf("wrong thaw height: expected %v got %v", thawHeight,
bobChannels[0].ThawHeight)
}
if thawHeight != aliceChannels[0].ThawHeight {
t.Fatalf("wrong thaw height: expected %v got %v", thawHeight,
aliceChannels[0].ThawHeight)
}
assertReservationDeleted(aliceChanReservation, t) assertReservationDeleted(aliceChanReservation, t)
assertReservationDeleted(bobChanReservation, t) assertReservationDeleted(bobChanReservation, t)
} }
@ -2546,8 +2564,8 @@ var walletTests = []walletTestCase{
testSingleFunderReservationWorkflow( testSingleFunderReservationWorkflow(
miner, alice, bob, t, miner, alice, bob, t,
lnwallet.CommitmentTypeLegacy, nil, nil, lnwallet.CommitmentTypeLegacy, nil,
[32]byte{}, nil, [32]byte{}, 0,
) )
}, },
}, },
@ -2558,8 +2576,8 @@ var walletTests = []walletTestCase{
testSingleFunderReservationWorkflow( testSingleFunderReservationWorkflow(
miner, alice, bob, t, miner, alice, bob, t,
lnwallet.CommitmentTypeTweakless, nil, nil, lnwallet.CommitmentTypeTweakless, nil,
[32]byte{}, nil, [32]byte{}, 0,
) )
}, },
}, },
@ -2777,12 +2795,13 @@ func testSingleFunderExternalFundingTx(miner *rpctest.Harness,
// Now that we have the fully constructed funding transaction, we'll // Now that we have the fully constructed funding transaction, we'll
// create a new shim external funder out of it for Alice, and prep a // create a new shim external funder out of it for Alice, and prep a
// shim intent for Bob. // shim intent for Bob.
thawHeight := uint32(200)
aliceExternalFunder := chanfunding.NewCannedAssembler( aliceExternalFunder := chanfunding.NewCannedAssembler(
*chanPoint, btcutil.Amount(chanAmt), &aliceFundingKey, thawHeight, *chanPoint, btcutil.Amount(chanAmt), &aliceFundingKey,
bobFundingKey.PubKey, true, bobFundingKey.PubKey, true,
) )
bobShimIntent, err := chanfunding.NewCannedAssembler( bobShimIntent, err := chanfunding.NewCannedAssembler(
*chanPoint, btcutil.Amount(chanAmt), &bobFundingKey, thawHeight, *chanPoint, btcutil.Amount(chanAmt), &bobFundingKey,
aliceFundingKey.PubKey, false, aliceFundingKey.PubKey, false,
).ProvisionChannel(&chanfunding.Request{ ).ProvisionChannel(&chanfunding.Request{
LocalAmt: btcutil.Amount(chanAmt), LocalAmt: btcutil.Amount(chanAmt),
@ -2816,7 +2835,7 @@ func testSingleFunderExternalFundingTx(miner *rpctest.Harness,
miner, alice, bob, t, lnwallet.CommitmentTypeTweakless, miner, alice, bob, t, lnwallet.CommitmentTypeTweakless,
aliceExternalFunder, func() *wire.MsgTx { aliceExternalFunder, func() *wire.MsgTx {
return fundingTx return fundingTx
}, pendingChanID, }, pendingChanID, thawHeight,
) )
} }

@ -171,7 +171,7 @@ func NewChannelReservation(capacity, localFundingAmt btcutil.Amount,
id uint64, pushMSat lnwire.MilliSatoshi, chainHash *chainhash.Hash, id uint64, pushMSat lnwire.MilliSatoshi, chainHash *chainhash.Hash,
flags lnwire.FundingFlag, commitType CommitmentType, flags lnwire.FundingFlag, commitType CommitmentType,
fundingAssembler chanfunding.Assembler, fundingAssembler chanfunding.Assembler,
pendingChanID [32]byte) (*ChannelReservation, error) { pendingChanID [32]byte, thawHeight uint32) (*ChannelReservation, error) {
var ( var (
ourBalance lnwire.MilliSatoshi ourBalance lnwire.MilliSatoshi
@ -306,6 +306,12 @@ func NewChannelReservation(capacity, localFundingAmt btcutil.Amount,
chanType |= channeldb.AnchorOutputsBit chanType |= channeldb.AnchorOutputsBit
} }
// If the channel is meant to be frozen, then we'll set the frozen bit
// now so once the channel is open, it can be interpreted properly.
if thawHeight != 0 {
chanType |= channeldb.FrozenBit
}
return &ChannelReservation{ return &ChannelReservation{
ourContribution: &ChannelContribution{ ourContribution: &ChannelContribution{
FundingAmount: ourBalance.ToSatoshis(), FundingAmount: ourBalance.ToSatoshis(),
@ -334,6 +340,7 @@ func NewChannelReservation(capacity, localFundingAmt btcutil.Amount,
FeePerKw: btcutil.Amount(commitFeePerKw), FeePerKw: btcutil.Amount(commitFeePerKw),
CommitFee: commitFee, CommitFee: commitFee,
}, },
ThawHeight: thawHeight,
Db: wallet.Cfg.Database, Db: wallet.Cfg.Database,
}, },
pushMSat: pushMSat, pushMSat: pushMSat,

@ -560,6 +560,26 @@ func (l *LightningWallet) handleFundingReserveRequest(req *InitFundingReserveMsg
remoteFundingAmt = fundingIntent.RemoteFundingAmt() remoteFundingAmt = fundingIntent.RemoteFundingAmt()
} }
// If this is a shim intent, then it may be attempting to use an
// existing set of keys for the funding workflow. In this case, we'll
// make a simple wrapper keychain.KeyRing that will proxy certain
// derivation calls to future callers.
var (
keyRing keychain.KeyRing = l.SecretKeyRing
thawHeight uint32
)
if shimIntent, ok := fundingIntent.(*chanfunding.ShimIntent); ok {
keyRing = &shimKeyRing{
KeyRing: keyRing,
ShimIntent: shimIntent,
}
// As this was a registered shim intent, we'll obtain the thaw
// height of the intent, if present at all. If this is
// non-zero, then we'll mark this as the proper channel type.
thawHeight = shimIntent.ThawHeight()
}
// The total channel capacity will be the size of the funding output we // The total channel capacity will be the size of the funding output we
// created plus the remote contribution. // created plus the remote contribution.
capacity := localFundingAmt + remoteFundingAmt capacity := localFundingAmt + remoteFundingAmt
@ -569,6 +589,7 @@ func (l *LightningWallet) handleFundingReserveRequest(req *InitFundingReserveMsg
capacity, localFundingAmt, req.CommitFeePerKw, l, id, capacity, localFundingAmt, req.CommitFeePerKw, l, id,
req.PushMSat, l.Cfg.NetParams.GenesisHash, req.Flags, req.PushMSat, l.Cfg.NetParams.GenesisHash, req.Flags,
req.CommitType, req.ChanFunder, req.PendingChanID, req.CommitType, req.ChanFunder, req.PendingChanID,
thawHeight,
) )
if err != nil { if err != nil {
if fundingIntent != nil { if fundingIntent != nil {
@ -580,19 +601,6 @@ func (l *LightningWallet) handleFundingReserveRequest(req *InitFundingReserveMsg
return return
} }
var keyRing keychain.KeyRing = l.SecretKeyRing
// If this is a shim intent, then it may be attempting to use an
// existing set of keys for the funding workflow. In this case, we'll
// make a simple wrapper keychain.KeyRing that will proxy certain
// derivation calls to future callers.
if shimIntent, ok := fundingIntent.(*chanfunding.ShimIntent); ok {
keyRing = &shimKeyRing{
KeyRing: keyRing,
ShimIntent: shimIntent,
}
}
err = l.initOurContribution( err = l.initOurContribution(
reservation, fundingIntent, req.NodeAddr, req.NodeID, keyRing, reservation, fundingIntent, req.NodeAddr, req.NodeID, keyRing,
) )