ln_gateway/gateway_module_v2/
send_sm.rsuse std::fmt;
use fedimint_client::sm::{ClientSMDatabaseTransaction, State, StateTransition};
use fedimint_client::transaction::{ClientInput, ClientInputBundle};
use fedimint_client::DynGlobalClientContext;
use fedimint_core::config::FederationId;
use fedimint_core::core::OperationId;
use fedimint_core::encoding::{Decodable, Encodable};
use fedimint_core::secp256k1::Keypair;
use fedimint_core::{Amount, OutPoint};
use fedimint_lnv2_common::contracts::{OutgoingContract, PaymentImage};
use fedimint_lnv2_common::{LightningInput, LightningInputV0, LightningInvoice, OutgoingWitness};
use serde::{Deserialize, Serialize};
use super::events::{OutgoingPaymentFailed, OutgoingPaymentSucceeded};
use super::FinalReceiveState;
use crate::gateway_module_v2::{GatewayClientContextV2, GatewayClientModuleV2};
#[derive(Debug, Clone, Eq, PartialEq, Hash, Decodable, Encodable)]
pub struct SendStateMachine {
pub common: SendSMCommon,
pub state: SendSMState,
}
impl SendStateMachine {
pub fn update(&self, state: SendSMState) -> Self {
Self {
common: self.common.clone(),
state,
}
}
}
impl fmt::Display for SendStateMachine {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(
f,
"Send State Machine Operation ID: {:?} State: {}",
self.common.operation_id, self.state
)
}
}
#[derive(Debug, Clone, Eq, PartialEq, Hash, Decodable, Encodable)]
pub struct SendSMCommon {
pub operation_id: OperationId,
pub contract: OutgoingContract,
pub max_delay: u64,
pub min_contract_amount: Amount,
pub invoice: LightningInvoice,
pub claim_keypair: Keypair,
}
#[derive(Debug, Clone, Eq, PartialEq, Hash, Decodable, Encodable)]
pub enum SendSMState {
Sending,
Claiming(Claiming),
Cancelled(Cancelled),
}
#[derive(Debug, Serialize, Deserialize)]
pub struct PaymentResponse {
preimage: [u8; 32],
target_federation: Option<FederationId>,
}
impl fmt::Display for SendSMState {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
match self {
SendSMState::Sending => write!(f, "Sending"),
SendSMState::Claiming(_) => write!(f, "Claiming"),
SendSMState::Cancelled(_) => write!(f, "Cancelled"),
}
}
}
#[derive(Debug, Clone, Eq, PartialEq, Hash, Decodable, Encodable)]
pub struct Claiming {
pub preimage: [u8; 32],
pub outpoints: Vec<OutPoint>,
}
#[derive(Debug, Clone, Eq, PartialEq, Hash, Decodable, Encodable, Serialize, Deserialize)]
pub enum Cancelled {
InvoiceExpired,
TimeoutTooClose,
Underfunded,
RegistrationError(String),
FinalizationError(String),
Rejected,
Refunded,
Failure,
LightningRpcError(String),
}
#[cfg_attr(doc, aquamarine::aquamarine)]
impl State for SendStateMachine {
type ModuleContext = GatewayClientContextV2;
fn transitions(
&self,
context: &Self::ModuleContext,
global_context: &DynGlobalClientContext,
) -> Vec<StateTransition<Self>> {
let gc = global_context.clone();
let gateway_context = context.clone();
match &self.state {
SendSMState::Sending => {
vec![StateTransition::new(
Self::send_payment(
context.clone(),
self.common.max_delay,
self.common.min_contract_amount,
self.common.invoice.clone(),
self.common.contract.clone(),
),
move |dbtx, result, old_state| {
Box::pin(Self::transition_send_payment(
dbtx,
old_state,
gc.clone(),
result,
gateway_context.clone(),
))
},
)]
}
SendSMState::Claiming(..) | SendSMState::Cancelled(..) => {
vec![]
}
}
}
fn operation_id(&self) -> OperationId {
self.common.operation_id
}
}
impl SendStateMachine {
async fn send_payment(
context: GatewayClientContextV2,
max_delay: u64,
min_contract_amount: Amount,
invoice: LightningInvoice,
contract: OutgoingContract,
) -> Result<PaymentResponse, Cancelled> {
let LightningInvoice::Bolt11(invoice) = invoice;
if invoice.is_expired() {
return Err(Cancelled::InvoiceExpired);
}
if max_delay == 0 {
return Err(Cancelled::TimeoutTooClose);
}
let Some(max_fee) = contract.amount.checked_sub(min_contract_amount) else {
return Err(Cancelled::Underfunded);
};
let lightning_context = context
.gateway
.get_lightning_context()
.await
.map_err(|e| Cancelled::LightningRpcError(e.to_string()))?;
if lightning_context.lightning_public_key == invoice.get_payee_pub_key() {
let (contract, client) = context
.gateway
.get_registered_incoming_contract_and_client_v2(
PaymentImage::Hash(*invoice.payment_hash()),
invoice
.amount_milli_satoshis()
.expect("The amount invoice has been checked previously"),
)
.await
.map_err(|e| Cancelled::RegistrationError(e.to_string()))?;
return match client
.get_first_module::<GatewayClientModuleV2>()
.expect("Must have client module")
.relay_direct_swap(
contract,
invoice
.amount_milli_satoshis()
.expect("amountless invoices are not supported"),
)
.await
{
Ok(final_receive_state) => match final_receive_state {
FinalReceiveState::Rejected => Err(Cancelled::Rejected),
FinalReceiveState::Success(preimage) => Ok(PaymentResponse {
preimage,
target_federation: Some(client.federation_id()),
}),
FinalReceiveState::Refunded => Err(Cancelled::Refunded),
FinalReceiveState::Failure => Err(Cancelled::Failure),
},
Err(e) => Err(Cancelled::FinalizationError(e.to_string())),
};
}
let preimage = lightning_context
.lnrpc
.pay(invoice, max_delay, max_fee)
.await
.map(|response| response.preimage.0)
.map_err(|e| Cancelled::LightningRpcError(e.to_string()))?;
Ok(PaymentResponse {
preimage,
target_federation: None,
})
}
async fn transition_send_payment(
dbtx: &mut ClientSMDatabaseTransaction<'_, '_>,
old_state: SendStateMachine,
global_context: DynGlobalClientContext,
result: Result<PaymentResponse, Cancelled>,
client_ctx: GatewayClientContextV2,
) -> SendStateMachine {
match result {
Ok(payment_response) => {
client_ctx
.module
.client_ctx
.log_event(
&mut dbtx.module_tx(),
OutgoingPaymentSucceeded {
payment_image: old_state.common.contract.payment_image.clone(),
target_federation: payment_response.target_federation,
},
)
.await;
let client_input = ClientInput::<LightningInput> {
input: LightningInput::V0(LightningInputV0::Outgoing(
old_state.common.contract.contract_id(),
OutgoingWitness::Claim(payment_response.preimage),
)),
amount: old_state.common.contract.amount,
keys: vec![old_state.common.claim_keypair],
};
let outpoints = global_context
.claim_inputs(dbtx, ClientInputBundle::new_no_sm(vec![client_input]))
.await
.expect("Cannot claim input, additional funding needed")
.into_iter()
.collect();
old_state.update(SendSMState::Claiming(Claiming {
preimage: payment_response.preimage,
outpoints,
}))
}
Err(e) => {
client_ctx
.module
.client_ctx
.log_event(
&mut dbtx.module_tx(),
OutgoingPaymentFailed {
payment_image: old_state.common.contract.payment_image.clone(),
error: e.clone(),
},
)
.await;
old_state.update(SendSMState::Cancelled(e))
}
}
}
}