Skip to content

Only mark all mon updates complete if there are no blocked updates #3907

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

Open
wants to merge 1 commit into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
69 changes: 56 additions & 13 deletions lightning/src/ln/chanmon_update_fail_tests.rs
Original file line number Diff line number Diff line change
Expand Up @@ -3410,20 +3410,22 @@ fn test_inbound_reload_without_init_mon() {
do_test_inbound_reload_without_init_mon(false, false);
}

#[test]
fn test_blocked_chan_preimage_release() {
fn do_test_blocked_chan_preimage_release(completion_mode: u8) {
// Test that even if a channel's `ChannelMonitorUpdate` flow is blocked waiting on an event to
// be handled HTLC preimage `ChannelMonitorUpdate`s will still go out.
let chanmon_cfgs = create_chanmon_cfgs(3);
let node_cfgs = create_node_cfgs(3, &chanmon_cfgs);
let persister;
let new_chain_mon;
let node_chanmgrs = create_node_chanmgrs(3, &node_cfgs, &[None, None, None]);
let nodes_1_reload;
let mut nodes = create_network(3, &node_cfgs, &node_chanmgrs);

let node_a_id = nodes[0].node.get_our_node_id();
let node_b_id = nodes[1].node.get_our_node_id();
let node_c_id = nodes[2].node.get_our_node_id();

create_announced_chan_between_nodes(&nodes, 0, 1);
let chan_id_1 = create_announced_chan_between_nodes(&nodes, 0, 1).2;
let chan_id_2 = create_announced_chan_between_nodes(&nodes, 1, 2).2;

send_payment(&nodes[0], &[&nodes[1], &nodes[2]], 5_000_000);
Expand Down Expand Up @@ -3456,29 +3458,63 @@ fn test_blocked_chan_preimage_release() {
expect_payment_claimed!(nodes[0], payment_hash_2, 1_000_000);

let as_htlc_fulfill_updates = get_htlc_update_msgs!(nodes[0], node_b_id);
if completion_mode != 0 {
// We use to incorrectly handle monitor update completion in cases where we completed a
// monitor update async or after reload. We test both based on the `completion_mode`.
chanmon_cfgs[1].persister.set_update_ret(ChannelMonitorUpdateStatus::InProgress);
}
nodes[1]
.node
.handle_update_fulfill_htlc(node_a_id, &as_htlc_fulfill_updates.update_fulfill_htlcs[0]);
check_added_monitors(&nodes[1], 1); // We generate only a preimage monitor update
assert!(get_monitor!(nodes[1], chan_id_2).get_stored_preimages().contains_key(&payment_hash_2));
assert!(nodes[1].node.get_and_clear_pending_msg_events().is_empty());
if completion_mode == 1 {
let node_ser = nodes[1].node.encode();
let chan_mon_0 = get_monitor!(nodes[1], chan_id_1).encode();
let chan_mon_1 = get_monitor!(nodes[1], chan_id_2).encode();

let mons = &[&chan_mon_0[..], &chan_mon_1[..]];
reload_node!(nodes[1], &node_ser, mons, persister, new_chain_mon, nodes_1_reload);

nodes[0].node.peer_disconnected(node_b_id);
nodes[2].node.peer_disconnected(node_b_id);

let mut a_b_reconnect = ReconnectArgs::new(&nodes[0], &nodes[1]);
a_b_reconnect.pending_htlc_claims.1 = 1;
// Note that we will expect no final RAA monitor update in
// `commitment_signed_dance_through_cp_raa` during the reconnect, matching the below case.
reconnect_nodes(a_b_reconnect);
reconnect_nodes(ReconnectArgs::new(&nodes[2], &nodes[1]));
} else if completion_mode == 2 {
let (latest_update, _) = get_latest_mon_update_id(&nodes[1], chan_id_2);
nodes[1]
.chain_monitor
.chain_monitor
.channel_monitor_updated(chan_id_2, latest_update)
.unwrap();
}

// Finish the CS dance between nodes[0] and nodes[1]. Note that until the event handling, the
// update_fulfill_htlc + CS is held, even though the preimage is already on disk for the
// channel.
nodes[1]
.node
.handle_commitment_signed_batch_test(node_a_id, &as_htlc_fulfill_updates.commitment_signed);
check_added_monitors(&nodes[1], 1);
let (a, raa) = do_main_commitment_signed_dance(&nodes[1], &nodes[0], false);
assert!(a.is_none());
// Note that in completion_mode 1 we completed the CS dance in `reconnect_nodes` above.
if completion_mode != 1 {
nodes[1].node.handle_commitment_signed_batch_test(
node_a_id,
&as_htlc_fulfill_updates.commitment_signed,
);
check_added_monitors(&nodes[1], 1);
let (a, raa) = do_main_commitment_signed_dance(&nodes[1], &nodes[0], false);
assert!(a.is_none());

nodes[1].node.handle_revoke_and_ack(node_a_id, &raa);
check_added_monitors(&nodes[1], 0);
assert!(nodes[1].node.get_and_clear_pending_msg_events().is_empty());
nodes[1].node.handle_revoke_and_ack(node_a_id, &raa);
check_added_monitors(&nodes[1], 0);
assert!(nodes[1].node.get_and_clear_pending_msg_events().is_empty());
}

let events = nodes[1].node.get_and_clear_pending_events();
assert_eq!(events.len(), 3);
assert_eq!(events.len(), 3, "{events:?}");
if let Event::PaymentSent { .. } = events[0] {
} else {
panic!();
Expand Down Expand Up @@ -3508,6 +3544,13 @@ fn test_blocked_chan_preimage_release() {
expect_payment_sent(&nodes[2], payment_preimage_2, None, true, true);
}

#[test]
fn test_blocked_chan_preimage_release() {
do_test_blocked_chan_preimage_release(0);
do_test_blocked_chan_preimage_release(1);
do_test_blocked_chan_preimage_release(2);
}

fn do_test_inverted_mon_completion_order(
with_latest_manager: bool, complete_bc_commitment_dance: bool,
) {
Expand Down
1 change: 1 addition & 0 deletions lightning/src/ln/channel.rs
Original file line number Diff line number Diff line change
Expand Up @@ -7925,6 +7925,7 @@ where
{
assert!(self.context.channel_state.is_monitor_update_in_progress());
self.context.channel_state.clear_monitor_update_in_progress();
assert_eq!(self.blocked_monitor_updates_pending(), 0);

// If we're past (or at) the AwaitingChannelReady stage on an outbound (or V2-established) channel,
// try to (re-)broadcast the funding transaction as we may have declined to broadcast it when we
Expand Down
8 changes: 6 additions & 2 deletions lightning/src/ln/channelmanager.rs
Original file line number Diff line number Diff line change
Expand Up @@ -6934,7 +6934,9 @@ where
.get_mut(&channel_id)
.and_then(Channel::as_funded_mut)
{
handle_monitor_update_completion!(self, peer_state_lock, peer_state, per_peer_state, chan);
if chan.blocked_monitor_updates_pending() == 0 {
handle_monitor_update_completion!(self, peer_state_lock, peer_state, per_peer_state, chan);
}
} else {
let update_actions = peer_state.monitor_update_blocked_actions
.remove(&channel_id).unwrap_or(Vec::new());
Expand Down Expand Up @@ -8227,7 +8229,9 @@ This indicates a bug inside LDK. Please report this error at https://github.com/
{
if chan.is_awaiting_monitor_update() {
log_trace!(logger, "Channel is open and awaiting update, resuming it");
handle_monitor_update_completion!(self, peer_state_lock, peer_state, per_peer_state, chan);
if chan.blocked_monitor_updates_pending() == 0 {
handle_monitor_update_completion!(self, peer_state_lock, peer_state, per_peer_state, chan);
}
} else {
log_trace!(logger, "Channel is open but not awaiting update");
}
Expand Down
17 changes: 15 additions & 2 deletions lightning/src/ln/quiescence_tests.rs
Original file line number Diff line number Diff line change
Expand Up @@ -260,9 +260,22 @@ fn test_quiescence_waits_for_async_signer_and_monitor_update() {
let chain_monitor = &nodes[0].chain_monitor.chain_monitor;
// One for the latest commitment transaction update from the last `revoke_and_ack`
chain_monitor.channel_monitor_updated(chan_id, latest_update).unwrap();
expect_payment_sent(&nodes[0], preimage, None, true, true);
expect_payment_sent(&nodes[0], preimage, None, false, true);

let chain_monitor = &nodes[0].chain_monitor;
let (_, new_latest_update) =
chain_monitor.latest_monitor_update_id.lock().unwrap().get(&chan_id).unwrap().clone();
assert_eq!(new_latest_update, latest_update + 1);
let chain_monitor = &nodes[0].chain_monitor.chain_monitor;
// One for the commitment secret update from the last `revoke_and_ack`
chain_monitor.channel_monitor_updated(chan_id, latest_update + 1).unwrap();
chain_monitor.channel_monitor_updated(chan_id, new_latest_update).unwrap();
// Once that update completes, we'll get the `PaymentPathSuccessful` event
let events = nodes[0].node.get_and_clear_pending_events();
assert_eq!(events.len(), 1);
if let Event::PaymentPathSuccessful { .. } = &events[0] {
} else {
panic!("{events:?}");
}
}

// With the updates completed, we can now become quiescent.
Expand Down
Loading