multi: add relative thaw height interpretation
This is useful when we wish to have a channel frozen for a specific amount of blocks after its confirmation. This could also be done with an absolute thaw height, but it does not suit cases where a strict block delta needs to be enforced, as it's not possible to know for certain when a channel will be included in the chain. To work around this, we add a relative interpretation of the field, where if its value is below 500,000, then it's interpreted as a relative height. This approach allows us to prevent further database modifications to account for a relative thaw height.
This commit is contained in:
parent
01ba4d0b59
commit
6075997ebc
@ -23,6 +23,13 @@ import (
|
|||||||
"github.com/lightningnetwork/lnd/shachain"
|
"github.com/lightningnetwork/lnd/shachain"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
// AbsoluteThawHeightThreshold is the threshold at which a thaw height
|
||||||
|
// begins to be interpreted as an absolute block height, rather than a
|
||||||
|
// relative one.
|
||||||
|
AbsoluteThawHeightThreshold uint32 = 500000
|
||||||
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
// closedChannelBucket stores summarization information concerning
|
// closedChannelBucket stores summarization information concerning
|
||||||
// previously open, but now closed channels.
|
// previously open, but now closed channels.
|
||||||
@ -654,7 +661,8 @@ type OpenChannel struct {
|
|||||||
|
|
||||||
// ThawHeight is the height when a frozen channel once again becomes a
|
// ThawHeight is the height when a frozen channel once again becomes a
|
||||||
// normal channel. If this is zero, then there're no restrictions on
|
// normal channel. If this is zero, then there're no restrictions on
|
||||||
// this channel.
|
// this channel. If the value is lower than 500,000, then it's
|
||||||
|
// interpreted as a relative height, or an absolute height otherwise.
|
||||||
ThawHeight uint32
|
ThawHeight uint32
|
||||||
|
|
||||||
// TODO(roasbeef): eww
|
// TODO(roasbeef): eww
|
||||||
@ -2766,6 +2774,28 @@ func (c *OpenChannel) RemoteRevocationStore() (shachain.Store, error) {
|
|||||||
return c.RevocationStore, nil
|
return c.RevocationStore, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// AbsoluteThawHeight determines a frozen channel's absolute thaw height. If the
|
||||||
|
// channel is not frozen, then 0 is returned.
|
||||||
|
func (c *OpenChannel) AbsoluteThawHeight() (uint32, error) {
|
||||||
|
// Only frozen channels have a thaw height.
|
||||||
|
if !c.ChanType.IsFrozen() {
|
||||||
|
return 0, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// If the channel's thaw height is below the absolute threshold, then
|
||||||
|
// it's interpreted as a relative height to the chain's current height.
|
||||||
|
if c.ThawHeight < AbsoluteThawHeightThreshold {
|
||||||
|
// We'll only known of the channel's short ID once it's
|
||||||
|
// confirmed.
|
||||||
|
if c.IsPending {
|
||||||
|
return 0, errors.New("cannot use relative thaw " +
|
||||||
|
"height for unconfirmed channel")
|
||||||
|
}
|
||||||
|
return c.ShortChannelID.BlockHeight + c.ThawHeight, nil
|
||||||
|
}
|
||||||
|
return c.ThawHeight, nil
|
||||||
|
}
|
||||||
|
|
||||||
func putChannelCloseSummary(tx kvdb.RwTx, chanID []byte,
|
func putChannelCloseSummary(tx kvdb.RwTx, chanID []byte,
|
||||||
summary *ChannelCloseSummary, lastChanState *OpenChannel) error {
|
summary *ChannelCloseSummary, lastChanState *OpenChannel) error {
|
||||||
|
|
||||||
|
@ -2891,7 +2891,9 @@ type Channel struct {
|
|||||||
//frozen channel doest not allow a cooperative channel close by the
|
//frozen channel doest not allow a cooperative channel close by the
|
||||||
//initiator. The thaw_height is the height that this restriction stops
|
//initiator. The thaw_height is the height that this restriction stops
|
||||||
//applying to the channel. This field is optional, not setting it or using a
|
//applying to the channel. This field is optional, not setting it or using a
|
||||||
//value of zero will mean the channel has no additional restrictions.
|
//value of zero will mean the channel has no additional restrictions. The
|
||||||
|
//height can be interpreted in two ways: as a relative height if the value is
|
||||||
|
//less than 500,000, or as an absolute height otherwise.
|
||||||
ThawHeight uint32 `protobuf:"varint,28,opt,name=thaw_height,json=thawHeight,proto3" json:"thaw_height,omitempty"`
|
ThawHeight uint32 `protobuf:"varint,28,opt,name=thaw_height,json=thawHeight,proto3" json:"thaw_height,omitempty"`
|
||||||
// List constraints for the local node.
|
// List constraints for the local node.
|
||||||
LocalConstraints *ChannelConstraints `protobuf:"bytes,29,opt,name=local_constraints,json=localConstraints,proto3" json:"local_constraints,omitempty"`
|
LocalConstraints *ChannelConstraints `protobuf:"bytes,29,opt,name=local_constraints,json=localConstraints,proto3" json:"local_constraints,omitempty"`
|
||||||
@ -5051,10 +5053,11 @@ type ChanPointShim struct {
|
|||||||
//channel ID.
|
//channel ID.
|
||||||
PendingChanId []byte `protobuf:"bytes,5,opt,name=pending_chan_id,json=pendingChanId,proto3" json:"pending_chan_id,omitempty"`
|
PendingChanId []byte `protobuf:"bytes,5,opt,name=pending_chan_id,json=pendingChanId,proto3" json:"pending_chan_id,omitempty"`
|
||||||
//
|
//
|
||||||
//This uint32 indicates if this channel is to be considered 'frozen'. A
|
//This uint32 indicates if this channel is to be considered 'frozen'. A frozen
|
||||||
//frozen channel does not allow a cooperative channel close by the
|
//channel does not allow a cooperative channel close by the initiator. The
|
||||||
//initiator. The thaw_height is the height that this restriction stops
|
//thaw_height is the height that this restriction stops applying to the
|
||||||
//applying to the channel.
|
//channel. The height can be interpreted in two ways: as a relative height if
|
||||||
|
//the value is less than 500,000, or as an absolute height otherwise.
|
||||||
ThawHeight uint32 `protobuf:"varint,6,opt,name=thaw_height,json=thawHeight,proto3" json:"thaw_height,omitempty"`
|
ThawHeight uint32 `protobuf:"varint,6,opt,name=thaw_height,json=thawHeight,proto3" json:"thaw_height,omitempty"`
|
||||||
XXX_NoUnkeyedLiteral struct{} `json:"-"`
|
XXX_NoUnkeyedLiteral struct{} `json:"-"`
|
||||||
XXX_unrecognized []byte `json:"-"`
|
XXX_unrecognized []byte `json:"-"`
|
||||||
|
@ -1157,7 +1157,9 @@ message Channel {
|
|||||||
frozen channel doest not allow a cooperative channel close by the
|
frozen channel doest not allow a cooperative channel close by the
|
||||||
initiator. The thaw_height is the height that this restriction stops
|
initiator. The thaw_height is the height that this restriction stops
|
||||||
applying to the channel. This field is optional, not setting it or using a
|
applying to the channel. This field is optional, not setting it or using a
|
||||||
value of zero will mean the channel has no additional restrictions.
|
value of zero will mean the channel has no additional restrictions. The
|
||||||
|
height can be interpreted in two ways: as a relative height if the value is
|
||||||
|
less than 500,000, or as an absolute height otherwise.
|
||||||
*/
|
*/
|
||||||
uint32 thaw_height = 28;
|
uint32 thaw_height = 28;
|
||||||
|
|
||||||
@ -1665,10 +1667,11 @@ message ChanPointShim {
|
|||||||
bytes pending_chan_id = 5;
|
bytes pending_chan_id = 5;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
This uint32 indicates if this channel is to be considered 'frozen'. A
|
This uint32 indicates if this channel is to be considered 'frozen'. A frozen
|
||||||
frozen channel does not allow a cooperative channel close by the
|
channel does not allow a cooperative channel close by the initiator. The
|
||||||
initiator. The thaw_height is the height that this restriction stops
|
thaw_height is the height that this restriction stops applying to the
|
||||||
applying to the channel.
|
channel. The height can be interpreted in two ways: as a relative height if
|
||||||
|
the value is less than 500,000, or as an absolute height otherwise.
|
||||||
*/
|
*/
|
||||||
uint32 thaw_height = 6;
|
uint32 thaw_height = 6;
|
||||||
}
|
}
|
||||||
|
@ -2466,7 +2466,7 @@
|
|||||||
"thaw_height": {
|
"thaw_height": {
|
||||||
"type": "integer",
|
"type": "integer",
|
||||||
"format": "int64",
|
"format": "int64",
|
||||||
"description": "This uint32 indicates if this channel is to be considered 'frozen'. A\nfrozen channel does not allow a cooperative channel close by the\ninitiator. The thaw_height is the height that this restriction stops\napplying to the channel."
|
"description": "This uint32 indicates if this channel is to be considered 'frozen'. A frozen\nchannel does not allow a cooperative channel close by the initiator. The\nthaw_height is the height that this restriction stops applying to the\nchannel. The height can be interpreted in two ways: as a relative height if\nthe value is less than 500,000, or as an absolute height otherwise."
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@ -2608,7 +2608,7 @@
|
|||||||
"thaw_height": {
|
"thaw_height": {
|
||||||
"type": "integer",
|
"type": "integer",
|
||||||
"format": "int64",
|
"format": "int64",
|
||||||
"description": "This uint32 indicates if this channel is to be considered 'frozen'. A\nfrozen channel doest not allow a cooperative channel close by the\ninitiator. The thaw_height is the height that this restriction stops\napplying to the channel. This field is optional, not setting it or using a\nvalue of zero will mean the channel has no additional restrictions."
|
"description": "This uint32 indicates if this channel is to be considered 'frozen'. A\nfrozen channel doest not allow a cooperative channel close by the\ninitiator. The thaw_height is the height that this restriction stops\napplying to the channel. This field is optional, not setting it or using a\nvalue of zero will mean the channel has no additional restrictions. The\nheight can be interpreted in two ways: as a relative height if the value is\nless than 500,000, or as an absolute height otherwise."
|
||||||
},
|
},
|
||||||
"local_constraints": {
|
"local_constraints": {
|
||||||
"$ref": "#/definitions/lnrpcChannelConstraints",
|
"$ref": "#/definitions/lnrpcChannelConstraints",
|
||||||
|
@ -14501,11 +14501,6 @@ func testExternalFundingChanPoint(net *lntest.NetworkHarness, t *harnessTest) {
|
|||||||
t.Fatalf("unable to gen pending chan ID: %v", err)
|
t.Fatalf("unable to gen pending chan ID: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
_, currentHeight, err := net.Miner.Node.GetBestBlock()
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("unable to get current blockheight %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Now that we have the pending channel ID, Dave (our responder) will
|
// Now that we have the pending channel ID, Dave (our responder) will
|
||||||
// register the intent to receive a new channel funding workflow using
|
// register the intent to receive a new channel funding workflow using
|
||||||
// the pending channel ID.
|
// the pending channel ID.
|
||||||
@ -14514,7 +14509,7 @@ func testExternalFundingChanPoint(net *lntest.NetworkHarness, t *harnessTest) {
|
|||||||
FundingTxidBytes: txid[:],
|
FundingTxidBytes: txid[:],
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
thawHeight := uint32(currentHeight + 10)
|
thawHeight := uint32(10)
|
||||||
chanPointShim := &lnrpc.ChanPointShim{
|
chanPointShim := &lnrpc.ChanPointShim{
|
||||||
Amt: int64(chanSize),
|
Amt: int64(chanSize),
|
||||||
ChanPoint: chanPoint,
|
ChanPoint: chanPoint,
|
||||||
|
@ -355,19 +355,25 @@ func (c *ChanCloser) ProcessCloseMsg(msg lnwire.Message) ([]lnwire.Message,
|
|||||||
"have %v", spew.Sdump(msg))
|
"have %v", spew.Sdump(msg))
|
||||||
}
|
}
|
||||||
|
|
||||||
// As we're the responder to this shutdown (the other party wants to
|
// As we're the responder to this shutdown (the other party
|
||||||
// close), we'll check if this is a frozen channel or not. If the
|
// wants to close), we'll check if this is a frozen channel or
|
||||||
// channel is frozen and we were not also the initiator of the channel
|
// not. If the channel is frozen and we were not also the
|
||||||
// opening, then we'll deny their close attempt.
|
// initiator of the channel opening, then we'll deny their close
|
||||||
|
// attempt.
|
||||||
chanInitiator := c.cfg.Channel.IsInitiator()
|
chanInitiator := c.cfg.Channel.IsInitiator()
|
||||||
chanState := c.cfg.Channel.State()
|
chanState := c.cfg.Channel.State()
|
||||||
if !chanInitiator && chanState.ChanType.IsFrozen() &&
|
if !chanInitiator {
|
||||||
c.negotiationHeight < chanState.ThawHeight {
|
absoluteThawHeight, err := chanState.AbsoluteThawHeight()
|
||||||
|
if err != nil {
|
||||||
return nil, false, fmt.Errorf("initiator attempting to co-op "+
|
return nil, false, err
|
||||||
"close frozen ChannelPoint(%v) (current_height=%v, "+
|
}
|
||||||
"thaw_height=%v)", c.chanPoint, c.negotiationHeight,
|
if c.negotiationHeight < absoluteThawHeight {
|
||||||
chanState.ThawHeight)
|
return nil, false, fmt.Errorf("initiator "+
|
||||||
|
"attempting to co-op close frozen "+
|
||||||
|
"ChannelPoint(%v) (current_height=%v, "+
|
||||||
|
"thaw_height=%v)", c.chanPoint,
|
||||||
|
c.negotiationHeight, absoluteThawHeight)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// If the remote node opened the channel with option upfront shutdown
|
// If the remote node opened the channel with option upfront shutdown
|
||||||
|
21
rpcserver.go
21
rpcserver.go
@ -2122,14 +2122,19 @@ func (r *rpcServer) CloseChannel(in *lnrpc.CloseChannelRequest,
|
|||||||
})
|
})
|
||||||
} else {
|
} else {
|
||||||
// If this is a frozen channel, then we only allow the co-op
|
// If this is a frozen channel, then we only allow the co-op
|
||||||
// close to proceed if we were the responder to this channel.
|
// close to proceed if we were the responder to this channel if
|
||||||
if channel.ChanType.IsFrozen() && channel.IsInitiator &&
|
// the absolute thaw height has not been met.
|
||||||
uint32(bestHeight) < channel.ThawHeight {
|
if channel.IsInitiator {
|
||||||
|
absoluteThawHeight, err := channel.AbsoluteThawHeight()
|
||||||
return fmt.Errorf("cannot co-op close frozen channel "+
|
if err != nil {
|
||||||
"as initiator until height=%v, "+
|
return err
|
||||||
"(current_height=%v)", channel.ThawHeight,
|
}
|
||||||
bestHeight)
|
if uint32(bestHeight) < absoluteThawHeight {
|
||||||
|
return fmt.Errorf("cannot co-op close frozen "+
|
||||||
|
"channel as initiator until height=%v, "+
|
||||||
|
"(current_height=%v)",
|
||||||
|
absoluteThawHeight, bestHeight)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// If the link is not known by the switch, we cannot gracefully close
|
// If the link is not known by the switch, we cannot gracefully close
|
||||||
|
Loading…
Reference in New Issue
Block a user