#![doc = include_str!("../README.md")]
#![warn(missing_docs)]
#![warn(rustdoc::bare_urls)]
use std::sync::Arc;
use anyhow::Result;
#[cfg(feature = "auth")]
use auth::create_auth_router;
use axum::middleware::from_fn;
use axum::response::Response;
use axum::routing::{get, post};
use axum::Router;
use cache::HttpCache;
use cdk::mint::Mint;
use router_handlers::*;
#[cfg(feature = "auth")]
mod auth;
pub mod cache;
mod router_handlers;
mod ws;
#[cfg(feature = "swagger")]
mod swagger_imports {
pub use cdk::amount::Amount;
pub use cdk::error::{ErrorCode, ErrorResponse};
pub use cdk::nuts::nut00::{
BlindSignature, BlindedMessage, CurrencyUnit, PaymentMethod, Proof, Witness,
};
pub use cdk::nuts::nut01::{Keys, KeysResponse, PublicKey, SecretKey};
pub use cdk::nuts::nut02::{KeySet, KeySetInfo, KeysetResponse};
pub use cdk::nuts::nut03::{SwapRequest, SwapResponse};
pub use cdk::nuts::nut04::{MintMethodSettings, MintRequest, MintResponse};
pub use cdk::nuts::nut05::{MeltMethodSettings, MeltRequest};
pub use cdk::nuts::nut06::{ContactInfo, MintInfo, MintVersion, Nuts, SupportedSettings};
pub use cdk::nuts::nut07::{CheckStateRequest, CheckStateResponse, ProofState, State};
pub use cdk::nuts::nut09::{RestoreRequest, RestoreResponse};
pub use cdk::nuts::nut11::P2PKWitness;
pub use cdk::nuts::nut12::{BlindSignatureDleq, ProofDleq};
pub use cdk::nuts::nut14::HTLCWitness;
pub use cdk::nuts::nut15::{Mpp, MppMethodSettings};
pub use cdk::nuts::nut23::{
MeltQuoteBolt11Request, MeltQuoteBolt11Response, MintQuoteBolt11Request,
MintQuoteBolt11Response,
};
pub use cdk::nuts::{nut04, nut05, nut15, MeltQuoteState, MintQuoteState};
}
#[cfg(feature = "swagger")]
use swagger_imports::*;
#[derive(Clone)]
pub struct MintState {
mint: Arc<Mint>,
cache: Arc<cache::HttpCache>,
}
#[cfg(feature = "swagger")]
#[derive(utoipa::OpenApi)]
#[openapi(
components(schemas(
Amount,
BlindedMessage,
BlindSignature,
BlindSignatureDleq,
CheckStateRequest,
CheckStateResponse,
ContactInfo,
CurrencyUnit,
ErrorCode,
ErrorResponse,
HTLCWitness,
Keys,
KeysResponse,
KeysetResponse,
KeySet,
KeySetInfo,
MeltRequest<String>,
MeltQuoteBolt11Request,
MeltQuoteBolt11Response<String>,
MeltQuoteState,
MeltMethodSettings,
MintRequest<String>,
MintResponse,
MintInfo,
MintQuoteBolt11Request,
MintQuoteBolt11Response<String>,
MintQuoteState,
MintMethodSettings,
MintVersion,
Mpp,
MppMethodSettings,
Nuts,
P2PKWitness,
PaymentMethod,
Proof,
ProofDleq,
ProofState,
PublicKey,
RestoreRequest,
RestoreResponse,
SecretKey,
State,
SupportedSettings,
SwapRequest,
SwapResponse,
Witness,
nut04::Settings,
nut05::Settings,
nut15::Settings
)),
info(description = "Cashu CDK mint APIs", title = "cdk-mintd",),
paths(
get_keys,
get_keyset_pubkeys,
get_keysets,
get_mint_info,
post_mint_bolt11_quote,
get_check_mint_bolt11_quote,
post_mint_bolt11,
post_melt_bolt11_quote,
get_check_melt_bolt11_quote,
post_melt_bolt11,
post_swap,
post_check,
post_restore
)
)]
pub struct ApiDocV1;
pub async fn create_mint_router(mint: Arc<Mint>) -> Result<Router> {
create_mint_router_with_custom_cache(mint, Default::default()).await
}
async fn cors_middleware(
req: axum::http::Request<axum::body::Body>,
next: axum::middleware::Next,
) -> Response {
if req.method() == axum::http::Method::OPTIONS {
let mut response = Response::new("".into());
response
.headers_mut()
.insert("Access-Control-Allow-Origin", "*".parse().unwrap());
response.headers_mut().insert(
"Access-Control-Allow-Methods",
"GET, POST, OPTIONS".parse().unwrap(),
);
response.headers_mut().insert(
"Access-Control-Allow-Headers",
"Content-Type".parse().unwrap(),
);
return response;
}
let mut response = next.run(req).await;
response
.headers_mut()
.insert("Access-Control-Allow-Origin", "*".parse().unwrap());
response.headers_mut().insert(
"Access-Control-Allow-Methods",
"GET, POST, OPTIONS".parse().unwrap(),
);
response.headers_mut().insert(
"Access-Control-Allow-Headers",
"Content-Type".parse().unwrap(),
);
response
}
pub async fn create_mint_router_with_custom_cache(
mint: Arc<Mint>,
cache: HttpCache,
) -> Result<Router> {
let state = MintState {
mint,
cache: Arc::new(cache),
};
let v1_router = Router::new()
.route("/keys", get(get_keys))
.route("/keysets", get(get_keysets))
.route("/keys/{keyset_id}", get(get_keyset_pubkeys))
.route("/swap", post(cache_post_swap))
.route("/mint/quote/bolt11", post(post_mint_bolt11_quote))
.route(
"/mint/quote/bolt11/{quote_id}",
get(get_check_mint_bolt11_quote),
)
.route("/mint/bolt11", post(cache_post_mint_bolt11))
.route("/melt/quote/bolt11", post(post_melt_bolt11_quote))
.route("/ws", get(ws_handler))
.route(
"/melt/quote/bolt11/{quote_id}",
get(get_check_melt_bolt11_quote),
)
.route("/melt/bolt11", post(cache_post_melt_bolt11))
.route("/checkstate", post(post_check))
.route("/info", get(get_mint_info))
.route("/restore", post(post_restore));
let mint_router = Router::new()
.nest("/v1", v1_router)
.layer(from_fn(cors_middleware));
#[cfg(feature = "auth")]
let mint_router = {
let auth_router = create_auth_router(state.clone());
mint_router.nest("/v1", auth_router)
};
let mint_router = mint_router.with_state(state);
Ok(mint_router)
}