cnct: add channel_arbitor test that exercises the htlcOutgoingContestResolver
This commit is contained in:
parent
6d759bd8d9
commit
f34ffab9ba
@ -15,16 +15,22 @@ import (
|
|||||||
|
|
||||||
type mockNotifier struct {
|
type mockNotifier struct {
|
||||||
spendChan chan *chainntnfs.SpendDetail
|
spendChan chan *chainntnfs.SpendDetail
|
||||||
|
epochChan chan *chainntnfs.BlockEpoch
|
||||||
|
confChan chan *chainntnfs.TxConfirmation
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *mockNotifier) RegisterConfirmationsNtfn(txid *chainhash.Hash, _ []byte, numConfs,
|
func (m *mockNotifier) RegisterConfirmationsNtfn(txid *chainhash.Hash, _ []byte, numConfs,
|
||||||
heightHint uint32) (*chainntnfs.ConfirmationEvent, error) {
|
heightHint uint32) (*chainntnfs.ConfirmationEvent, error) {
|
||||||
return nil, nil
|
return &chainntnfs.ConfirmationEvent{
|
||||||
|
Confirmed: m.confChan,
|
||||||
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *mockNotifier) RegisterBlockEpochNtfn(
|
func (m *mockNotifier) RegisterBlockEpochNtfn(
|
||||||
bestBlock *chainntnfs.BlockEpoch) (*chainntnfs.BlockEpochEvent, error) {
|
bestBlock *chainntnfs.BlockEpoch) (*chainntnfs.BlockEpochEvent, error) {
|
||||||
|
|
||||||
return &chainntnfs.BlockEpochEvent{
|
return &chainntnfs.BlockEpochEvent{
|
||||||
Epochs: make(chan *chainntnfs.BlockEpoch),
|
Epochs: m.epochChan,
|
||||||
Cancel: func() {},
|
Cancel: func() {},
|
||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
@ -38,6 +44,7 @@ func (m *mockNotifier) Stop() error {
|
|||||||
}
|
}
|
||||||
func (m *mockNotifier) RegisterSpendNtfn(outpoint *wire.OutPoint, _ []byte,
|
func (m *mockNotifier) RegisterSpendNtfn(outpoint *wire.OutPoint, _ []byte,
|
||||||
heightHint uint32) (*chainntnfs.SpendEvent, error) {
|
heightHint uint32) (*chainntnfs.SpendEvent, error) {
|
||||||
|
|
||||||
return &chainntnfs.SpendEvent{
|
return &chainntnfs.SpendEvent{
|
||||||
Spend: m.spendChan,
|
Spend: m.spendChan,
|
||||||
Cancel: func() {},
|
Cancel: func() {},
|
||||||
|
@ -2,6 +2,7 @@ package contractcourt
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"sync"
|
||||||
"testing"
|
"testing"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
@ -20,6 +21,11 @@ type mockArbitratorLog struct {
|
|||||||
failFetch error
|
failFetch error
|
||||||
failCommit bool
|
failCommit bool
|
||||||
failCommitState ArbitratorState
|
failCommitState ArbitratorState
|
||||||
|
resolutions *ContractResolutions
|
||||||
|
chainActions ChainActionMap
|
||||||
|
resolvers map[ContractResolver]struct{}
|
||||||
|
|
||||||
|
sync.Mutex
|
||||||
}
|
}
|
||||||
|
|
||||||
// A compile time check to ensure mockArbitratorLog meets the ArbitratorLog
|
// A compile time check to ensure mockArbitratorLog meets the ArbitratorLog
|
||||||
@ -40,20 +46,48 @@ func (b *mockArbitratorLog) CommitState(s ArbitratorState) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *mockArbitratorLog) FetchUnresolvedContracts() ([]ContractResolver, error) {
|
func (b *mockArbitratorLog) FetchUnresolvedContracts() ([]ContractResolver,
|
||||||
var contracts []ContractResolver
|
error) {
|
||||||
return contracts, nil
|
|
||||||
|
b.Lock()
|
||||||
|
v := make([]ContractResolver, len(b.resolvers))
|
||||||
|
idx := 0
|
||||||
|
for resolver := range b.resolvers {
|
||||||
|
v[idx] = resolver
|
||||||
|
idx++
|
||||||
|
}
|
||||||
|
b.Unlock()
|
||||||
|
|
||||||
|
return v, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *mockArbitratorLog) InsertUnresolvedContracts(resolvers ...ContractResolver) error {
|
func (b *mockArbitratorLog) InsertUnresolvedContracts(
|
||||||
|
resolvers ...ContractResolver) error {
|
||||||
|
|
||||||
|
b.Lock()
|
||||||
|
for _, resolver := range resolvers {
|
||||||
|
b.resolvers[resolver] = struct{}{}
|
||||||
|
}
|
||||||
|
b.Unlock()
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *mockArbitratorLog) SwapContract(oldContract, newContract ContractResolver) error {
|
func (b *mockArbitratorLog) SwapContract(oldContract,
|
||||||
|
newContract ContractResolver) error {
|
||||||
|
|
||||||
|
b.Lock()
|
||||||
|
delete(b.resolvers, oldContract)
|
||||||
|
b.resolvers[newContract] = struct{}{}
|
||||||
|
b.Unlock()
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *mockArbitratorLog) ResolveContract(res ContractResolver) error {
|
func (b *mockArbitratorLog) ResolveContract(res ContractResolver) error {
|
||||||
|
b.Lock()
|
||||||
|
delete(b.resolvers, res)
|
||||||
|
b.Unlock()
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -61,6 +95,7 @@ func (b *mockArbitratorLog) LogContractResolutions(c *ContractResolutions) error
|
|||||||
if b.failLog {
|
if b.failLog {
|
||||||
return fmt.Errorf("intentional log failure")
|
return fmt.Errorf("intentional log failure")
|
||||||
}
|
}
|
||||||
|
b.resolutions = c
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -68,17 +103,17 @@ func (b *mockArbitratorLog) FetchContractResolutions() (*ContractResolutions, er
|
|||||||
if b.failFetch != nil {
|
if b.failFetch != nil {
|
||||||
return nil, b.failFetch
|
return nil, b.failFetch
|
||||||
}
|
}
|
||||||
c := &ContractResolutions{}
|
|
||||||
|
|
||||||
return c, nil
|
return b.resolutions, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *mockArbitratorLog) LogChainActions(actions ChainActionMap) error {
|
func (b *mockArbitratorLog) LogChainActions(actions ChainActionMap) error {
|
||||||
|
b.chainActions = actions
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *mockArbitratorLog) FetchChainActions() (ChainActionMap, error) {
|
func (b *mockArbitratorLog) FetchChainActions() (ChainActionMap, error) {
|
||||||
actionsMap := make(ChainActionMap)
|
actionsMap := b.chainActions
|
||||||
return actionsMap, nil
|
return actionsMap, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -126,6 +161,20 @@ func createTestChannelArbitrator(log ArbitratorLog) (*ChannelArbitrator,
|
|||||||
PublishTx: func(*wire.MsgTx) error {
|
PublishTx: func(*wire.MsgTx) error {
|
||||||
return nil
|
return nil
|
||||||
},
|
},
|
||||||
|
DeliverResolutionMsg: func(...ResolutionMsg) error {
|
||||||
|
return nil
|
||||||
|
},
|
||||||
|
BroadcastDelta: 5,
|
||||||
|
Notifier: &mockNotifier{
|
||||||
|
epochChan: make(chan *chainntnfs.BlockEpoch),
|
||||||
|
spendChan: make(chan *chainntnfs.SpendDetail),
|
||||||
|
confChan: make(chan *chainntnfs.TxConfirmation),
|
||||||
|
},
|
||||||
|
IncubateOutputs: func(wire.OutPoint, *lnwallet.CommitOutputResolution,
|
||||||
|
*lnwallet.OutgoingHtlcResolution,
|
||||||
|
*lnwallet.IncomingHtlcResolution) error {
|
||||||
|
return nil
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
// We'll use the resolvedChan to synchronize on call to
|
// We'll use the resolvedChan to synchronize on call to
|
||||||
@ -398,6 +447,178 @@ func TestChannelArbitratorLocalForceClose(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TestChannelArbitratorLocalForceClosePendingHtlc tests that the
|
||||||
|
// ChannelArbitrator goes through the expected states in case we request it to
|
||||||
|
// force close a channel that still has an HTLC pending.
|
||||||
|
func TestChannelArbitratorLocalForceClosePendingHtlc(t *testing.T) {
|
||||||
|
arbLog := &mockArbitratorLog{
|
||||||
|
state: StateDefault,
|
||||||
|
newStates: make(chan ArbitratorState, 5),
|
||||||
|
resolvers: make(map[ContractResolver]struct{}),
|
||||||
|
}
|
||||||
|
|
||||||
|
chanArb, resolved, err := createTestChannelArbitrator(arbLog)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("unable to create ChannelArbitrator: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
incubateChan := make(chan struct{})
|
||||||
|
chanArb.cfg.IncubateOutputs = func(outPoint wire.OutPoint,
|
||||||
|
commitOutputRes *lnwallet.CommitOutputResolution,
|
||||||
|
outgoingHtlcRes *lnwallet.OutgoingHtlcResolution,
|
||||||
|
incomingHtlcRes *lnwallet.IncomingHtlcResolution) error {
|
||||||
|
|
||||||
|
incubateChan <- struct{}{}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := chanArb.Start(); err != nil {
|
||||||
|
t.Fatalf("unable to start ChannelArbitrator: %v", err)
|
||||||
|
}
|
||||||
|
defer chanArb.Stop()
|
||||||
|
|
||||||
|
// Create htlcUpdates channel.
|
||||||
|
htlcUpdates := make(chan []channeldb.HTLC)
|
||||||
|
|
||||||
|
signals := &ContractSignals{
|
||||||
|
HtlcUpdates: htlcUpdates,
|
||||||
|
ShortChanID: lnwire.ShortChannelID{},
|
||||||
|
}
|
||||||
|
chanArb.UpdateContractSignals(signals)
|
||||||
|
|
||||||
|
// Add HTLC to channel arbitrator.
|
||||||
|
htlc := channeldb.HTLC{
|
||||||
|
Incoming: false,
|
||||||
|
Amt: 10000,
|
||||||
|
HtlcIndex: 0,
|
||||||
|
}
|
||||||
|
|
||||||
|
htlcUpdates <- []channeldb.HTLC{
|
||||||
|
htlc,
|
||||||
|
}
|
||||||
|
|
||||||
|
errChan := make(chan error, 1)
|
||||||
|
respChan := make(chan *wire.MsgTx, 1)
|
||||||
|
|
||||||
|
// With the channel found, and the request crafted, we'll send over a
|
||||||
|
// force close request to the arbitrator that watches this channel.
|
||||||
|
chanArb.forceCloseReqs <- &forceCloseReq{
|
||||||
|
errResp: errChan,
|
||||||
|
closeTx: respChan,
|
||||||
|
}
|
||||||
|
|
||||||
|
// The force close request should trigger broadcast of the commitment
|
||||||
|
// transaction.
|
||||||
|
assertStateTransitions(t, arbLog.newStates,
|
||||||
|
StateBroadcastCommit, StateCommitmentBroadcasted)
|
||||||
|
select {
|
||||||
|
case <-respChan:
|
||||||
|
case <-time.After(5 * time.Second):
|
||||||
|
t.Fatalf("no response received")
|
||||||
|
}
|
||||||
|
|
||||||
|
select {
|
||||||
|
case err := <-errChan:
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("error force closing channel: %v", err)
|
||||||
|
}
|
||||||
|
case <-time.After(5 * time.Second):
|
||||||
|
t.Fatalf("no response received")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Now notify about the local force close getting confirmed.
|
||||||
|
closeTx := &wire.MsgTx{}
|
||||||
|
|
||||||
|
htlcOp := wire.OutPoint{
|
||||||
|
Hash: closeTx.TxHash(),
|
||||||
|
Index: 0,
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set up the outgoing resolution. Populate SignedTimeoutTx because
|
||||||
|
// our commitment transaction got confirmed.
|
||||||
|
outgoingRes := lnwallet.OutgoingHtlcResolution{
|
||||||
|
Expiry: 10,
|
||||||
|
SweepSignDesc: lnwallet.SignDescriptor{
|
||||||
|
Output: &wire.TxOut{},
|
||||||
|
},
|
||||||
|
SignedTimeoutTx: &wire.MsgTx{
|
||||||
|
TxIn: []*wire.TxIn{
|
||||||
|
{
|
||||||
|
PreviousOutPoint: htlcOp,
|
||||||
|
Witness: [][]byte{{}},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
TxOut: []*wire.TxOut{
|
||||||
|
{},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
chanArb.cfg.ChainEvents.LocalUnilateralClosure <- &LocalUnilateralCloseInfo{
|
||||||
|
&chainntnfs.SpendDetail{},
|
||||||
|
&lnwallet.LocalForceCloseSummary{
|
||||||
|
CloseTx: closeTx,
|
||||||
|
HtlcResolutions: &lnwallet.HtlcResolutions{
|
||||||
|
OutgoingHTLCs: []lnwallet.OutgoingHtlcResolution{
|
||||||
|
outgoingRes,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
&channeldb.ChannelCloseSummary{},
|
||||||
|
}
|
||||||
|
|
||||||
|
assertStateTransitions(t, arbLog.newStates, StateContractClosed,
|
||||||
|
StateWaitingFullResolution)
|
||||||
|
|
||||||
|
// htlcOutgoingContestResolver is now active and waiting for the HTLC to
|
||||||
|
// expire. It should not yet have passed it on for incubation.
|
||||||
|
select {
|
||||||
|
case <-incubateChan:
|
||||||
|
t.Fatalf("contract should not be incubated yet")
|
||||||
|
default:
|
||||||
|
}
|
||||||
|
|
||||||
|
// Send a notification that the expiry height has been reached.
|
||||||
|
notifier := chanArb.cfg.Notifier.(*mockNotifier)
|
||||||
|
notifier.epochChan <- &chainntnfs.BlockEpoch{Height: 10}
|
||||||
|
|
||||||
|
// htlcOutgoingContestResolver is now transforming into a
|
||||||
|
// htlcTimeoutResolver and should send the contract off for incubation.
|
||||||
|
select {
|
||||||
|
case <-incubateChan:
|
||||||
|
case <-time.After(5 * time.Second):
|
||||||
|
t.Fatalf("no response received")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Notify resolver that output of the commitment has been spent.
|
||||||
|
notifier.confChan <- &chainntnfs.TxConfirmation{}
|
||||||
|
|
||||||
|
// As this is our own commitment transaction, the HTLC will go through
|
||||||
|
// to the second level. Channel arbitrator should still not be marked as
|
||||||
|
// resolved.
|
||||||
|
select {
|
||||||
|
case <-resolved:
|
||||||
|
t.Fatalf("channel resolved prematurely")
|
||||||
|
default:
|
||||||
|
}
|
||||||
|
|
||||||
|
// Notify resolver that the second level transaction is spent.
|
||||||
|
notifier.spendChan <- &chainntnfs.SpendDetail{}
|
||||||
|
|
||||||
|
// At this point channel should be marked as resolved.
|
||||||
|
|
||||||
|
// It should transition StateWaitingFullResolution ->
|
||||||
|
// StateFullyResolved, but this isn't happening.
|
||||||
|
// assertStateTransitions(t, arbLog.newStates, StateFullyResolved)
|
||||||
|
|
||||||
|
select {
|
||||||
|
case <-resolved:
|
||||||
|
case <-time.After(5 * time.Second):
|
||||||
|
t.Fatalf("contract was not resolved")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// TestChannelArbitratorLocalForceCloseRemoteConfiremd tests that the
|
// TestChannelArbitratorLocalForceCloseRemoteConfiremd tests that the
|
||||||
// ChannelArbitrator behaves as expected in the case where we request a local
|
// ChannelArbitrator behaves as expected in the case where we request a local
|
||||||
// force close, but a remote commitment ends up being confirmed in chain.
|
// force close, but a remote commitment ends up being confirmed in chain.
|
||||||
|
Loading…
Reference in New Issue
Block a user