-
Notifications
You must be signed in to change notification settings - Fork 377
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Zero Conf Channels #1401
Zero Conf Channels #1401
Conversation
Codecov Report
@@ Coverage Diff @@
## main #1401 +/- ##
==========================================
+ Coverage 90.89% 91.80% +0.91%
==========================================
Files 76 80 +4
Lines 42075 50226 +8151
Branches 42075 50226 +8151
==========================================
+ Hits 38244 46111 +7867
- Misses 3831 4115 +284
Continue to review full report at Codecov.
|
fabf567
to
6e30722
Compare
Note things not implemented that we may want eventually (will open followup issues after this makes some progress) - (a) always send a funding_locked for outbound channels immediately whether its 0conf or not, (b) allowing users to lock in an inbound channel before the min-depth even if they didn't follow the manual-acceptance flow (assuming the peer implements (a)). |
6e30722
to
3083a30
Compare
/// If this is set, we assume that our own funding transactions are *never* double-spent, and | ||
/// thus we can trust them without any confirmations. This is generally a reasonable | ||
/// assumption, given we're the only ones who could ever double-spend it (assuming we have sole | ||
/// control of the signing keys). |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Should we make the comment even more conservative and say "The funding transaction should not have inputs spendable by a non-trusted third-party" ? The notion of "sole control of the signing keys" can be quickly confusing as there could be an alternative script branch spendable. The "sole knowledge of any satisfying witness of the inputs" would be better imo.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hmm, it already is very explicit - "given we're the only ones who could ever double-spend it". The comment about control of keys is just a parenthetical to make it concrete. I'm not sure its worth clarifying a clarification example.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
You might have an output locked by a Taproot with 2 script path spends : you own key in the script path but not the ones in the second script path, or they're also locked with a timelock and you think it's safe but the timelock expires. I find the "sole knowledge of any satisfying witness of the inputs" a stronger requirement.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Its also much less readable for people not super deep in Bitcoin - most folks looking at LDK know what a private key is, but not all of them know what "any satisfying witness of the inputs" would mean, or at least would have to think hard about it.
lightning/src/util/config.rs
Outdated
/// | ||
/// You may wish to un-set this if you allow the user to (or do in an automated fashion) | ||
/// double-spend the funding transaction with an alternative channel open to RBF a channel | ||
/// opening transaction. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
If a 0conf user is willing to fee-bump, should we advice them to CPFP the funding transaction through a change output, if any ?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Sure, that seems implied, no? You can of course RBF a funding transaction by simply opening a new channel and closing the original, but that requires non-0-conf.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hmmm you might be willing to use CPFP to get an instant usage effect that you loose with RBF as the replacement transaction invalids the commitment transactions.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm not sure what specifically you want changed here - it talks explicitly about RBF, implying that CPFP is still allowed, IMO.
lightning/src/ln/channelmanager.rs
Outdated
/// | ||
/// [`Event::OpenChannelRequest`]: events::Event::OpenChannelRequest | ||
/// [`Event::ChannelClosed::user_channel_id`]: events::Event::ChannelClosed::user_channel_id | ||
pub fn accept_inbound_channel_from_trusted_peer_0conf(&self, temporary_channel_id: &[u8; 32], user_channel_id: u64) -> Result<(), APIError> { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think one step to reduce trust peer with 0conf inbound channel would be to only authorize payment forwarding decreasing our own balance, then authorize payment reception once we have 1 conf.
If we think it's relevant, we could have another config boolean 0conf_authorize_forward_until_conf
.
(iirc there is an issue somewhere we're already discussed this but can't find it rn..)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
That feels like its somewhat useless, no? Unless you have push_msat you won't have any balance to forward out over.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Well, in that precise case where you've a substantial push_msat though dunno if we see any instant outbound liquidity channel in the wild.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yea, I just don't think that's all that common...Unless we have reason to believe that will change I'm not sure we should really cater to it.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We're much better off letting users manually decide what to do with each HTLC for such cases, I think.
3083a30
to
c508890
Compare
lightning/src/util/config.rs
Outdated
@@ -128,6 +128,23 @@ pub struct ChannelHandshakeLimits { | |||
/// | |||
/// Default value: 144, or roughly one day and only applies to outbound channels. | |||
pub max_minimum_depth: u32, | |||
/// Whether we implicitly trust our own funding transactions to not be double-spent. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
"our own funding transaction" = a transaction where we're the channel initiator?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Tried some tweaked wording.
// (re-)broadcast the funding transaction as we may have declined to broadcast it when we | ||
// first received the funding_signed. | ||
let mut funding_broadcastable = | ||
if self.is_outbound() && self.channel_state & !MULTI_STATE_FLAGS >= ChannelState::FundingSent as u32 { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This change almost seems like it belongs in a separate commit or be documented in the commit message, seemed a bit unrelated at first
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Updated the commit message.
lightning/src/util/config.rs
Outdated
/// A lower-bound of 1 is applied, requiring all channels to have a confirmed commitment | ||
/// transaction before operation. If you wish to accept channels with zero confirmations, see | ||
/// [`UserConfig::manually_accept_inbound_channels`] and | ||
/// [`ChannelManager::accept_inbound_channel`]. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Should we link to accept_inbound_channel_from_trusted_peer_0conf
? I think the API might be cleaner if we aggregate the two methods .. Also so weird to me that you can't initiate 0-conf channels, only accept them 🤔
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is "applied only for inbound channels" :)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
As an improvement, perhaps the struct-level docs could be expanded to be on par with ChannelHandshakeLimits
docs.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm not really sure what we could add to make it more detailed. Do you have a specific suggestion?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I guess not entirely applicable to the comment, but perhaps explaining why a config parameter would go in ChannelHandshakeConfig
vs ChannelHandshakeLimits
. The comments on the fields in the UserConfig
make it clear, but ChannelHandshakeConfig
's docs are super vague. Feel free to disregard as it is unrelated to this PR.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@@ -4370,6 +4422,7 @@ impl<Signer: Sign, M: Deref, T: Deref, K: Deref, F: Deref, L: Deref> ChannelMana | |||
if !channel.get().inbound_is_awaiting_accept() { | |||
return Err(APIError::APIMisuseError { err: "The channel isn't currently awaiting to be accepted.".to_owned() }); | |||
} | |||
if accept_0conf { channel.get_mut().set_0conf(); } |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Curious why we don't check the channel features before setting it to 0-conf?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The 0conf feature is a "I DEMAND ZEROCONF" feature not "I must set this in order to use the channel at zeroconf". Originally we had no such feature at all, but lnd demanded it for reasons that are still somewhat unclear to me. You're welcome to send a funding_locked at 0 conf without the zeroconf feature bit being set.
lightning/src/ln/channel.rs
Outdated
let mut funding_broadcastable = | ||
if self.is_outbound() && self.channel_state & !MULTI_STATE_FLAGS >= ChannelState::FundingSent as u32 { | ||
self.funding_transaction.take() | ||
} else { None }; | ||
// That said, if the funding transaction is already confirmed (ie we're active with a | ||
// minimum_depth over 0, don't bother re-broadcasting the confirmed funding tx. | ||
if self.channel_state & !MULTI_STATE_FLAGS >= ChannelState::ChannelFunded as u32 && self.minimum_depth != Some(0) { | ||
funding_broadcastable = None; | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
These checks seem like they can be combined?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Actually it also addresses a subtle bug - if we confirmed the channel a week ago, we should be removing the funding_transaction
field and no longer re-serializing it in the Channel.
093d75e
to
fb24768
Compare
Hmm, intuitively the 0-conf feature seems incomplete to me until we add the ability to call |
Seems we'll currently reject inbound channels with |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Left some questions, but this looks good and the test coverage seems solid (tried commenting out different stuff, all caused failures). Would prefer to take a last look after another reviewer
@@ -4908,7 +4965,7 @@ impl<Signer: Sign, M: Deref, T: Deref, K: Deref, F: Deref, L: Deref> ChannelMana | |||
break Ok((raa_updates.accepted_htlcs, raa_updates.failed_htlcs, | |||
raa_updates.finalized_claimed_htlcs, | |||
chan.get().get_short_channel_id() | |||
.expect("RAA should only work on a short-id-available channel"), | |||
.unwrap_or(chan.get().outbound_scid_alias()), |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It looks like this could cause us to set short_channel_id
in PaymentPathFailed
to the alias (maybe elsewhere too), may want to document
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I don't think this causes that? I added two comments on the fields I see getting copied from here, but I don't see how it ends up in PaymentPathFailed
. That said, PaymentPathFailed
can just copy the SCID from the first hop in the route into short_channel_id
, which would indeed be an alias.
Gonna let #1461 go first as it will fix #1401 (comment) |
fb24768
to
f894386
Compare
Rebased after #1461, added a new test and addressed feedback. |
I don't think there is ever a reason anyone should use the 0-conf channel type. It really just exists cause lnd demanded it really vocally, I'm really not at all sure why.
Oh, good point lol, yea, we should support this. Maybe we can do this as a follow-up? |
I'm just wondering, couldn't there be a scenario where I want to open a channel but only if I can use it immediately? Is there any way to do that with the way this is implemented?
Good with a follow-up for this! |
The feature doesn't really provide that, though, you're never going to just open a channel with someone and have them accept zeroconf, you're always going to do some kind of out-of-band communication with the peer before you go to open the channel, negotiating zeroconf and that the counterparty is willing to accept zeroconf from you specifically. Once you've done that, the feature doesn't really add anything... |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
High-level comment: not sure if I have the whole context around feature bits, but do we at least want to advertise that we support option_zeroconf
?
lightning/src/ln/functional_tests.rs
Outdated
let mut as_chan = a_channel_lock.by_id.get_mut(&open_chan_2_msg.temporary_channel_id).unwrap(); | ||
// Note that we drop the channel entirely here as its in a somewhat nonsense state (leaving | ||
// the channelmanager in a possibly nonsense state instead). | ||
let mut as_chan = a_channel_lock.by_id.remove(&open_chan_2_msg.temporary_channel_id).unwrap(); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Not sure I understand what's happening here and how it relates the other changes in the commit.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It only tangentially has to do with the commit - previously we'd not even consider the funding transaction because we weren't in the right state, but now we do, and thus get into an even more nonsensical state with the changes in the commit. I updated the comment to be a bit more explicit.
// should fly immediately and the channel should be available for use as soon as they are | ||
// received. | ||
// Receiver must have been initialized with manually_accept_inbound_channels set to true. | ||
fn open_zero_conf_channel<'a, 'b, 'c, 'd>(initiator: &'a Node<'b, 'c, 'd>, receiver: &'a Node<'b, 'c, 'd>, initiator_config: Option<UserConfig>) -> bitcoin::Transaction { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Is this general enough to move to functional_test_utils
?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I guess? Its not currently used outside of this file, and functional_test_utils is already 2300 LoC, so I'd kinda prefer to leave it here if possible.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I guess this file is already near 10k lines anyway lol
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hmm? priv_short_conf_tests
is 1k lines?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Whoops, confused this for functional_tests
😅
// At this point the channel no longer has an SCID again. In the future we should likely | ||
// support simply un-setting the SCID and waiting until the channel gets re-confirmed, but for | ||
// now we force-close the channel here. | ||
check_closed_event!(&nodes[0], 1, ClosureReason::ProcessingError { | ||
err: "Funding transaction was un-confirmed. Locked at 0 confs, now have 0 confs.".to_owned() | ||
}); | ||
check_closed_broadcast!(nodes[0], true); | ||
check_closed_event!(&nodes[1], 1, ClosureReason::ProcessingError { | ||
err: "Funding transaction was un-confirmed. Locked at 0 confs, now have 0 confs.".to_owned() | ||
}); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Would this be difficult to support? Don't need to do it here but probably could open an issue once merged.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
#867 :)
f894386
to
6ffb633
Compare
7415bb3
to
433e487
Compare
Rebased to address merge conflict, lmk if I should squash. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Still reviewing - would like to look at test coverage in depth at least.
/// control of the signing keys). | ||
/// | ||
/// You may wish to un-set this if you allow the user to (or do in an automated fashion) | ||
/// double-spend the funding transaction to RBF with an alternative channel open. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
In fact replace-by-fee is a node policy allowing to replace an original transaction with a newer one, implying a change in the txid and thus nullifying the first commitment transactions validity. However, this is only one fee-bumping technique destructive of the txid, you could imagine a (not) smart user relying on some out-of-band miners API for their fee-bumping, shrugging about our RBF warning as not understanding they're concerned.
"You may wish to un-set this if you allow the user a) to halt the channel open by double-spending the funding inputs or b) to fee-bump the channel open with a replacement transaction (e.g RBF)". Or another wording, but current one is weird as RBF implies a double-spend.
Yes I'm picky but as we're not covering the funding transaction, there are margins for the users to screw up there.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
elying on some out-of-band miners API for their fee-bumping, shrugging about our RBF warning as not understanding they're concerned.
sure, but if they're doing that its totally safe to use 0conf? I'm not sure why we'd want to specify that? Or are you saying we'd also like to include the "cancel a channel open with an RBF" bit?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Well they can still propose a newer transaction through out-of-band miners API, that's just a communication channel. What I would like to underscore is that you have more ways to screw up your 0-conf fee-bumping than leveraging RBF like using ANYONECANPAY...so warning : Be sure of you know well what you're doing if you fee-bump !
Thinking more I'm not even sure we explain somewhere how to insert RBF in our funding flow, as we assume the funding_txid
to be stable (e.g in transactions_confirmed
) so a user would have to re-start from scratch the funding sequence and force-close the previous channel open (though you would have to wait at least 1 conf as a previous channel open might end up confirmed). Maybe to open an issue that we should document cleanly "how to implement RBF with our current API".
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Well they can still propose a newer transaction through out-of-band miners API, that's just a communication channel.
Right, but that's still RBF? No matter how it ends up there its replace-by-fee, even if the fee is out-of-band. Would you prefer if I just remove the "to RBF..." part or say "e.g. to RBF"? That way its just a more general "double-spend".
Thinking more I'm not even sure we explain somewhere how to insert RBF in our funding flow,
Right, the only way to do funding with RBF is to literally open a new channel with a conflicting tx, and then force-close the old one once the rbf'd one confirms. Indeed, I don't think its documented anywhere, and its pretty gross.
// un-confirmed we force-close the channel, ensuring short_to_id is | ||
// always consistent. | ||
let scid_insert = short_to_id.insert(real_scid, channel.channel_id()); | ||
assert!(scid_insert.is_none() || scid_insert.unwrap() == channel.channel_id(), |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Could be nice to call the variable name fake_scid
in clear opposition to real_scid
and stick to it as a convention. In that direction the error message could also be "fake SCID should never collide with real SCID"
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Not sure what you want to rename fake_scid
here? scid_insert
is a result, not an SCID specifically.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ah yep.
/// If this is set, we assume that our own funding transactions are *never* double-spent, and | ||
/// thus we can trust them without any confirmations. This is generally a reasonable | ||
/// assumption, given we're the only ones who could ever double-spend it (assuming we have sole | ||
/// control of the signing keys). |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
You might have an output locked by a Taproot with 2 script path spends : you own key in the script path but not the ones in the second script path, or they're also locked with a timelock and you think it's safe but the timelock expires. I find the "sole knowledge of any satisfying witness of the inputs" a stronger requirement.
lightning/src/util/config.rs
Outdated
/// | ||
/// You may wish to un-set this if you allow the user to (or do in an automated fashion) | ||
/// double-spend the funding transaction with an alternative channel open to RBF a channel | ||
/// opening transaction. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hmmm you might be willing to use CPFP to get an instant usage effect that you loose with RBF as the replacement transaction invalids the commitment transactions.
lightning/src/ln/channelmanager.rs
Outdated
/// | ||
/// [`Event::OpenChannelRequest`]: events::Event::OpenChannelRequest | ||
/// [`Event::ChannelClosed::user_channel_id`]: events::Event::ChannelClosed::user_channel_id | ||
pub fn accept_inbound_channel_from_trusted_peer_0conf(&self, temporary_channel_id: &[u8; 32], user_channel_id: u64) -> Result<(), APIError> { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Well, in that precise case where you've a substantial push_msat though dunno if we see any instant outbound liquidity channel in the wild.
if peer_limits.trust_own_funding_0conf { | ||
self.minimum_depth = Some(msg.minimum_depth); | ||
} else { | ||
self.minimum_depth = Some(cmp::max(1, msg.minimum_depth)); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This would previously return an error when min_depth == 0
, whereas now we'll enforce a minimum of one, so they may think we've agreed to proceed with zero confs once we send funding_created
back.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Right, this is kinda a 0conf philosophy question. This PR doesn't actually support the 0conf channel type (as it was written before the 0conf channel type existed) which is the "I require 0conf" option, but we'll add that in a followup.
In general, if you're doing a 0conf channel, you've proably pre-negotiated it in some way - most likely the recipient of the channel is pre-configured to trust an LSP. There's arguably not a lot of value in negotiating 0conf in the open/accept channel flow - by the time you're there both sides of the channel should really already know if the channel should be 0conf or not.
More generally, nodes can always delay channel_ready
/funding_locked
as long as they want - the minimum_depth
field is just a hint, and the protocol can't enforce it in any way. We're not strictly-speaking "out of spec" by delaying more than the channel recipient wanted.
32df62a
to
2269c40
Compare
Squashed fixups without other changes.
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
See potential test coverage holes. Otherwise, SGTM, I don't think any comment is a blocker.
@@ -136,6 +136,8 @@ pub(super) enum HTLCForwardInfo { | |||
// `process_pending_htlc_forwards()` for constructing the | |||
// `HTLCSource::PreviousHopData` for failed and forwarded | |||
// HTLCs. | |||
// | |||
// Note that this may be an outbound SCID alias for the associated channel. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I believe that comment should be dup on PendingHTLCRouting::Forward
's short_channel_id
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ah, a similar comment applies there as well, yes.
@@ -1398,7 +1401,7 @@ macro_rules! handle_chan_restoration_locked { | |||
let res = loop { | |||
let forwards: Vec<(PendingHTLCInfo, u64)> = $pending_forwards; // Force type-checking to resolve | |||
if !forwards.is_empty() { | |||
htlc_forwards = Some(($channel_entry.get().get_short_channel_id().expect("We can't have pending forwards before funding confirmation"), | |||
htlc_forwards = Some(($channel_entry.get().get_short_channel_id().unwrap_or($channel_entry.get().outbound_scid_alias()), |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Not covering the get_short_channel_id()
in list_channels_with_filter
, deliberate ?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm not sure what this is in reference to?
// inconsistent, so we currently have to. | ||
// Note that ideally we wouldn't force-close if we see *any* reorg on a 1-conf or | ||
// 0-conf channel, but not doing so may lead to the `ChannelManager::short_to_id` map | ||
// being inconsistent, so we currently have to. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Apart of more code complexity, I don't understand why we would force-close a 0-conf channel as we already assume it's safe without block inclusion ? And I'm not sure there is a real inconsistency risk with the short_to_id
being inserted at funding_locked
being a fake SCID ? Though one concern would be the funding inputs unconfirmed themselves.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
No, we should fix that eventually - #867
if channel.is_our_funding_locked() { | ||
if let Some(real_scid) = channel.get_short_channel_id() { | ||
// If we sent a 0conf funding_locked, and now have an SCID, we add it | ||
// to the short_to_id map here. Note that we check whether we can relay |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Is that true of option_scid_alias
private chan where if the short_channel_id
is different from the outbound scid alias we fail the forward ?
rust-lightning/lightning/src/ln/channelmanager.rs
Line 2196 in 46e58ae
if chan.get_channel_type().supports_scid_privacy() && *short_channel_id != chan.outbound_scid_alias() { |
Do you have test coverage of combining a private 0-conf and fake SCID routing over it ?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Is that true of option_scid_alias private chan where if the short_channel_id is different from the outbound scid alias we fail the forward ?
Yes, I updated the comment to be clearer.
Do you have test coverage of combining a private 0-conf and fake SCID routing over it ?
Yes? I'm not sure exactly what combo you're asking for but there's good testing of the reject-real-scid-on-aliases.
@@ -2291,6 +2291,9 @@ impl<Signer: Sign, M: Deref, T: Deref, K: Deref, F: Deref, L: Deref> ChannelMana | |||
action: msgs::ErrorAction::IgnoreError | |||
}); | |||
} | |||
if chan.get_short_channel_id().is_none() { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Doesn't seem to have test coverage ? Commenting out doesn't break anything.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hmm, good point, its kinda hard to hit, though - its only used on force-close (which I added a trivial test for) and in broadcasting our node announcement (which we can't really hit from here because we don't have our counterparty announcement sigs).
2269c40
to
d58b8b0
Compare
In the next few commits we add support for 0conf channels, allowing us to have an active channel with HTLC and other updates flying prior to having an SCID available. This would break several assumptions made in `ChannelManager`, which we address here by looking at SCID aliases in addition to SCIDs.
If our peer sets a minimum depth of 0, and we're set to trusting ourselves to not double-spend our own funding transactions, send a funding_locked message immediately after funding signed. Note that some special care has to be taken around the `channel_state` values - `ChannelFunded` no longer implies the funding transaction is confirmed on-chain. Thus, for example, the should-we-re-broadcast logic has to now accept `channel_state` values greater than `ChannelFunded` as indicating we may still need to re-broadcast our funding tranasction, unless `minimum_depth` is greater than 0. Further note that this starts writing `Channel` objects with a `MIN_SERIALIZATION_VERSION` of 2. Thus, LDK versions prior to 0.0.99 (July 2021) will now refuse to read serialized Channels/ChannelManagers.
This supports routing outbound over 0-conf channels by utilizing the outbound SCID alias that we assign to all channels to refer to the selected channel when routing.
This tests a few cases of monitor failure updates that were broken in earlier versions of the 0conf patchset.
In a previous version of the 0-conf code we did not correctly handle 0-conf channels getting the funding transaction reorg'd out (and the real SCID possibly changing on us).
This isn't specific to 0-conf but is useful for users to note as the network moves towards more SCID aliases.
d58b8b0
to
ffaf9bc
Compare
Squashed without diff:
|
if non_shutdown_state & !(ChannelState::TheirFundingLocked as u32) == ChannelState::FundingSent as u32 { | ||
// Check if the transaction is the expected funding transaction, and if it is, | ||
// check that it pays the right amount to the right script. | ||
if self.funding_tx_confirmation_height == 0 { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I guess this predated the PR, but it's unclear to me what funding_tx_confirmation_height
of 0 means.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ah, it means "not yet confirmed", as certainly no channel is going to be confirmed at height 0 :) I kinda figured it was an obvious default but I can add docs if you prefer?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Meh, maybe an is_funding_confirmed
function would be more readable, but no need to do anything here.
🎉 not sure what else to say. See individual commits.