use serde::de::DeserializeOwned;
use serde::{Deserialize, Serialize};
use crate::nuts::nut00::KnownMethod;
use crate::nuts::nut04::{MintQuoteCustomRequest, MintQuoteCustomResponse};
use crate::nuts::nut23::{MintQuoteBolt11Request, MintQuoteBolt11Response, QuoteState};
use crate::nuts::nut25::{MintQuoteBolt12Request, MintQuoteBolt12Response};
use crate::nuts::nut_onchain::{MintQuoteOnchainRequest, MintQuoteOnchainResponse};
use crate::{Amount, CurrencyUnit, PaymentMethod, PublicKey};
#[derive(Debug, Clone, Serialize, Deserialize)]
pub enum MintQuoteRequest {
Bolt11(MintQuoteBolt11Request),
Bolt12(MintQuoteBolt12Request),
Onchain(MintQuoteOnchainRequest),
Custom {
method: PaymentMethod,
request: MintQuoteCustomRequest,
},
}
impl From<MintQuoteBolt11Request> for MintQuoteRequest {
fn from(request: MintQuoteBolt11Request) -> Self {
MintQuoteRequest::Bolt11(request)
}
}
impl From<MintQuoteBolt12Request> for MintQuoteRequest {
fn from(request: MintQuoteBolt12Request) -> Self {
MintQuoteRequest::Bolt12(request)
}
}
impl From<MintQuoteOnchainRequest> for MintQuoteRequest {
fn from(request: MintQuoteOnchainRequest) -> Self {
MintQuoteRequest::Onchain(request)
}
}
impl MintQuoteRequest {
pub fn method(&self) -> PaymentMethod {
match self {
Self::Bolt11(_) => PaymentMethod::Known(KnownMethod::Bolt11),
Self::Bolt12(_) => PaymentMethod::Known(KnownMethod::Bolt12),
Self::Onchain(_) => PaymentMethod::Known(KnownMethod::Onchain),
Self::Custom { method, .. } => method.clone(),
}
}
pub fn amount(&self) -> Option<Amount> {
match self {
Self::Bolt11(request) => Some(request.amount),
Self::Bolt12(request) => request.amount,
Self::Onchain(_) => None,
Self::Custom { request, .. } => Some(request.amount),
}
}
pub fn unit(&self) -> CurrencyUnit {
match self {
Self::Bolt11(request) => request.unit.clone(),
Self::Bolt12(request) => request.unit.clone(),
Self::Onchain(request) => request.unit.clone(),
Self::Custom { request, .. } => request.unit.clone(),
}
}
pub fn payment_method(&self) -> PaymentMethod {
self.method()
}
pub fn pubkey(&self) -> Option<PublicKey> {
match self {
Self::Bolt11(request) => request.pubkey,
Self::Bolt12(request) => Some(request.pubkey),
Self::Onchain(request) => Some(request.pubkey),
Self::Custom { request, .. } => request.pubkey,
}
}
}
#[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(bound = "Q: Serialize + DeserializeOwned")]
pub enum MintQuoteResponse<Q> {
Bolt11(MintQuoteBolt11Response<Q>),
Bolt12(MintQuoteBolt12Response<Q>),
Onchain(MintQuoteOnchainResponse<Q>),
Custom {
method: PaymentMethod,
response: MintQuoteCustomResponse<Q>,
},
}
impl<Q> MintQuoteResponse<Q> {
pub fn method(&self) -> PaymentMethod {
match self {
Self::Bolt11(_) => PaymentMethod::Known(KnownMethod::Bolt11),
Self::Bolt12(_) => PaymentMethod::Known(KnownMethod::Bolt12),
Self::Onchain(_) => PaymentMethod::Known(KnownMethod::Onchain),
Self::Custom { method, .. } => method.clone(),
}
}
pub fn quote(&self) -> &Q {
match self {
Self::Bolt11(r) => &r.quote,
Self::Bolt12(r) => &r.quote,
Self::Onchain(r) => &r.quote,
Self::Custom { response: r, .. } => &r.quote,
}
}
pub fn request(&self) -> &str {
match self {
Self::Bolt11(r) => &r.request,
Self::Bolt12(r) => &r.request,
Self::Onchain(r) => &r.request,
Self::Custom { response: r, .. } => &r.request,
}
}
pub fn state(&self) -> Option<QuoteState> {
match self {
Self::Bolt11(r) => Some(r.state),
Self::Bolt12(r) => Some(quote_state_from_amounts(r.amount_paid, r.amount_issued)),
Self::Onchain(r) => Some(quote_state_from_amounts(r.amount_paid, r.amount_issued)),
Self::Custom { response, .. } => Some(quote_state_from_amounts(
response.amount_paid,
response.amount_issued,
)),
}
}
pub fn expiry(&self) -> Option<u64> {
match self {
Self::Bolt11(r) => r.expiry,
Self::Bolt12(r) => r.expiry,
Self::Onchain(r) => r.expiry,
Self::Custom { response: r, .. } => r.expiry,
}
}
}
pub(crate) fn quote_state_from_amounts(amount_paid: Amount, amount_issued: Amount) -> QuoteState {
if amount_paid == Amount::ZERO && amount_issued == Amount::ZERO {
return QuoteState::Unpaid;
}
match amount_paid.cmp(&amount_issued) {
std::cmp::Ordering::Less | std::cmp::Ordering::Equal => QuoteState::Issued,
std::cmp::Ordering::Greater => QuoteState::Paid,
}
}
#[cfg(test)]
mod tests {
use super::*;
fn custom_response(amount_paid: Amount, amount_issued: Amount) -> MintQuoteResponse<String> {
MintQuoteResponse::Custom {
method: PaymentMethod::Custom("custom".to_string()),
response: MintQuoteCustomResponse {
quote: "quote".to_string(),
request: "custom-request".to_string(),
amount: Some(Amount::from(100)),
amount_paid,
amount_issued,
unit: Some(CurrencyUnit::Sat),
expiry: None,
pubkey: None,
extra: serde_json::Value::Null,
},
}
}
#[test]
fn custom_state_is_derived_from_amount_counters() {
assert_eq!(
custom_response(Amount::ZERO, Amount::ZERO).state(),
Some(QuoteState::Unpaid)
);
assert_eq!(
custom_response(Amount::from(100), Amount::ZERO).state(),
Some(QuoteState::Paid)
);
assert_eq!(
custom_response(Amount::from(100), Amount::from(100)).state(),
Some(QuoteState::Issued)
);
}
#[test]
fn bolt12_state_uses_unissued_amount() {
let response = MintQuoteResponse::Bolt12(MintQuoteBolt12Response {
quote: "quote".to_string(),
request: "bolt12-request".to_string(),
amount: Some(Amount::from(100)),
unit: CurrencyUnit::Sat,
expiry: None,
pubkey: PublicKey::from_hex(
"02a8cda4cf448bfce9a9e46e588c06ea1780fcb94e3bbdf3277f42995d403a8b0c",
)
.expect("valid public key"),
amount_paid: Amount::from(100),
amount_issued: Amount::from(40),
});
assert_eq!(response.state(), Some(QuoteState::Paid));
}
}