Skip to content
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
33 changes: 28 additions & 5 deletions src/builder.rs
Original file line number Diff line number Diff line change
Expand Up @@ -64,7 +64,9 @@ use crate::io::utils::{
};
use crate::io::vss_store::VssStoreBuilder;
use crate::io::{
self, PAYMENT_INFO_PERSISTENCE_PRIMARY_NAMESPACE, PAYMENT_INFO_PERSISTENCE_SECONDARY_NAMESPACE,
self, CHANNEL_RECORD_PERSISTENCE_PRIMARY_NAMESPACE,
CHANNEL_RECORD_PERSISTENCE_SECONDARY_NAMESPACE, PAYMENT_INFO_PERSISTENCE_PRIMARY_NAMESPACE,
PAYMENT_INFO_PERSISTENCE_SECONDARY_NAMESPACE,
PENDING_PAYMENT_INFO_PERSISTENCE_PRIMARY_NAMESPACE,
PENDING_PAYMENT_INFO_PERSISTENCE_SECONDARY_NAMESPACE,
};
Expand All @@ -77,9 +79,9 @@ use crate::peer_store::PeerStore;
use crate::runtime::{Runtime, RuntimeSpawner};
use crate::tx_broadcaster::TransactionBroadcaster;
use crate::types::{
AsyncPersister, ChainMonitor, ChannelManager, DynStore, DynStoreRef, DynStoreWrapper,
GossipSync, Graph, HRNResolver, KeysManager, MessageRouter, OnionMessenger, PaymentStore,
PeerManager, PendingPaymentStore,
AsyncPersister, ChainMonitor, ChannelManager, ChannelRecordStore, DynStore, DynStoreRef,
DynStoreWrapper, GossipSync, Graph, HRNResolver, KeysManager, MessageRouter, OnionMessenger,
PaymentStore, PeerManager, PendingPaymentStore,
};
use crate::wallet::persist::KVStoreWalletPersister;
use crate::wallet::Wallet;
Expand Down Expand Up @@ -1379,7 +1381,7 @@ fn build_with_store_internal(

let kv_store_ref = Arc::clone(&kv_store);
let logger_ref = Arc::clone(&logger);
let (payment_store_res, node_metris_res, pending_payment_store_res) =
let (payment_store_res, node_metris_res, pending_payment_store_res, channel_record_store_res) =
runtime.block_on(async move {
tokio::join!(
read_all_objects(
Expand All @@ -1394,6 +1396,12 @@ fn build_with_store_internal(
PENDING_PAYMENT_INFO_PERSISTENCE_PRIMARY_NAMESPACE,
PENDING_PAYMENT_INFO_PERSISTENCE_SECONDARY_NAMESPACE,
Arc::clone(&logger_ref),
),
read_all_objects(
&*kv_store_ref,
CHANNEL_RECORD_PERSISTENCE_PRIMARY_NAMESPACE,
CHANNEL_RECORD_PERSISTENCE_SECONDARY_NAMESPACE,
Arc::clone(&logger_ref),
)
)
});
Expand Down Expand Up @@ -1605,6 +1613,20 @@ fn build_with_store_internal(
},
};

let channel_record_store = match channel_record_store_res {
Ok(channel_records) => Arc::new(ChannelRecordStore::new(
channel_records,
CHANNEL_RECORD_PERSISTENCE_PRIMARY_NAMESPACE.to_string(),
CHANNEL_RECORD_PERSISTENCE_SECONDARY_NAMESPACE.to_string(),
Arc::clone(&kv_store),
Arc::clone(&logger),
)),
Err(e) => {
log_error!(logger, "Failed to read channel record data from store: {}", e);
return Err(BuildError::ReadFailed);
},
};

let wallet = Arc::new(Wallet::new(
bdk_wallet,
wallet_persister,
Expand Down Expand Up @@ -2149,6 +2171,7 @@ fn build_with_store_internal(
scorer,
peer_store,
payment_store,
channel_record_store,
lnurl_auth,
is_running,
node_metrics,
Expand Down
10 changes: 10 additions & 0 deletions src/channel/mod.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
// 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.

//! Per-channel state tracking.
pub(crate) mod store;
125 changes: 125 additions & 0 deletions src/channel/store.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,125 @@
// 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 bitcoin::secp256k1::PublicKey;
use lightning::impl_writeable_tlv_based_enum;
use lightning::ln::types::ChannelId;

use crate::data_store::{StorableObject, StorableObjectId, StorableObjectUpdate};
use crate::hex_utils;
use crate::types::UserChannelId;

/// Persistent per-channel state tracked by LDK Node, keyed by [`UserChannelId`].
///
/// Tracks where a channel sits in its lifecycle. Each variant currently holds only the channel's
/// identity; the per-feature state and the transitions between lifecycle states are added by the
/// work that needs them.
#[derive(Clone, Debug, PartialEq, Eq)]
pub(crate) enum ChannelRecord {
/// A channel whose funding transaction has not yet been observed.
Unfunded {
user_channel_id: UserChannelId,
counterparty_node_id: PublicKey,
channel_id: ChannelId,
},
/// A channel whose funding transaction exists.
Funded {
user_channel_id: UserChannelId,
counterparty_node_id: PublicKey,
channel_id: ChannelId,
},
/// A channel that has been closed.
Closed {
user_channel_id: UserChannelId,
counterparty_node_id: PublicKey,
channel_id: ChannelId,
},
}

impl_writeable_tlv_based_enum!(ChannelRecord,
(0, Unfunded) => {
(0, user_channel_id, required),
(2, counterparty_node_id, required),
(4, channel_id, required),
},
(2, Funded) => {
(0, user_channel_id, required),
(2, counterparty_node_id, required),
(4, channel_id, required),
},
(4, Closed) => {
(0, user_channel_id, required),
(2, counterparty_node_id, required),
(4, channel_id, required),
},
);

impl StorableObjectId for UserChannelId {
fn encode_to_hex_str(&self) -> String {
hex_utils::to_string(&self.0.to_be_bytes())
}
}

#[derive(Clone, Debug, PartialEq, Eq)]
pub(crate) struct ChannelRecordUpdate {
pub user_channel_id: UserChannelId,
}

impl StorableObject for ChannelRecord {
type Id = UserChannelId;
type Update = ChannelRecordUpdate;

fn id(&self) -> Self::Id {
match self {
ChannelRecord::Unfunded { user_channel_id, .. }
| ChannelRecord::Funded { user_channel_id, .. }
| ChannelRecord::Closed { user_channel_id, .. } => *user_channel_id,
}
}

fn update(&mut self, _update: Self::Update) -> bool {
// Records currently carry only the channel's identity, so there is nothing to mutate in
// place. Lifecycle transitions are performed by replacing the record once a consumer of
// this store exists.
false
}

fn to_update(&self) -> Self::Update {
Self::Update { user_channel_id: self.id() }
}
}

impl StorableObjectUpdate<ChannelRecord> for ChannelRecordUpdate {
fn id(&self) -> <ChannelRecord as StorableObject>::Id {
self.user_channel_id
}
}

#[cfg(test)]
mod tests {
use lightning::util::ser::{Readable, Writeable};

use super::*;

#[test]
fn channel_record_is_serializable() {
let user_channel_id = UserChannelId(42);
let counterparty_node_id = bitcoin::secp256k1::PublicKey::from_slice(&[2u8; 33]).unwrap();
let channel_id = ChannelId([3u8; 32]);

for record in [
ChannelRecord::Unfunded { user_channel_id, counterparty_node_id, channel_id },
ChannelRecord::Funded { user_channel_id, counterparty_node_id, channel_id },
ChannelRecord::Closed { user_channel_id, counterparty_node_id, channel_id },
] {
let encoded = record.encode();
let decoded = ChannelRecord::read(&mut &encoded[..]).unwrap();
assert_eq!(record, decoded);
assert_eq!(decoded.id(), user_channel_id);
}
}
}
4 changes: 4 additions & 0 deletions src/io/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -84,3 +84,7 @@ pub(crate) const STATIC_INVOICE_STORE_PRIMARY_NAMESPACE: &str = "static_invoices
/// The pending payment information will be persisted under this prefix.
pub(crate) const PENDING_PAYMENT_INFO_PERSISTENCE_PRIMARY_NAMESPACE: &str = "pending_payments";
pub(crate) const PENDING_PAYMENT_INFO_PERSISTENCE_SECONDARY_NAMESPACE: &str = "";

/// The per-channel records will be persisted under this prefix.
pub(crate) const CHANNEL_RECORD_PERSISTENCE_PRIMARY_NAMESPACE: &str = "channel_records";
pub(crate) const CHANNEL_RECORD_PERSISTENCE_SECONDARY_NAMESPACE: &str = "";
11 changes: 8 additions & 3 deletions src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -83,6 +83,7 @@
mod balance;
mod builder;
mod chain;
mod channel;
pub mod config;
mod connection;
mod data_store;
Expand Down Expand Up @@ -175,9 +176,9 @@ use peer_store::{PeerInfo, PeerStore};
use runtime::Runtime;
pub use tokio;
use types::{
Broadcaster, BumpTransactionEventHandler, ChainMonitor, ChannelManager, DynStore, Graph,
HRNResolver, KeysManager, OnionMessenger, PaymentStore, PeerManager, Router, Scorer, Sweeper,
Wallet,
Broadcaster, BumpTransactionEventHandler, ChainMonitor, ChannelManager, ChannelRecordStore,
DynStore, Graph, HRNResolver, KeysManager, OnionMessenger, PaymentStore, PeerManager, Router,
Scorer, Sweeper, Wallet,
};
pub use types::{ChannelDetails, CustomTlvRecord, PeerDetails, UserChannelId};
pub use vss_client;
Expand Down Expand Up @@ -242,6 +243,10 @@ pub struct Node {
scorer: Arc<Mutex<Scorer>>,
peer_store: Arc<PeerStore<Arc<Logger>>>,
payment_store: Arc<PaymentStore>,
// Foundational per-channel record store, loaded and persisted here so that upcoming
// per-channel features can build on it. Not yet read within this crate.
#[allow(dead_code)]

Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please drop this allow(dead_code). IMO it's fine to have the warnings on main temporarily, better than hiding and potentially never getting back to fixing them.

channel_record_store: Arc<ChannelRecordStore>,
lnurl_auth: Arc<LnurlAuth>,
is_running: Arc<RwLock<bool>>,
node_metrics: Arc<PersistedNodeMetrics>,
Expand Down
5 changes: 4 additions & 1 deletion src/types.rs
Original file line number Diff line number Diff line change
Expand Up @@ -38,6 +38,7 @@ use lightning_net_tokio::SocketDescriptor;

use crate::chain::bitcoind::UtxoSourceClient;
use crate::chain::ChainSource;
use crate::channel::store::ChannelRecord;
use crate::config::ChannelConfig;
use crate::data_store::DataStore;
use crate::fee_estimator::OnchainFeeEstimator;
Expand Down Expand Up @@ -318,7 +319,7 @@ pub(crate) type PaymentStore = DataStore<PaymentDetails, Arc<Logger>>;
/// A local, potentially user-provided, identifier of a channel.
///
/// By default, this will be randomly generated for the user to ensure local uniqueness.
#[derive(Debug, Copy, Clone, PartialEq, Eq)]
#[derive(Debug, Copy, Clone, PartialEq, Eq, Hash)]
pub struct UserChannelId(pub u128);

impl Writeable for UserChannelId {
Expand Down Expand Up @@ -628,3 +629,5 @@ impl From<&(u64, Vec<u8>)> for CustomTlvRecord {
}

pub(crate) type PendingPaymentStore = DataStore<PendingPaymentDetails, Arc<Logger>>;

pub(crate) type ChannelRecordStore = DataStore<ChannelRecord, Arc<Logger>>;