-
Notifications
You must be signed in to change notification settings - Fork 399
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
Add ChannelContext::get_commitment_stats
#3682
base: main
Are you sure you want to change the base?
Add ChannelContext::get_commitment_stats
#3682
Conversation
👋 Thanks for assigning @wpaulino as a reviewer! |
0af1b43
to
31300c3
Compare
Why is this draft? |
@TheBlueMatt I don't feel great about the code duplication between Let me know what you think. |
I'm working on further cleanups now. |
d6b1f51
to
c540347
Compare
@wpaulino the PR is in a better spot now, let me know what you think. |
c540347
to
c9b30cb
Compare
e637050
to
4d6bf13
Compare
5a212d8
to
d682615
Compare
Codecov ReportAttention: Patch coverage is
Additional details and impacted files@@ Coverage Diff @@
## main #3682 +/- ##
==========================================
- Coverage 89.18% 89.16% -0.03%
==========================================
Files 155 155
Lines 120796 120861 +65
Branches 120796 120861 +65
==========================================
+ Hits 107731 107760 +29
- Misses 10415 10446 +31
- Partials 2650 2655 +5 ☔ View full report in Codecov by Sentry. 🚀 New features to boost your workflow:
|
d682615
to
6f80f99
Compare
@TheBlueMatt take a look when you can thank you! |
lightning/src/ln/channel.rs
Outdated
@@ -223,6 +223,25 @@ impl From<&InboundHTLCState> for Option<InboundHTLCStateDetails> { | |||
} | |||
} | |||
|
|||
impl InboundHTLCState { | |||
fn as_str(&self) -> &str { |
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.
Heh, if its just for logging I think we can #[derive(Debug)]
:)
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.
Debug
would log all their inner contents too - don't think that's what we want here ?
I would have to derive Debug
in a bunch more places.
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 I'll just move that function to an impl fmt::Debug for InboundHTLCState
:)
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.
Went with an impl fmt::Display
here, seems this corresponds most to what we want.
lightning/src/ln/channel.rs
Outdated
/// have not yet committed it. Such HTLCs will only be included in transactions which are being | ||
/// generated by the peer which proposed adding the HTLCs, and thus we need to understand both | ||
/// which peer generated this transaction and "to whom" this transaction flows. | ||
fn for_each_inbound<F>(&self, generated_by_local: bool, mut for_each: F) |
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 fn name here doesn't really communicate what it does/is used for. Maybe inbound_htlcs_in_commitment
? It might also be more Rust-y to return an impl Iterator
and let the callsite do the for, rather than doing it as a callback.
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 initially had something like this is it closer to what you have in mind?
for htlc in self.pending_inbound_htlcs.iter() {
if htlc.state.included_in_commitment(generated_by_local) {
..
} else {
..
}
}
some other options for that specific function's return value:
- one iterator yielding (included, htlc)
- two iterators each yielding htlc, one for included, one for excluded.
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.
Went with the very first option above.
a6c6eac
to
71df614
Compare
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.
Basically LGTM, one nit. Sorry somehow I lost track of this one.
lightning/src/ln/channel.rs
Outdated
macro_rules! count_nondust_htlc { | ||
($htlc: expr, $outbound: expr) => { | ||
if $htlc.is_dust(local, feerate_per_kw, broadcaster_dust_limit_sat, self.get_channel_type()) { | ||
log_trace!(logger, " ...not counting {} {} dust HTLC {} (hash {}) with value {} due to dust limit", if $outbound { "outbound" } else { "inbound" }, $htlc.state, $htlc.htlc_id, $htlc.payment_hash, $htlc.amount_msat); |
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, this is gonna end up doubling the amount we log when we build a commitment - instead of logging once per HTLC we're now logging twice, once when building the stats and once when we build the commitment itself. IMO we should make one of the two methods silent (which depends a bit on how you intend to use commitment_stats
going forward, which isn't clear from the commit msg.
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 problem ! Thank you I added the motivation for commitment_stats
in the commit message, and removed all logs from that function. I prefer to let callers of this function do the logging depending on the use case.
71df614
to
6d8fb6e
Compare
} else { | ||
htlc_success_tx_weight(features) | ||
}; | ||
feerate_per_kw as u64 * htlc_tx_weight / 1000 |
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.
Nit: could note that we intentionally round down here as required by the spec
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.
thanks will do
lightning/src/ln/channel.rs
Outdated
value_to_self_msat_offset += htlc.amount_msat as i64; | ||
}, | ||
_ => {}, | ||
if let InboundHTLCState::LocalRemoved(InboundHTLCRemovalReason::Fulfill(_preimage)) = htlc.state { |
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.
Nit: use preimage()
instead and check for Some
?
lightning/src/ln/channel.rs
Outdated
value_to_self_msat_offset -= htlc.amount_msat as i64; | ||
}, | ||
_ => {}, | ||
OutboundHTLCState::RemoteRemoved(OutboundHTLCOutcome::Success(_)) |
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.
Same here regarding preimage
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.
Both of these work so far - I just need to check that this comment here is true for the outbound states:
enum OutboundHTLCOutcome {
/// LDK version 0.0.105+ will always fill in the preimage here.
Success(Option<PaymentPreimage>),
Failure(HTLCFailReason),
}
``
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.
@TheBlueMatt what do you think of replacing this if let ...
with a if htlc.state.preimage().is_some()
call ?
- Would assume no upgrades from 0.0.104 and below straight to 0.2
- We would now rely heavily on this statement being true:
OutboundHTLCOutcome::Success
always has a preimage filled in. - Right now if this is violated, a VLS will be missing a preimage.
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 assume no upgrades from 0.0.104 and below straight to 0.2
I believe we already require that for various things from some of wilmer's splicing work to, almost, the 0.1 "Nodes with pending forwarded HTLCs or unclaimed payments cannot be upgraded directly from 0.0.123 or earlier to 0.1"
We would now rely heavily on this statement being true: OutboundHTLCOutcome::Success always has a preimage filled in.
We should probably just enforce it in the deserialization step.
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.
Needs rebase, it seems
6d8fb6e
to
793d4d2
Compare
A question for you in an above comment Matt thanks :) |
It can be useful to get the stats on a potential commitment transaction without actually building it. Therefore, this commit splits the stats calculations from the actual build of a commitment transaction. This introduces an extra loop over the pending htlcs when actually building a commitment transaction, but current network behavior produces very few concurrent htlcs on channels. Furthermore, each iteration of the loop in the stats calculation is very cheap. The motivating use case for `build_commitment_stats` is to calculate the balances of the channel parties in order to validate the `funding_contribution_satoshis` field of `splice_init` and `splice_ack` messages without building a full commitment transaction.
This allows us to DRY the code that calculates the `value_to_self_msat_offset` in `ChannelContext::build_commitment_stats`. HTLC success states have held their corresponding preimage since 0.0.105, and the release notes of 0.1 already require users running 0.0.123 and earlier to resolve their HTLCs before upgrading to 0.1. So this change is fully compatible with existing upgrade paths to the yet-to-be-shipped 0.2 release.
0c7bab6
to
285f1b9
Compare
OutboundHTLCState::RemoteRemoved(OutboundHTLCOutcome::Success(preimage)) | | ||
OutboundHTLCState::AwaitingRemoteRevokeToRemove(OutboundHTLCOutcome::Success(preimage)) | | ||
OutboundHTLCState::AwaitingRemovedRemoteRevoke(OutboundHTLCOutcome::Success(preimage)) => { | ||
debug_assert!(preimage.is_some()); |
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 am not 100 sure if this debug assert and those below should be hard asserts. I went with debug because from what I can see so far, lacking preimages can lead to force closes, but not loss of funds.
🔔 1st Reminder Hey @wpaulino! This PR has been waiting for your review. |
Requested by @wpaulino in #3641