-
Notifications
You must be signed in to change notification settings - Fork 100
/
Copy pathbalance.rs
387 lines (378 loc) · 16.3 KB
/
balance.rs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
// This file is Copyright its original authors, visible in version control history.
//
// This file is licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license <LICENSE-MIT or
// http://opensource.org/licenses/MIT>, at your option. You may not use this file except in
// accordance with one or both of these licenses.
use crate::sweep::value_from_descriptor;
use lightning::chain::channelmonitor::Balance as LdkBalance;
use lightning::chain::channelmonitor::BalanceSource;
use lightning::ln::types::ChannelId;
use lightning::util::sweep::{OutputSpendStatus, TrackedSpendableOutput};
use lightning_types::payment::{PaymentHash, PaymentPreimage};
use bitcoin::secp256k1::PublicKey;
use bitcoin::{BlockHash, Txid};
/// Details of the known available balances returned by [`Node::list_balances`].
///
/// [`Node::list_balances`]: crate::Node::list_balances
#[derive(Debug, Clone)]
pub struct BalanceDetails {
/// The total balance of our on-chain wallet.
pub total_onchain_balance_sats: u64,
/// The currently spendable balance of our on-chain wallet.
///
/// This includes any sufficiently confirmed funds, minus
/// [`total_anchor_channels_reserve_sats`].
///
/// [`total_anchor_channels_reserve_sats`]: Self::total_anchor_channels_reserve_sats
pub spendable_onchain_balance_sats: u64,
/// The share of our total balance that we retain as an emergency reserve to (hopefully) be
/// able to spend the Anchor outputs when one of our channels is closed.
pub total_anchor_channels_reserve_sats: u64,
/// The total balance that we would be able to claim across all our Lightning channels.
///
/// Note this excludes balances that we are unsure if we are able to claim (e.g., as we are
/// waiting for a preimage or for a timeout to expire). These balances will however be included
/// as [`MaybePreimageClaimableHTLC`] and
/// [`MaybeTimeoutClaimableHTLC`] in [`lightning_balances`].
///
/// [`MaybePreimageClaimableHTLC`]: LightningBalance::MaybePreimageClaimableHTLC
/// [`MaybeTimeoutClaimableHTLC`]: LightningBalance::MaybeTimeoutClaimableHTLC
/// [`lightning_balances`]: Self::lightning_balances
pub total_lightning_balance_sats: u64,
/// A detailed list of all known Lightning balances that would be claimable on channel closure.
///
/// Note that less than the listed amounts are spendable over lightning as further reserve
/// restrictions apply. Please refer to [`ChannelDetails::outbound_capacity_msat`] and
/// [`ChannelDetails::next_outbound_htlc_limit_msat`] as returned by [`Node::list_channels`]
/// for a better approximation of the spendable amounts.
///
/// [`ChannelDetails::outbound_capacity_msat`]: crate::ChannelDetails::outbound_capacity_msat
/// [`ChannelDetails::next_outbound_htlc_limit_msat`]: crate::ChannelDetails::next_outbound_htlc_limit_msat
/// [`Node::list_channels`]: crate::Node::list_channels
pub lightning_balances: Vec<LightningBalance>,
/// A detailed list of balances currently being swept from the Lightning to the on-chain
/// wallet.
///
/// These are balances resulting from channel closures that may have been encumbered by a
/// delay, but are now being claimed and useable once sufficiently confirmed on-chain.
///
/// Note that, depending on the sync status of the wallets, swept balances listed here might or
/// might not already be accounted for in [`total_onchain_balance_sats`].
///
/// [`total_onchain_balance_sats`]: Self::total_onchain_balance_sats
pub pending_balances_from_channel_closures: Vec<PendingSweepBalance>,
}
/// Details about the status of a known Lightning balance.
#[derive(Debug, Clone)]
pub enum LightningBalance {
/// The channel is not yet closed (or the commitment or closing transaction has not yet
/// appeared in a block). The given balance is claimable (less on-chain fees) if the channel is
/// force-closed now.
ClaimableOnChannelClose {
/// The identifier of the channel this balance belongs to.
channel_id: ChannelId,
/// The identifier of our channel counterparty.
counterparty_node_id: PublicKey,
/// The amount available to claim, in satoshis, excluding the on-chain fees which will be
/// required to do so.
amount_satoshis: u64,
/// The transaction fee we pay for the closing commitment transaction. This amount is not
/// included in the `amount_satoshis` value.
///
/// Note that if this channel is inbound (and thus our counterparty pays the commitment
/// transaction fee) this value will be zero. For channels created prior to LDK Node 0.4
/// the channel is always treated as outbound (and thus this value is never zero).
transaction_fee_satoshis: u64,
/// The amount of millisatoshis which has been burned to fees from HTLCs which are outbound
/// from us and are related to a payment which was sent by us. This is the sum of the
/// millisatoshis part of all HTLCs which are otherwise represented by
/// [`LightningBalance::MaybeTimeoutClaimableHTLC`] with their
/// [`LightningBalance::MaybeTimeoutClaimableHTLC::outbound_payment`] flag set, as well as
/// any dust HTLCs which would otherwise be represented the same.
///
/// This amount (rounded up to a whole satoshi value) will not be included in `amount_satoshis`.
outbound_payment_htlc_rounded_msat: u64,
/// The amount of millisatoshis which has been burned to fees from HTLCs which are outbound
/// from us and are related to a forwarded HTLC. This is the sum of the millisatoshis part
/// of all HTLCs which are otherwise represented by
/// [`LightningBalance::MaybeTimeoutClaimableHTLC`] with their
/// [`LightningBalance::MaybeTimeoutClaimableHTLC::outbound_payment`] flag *not* set, as
/// well as any dust HTLCs which would otherwise be represented the same.
///
/// This amount (rounded up to a whole satoshi value) will not be included in `amount_satoshis`.
outbound_forwarded_htlc_rounded_msat: u64,
/// The amount of millisatoshis which has been burned to fees from HTLCs which are inbound
/// to us and for which we know the preimage. This is the sum of the millisatoshis part of
/// all HTLCs which would be represented by [`LightningBalance::ContentiousClaimable`] on
/// channel close, but whose current value is included in `amount_satoshis`, as well as any
/// dust HTLCs which would otherwise be represented the same.
///
/// This amount (rounded up to a whole satoshi value) will not be included in the counterparty's
/// `amount_satoshis`.
inbound_claiming_htlc_rounded_msat: u64,
/// The amount of millisatoshis which has been burned to fees from HTLCs which are inbound
/// to us and for which we do not know the preimage. This is the sum of the millisatoshis
/// part of all HTLCs which would be represented by
/// [`LightningBalance::MaybePreimageClaimableHTLC`] on channel close, as well as any dust
/// HTLCs which would otherwise be represented the same.
///
/// This amount (rounded up to a whole satoshi value) will not be included in the
/// counterparty's `amount_satoshis`.
inbound_htlc_rounded_msat: u64,
},
/// The channel has been closed, and the given balance is ours but awaiting confirmations until
/// we consider it spendable.
ClaimableAwaitingConfirmations {
/// The identifier of the channel this balance belongs to.
channel_id: ChannelId,
/// The identifier of our channel counterparty.
counterparty_node_id: PublicKey,
/// The amount available to claim, in satoshis, possibly excluding the on-chain fees which
/// were spent in broadcasting the transaction.
amount_satoshis: u64,
/// The height at which an [`Event::SpendableOutputs`] event will be generated for this
/// amount.
///
/// [`Event::SpendableOutputs`]: lightning::events::Event::SpendableOutputs
confirmation_height: u32,
/// Whether this balance is a result of cooperative close, a force-close, or an HTLC.
source: BalanceSource,
},
/// The channel has been closed, and the given balance should be ours but awaiting spending
/// transaction confirmation. If the spending transaction does not confirm in time, it is
/// possible our counterparty can take the funds by broadcasting an HTLC timeout on-chain.
///
/// Once the spending transaction confirms, before it has reached enough confirmations to be
/// considered safe from chain reorganizations, the balance will instead be provided via
/// [`LightningBalance::ClaimableAwaitingConfirmations`].
ContentiousClaimable {
/// The identifier of the channel this balance belongs to.
channel_id: ChannelId,
/// The identifier of our channel counterparty.
counterparty_node_id: PublicKey,
/// The amount available to claim, in satoshis, excluding the on-chain fees which will be
/// required to do so.
amount_satoshis: u64,
/// The height at which the counterparty may be able to claim the balance if we have not
/// done so.
timeout_height: u32,
/// The payment hash that locks this HTLC.
payment_hash: PaymentHash,
/// The preimage that can be used to claim this HTLC.
payment_preimage: PaymentPreimage,
},
/// HTLCs which we sent to our counterparty which are claimable after a timeout (less on-chain
/// fees) if the counterparty does not know the preimage for the HTLCs. These are somewhat
/// likely to be claimed by our counterparty before we do.
MaybeTimeoutClaimableHTLC {
/// The identifier of the channel this balance belongs to.
channel_id: ChannelId,
/// The identifier of our channel counterparty.
counterparty_node_id: PublicKey,
/// The amount potentially available to claim, in satoshis, excluding the on-chain fees
/// which will be required to do so.
amount_satoshis: u64,
/// The height at which we will be able to claim the balance if our counterparty has not
/// done so.
claimable_height: u32,
/// The payment hash whose preimage our counterparty needs to claim this HTLC.
payment_hash: PaymentHash,
/// Indicates whether this HTLC represents a payment which was sent outbound from us.
outbound_payment: bool,
},
/// HTLCs which we received from our counterparty which are claimable with a preimage which we
/// do not currently have. This will only be claimable if we receive the preimage from the node
/// to which we forwarded this HTLC before the timeout.
MaybePreimageClaimableHTLC {
/// The identifier of the channel this balance belongs to.
channel_id: ChannelId,
/// The identifier of our channel counterparty.
counterparty_node_id: PublicKey,
/// The amount potentially available to claim, in satoshis, excluding the on-chain fees
/// which will be required to do so.
amount_satoshis: u64,
/// The height at which our counterparty will be able to claim the balance if we have not
/// yet received the preimage and claimed it ourselves.
expiry_height: u32,
/// The payment hash whose preimage we need to claim this HTLC.
payment_hash: PaymentHash,
},
/// The channel has been closed, and our counterparty broadcasted a revoked commitment
/// transaction.
///
/// Thus, we're able to claim all outputs in the commitment transaction, one of which has the
/// following amount.
CounterpartyRevokedOutputClaimable {
/// The identifier of the channel this balance belongs to.
channel_id: ChannelId,
/// The identifier of our channel counterparty.
counterparty_node_id: PublicKey,
/// The amount, in satoshis, of the output which we can claim.
amount_satoshis: u64,
},
}
impl LightningBalance {
pub(crate) fn from_ldk_balance(
channel_id: ChannelId, counterparty_node_id: PublicKey, balance: LdkBalance,
) -> Self {
match balance {
LdkBalance::ClaimableOnChannelClose {
amount_satoshis,
transaction_fee_satoshis,
outbound_payment_htlc_rounded_msat,
outbound_forwarded_htlc_rounded_msat,
inbound_claiming_htlc_rounded_msat,
inbound_htlc_rounded_msat,
} => Self::ClaimableOnChannelClose {
channel_id,
counterparty_node_id,
amount_satoshis,
transaction_fee_satoshis,
outbound_payment_htlc_rounded_msat,
outbound_forwarded_htlc_rounded_msat,
inbound_claiming_htlc_rounded_msat,
inbound_htlc_rounded_msat,
},
LdkBalance::ClaimableAwaitingConfirmations {
amount_satoshis,
confirmation_height,
source,
} => Self::ClaimableAwaitingConfirmations {
channel_id,
counterparty_node_id,
amount_satoshis,
confirmation_height,
source,
},
LdkBalance::ContentiousClaimable {
amount_satoshis,
timeout_height,
payment_hash,
payment_preimage,
} => Self::ContentiousClaimable {
channel_id,
counterparty_node_id,
amount_satoshis,
timeout_height,
payment_hash,
payment_preimage,
},
LdkBalance::MaybeTimeoutClaimableHTLC {
amount_satoshis,
claimable_height,
payment_hash,
outbound_payment,
} => Self::MaybeTimeoutClaimableHTLC {
channel_id,
counterparty_node_id,
amount_satoshis,
claimable_height,
payment_hash,
outbound_payment,
},
LdkBalance::MaybePreimageClaimableHTLC {
amount_satoshis,
expiry_height,
payment_hash,
} => Self::MaybePreimageClaimableHTLC {
channel_id,
counterparty_node_id,
amount_satoshis,
expiry_height,
payment_hash,
},
LdkBalance::CounterpartyRevokedOutputClaimable { amount_satoshis } => {
Self::CounterpartyRevokedOutputClaimable {
channel_id,
counterparty_node_id,
amount_satoshis,
}
},
}
}
}
/// Details about the status of a known balance currently being swept to our on-chain wallet.
#[derive(Debug, Clone)]
pub enum PendingSweepBalance {
/// The spendable output is about to be swept, but a spending transaction has yet to be generated and
/// broadcast.
PendingBroadcast {
/// The identifier of the channel this balance belongs to.
channel_id: Option<ChannelId>,
/// The amount, in satoshis, of the output being swept.
amount_satoshis: u64,
},
/// A spending transaction has been generated and broadcast and is awaiting confirmation
/// on-chain.
BroadcastAwaitingConfirmation {
/// The identifier of the channel this balance belongs to.
channel_id: Option<ChannelId>,
/// The best height when we last broadcast a transaction spending the output being swept.
latest_broadcast_height: u32,
/// The identifier of the transaction spending the swept output we last broadcast.
latest_spending_txid: Txid,
/// The amount, in satoshis, of the output being swept.
amount_satoshis: u64,
},
/// A spending transaction has been confirmed on-chain and is awaiting threshold confirmations.
///
/// It will be pruned after reaching [`PRUNE_DELAY_BLOCKS`] confirmations.
///
/// [`PRUNE_DELAY_BLOCKS`]: lightning::util::sweep::PRUNE_DELAY_BLOCKS
AwaitingThresholdConfirmations {
/// The identifier of the channel this balance belongs to.
channel_id: Option<ChannelId>,
/// The identifier of the confirmed transaction spending the swept output.
latest_spending_txid: Txid,
/// The hash of the block in which the spending transaction was confirmed.
confirmation_hash: BlockHash,
/// The height at which the spending transaction was confirmed.
confirmation_height: u32,
/// The amount, in satoshis, of the output being swept.
amount_satoshis: u64,
},
}
impl PendingSweepBalance {
pub(crate) fn from_tracked_spendable_output(output_info: TrackedSpendableOutput) -> Self {
match output_info.status {
OutputSpendStatus::PendingInitialBroadcast { .. } => {
let channel_id = output_info.channel_id;
let amount_satoshis = value_from_descriptor(&output_info.descriptor).to_sat();
Self::PendingBroadcast { channel_id, amount_satoshis }
},
OutputSpendStatus::PendingFirstConfirmation {
latest_broadcast_height,
latest_spending_tx,
..
} => {
let channel_id = output_info.channel_id;
let amount_satoshis = value_from_descriptor(&output_info.descriptor).to_sat();
let latest_spending_txid = latest_spending_tx.compute_txid();
Self::BroadcastAwaitingConfirmation {
channel_id,
latest_broadcast_height,
latest_spending_txid,
amount_satoshis,
}
},
OutputSpendStatus::PendingThresholdConfirmations {
latest_spending_tx,
confirmation_height,
confirmation_hash,
..
} => {
let channel_id = output_info.channel_id;
let amount_satoshis = value_from_descriptor(&output_info.descriptor).to_sat();
let latest_spending_txid = latest_spending_tx.compute_txid();
Self::AwaitingThresholdConfirmations {
channel_id,
latest_spending_txid,
confirmation_hash,
confirmation_height,
amount_satoshis,
}
},
}
}
}