contractcourt/channel_arbitrator: override trigger on startup if IsClosing
At ChannelArbitrator startup we now check the database close status of the channel. If we detect that the channel is closed, but our state machine hasn't advanced to reflect that (possibly because of a shutdown before the state transition was finished), we manually trigger the state transition to recover.
This commit is contained in:
parent
1758ad829f
commit
f2a033e965
@ -262,12 +262,38 @@ func (c *ChannelArbitrator) Start() error {
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// If the channel has been marked pending close in the database, and we
|
||||||
|
// haven't transitioned the state machine to StateContractClosed (or a
|
||||||
|
// suceeding state), then a state transition most likely failed. We'll
|
||||||
|
// try to recover from this by manually advancing the state by setting
|
||||||
|
// the corresponding close trigger.
|
||||||
|
trigger := chainTrigger
|
||||||
|
triggerHeight := uint32(bestHeight)
|
||||||
|
if c.cfg.IsPendingClose {
|
||||||
|
switch c.state {
|
||||||
|
case StateDefault:
|
||||||
|
fallthrough
|
||||||
|
case StateBroadcastCommit:
|
||||||
|
fallthrough
|
||||||
|
case StateCommitmentBroadcasted:
|
||||||
|
switch c.cfg.CloseType {
|
||||||
|
case channeldb.LocalForceClose:
|
||||||
|
trigger = localCloseTrigger
|
||||||
|
case channeldb.RemoteForceClose:
|
||||||
|
trigger = remoteCloseTrigger
|
||||||
|
}
|
||||||
|
triggerHeight = c.cfg.ClosingHeight
|
||||||
|
|
||||||
|
log.Warnf("ChannelArbitrator(%v): detected stalled "+
|
||||||
|
"state=%v for closed channel, using "+
|
||||||
|
"trigger=%v", c.cfg.ChanPoint, c.state, trigger)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// We'll now attempt to advance our state forward based on the current
|
// We'll now attempt to advance our state forward based on the current
|
||||||
// on-chain state, and our set of active contracts.
|
// on-chain state, and our set of active contracts.
|
||||||
startingState := c.state
|
startingState := c.state
|
||||||
nextState, _, err := c.advanceState(
|
nextState, _, err := c.advanceState(triggerHeight, trigger)
|
||||||
uint32(bestHeight), chainTrigger,
|
|
||||||
)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
c.cfg.BlockEpochs.Cancel()
|
c.cfg.BlockEpochs.Cancel()
|
||||||
return err
|
return err
|
||||||
|
Loading…
Reference in New Issue
Block a user