htlcswitch/test: move test functions into context

This commit is contained in:
Joost Jager 2019-09-17 17:41:22 +02:00
parent 23617bb9fd
commit 426ee28895
No known key found for this signature in database
GPG Key ID: A61B9D4C393C59C7
2 changed files with 245 additions and 197 deletions

@ -10,30 +10,36 @@ import (
"github.com/lightningnetwork/lnd/lnwire" "github.com/lightningnetwork/lnd/lnwire"
) )
type linkTestContext struct {
t *testing.T
aliceLink ChannelLink
bobChannel *lnwallet.LightningChannel
aliceMsgs <-chan lnwire.Message
}
// sendHtlcBobToAlice sends an HTLC from Bob to Alice, that pays to a preimage // sendHtlcBobToAlice sends an HTLC from Bob to Alice, that pays to a preimage
// already in Alice's registry. // already in Alice's registry.
func sendHtlcBobToAlice(t *testing.T, aliceLink ChannelLink, func (l *linkTestContext) sendHtlcBobToAlice(htlc *lnwire.UpdateAddHTLC) {
bobChannel *lnwallet.LightningChannel, htlc *lnwire.UpdateAddHTLC) { l.t.Helper()
t.Helper() _, err := l.bobChannel.AddHTLC(htlc, nil)
_, err := bobChannel.AddHTLC(htlc, nil)
if err != nil { if err != nil {
t.Fatalf("bob failed adding htlc: %v", err) l.t.Fatalf("bob failed adding htlc: %v", err)
} }
aliceLink.HandleChannelUpdate(htlc) l.aliceLink.HandleChannelUpdate(htlc)
} }
// sendHtlcAliceToBob sends an HTLC from Alice to Bob, by first committing the // sendHtlcAliceToBob sends an HTLC from Alice to Bob, by first committing the
// HTLC in the circuit map, then delivering the outgoing packet to Alice's link. // HTLC in the circuit map, then delivering the outgoing packet to Alice's link.
// The HTLC will be sent to Bob via Alice's message stream. // The HTLC will be sent to Bob via Alice's message stream.
func sendHtlcAliceToBob(t *testing.T, aliceLink ChannelLink, htlcID int, func (l *linkTestContext) sendHtlcAliceToBob(htlcID int,
htlc *lnwire.UpdateAddHTLC) { htlc *lnwire.UpdateAddHTLC) {
t.Helper() l.t.Helper()
circuitMap := aliceLink.(*channelLink).cfg.Switch.circuits circuitMap := l.aliceLink.(*channelLink).cfg.Switch.circuits
fwdActions, err := circuitMap.CommitCircuits( fwdActions, err := circuitMap.CommitCircuits(
&PaymentCircuit{ &PaymentCircuit{
Incoming: CircuitKey{ Incoming: CircuitKey{
@ -43,55 +49,53 @@ func sendHtlcAliceToBob(t *testing.T, aliceLink ChannelLink, htlcID int,
}, },
) )
if err != nil { if err != nil {
t.Fatalf("unable to commit circuit: %v", err) l.t.Fatalf("unable to commit circuit: %v", err)
} }
if len(fwdActions.Adds) != 1 { if len(fwdActions.Adds) != 1 {
t.Fatalf("expected 1 adds, found %d", len(fwdActions.Adds)) l.t.Fatalf("expected 1 adds, found %d", len(fwdActions.Adds))
} }
aliceLink.HandleSwitchPacket(&htlcPacket{ err = l.aliceLink.HandleSwitchPacket(&htlcPacket{
incomingHTLCID: uint64(htlcID), incomingHTLCID: uint64(htlcID),
htlc: htlc, htlc: htlc,
}) })
if err != nil {
l.t.Fatal(err)
}
} }
// receiveHtlcAliceToBob pulls the next message from Alice's message stream, // receiveHtlcAliceToBob pulls the next message from Alice's message stream,
// asserts that it is an UpdateAddHTLC, then applies it to Bob's state machine. // asserts that it is an UpdateAddHTLC, then applies it to Bob's state machine.
func receiveHtlcAliceToBob(t *testing.T, aliceMsgs <-chan lnwire.Message, func (l *linkTestContext) receiveHtlcAliceToBob() {
bobChannel *lnwallet.LightningChannel) { l.t.Helper()
t.Helper()
var msg lnwire.Message var msg lnwire.Message
select { select {
case msg = <-aliceMsgs: case msg = <-l.aliceMsgs:
case <-time.After(15 * time.Second): case <-time.After(15 * time.Second):
t.Fatalf("did not received htlc from alice") l.t.Fatalf("did not received htlc from alice")
} }
htlcAdd, ok := msg.(*lnwire.UpdateAddHTLC) htlcAdd, ok := msg.(*lnwire.UpdateAddHTLC)
if !ok { if !ok {
t.Fatalf("expected UpdateAddHTLC, got %T", msg) l.t.Fatalf("expected UpdateAddHTLC, got %T", msg)
} }
_, err := bobChannel.ReceiveHTLC(htlcAdd) _, err := l.bobChannel.ReceiveHTLC(htlcAdd)
if err != nil { if err != nil {
t.Fatalf("bob failed receiving htlc: %v", err) l.t.Fatalf("bob failed receiving htlc: %v", err)
} }
} }
// sendCommitSigBobToAlice makes Bob sign a new commitment and send it to // sendCommitSigBobToAlice makes Bob sign a new commitment and send it to
// Alice, asserting that it signs expHtlcs number of HTLCs. // Alice, asserting that it signs expHtlcs number of HTLCs.
func sendCommitSigBobToAlice(t *testing.T, aliceLink ChannelLink, func (l *linkTestContext) sendCommitSigBobToAlice(expHtlcs int) {
bobChannel *lnwallet.LightningChannel, expHtlcs int) { l.t.Helper()
t.Helper() sig, htlcSigs, _, err := l.bobChannel.SignNextCommitment()
sig, htlcSigs, _, err := bobChannel.SignNextCommitment()
if err != nil { if err != nil {
t.Fatalf("error signing commitment: %v", err) l.t.Fatalf("error signing commitment: %v", err)
} }
commitSig := &lnwire.CommitSig{ commitSig := &lnwire.CommitSig{
@ -100,122 +104,111 @@ func sendCommitSigBobToAlice(t *testing.T, aliceLink ChannelLink,
} }
if len(commitSig.HtlcSigs) != expHtlcs { if len(commitSig.HtlcSigs) != expHtlcs {
t.Fatalf("Expected %d htlc sigs, got %d", expHtlcs, l.t.Fatalf("Expected %d htlc sigs, got %d", expHtlcs,
len(commitSig.HtlcSigs)) len(commitSig.HtlcSigs))
} }
aliceLink.HandleChannelUpdate(commitSig) l.aliceLink.HandleChannelUpdate(commitSig)
} }
// receiveRevAndAckAliceToBob waits for Alice to send a RevAndAck to Bob, then // receiveRevAndAckAliceToBob waits for Alice to send a RevAndAck to Bob, then
// hands this to Bob. // hands this to Bob.
func receiveRevAndAckAliceToBob(t *testing.T, aliceMsgs chan lnwire.Message, func (l *linkTestContext) receiveRevAndAckAliceToBob() {
aliceLink ChannelLink, l.t.Helper()
bobChannel *lnwallet.LightningChannel) {
t.Helper()
var msg lnwire.Message var msg lnwire.Message
select { select {
case msg = <-aliceMsgs: case msg = <-l.aliceMsgs:
case <-time.After(15 * time.Second): case <-time.After(15 * time.Second):
t.Fatalf("did not receive message") l.t.Fatalf("did not receive message")
} }
rev, ok := msg.(*lnwire.RevokeAndAck) rev, ok := msg.(*lnwire.RevokeAndAck)
if !ok { if !ok {
t.Fatalf("expected RevokeAndAck, got %T", msg) l.t.Fatalf("expected RevokeAndAck, got %T", msg)
} }
_, _, _, _, err := bobChannel.ReceiveRevocation(rev) _, _, _, _, err := l.bobChannel.ReceiveRevocation(rev)
if err != nil { if err != nil {
t.Fatalf("bob failed receiving revocation: %v", err) l.t.Fatalf("bob failed receiving revocation: %v", err)
} }
} }
// receiveCommitSigAliceToBob waits for Alice to send a CommitSig to Bob, // receiveCommitSigAliceToBob waits for Alice to send a CommitSig to Bob,
// signing expHtlcs numbers of HTLCs, then hands this to Bob. // signing expHtlcs numbers of HTLCs, then hands this to Bob.
func receiveCommitSigAliceToBob(t *testing.T, aliceMsgs chan lnwire.Message, func (l *linkTestContext) receiveCommitSigAliceToBob(expHtlcs int) {
aliceLink ChannelLink, bobChannel *lnwallet.LightningChannel, l.t.Helper()
expHtlcs int) {
t.Helper()
var msg lnwire.Message var msg lnwire.Message
select { select {
case msg = <-aliceMsgs: case msg = <-l.aliceMsgs:
case <-time.After(15 * time.Second): case <-time.After(15 * time.Second):
t.Fatalf("did not receive message") l.t.Fatalf("did not receive message")
} }
comSig, ok := msg.(*lnwire.CommitSig) comSig, ok := msg.(*lnwire.CommitSig)
if !ok { if !ok {
t.Fatalf("expected CommitSig, got %T", msg) l.t.Fatalf("expected CommitSig, got %T", msg)
} }
if len(comSig.HtlcSigs) != expHtlcs { if len(comSig.HtlcSigs) != expHtlcs {
t.Fatalf("expected %d htlc sigs, got %d", expHtlcs, l.t.Fatalf("expected %d htlc sigs, got %d", expHtlcs,
len(comSig.HtlcSigs)) len(comSig.HtlcSigs))
} }
err := bobChannel.ReceiveNewCommitment(comSig.CommitSig, err := l.bobChannel.ReceiveNewCommitment(comSig.CommitSig,
comSig.HtlcSigs) comSig.HtlcSigs)
if err != nil { if err != nil {
t.Fatalf("bob failed receiving commitment: %v", err) l.t.Fatalf("bob failed receiving commitment: %v", err)
} }
} }
// sendRevAndAckBobToAlice make Bob revoke his current commitment, then hand // sendRevAndAckBobToAlice make Bob revoke his current commitment, then hand
// the RevokeAndAck to Alice. // the RevokeAndAck to Alice.
func sendRevAndAckBobToAlice(t *testing.T, aliceLink ChannelLink, func (l *linkTestContext) sendRevAndAckBobToAlice() {
bobChannel *lnwallet.LightningChannel) { l.t.Helper()
t.Helper() rev, _, err := l.bobChannel.RevokeCurrentCommitment()
rev, _, err := bobChannel.RevokeCurrentCommitment()
if err != nil { if err != nil {
t.Fatalf("unable to revoke commitment: %v", err) l.t.Fatalf("unable to revoke commitment: %v", err)
} }
aliceLink.HandleChannelUpdate(rev) l.aliceLink.HandleChannelUpdate(rev)
} }
// receiveSettleAliceToBob waits for Alice to send a HTLC settle message to // receiveSettleAliceToBob waits for Alice to send a HTLC settle message to
// Bob, then hands this to Bob. // Bob, then hands this to Bob.
func receiveSettleAliceToBob(t *testing.T, aliceMsgs chan lnwire.Message, func (l *linkTestContext) receiveSettleAliceToBob() {
aliceLink ChannelLink, bobChannel *lnwallet.LightningChannel) { l.t.Helper()
t.Helper()
var msg lnwire.Message var msg lnwire.Message
select { select {
case msg = <-aliceMsgs: case msg = <-l.aliceMsgs:
case <-time.After(15 * time.Second): case <-time.After(15 * time.Second):
t.Fatalf("did not receive message") l.t.Fatalf("did not receive message")
} }
settleMsg, ok := msg.(*lnwire.UpdateFulfillHTLC) settleMsg, ok := msg.(*lnwire.UpdateFulfillHTLC)
if !ok { if !ok {
t.Fatalf("expected UpdateFulfillHTLC, got %T", msg) l.t.Fatalf("expected UpdateFulfillHTLC, got %T", msg)
} }
err := bobChannel.ReceiveHTLCSettle(settleMsg.PaymentPreimage, err := l.bobChannel.ReceiveHTLCSettle(settleMsg.PaymentPreimage,
settleMsg.ID) settleMsg.ID)
if err != nil { if err != nil {
t.Fatalf("failed settling htlc: %v", err) l.t.Fatalf("failed settling htlc: %v", err)
} }
} }
// sendSettleBobToAlice settles an HTLC on Bob's state machine, then sends an // sendSettleBobToAlice settles an HTLC on Bob's state machine, then sends an
// UpdateFulfillHTLC message to Alice's upstream inbox. // UpdateFulfillHTLC message to Alice's upstream inbox.
func sendSettleBobToAlice(t *testing.T, aliceLink ChannelLink, func (l *linkTestContext) sendSettleBobToAlice(htlcID uint64,
bobChannel *lnwallet.LightningChannel, htlcID uint64,
preimage lntypes.Preimage) { preimage lntypes.Preimage) {
t.Helper() l.t.Helper()
err := bobChannel.SettleHTLC(preimage, htlcID, nil, nil, nil) err := l.bobChannel.SettleHTLC(preimage, htlcID, nil, nil, nil)
if err != nil { if err != nil {
t.Fatalf("alice failed settling htlc id=%d hash=%x", l.t.Fatalf("alice failed settling htlc id=%d hash=%x",
htlcID, sha256.Sum256(preimage[:])) htlcID, sha256.Sum256(preimage[:]))
} }
@ -224,30 +217,28 @@ func sendSettleBobToAlice(t *testing.T, aliceLink ChannelLink,
PaymentPreimage: preimage, PaymentPreimage: preimage,
} }
aliceLink.HandleChannelUpdate(settle) l.aliceLink.HandleChannelUpdate(settle)
} }
// receiveSettleAliceToBob waits for Alice to send a HTLC settle message to // receiveSettleAliceToBob waits for Alice to send a HTLC settle message to
// Bob, then hands this to Bob. // Bob, then hands this to Bob.
func receiveFailAliceToBob(t *testing.T, aliceMsgs chan lnwire.Message, func (l *linkTestContext) receiveFailAliceToBob() {
aliceLink ChannelLink, bobChannel *lnwallet.LightningChannel) { l.t.Helper()
t.Helper()
var msg lnwire.Message var msg lnwire.Message
select { select {
case msg = <-aliceMsgs: case msg = <-l.aliceMsgs:
case <-time.After(15 * time.Second): case <-time.After(15 * time.Second):
t.Fatalf("did not receive message") l.t.Fatalf("did not receive message")
} }
failMsg, ok := msg.(*lnwire.UpdateFailHTLC) failMsg, ok := msg.(*lnwire.UpdateFailHTLC)
if !ok { if !ok {
t.Fatalf("expected UpdateFailHTLC, got %T", msg) l.t.Fatalf("expected UpdateFailHTLC, got %T", msg)
} }
err := bobChannel.ReceiveFailHTLC(failMsg.ID, failMsg.Reason) err := l.bobChannel.ReceiveFailHTLC(failMsg.ID, failMsg.Reason)
if err != nil { if err != nil {
t.Fatalf("unable to apply received fail htlc: %v", err) l.t.Fatalf("unable to apply received fail htlc: %v", err)
} }
} }

@ -4405,6 +4405,13 @@ func TestChannelLinkNoMoreUpdates(t *testing.T) {
htlc1 := generateHtlc(t, coreLink, bobChannel, 0) htlc1 := generateHtlc(t, coreLink, bobChannel, 0)
htlc2 := generateHtlc(t, coreLink, bobChannel, 1) htlc2 := generateHtlc(t, coreLink, bobChannel, 1)
ctx := linkTestContext{
t: t,
aliceLink: aliceLink,
aliceMsgs: aliceMsgs,
bobChannel: bobChannel,
}
// We now play out the following scanario: // We now play out the following scanario:
// //
// (1) Alice receives htlc1 from Bob. // (1) Alice receives htlc1 from Bob.
@ -4435,29 +4442,29 @@ func TestChannelLinkNoMoreUpdates(t *testing.T) {
// Alice sent. // Alice sent.
// (17) Send a signature for the empty state. No HTLCs are left. // (17) Send a signature for the empty state. No HTLCs are left.
// (18) Alice will revoke her previous state. // (18) Alice will revoke her previous state.
// Alice Bob // Alice Bob
// | | // | |
// | ... | // | ... |
// | | <--- idle (no htlc on either side) // | | <--- idle (no htlc on either side)
// | | // | |
sendHtlcBobToAlice(t, aliceLink, bobChannel, htlc1) // |<----- add-1 ------| (1) ctx.sendHtlcBobToAlice(htlc1) // |<----- add-1 ------| (1)
sendCommitSigBobToAlice(t, aliceLink, bobChannel, 1) // |<------ sig -------| (2) ctx.sendCommitSigBobToAlice(1) // |<------ sig -------| (2)
sendHtlcBobToAlice(t, aliceLink, bobChannel, htlc2) // |<----- add-2 ------| (3) ctx.sendHtlcBobToAlice(htlc2) // |<----- add-2 ------| (3)
receiveRevAndAckAliceToBob(t, aliceMsgs, aliceLink, bobChannel) // |------- rev ------>| (4) <--- Alice acks add-1 ctx.receiveRevAndAckAliceToBob() // |------- rev ------>| (4) <--- Alice acks add-1
receiveCommitSigAliceToBob(t, aliceMsgs, aliceLink, bobChannel, 1) // |------- sig ------>| (5) <--- Alice signs add-1 ctx.receiveCommitSigAliceToBob(1) // |------- sig ------>| (5) <--- Alice signs add-1
sendCommitSigBobToAlice(t, aliceLink, bobChannel, 2) // |<------ sig -------| (6) ctx.sendCommitSigBobToAlice(2) // |<------ sig -------| (6)
receiveRevAndAckAliceToBob(t, aliceMsgs, aliceLink, bobChannel) // |------- rev ------>| (7) <--- Alice acks add-2 ctx.receiveRevAndAckAliceToBob() // |------- rev ------>| (7) <--- Alice acks add-2
sendRevAndAckBobToAlice(t, aliceLink, bobChannel) // |<------ rev -------| (8) ctx.sendRevAndAckBobToAlice() // |<------ rev -------| (8)
receiveSettleAliceToBob(t, aliceMsgs, aliceLink, bobChannel) // |------ ful-1 ----->| (9) ctx.receiveSettleAliceToBob() // |------ ful-1 ----->| (9)
receiveCommitSigAliceToBob(t, aliceMsgs, aliceLink, bobChannel, 1) // |------- sig ------>| (10) <--- Alice signs add-1 + add-2 + ful-1 = add-2 ctx.receiveCommitSigAliceToBob(1) // |------- sig ------>| (10) <--- Alice signs add-1 + add-2 + ful-1 = add-2
sendRevAndAckBobToAlice(t, aliceLink, bobChannel) // |<------ rev -------| (11) ctx.sendRevAndAckBobToAlice() // |<------ rev -------| (11)
sendCommitSigBobToAlice(t, aliceLink, bobChannel, 1) // |<------ sig -------| (12) ctx.sendCommitSigBobToAlice(1) // |<------ sig -------| (12)
receiveSettleAliceToBob(t, aliceMsgs, aliceLink, bobChannel) // |------ ful-2 ----->| (13) ctx.receiveSettleAliceToBob() // |------ ful-2 ----->| (13)
receiveCommitSigAliceToBob(t, aliceMsgs, aliceLink, bobChannel, 0) // |------- sig ------>| (14) <--- Alice signs add-2 + ful-2 = no htlcs ctx.receiveCommitSigAliceToBob(0) // |------- sig ------>| (14) <--- Alice signs add-2 + ful-2 = no htlcs
receiveRevAndAckAliceToBob(t, aliceMsgs, aliceLink, bobChannel) // |------- rev ------>| (15) ctx.receiveRevAndAckAliceToBob() // |------- rev ------>| (15)
sendRevAndAckBobToAlice(t, aliceLink, bobChannel) // |<------ rev -------| (16) <--- Bob acks that there are no more htlcs ctx.sendRevAndAckBobToAlice() // |<------ rev -------| (16) <--- Bob acks that there are no more htlcs
sendCommitSigBobToAlice(t, aliceLink, bobChannel, 0) // |<------ sig -------| (17) ctx.sendCommitSigBobToAlice(0) // |<------ sig -------| (17)
receiveRevAndAckAliceToBob(t, aliceMsgs, aliceLink, bobChannel) // |------- rev ------>| (18) <--- Alice acks that there are no htlcs on Alice's side ctx.receiveRevAndAckAliceToBob() // |------- rev ------>| (18) <--- Alice acks that there are no htlcs on Alice's side
// No there are no more changes to ACK or sign, make sure Alice doesn't // No there are no more changes to ACK or sign, make sure Alice doesn't
// attempt to send any more messages. // attempt to send any more messages.
@ -4521,6 +4528,21 @@ func TestChannelLinkWaitForRevocation(t *testing.T) {
htlcs = append(htlcs, htlc) htlcs = append(htlcs, htlc)
} }
ctx := linkTestContext{
t: t,
aliceLink: aliceLink,
aliceMsgs: aliceMsgs,
bobChannel: bobChannel,
}
assertNoMsgFromAlice := func() {
select {
case <-aliceMsgs:
t.Fatalf("did not expect message from Alice")
case <-time.After(50 * time.Millisecond):
}
}
// We play out the following scenario: // We play out the following scenario:
// //
// (1) Add the first HTLC. // (1) Add the first HTLC.
@ -4559,45 +4581,41 @@ func TestChannelLinkWaitForRevocation(t *testing.T) {
// that now. Since Bob ACKed all settles, no HTLCs should be left on // that now. Since Bob ACKed all settles, no HTLCs should be left on
// the commitment. // the commitment.
// (18) Alice will revoke her previous state. // (18) Alice will revoke her previous state.
// Alice Bob // Alice Bob
// | | // | |
// | ... | // | ... |
// | | <--- idle (no htlc on either side) // | | <--- idle (no htlc on either side)
// | | // | |
sendHtlcBobToAlice(t, aliceLink, bobChannel, htlcs[0]) // |<----- add-1 ------| (1) ctx.sendHtlcBobToAlice(htlcs[0]) // |<----- add-1 ------| (1)
sendCommitSigBobToAlice(t, aliceLink, bobChannel, 1) // |<------ sig -------| (2) ctx.sendCommitSigBobToAlice(1) // |<------ sig -------| (2)
receiveRevAndAckAliceToBob(t, aliceMsgs, aliceLink, bobChannel) // |------- rev ------>| (3) <--- Alice acks add-1 ctx.receiveRevAndAckAliceToBob() // |------- rev ------>| (3) <--- Alice acks add-1
receiveCommitSigAliceToBob(t, aliceMsgs, aliceLink, bobChannel, 1) // |------- sig ------>| (4) <--- Alice signs add-1 ctx.receiveCommitSigAliceToBob(1) // |------- sig ------>| (4) <--- Alice signs add-1
for i := 1; i < numHtlcs; i++ { // | | for i := 1; i < numHtlcs; i++ { // | |
sendHtlcBobToAlice(t, aliceLink, bobChannel, htlcs[i]) // |<----- add-i ------| (5.i) ctx.sendHtlcBobToAlice(htlcs[i]) // |<----- add-i ------| (5.i)
sendCommitSigBobToAlice(t, aliceLink, bobChannel, i+1) // |<------ sig -------| (6.i) ctx.sendCommitSigBobToAlice(i + 1) // |<------ sig -------| (6.i)
receiveRevAndAckAliceToBob(t, aliceMsgs, aliceLink, bobChannel) // |------- rev ------>| (7.i) <--- Alice acks add-i ctx.receiveRevAndAckAliceToBob() // |------- rev ------>| (7.i) <--- Alice acks add-i
select { // | | assertNoMsgFromAlice() // | |
case <-aliceMsgs: // | | Alice should not send a sig for // | | Alice should not send a sig for
t.Fatalf("unexpectedly received msg from Alice") // | | Bob's last state, since she is // | | Bob's last state, since she is
default: // | | still waiting for a revocation // | | still waiting for a revocation
} // | | for the previous one. // | | for the previous one.
} // | | } // | |
sendRevAndAckBobToAlice(t, aliceLink, bobChannel) // |<------ rev -------| (8) Finally let Bob send rev ctx.sendRevAndAckBobToAlice() // |<------ rev -------| (8) Finally let Bob send rev
receiveSettleAliceToBob(t, aliceMsgs, aliceLink, bobChannel) // |------ ful-1 ----->| (9) ctx.receiveSettleAliceToBob() // |------ ful-1 ----->| (9)
receiveCommitSigAliceToBob(t, aliceMsgs, aliceLink, bobChannel, numHtlcs-1) // |------- sig ------>| (10) <--- Alice signs add-i ctx.receiveCommitSigAliceToBob(numHtlcs - 1) // |------- sig ------>| (10) <--- Alice signs add-i
sendRevAndAckBobToAlice(t, aliceLink, bobChannel) // |<------ rev -------| (11) ctx.sendRevAndAckBobToAlice() // |<------ rev -------| (11)
for i := 1; i < numHtlcs; i++ { // | | for i := 1; i < numHtlcs; i++ { // | |
receiveSettleAliceToBob(t, aliceMsgs, aliceLink, bobChannel) // |------ ful-1 ----->| (12.i) ctx.receiveSettleAliceToBob() // |------ ful-1 ----->| (12.i)
} // | | } // | |
sendCommitSigBobToAlice(t, aliceLink, bobChannel, numHtlcs-1) // |<------ sig -------| (13) ctx.sendCommitSigBobToAlice(numHtlcs - 1) // |<------ sig -------| (13)
receiveCommitSigAliceToBob(t, aliceMsgs, aliceLink, bobChannel, 0) // |------- sig ------>| (14) ctx.receiveCommitSigAliceToBob(0) // |------- sig ------>| (14)
sendRevAndAckBobToAlice(t, aliceLink, bobChannel) // |<------ rev -------| (15) ctx.sendRevAndAckBobToAlice() // |<------ rev -------| (15)
receiveRevAndAckAliceToBob(t, aliceMsgs, aliceLink, bobChannel) // |------- rev ------>| (16) ctx.receiveRevAndAckAliceToBob() // |------- rev ------>| (16)
sendCommitSigBobToAlice(t, aliceLink, bobChannel, 0) // |<------ sig -------| (17) ctx.sendCommitSigBobToAlice(0) // |<------ sig -------| (17)
receiveRevAndAckAliceToBob(t, aliceMsgs, aliceLink, bobChannel) // |------- rev ------>| (18) ctx.receiveRevAndAckAliceToBob() // |------- rev ------>| (18)
// Both side's state is now updated, no more messages should be sent. // Both side's state is now updated, no more messages should be sent.
select { assertNoMsgFromAlice()
case <-aliceMsgs:
t.Fatalf("did not expect message from Alice")
case <-time.After(50 * time.Millisecond):
}
} }
// TestChannelLinkBatchPreimageWrite asserts that a link will batch preimage // TestChannelLinkBatchPreimageWrite asserts that a link will batch preimage
@ -4657,10 +4675,17 @@ func testChannelLinkBatchPreimageWrite(t *testing.T, disconnect bool) {
invoices = append(invoices, invoice) invoices = append(invoices, invoice)
} }
ctx := linkTestContext{
t: t,
aliceLink: aliceLink,
aliceMsgs: aliceMsgs,
bobChannel: bobChannel,
}
// First, send a batch of Adds from Alice to Bob. // First, send a batch of Adds from Alice to Bob.
for i, htlc := range htlcs { for i, htlc := range htlcs {
sendHtlcAliceToBob(t, aliceLink, i, htlc) ctx.sendHtlcAliceToBob(i, htlc)
receiveHtlcAliceToBob(t, aliceMsgs, bobChannel) ctx.receiveHtlcAliceToBob()
} }
// Assert that no preimages exist for these htlcs in Alice's cache. // Assert that no preimages exist for these htlcs in Alice's cache.
@ -4676,12 +4701,10 @@ func testChannelLinkBatchPreimageWrite(t *testing.T, disconnect bool) {
// Do a commitment dance to lock in the Adds, we expect numHtlcs htlcs // Do a commitment dance to lock in the Adds, we expect numHtlcs htlcs
// to be on each party's commitment transactions. // to be on each party's commitment transactions.
receiveCommitSigAliceToBob( ctx.receiveCommitSigAliceToBob(numHtlcs)
t, aliceMsgs, aliceLink, bobChannel, numHtlcs, ctx.sendRevAndAckBobToAlice()
) ctx.sendCommitSigBobToAlice(numHtlcs)
sendRevAndAckBobToAlice(t, aliceLink, bobChannel) ctx.receiveRevAndAckAliceToBob()
sendCommitSigBobToAlice(t, aliceLink, bobChannel, numHtlcs)
receiveRevAndAckAliceToBob(t, aliceMsgs, aliceLink, bobChannel)
// Check again that no preimages exist for these htlcs in Alice's cache. // Check again that no preimages exist for these htlcs in Alice's cache.
checkHasPreimages(t, coreLink, htlcs, false) checkHasPreimages(t, coreLink, htlcs, false)
@ -4689,8 +4712,8 @@ func testChannelLinkBatchPreimageWrite(t *testing.T, disconnect bool) {
// Now, have Bob settle the HTLCs back to Alice using the preimages in // Now, have Bob settle the HTLCs back to Alice using the preimages in
// the invoice corresponding to each of the HTLCs. // the invoice corresponding to each of the HTLCs.
for i, invoice := range invoices { for i, invoice := range invoices {
sendSettleBobToAlice( ctx.sendSettleBobToAlice(
t, aliceLink, bobChannel, uint64(i), uint64(i),
invoice.Terms.PaymentPreimage, invoice.Terms.PaymentPreimage,
) )
} }
@ -4711,7 +4734,7 @@ func testChannelLinkBatchPreimageWrite(t *testing.T, disconnect bool) {
// Otherwise, we are testing that Alice commits the preimages after // Otherwise, we are testing that Alice commits the preimages after
// receiving a CommitSig from Bob. Bob's commitment should now have 0 // receiving a CommitSig from Bob. Bob's commitment should now have 0
// HTLCs. // HTLCs.
sendCommitSigBobToAlice(t, aliceLink, bobChannel, 0) ctx.sendCommitSigBobToAlice(0)
// Since Alice will process the CommitSig asynchronously, we wait until // Since Alice will process the CommitSig asynchronously, we wait until
// she replies with her RevokeAndAck to ensure the tests reliably // she replies with her RevokeAndAck to ensure the tests reliably
@ -4762,6 +4785,13 @@ func TestChannelLinkCleanupSpuriousResponses(t *testing.T) {
htlc1 := generateHtlc(t, coreLink, bobChannel, 0) htlc1 := generateHtlc(t, coreLink, bobChannel, 0)
htlc2 := generateHtlc(t, coreLink, bobChannel, 1) htlc2 := generateHtlc(t, coreLink, bobChannel, 1)
ctx := linkTestContext{
t: t,
aliceLink: aliceLink,
aliceMsgs: aliceMsgs,
bobChannel: bobChannel,
}
// We start with he following scenario: Bob sends Alice two HTLCs, and a // We start with he following scenario: Bob sends Alice two HTLCs, and a
// commitment dance ensures, leaving two HTLCs that Alice can respond // commitment dance ensures, leaving two HTLCs that Alice can respond
// to. Since Alice is in ExitSettle mode, we will then take over and // to. Since Alice is in ExitSettle mode, we will then take over and
@ -4775,12 +4805,12 @@ func TestChannelLinkCleanupSpuriousResponses(t *testing.T) {
// |<----- rev ------| // |<----- rev ------|
// |<----- sig ------| commits add-1 + add-2 // |<----- sig ------| commits add-1 + add-2
// |------ rev ----->| // |------ rev ----->|
sendHtlcBobToAlice(t, aliceLink, bobChannel, htlc1) ctx.sendHtlcBobToAlice(htlc1)
sendHtlcBobToAlice(t, aliceLink, bobChannel, htlc2) ctx.sendHtlcBobToAlice(htlc2)
sendCommitSigBobToAlice(t, aliceLink, bobChannel, 2) ctx.sendCommitSigBobToAlice(2)
receiveRevAndAckAliceToBob(t, aliceMsgs, aliceLink, bobChannel) ctx.receiveRevAndAckAliceToBob()
receiveCommitSigAliceToBob(t, aliceMsgs, aliceLink, bobChannel, 2) ctx.receiveCommitSigAliceToBob(2)
sendRevAndAckBobToAlice(t, aliceLink, bobChannel) ctx.sendRevAndAckBobToAlice()
// Give Alice to time to process the revocation. // Give Alice to time to process the revocation.
time.Sleep(time.Second) time.Sleep(time.Second)
@ -4829,8 +4859,8 @@ func TestChannelLinkCleanupSpuriousResponses(t *testing.T) {
// Bob Alice // Bob Alice
// |<----- fal-1 ------| // |<----- fal-1 ------|
// |<----- sig ------| commits fal-1 // |<----- sig ------| commits fal-1
receiveFailAliceToBob(t, aliceMsgs, aliceLink, bobChannel) ctx.receiveFailAliceToBob()
receiveCommitSigAliceToBob(t, aliceMsgs, aliceLink, bobChannel, 1) ctx.receiveCommitSigAliceToBob(1)
aliceFwdPkgs, err = coreLink.channel.LoadFwdPkgs() aliceFwdPkgs, err = coreLink.channel.LoadFwdPkgs()
if err != nil { if err != nil {
@ -4866,9 +4896,9 @@ func TestChannelLinkCleanupSpuriousResponses(t *testing.T) {
// |------ rev ----->| // |------ rev ----->|
// |------ sig ----->| // |------ sig ----->|
// |<----- rev ------| // |<----- rev ------|
sendRevAndAckBobToAlice(t, aliceLink, bobChannel) ctx.sendRevAndAckBobToAlice()
sendCommitSigBobToAlice(t, aliceLink, bobChannel, 1) ctx.sendCommitSigBobToAlice(1)
receiveRevAndAckAliceToBob(t, aliceMsgs, aliceLink, bobChannel) ctx.receiveRevAndAckAliceToBob()
// Next, we'll construct a fail packet for add-2 (index 1), which we'll // Next, we'll construct a fail packet for add-2 (index 1), which we'll
// send to Bob and lock in. Since the AddRef is set on this instance, we // send to Bob and lock in. Since the AddRef is set on this instance, we
@ -4889,8 +4919,8 @@ func TestChannelLinkCleanupSpuriousResponses(t *testing.T) {
// Bob Alice // Bob Alice
// |<----- fal-1 ------| // |<----- fal-1 ------|
// |<----- sig ------| commits fal-1 // |<----- sig ------| commits fal-1
receiveFailAliceToBob(t, aliceMsgs, aliceLink, bobChannel) ctx.receiveFailAliceToBob()
receiveCommitSigAliceToBob(t, aliceMsgs, aliceLink, bobChannel, 0) ctx.receiveCommitSigAliceToBob(0)
aliceFwdPkgs, err = coreLink.channel.LoadFwdPkgs() aliceFwdPkgs, err = coreLink.channel.LoadFwdPkgs()
if err != nil { if err != nil {
@ -4932,9 +4962,9 @@ func TestChannelLinkCleanupSpuriousResponses(t *testing.T) {
// |------ rev ----->| // |------ rev ----->|
// |------ sig ----->| // |------ sig ----->|
// |<----- rev ------| // |<----- rev ------|
sendRevAndAckBobToAlice(t, aliceLink, bobChannel) ctx.sendRevAndAckBobToAlice()
sendCommitSigBobToAlice(t, aliceLink, bobChannel, 0) ctx.sendCommitSigBobToAlice(0)
receiveRevAndAckAliceToBob(t, aliceMsgs, aliceLink, bobChannel) ctx.receiveRevAndAckAliceToBob()
// We'll do a quick sanity check, and blindly send the same fail packet // We'll do a quick sanity check, and blindly send the same fail packet
// for the first HTLC. Since this HTLC index has already been settled, // for the first HTLC. Since this HTLC index has already been settled,
@ -5139,7 +5169,12 @@ func TestChannelLinkFail(t *testing.T) {
// Generate an HTLC and send to the link. // Generate an HTLC and send to the link.
htlc1 := generateHtlc(t, c, remoteChannel, 0) htlc1 := generateHtlc(t, c, remoteChannel, 0)
sendHtlcBobToAlice(t, c, remoteChannel, htlc1) ctx := linkTestContext{
t: t,
aliceLink: c,
bobChannel: remoteChannel,
}
ctx.sendHtlcBobToAlice(htlc1)
// Sign a commitment that will include // Sign a commitment that will include
// signature for the HTLC just sent. // signature for the HTLC just sent.
@ -5171,7 +5206,13 @@ func TestChannelLinkFail(t *testing.T) {
// Generate an HTLC and send to the link. // Generate an HTLC and send to the link.
htlc1 := generateHtlc(t, c, remoteChannel, 0) htlc1 := generateHtlc(t, c, remoteChannel, 0)
sendHtlcBobToAlice(t, c, remoteChannel, htlc1) ctx := linkTestContext{
t: t,
aliceLink: c,
bobChannel: remoteChannel,
}
ctx.sendHtlcBobToAlice(htlc1)
// Sign a commitment that will include // Sign a commitment that will include
// signature for the HTLC just sent. // signature for the HTLC just sent.
@ -5714,12 +5755,19 @@ func TestChannelLinkHoldInvoiceRestart(t *testing.T) {
t.Fatalf("unable to add invoice to registry: %v", err) t.Fatalf("unable to add invoice to registry: %v", err)
} }
ctx := linkTestContext{
t: t,
aliceLink: alice.link,
aliceMsgs: alice.msgs,
bobChannel: bobChannel,
}
// Lock in htlc paying the hodl invoice. // Lock in htlc paying the hodl invoice.
sendHtlcBobToAlice(t, alice.link, bobChannel, htlc) ctx.sendHtlcBobToAlice(htlc)
sendCommitSigBobToAlice(t, alice.link, bobChannel, 1) ctx.sendCommitSigBobToAlice(1)
receiveRevAndAckAliceToBob(t, alice.msgs, alice.link, bobChannel) ctx.receiveRevAndAckAliceToBob()
receiveCommitSigAliceToBob(t, alice.msgs, alice.link, bobChannel, 1) ctx.receiveCommitSigAliceToBob(1)
sendRevAndAckBobToAlice(t, alice.link, bobChannel) ctx.sendRevAndAckBobToAlice()
// We expect a call to the invoice registry to notify the arrival of the // We expect a call to the invoice registry to notify the arrival of the
// htlc. // htlc.
@ -5731,6 +5779,8 @@ func TestChannelLinkHoldInvoiceRestart(t *testing.T) {
// Restart link. // Restart link.
alice.restart(false) alice.restart(false)
ctx.aliceLink = alice.link
ctx.aliceMsgs = alice.msgs
// Expect htlc to be reprocessed. // Expect htlc to be reprocessed.
<-registry.settleChan <-registry.settleChan
@ -5739,8 +5789,8 @@ func TestChannelLinkHoldInvoiceRestart(t *testing.T) {
registry.SettleHodlInvoice(preimage) registry.SettleHodlInvoice(preimage)
// Expect alice to send a settle and commitsig message to bob. // Expect alice to send a settle and commitsig message to bob.
receiveSettleAliceToBob(t, alice.msgs, alice.link, bobChannel) ctx.receiveSettleAliceToBob()
receiveCommitSigAliceToBob(t, alice.msgs, alice.link, bobChannel, 0) ctx.receiveCommitSigAliceToBob(0)
// Stop the link // Stop the link
alice.link.Stop() alice.link.Stop()
@ -5806,12 +5856,19 @@ func TestChannelLinkRevocationWindowHodl(t *testing.T) {
t.Fatalf("unable to add invoice to registry: %v", err) t.Fatalf("unable to add invoice to registry: %v", err)
} }
ctx := linkTestContext{
t: t,
aliceLink: aliceLink,
aliceMsgs: aliceMsgs,
bobChannel: bobChannel,
}
// Lock in htlc 1 on both sides. // Lock in htlc 1 on both sides.
sendHtlcBobToAlice(t, aliceLink, bobChannel, htlc1) ctx.sendHtlcBobToAlice(htlc1)
sendCommitSigBobToAlice(t, aliceLink, bobChannel, 1) ctx.sendCommitSigBobToAlice(1)
receiveRevAndAckAliceToBob(t, aliceMsgs, aliceLink, bobChannel) ctx.receiveRevAndAckAliceToBob()
receiveCommitSigAliceToBob(t, aliceMsgs, aliceLink, bobChannel, 1) ctx.receiveCommitSigAliceToBob(1)
sendRevAndAckBobToAlice(t, aliceLink, bobChannel) ctx.sendRevAndAckBobToAlice()
// We expect a call to the invoice registry to notify the arrival of // We expect a call to the invoice registry to notify the arrival of
// htlc 1. // htlc 1.
@ -5822,11 +5879,11 @@ func TestChannelLinkRevocationWindowHodl(t *testing.T) {
} }
// Lock in htlc 2 on both sides. // Lock in htlc 2 on both sides.
sendHtlcBobToAlice(t, aliceLink, bobChannel, htlc2) ctx.sendHtlcBobToAlice(htlc2)
sendCommitSigBobToAlice(t, aliceLink, bobChannel, 2) ctx.sendCommitSigBobToAlice(2)
receiveRevAndAckAliceToBob(t, aliceMsgs, aliceLink, bobChannel) ctx.receiveRevAndAckAliceToBob()
receiveCommitSigAliceToBob(t, aliceMsgs, aliceLink, bobChannel, 2) ctx.receiveCommitSigAliceToBob(2)
sendRevAndAckBobToAlice(t, aliceLink, bobChannel) ctx.sendRevAndAckBobToAlice()
select { select {
case <-registry.settleChan: case <-registry.settleChan:
@ -5839,14 +5896,14 @@ func TestChannelLinkRevocationWindowHodl(t *testing.T) {
// Expect alice to send a settle and commitsig message to bob. Bob does // Expect alice to send a settle and commitsig message to bob. Bob does
// not yet send the revocation. // not yet send the revocation.
receiveSettleAliceToBob(t, aliceMsgs, aliceLink, bobChannel) ctx.receiveSettleAliceToBob()
receiveCommitSigAliceToBob(t, aliceMsgs, aliceLink, bobChannel, 1) ctx.receiveCommitSigAliceToBob(1)
// Settle invoice 2 with the preimage. // Settle invoice 2 with the preimage.
registry.SettleHodlInvoice(preimage2) registry.SettleHodlInvoice(preimage2)
// Expect alice to send a settle for htlc 2. // Expect alice to send a settle for htlc 2.
receiveSettleAliceToBob(t, aliceMsgs, aliceLink, bobChannel) ctx.receiveSettleAliceToBob()
// At this point, Alice cannot send a new commit sig to bob because the // At this point, Alice cannot send a new commit sig to bob because the
// revocation window is exhausted. // revocation window is exhausted.
@ -5862,7 +5919,7 @@ func TestChannelLinkRevocationWindowHodl(t *testing.T) {
} }
// Bob sends revocation and signs commit with htlc 1 settled. // Bob sends revocation and signs commit with htlc 1 settled.
sendRevAndAckBobToAlice(t, aliceLink, bobChannel) ctx.sendRevAndAckBobToAlice()
// Allow some time for it to be processed by the link. // Allow some time for it to be processed by the link.
time.Sleep(time.Second) time.Sleep(time.Second)
@ -5873,12 +5930,12 @@ func TestChannelLinkRevocationWindowHodl(t *testing.T) {
// After the revocation, it is again possible for Alice to send a commit // After the revocation, it is again possible for Alice to send a commit
// sig no more htlcs. Bob acks the update. // sig no more htlcs. Bob acks the update.
receiveCommitSigAliceToBob(t, aliceMsgs, aliceLink, bobChannel, 0) ctx.receiveCommitSigAliceToBob(0)
sendRevAndAckBobToAlice(t, aliceLink, bobChannel) ctx.sendRevAndAckBobToAlice()
// Bob updates his remote commit tx. // Bob updates his remote commit tx.
sendCommitSigBobToAlice(t, aliceLink, bobChannel, 0) ctx.sendCommitSigBobToAlice(0)
receiveRevAndAckAliceToBob(t, aliceMsgs, aliceLink, bobChannel) ctx.receiveRevAndAckAliceToBob()
// Stop the link // Stop the link
aliceLink.Stop() aliceLink.Stop()