peer: return an error from updateNextRevocation and patch unit tests

This commit makes the `updateNextRevocation` to return an error and
further feed it through the request's error chan so it'd be handled by
the caller.
This commit is contained in:
yyforyongyu 2023-06-12 23:41:20 +08:00
parent 927572583b
commit d28242c664
No known key found for this signature in database
GPG Key ID: 9BCD95C4FF296868
2 changed files with 68 additions and 8 deletions

@ -3734,7 +3734,7 @@ func (p *Brontide) attachChannelEventSubscription() error {
// updateNextRevocation updates the existing channel's next revocation if it's
// nil.
func (p *Brontide) updateNextRevocation(c *channeldb.OpenChannel) {
func (p *Brontide) updateNextRevocation(c *channeldb.OpenChannel) error {
chanPoint := &c.FundingOutpoint
chanID := lnwire.NewChanIDFromOutPoint(chanPoint)
@ -3744,32 +3744,35 @@ func (p *Brontide) updateNextRevocation(c *channeldb.OpenChannel) {
// currentChan should exist, but we perform a check anyway to avoid nil
// pointer dereference.
if !loaded {
p.log.Errorf("missing active channel with chanID=%v", chanID)
return
return fmt.Errorf("missing active channel with chanID=%v",
chanID)
}
// currentChan should not be nil, but we perform a check anyway to
// avoid nil pointer dereference.
if currentChan == nil {
p.log.Errorf("found nil active channel with chanID=%v", chanID)
return
return fmt.Errorf("found nil active channel with chanID=%v",
chanID)
}
// If we're being sent a new channel, and our existing channel doesn't
// have the next revocation, then we need to update the current
// existing channel.
if currentChan.RemoteNextRevocation() != nil {
return
return nil
}
p.log.Infof("Processing retransmitted ChannelReady for "+
"ChannelPoint(%v)", chanPoint)
nextRevoke := c.RemoteNextRevocation
err := currentChan.InitNextRevocation(nextRevoke)
if err != nil {
p.log.Errorf("unable to init chan revocation: %v", err)
return fmt.Errorf("unable to init next revocation: %w", err)
}
return nil
}
// addActiveChannel adds a new active channel to the `activeChannels` map. It
@ -3855,7 +3858,11 @@ func (p *Brontide) handleNewActiveChannel(req *newChannelMsg) {
// Handle it and close the err chan on the request.
close(req.err)
p.updateNextRevocation(newChan)
// Update the next revocation point.
if err := p.updateNextRevocation(newChan); err != nil {
p.log.Errorf(err.Error())
}
return
}

@ -1118,3 +1118,56 @@ func TestPeerCustomMessage(t *testing.T) {
require.Equal(t, remoteKey, receivedCustom.peer)
require.Equal(t, receivedCustomMsg, &receivedCustom.msg)
}
// TestUpdateNextRevocation checks that the method `updateNextRevocation` is
// behave as expected.
func TestUpdateNextRevocation(t *testing.T) {
t.Parallel()
require := require.New(t)
// TODO(yy): create interface for lnwallet.LightningChannel so we can
// easily mock it without the following setups.
notifier := &mock.ChainNotifier{
SpendChan: make(chan *chainntnfs.SpendDetail),
EpochChan: make(chan *chainntnfs.BlockEpoch),
ConfChan: make(chan *chainntnfs.TxConfirmation),
}
broadcastTxChan := make(chan *wire.MsgTx)
mockSwitch := &mockMessageSwitch{}
alicePeer, bobChan, err := createTestPeer(
t, notifier, broadcastTxChan, noUpdate, mockSwitch,
)
require.NoError(err, "unable to create test channels")
// testChannel is used to test the updateNextRevocation function.
testChannel := bobChan.State()
// Update the next revocation for a known channel should give us no
// error.
err = alicePeer.updateNextRevocation(testChannel)
require.NoError(err, "expected no error")
// Test an error is returned when the chanID cannot be found in
// `activeChannels` map.
testChannel.FundingOutpoint = wire.OutPoint{Index: 0}
err = alicePeer.updateNextRevocation(testChannel)
require.Error(err, "expected an error")
// Test an error is returned when the chanID's corresponding channel is
// nil.
testChannel.FundingOutpoint = wire.OutPoint{Index: 1}
chanID := lnwire.NewChanIDFromOutPoint(&testChannel.FundingOutpoint)
alicePeer.activeChannels.Store(chanID, nil)
err = alicePeer.updateNextRevocation(testChannel)
require.Error(err, "expected an error")
// TODO(yy): should also test `InitNextRevocation` is called on
// `lnwallet.LightningWallet` once it's interfaced.
}
// TODO(yy): add test for `addActiveChannel` and `handleNewActiveChannel` once
// we have interfaced `lnwallet.LightningChannel` and
// `*contractcourt.ChainArbitrator`.