Skip to content

Commit d137df1

Browse files
committed
Open LN channel from incoming Payjoin tx
This commit allows users to schedule a channel that will opened once a Payjoin request received. This can save users 1 extra onchain transaction fees. The Payjoin flow is normal with the following caveats: 1. We use `Payjoin::ProvisionalProposal::substitue_output_address` to point to the multisig output script as retrived from `LdkEvent::FundingGeneratingReady`. 2. We dont try to preserve privacy in Payjoin channel opening transactions. 3. We wait with our response to the Payjoin sender until a `Ldk::Event::FundingTxBroadcastSafe` event is received.
1 parent cd8d1ef commit d137df1

9 files changed

+640
-31
lines changed

Cargo.toml

+19-19
Original file line numberDiff line numberDiff line change
@@ -28,23 +28,23 @@ panic = 'abort' # Abort on panic
2828
default = []
2929

3030
[dependencies]
31-
lightning = { version = "0.0.123", features = ["std"] }
32-
lightning-invoice = { version = "0.31.0" }
33-
lightning-net-tokio = { version = "0.0.123" }
34-
lightning-persister = { version = "0.0.123" }
35-
lightning-background-processor = { version = "0.0.123", features = ["futures"] }
36-
lightning-rapid-gossip-sync = { version = "0.0.123" }
37-
lightning-transaction-sync = { version = "0.0.123", features = ["esplora-async-https", "time"] }
38-
lightning-liquidity = { version = "0.1.0-alpha.4", features = ["std"] }
39-
40-
#lightning = { git = "https://github.com/lightningdevkit/rust-lightning", branch="main", features = ["std"] }
41-
#lightning-invoice = { git = "https://github.com/lightningdevkit/rust-lightning", branch="main" }
42-
#lightning-net-tokio = { git = "https://github.com/lightningdevkit/rust-lightning", branch="main" }
43-
#lightning-persister = { git = "https://github.com/lightningdevkit/rust-lightning", branch="main" }
44-
#lightning-background-processor = { git = "https://github.com/lightningdevkit/rust-lightning", branch="main", features = ["futures"] }
45-
#lightning-rapid-gossip-sync = { git = "https://github.com/lightningdevkit/rust-lightning", branch="main" }
46-
#lightning-transaction-sync = { git = "https://github.com/lightningdevkit/rust-lightning", branch="main", features = ["esplora-async"] }
47-
#lightning-liquidity = { git = "https://github.com/lightningdevkit/lightning-liquidity", branch="main", features = ["std"] }
31+
# lightning = { version = "0.0.123", features = ["std"] }
32+
# lightning-invoice = { version = "0.31.0" }
33+
# lightning-net-tokio = { version = "0.0.123" }
34+
# lightning-persister = { version = "0.0.123" }
35+
# lightning-background-processor = { version = "0.0.123", features = ["futures"] }
36+
# lightning-rapid-gossip-sync = { version = "0.0.123" }
37+
# lightning-transaction-sync = { version = "0.0.123", features = ["esplora-async-https", "time"] }
38+
# lightning-liquidity = { version = "0.1.0-alpha.4", features = ["std"] }
39+
40+
lightning = { git = "https://github.com/jbesraa/rust-lightning.git", branch="0.0.123-with-funding-brodsafe-event", features = ["std"] }
41+
lightning-invoice = { git = "https://github.com/jbesraa/rust-lightning.git", branch="0.0.123-with-funding-brodsafe-event" }
42+
lightning-net-tokio = { git = "https://github.com/jbesraa/rust-lightning.git", branch="0.0.123-with-funding-brodsafe-event" }
43+
lightning-persister = { git = "https://github.com/jbesraa/rust-lightning.git", branch="0.0.123-with-funding-brodsafe-event" }
44+
lightning-background-processor = { git = "https://github.com/jbesraa/rust-lightning.git", branch="0.0.123-with-funding-brodsafe-event", features = ["futures"] }
45+
lightning-rapid-gossip-sync = { git = "https://github.com/jbesraa/rust-lightning.git", branch="0.0.123-with-funding-brodsafe-event" }
46+
lightning-transaction-sync = { git = "https://github.com/jbesraa/rust-lightning.git", branch="0.0.123-with-funding-brodsafe-event", features = ["esplora-async"] }
47+
lightning-liquidity = { git = "https://github.com/jbesraa/lightning-liquidity", branch="pj-fixes", features = ["std"] }
4848

