Skip to content

Commit df20f5d

Browse files
committed
Check for invoice expiry in InvoicePayer before we send any HTLCs
1 parent 3f74857 commit df20f5d

File tree

1 file changed

+42
-0
lines changed

1 file changed

+42
-0
lines changed

lightning-invoice/src/payment.rs

Lines changed: 42 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -257,6 +257,10 @@ where
257257
final_value_msat: invoice.amount_milli_satoshis().or(amount_msats).unwrap(),
258258
final_cltv_expiry_delta: invoice.min_final_cltv_expiry() as u32,
259259
};
260+
if has_expired(&params) {
261+
log_trace!(self.logger, "Invoice expired prior to first send for payment {}", log_bytes!(payment_hash.0));
262+
return Err(PaymentError::Invoice("Invoice expired prior to send"));
263+
}
260264
let first_hops = self.payer.first_hops();
261265
let route = self.router.find_route(
262266
&payer,
@@ -502,6 +506,25 @@ mod tests {
502506
.unwrap()
503507
}
504508

509+
fn will_expire_in_1_sec_invoice(payment_preimage: PaymentPreimage) -> Invoice {
510+
let payment_hash = Sha256::hash(&payment_preimage.0);
511+
let private_key = SecretKey::from_slice(&[42; 32]).unwrap();
512+
let timestamp = SystemTime::now()
513+
.checked_sub(Duration::from_secs(DEFAULT_EXPIRY_TIME - 1))
514+
.unwrap();
515+
InvoiceBuilder::new(Currency::Bitcoin)
516+
.description("test".into())
517+
.payment_hash(payment_hash)
518+
.payment_secret(PaymentSecret([0; 32]))
519+
.timestamp(timestamp)
520+
.min_final_cltv_expiry(144)
521+
.amount_milli_satoshis(128)
522+
.build_signed(|hash| {
523+
Secp256k1::new().sign_recoverable(hash, &private_key)
524+
})
525+
.unwrap()
526+
}
527+
505528
#[test]
506529
fn pays_invoice_on_first_attempt() {
507530
let event_handled = core::cell::RefCell::new(false);
@@ -717,7 +740,26 @@ mod tests {
717740

718741
let payment_preimage = PaymentPreimage([1; 32]);
719742
let invoice = expired_invoice(payment_preimage);
743+
if let PaymentError::Invoice(msg) = invoice_payer.pay_invoice(&invoice).unwrap_err() {
744+
assert_eq!(msg, "Invoice expired prior to send");
745+
} else { panic!("Expected Invoice Error"); }
746+
}
747+
748+
#[test]
749+
fn fails_retrying_invoice_after_expiration() {
750+
let event_handled = core::cell::RefCell::new(false);
751+
let event_handler = |_: &_| { *event_handled.borrow_mut() = true; };
752+
753+
let payer = TestPayer::new();
754+
let router = TestRouter {};
755+
let logger = TestLogger::new();
756+
let invoice_payer =
757+
InvoicePayer::new(&payer, router, &logger, event_handler, RetryAttempts(2));
758+
759+
let payment_preimage = PaymentPreimage([1; 32]);
760+
let invoice = will_expire_in_1_sec_invoice(payment_preimage);
720761
let payment_id = Some(invoice_payer.pay_invoice(&invoice).unwrap());
762+
std::thread::sleep(Duration::from_secs(2));
721763
assert_eq!(*payer.attempts.borrow(), 1);
722764

723765
let event = Event::PaymentPathFailed {

0 commit comments

Comments
 (0)