4949
#lightning = { path = "../rust-lightning/lightning", features = ["std"] }
5050
#lightning-invoice = { path = "../rust-lightning/lightning-invoice" }
@@ -78,8 +78,8 @@ prost = { version = "0.11.6", default-features = false}
7878
winapi = { version = "0.3", features = ["winbase"] }
7979

8080
[dev-dependencies]
81-
lightning = { version = "0.0.123", features = ["std", "_test_utils"] }
82-
#lightning = { git = "https://github.com/lightningdevkit/rust-lightning", branch="main", features = ["std", "_test_utils"] }
81+
# lightning = { version = "0.0.123", features = ["std", "_test_utils"] }
82+
lightning = { git = "https://github.com/jbesraa/rust-lightning.git", branch="0.0.123-with-funding-brodsafe-event", features = ["std", "_test_utils"] }
8383
electrum-client = { version = "0.15.1", default-features = true }
8484
bitcoincore-rpc = { version = "0.17.0", default-features = false }
8585
proptest = "1.0.0"

src/builder.rs

+1
Original file line numberDiff line numberDiff line change
@@ -1070,6 +1070,7 @@ fn build_with_store_internal(
10701070
Some(Arc::new(PayjoinReceiver::new(
10711071
Arc::clone(&logger),
10721072
Arc::clone(&wallet),
1073+
Arc::clone(&channel_manager),
10731074
directory,
10741075
relay,
10751076
ohttp_keys,

src/event.rs

+57-2
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,4 @@
1+
use crate::payjoin_receiver::PayjoinReceiver;
12
use crate::types::{DynStore, Sweeper, Wallet};
23

34
use crate::{
@@ -323,6 +324,7 @@ where
323324
network_graph: Arc<NetworkGraph>,
324325
payment_store: Arc<PaymentStore<L>>,
325326
peer_store: Arc<PeerStore<L>>,
327+
payjoin_receiver: Option<Arc<PayjoinReceiver>>,
326328
runtime: Arc<RwLock<Option<tokio::runtime::Runtime>>>,
327329
logger: L,
328330
config: Arc<Config>,
@@ -336,8 +338,8 @@ where
336338
event_queue: Arc<EventQueue<L>>, wallet: Arc<Wallet>, channel_manager: Arc<ChannelManager>,
337339
connection_manager: Arc<ConnectionManager<L>>, output_sweeper: Arc<Sweeper>,
338340
network_graph: Arc<NetworkGraph>, payment_store: Arc<PaymentStore<L>>,
339-
peer_store: Arc<PeerStore<L>>, runtime: Arc<RwLock<Option<tokio::runtime::Runtime>>>,
340-
logger: L, config: Arc<Config>,
341+
payjoin_receiver: Option<Arc<PayjoinReceiver>>, peer_store: Arc<PeerStore<L>>,
342+
runtime: Arc<RwLock<Option<tokio::runtime::Runtime>>>, logger: L, config: Arc<Config>,
341343
) -> Self {
342344
Self {
343345
event_queue,
@@ -347,6 +349,7 @@ where
347349
output_sweeper,
348350
network_graph,
349351
payment_store,
352+
payjoin_receiver,
350353
peer_store,
351354
logger,
352355
runtime,
@@ -361,6 +364,7 @@ where
361364
counterparty_node_id,
362365
channel_value_satoshis,
363366
output_script,
367+
user_channel_id,
364368
..
365369
} => {
366370
// Construct the raw transaction with the output that is paid the amount of the
@@ -371,6 +375,18 @@ where
371375
let cur_height = self.channel_manager.current_best_block().height;
372376
let locktime = LockTime::from_height(cur_height).unwrap_or(LockTime::ZERO);
373377

378+
if let Some(payjoin_receiver) = self.payjoin_receiver.clone() {
379+
if payjoin_receiver
380+
.set_channel_accepted(
381+
user_channel_id,
382+
&output_script,
383+
temporary_channel_id.0,
384+
)
385+
.await
386+
{
387+
return;
388+
}
389+
}
374390
// Sign the final funding transaction and broadcast it.
375391
match self.wallet.create_funding_transaction(
376392
output_script,
@@ -918,6 +934,45 @@ where
918934
);
919935
}
920936
},
937+
LdkEvent::FundingTxBroadcastSafe { funding_tx, .. } => {
938+
use crate::io::utils::ohttp_headers;
939+
if let Some(payjoin_receiver) = self.payjoin_receiver.clone() {
940+
let is_payjoin_channel =
941+
payjoin_receiver.set_funding_tx_signed(funding_tx.clone()).await;
942+
if let Some((url, body)) = is_payjoin_channel {
943+
log_info!(
944+
self.logger,
945+
"Detected payjoin channel transaction. Sending payjoin sender request for transaction {}",
946+
funding_tx.txid()
947+
);
948+
let headers = ohttp_headers();
949+
let client = reqwest::Client::builder().build().unwrap();
950+
match client.post(url).body(body).headers(headers).send().await {
951+
Ok(response) => {
952+
if response.status().is_success() {
953+
log_info!(
954+
self.logger,
955+
"Responded to 'Payjoin Sender' successfuly"
956+
);
957+
} else {
958+
log_info!(
959+
self.logger,
960+
"Got unsuccessful response from 'Payjoin Sender': {}",
961+
response.status()
962+
);
963+
}
964+
},
965+
Err(e) => {
966+
log_error!(
967+
self.logger,
968+
"Failed to send a response to 'Payjoin Sender': {}",
969+
e
970+
);
971+
},
972+
};
973+
}
974+
}
975+
},
921976
LdkEvent::ChannelPending {
922977
channel_id,
923978
user_channel_id,

src/lib.rs

+8
Original file line numberDiff line numberDiff line change
@@ -88,6 +88,7 @@ pub mod io;
8888
mod liquidity;
8989
mod logger;
9090
mod message_handler;
91+
mod payjoin_channel_scheduler;
9192
mod payjoin_receiver;
9293
mod payjoin_sender;
9394
pub mod payment;
@@ -661,6 +662,7 @@ impl Node {
661662
Arc::clone(&self.output_sweeper),
662663
Arc::clone(&self.network_graph),
663664
Arc::clone(&self.payment_store),
665+
self.payjoin_receiver.clone(),
664666
Arc::clone(&self.peer_store),
665667
Arc::clone(&self.runtime),
666668
Arc::clone(&self.logger),
@@ -977,6 +979,9 @@ impl Node {
977979
payjoin_sender.map(Arc::clone),
978980
payjoin_receiver.map(Arc::clone),
979981
Arc::clone(&self.config),
982+
Arc::clone(&self.peer_store),
983+
Arc::clone(&self.channel_manager),
984+
Arc::clone(&self.connection_manager),
980985
)
981986
}
982987

@@ -1003,6 +1008,9 @@ impl Node {
10031008
payjoin_sender.map(Arc::clone),
10041009
payjoin_receiver.map(Arc::clone),
10051010
Arc::clone(&self.config),
1011+
Arc::clone(&self.peer_store),
1012+
Arc::clone(&self.channel_manager),
1013+
Arc::clone(&self.connection_manager),
10061014
)
10071015
}
10081016

0 commit comments

Comments
 (0)