mirror of
https://github.com/bitcoinresearchkit/brk.git
synced 2026-04-24 06:39:58 -07:00
259 lines
11 KiB
Rust
259 lines
11 KiB
Rust
use aide::axum::{
|
|
ApiRouter,
|
|
routing::{get_with, post_with},
|
|
};
|
|
use axum::{
|
|
extract::{Path, State},
|
|
http::{HeaderMap, Uri},
|
|
};
|
|
use brk_types::{
|
|
CpfpInfo, MerkleProof, Transaction, TxOutspend, TxStatus, Txid, TxidParam, TxidVout,
|
|
TxidsParam,
|
|
};
|
|
|
|
use crate::{CacheStrategy, extended::TransformResponseExtended};
|
|
|
|
use super::AppState;
|
|
|
|
pub trait TxRoutes {
|
|
fn add_tx_routes(self) -> Self;
|
|
}
|
|
|
|
impl TxRoutes for ApiRouter<AppState> {
|
|
fn add_tx_routes(self) -> Self {
|
|
self
|
|
.api_route(
|
|
"/api/v1/cpfp/{txid}",
|
|
get_with(
|
|
async |uri: Uri, headers: HeaderMap, Path(txid): Path<TxidParam>, State(state): State<AppState>| {
|
|
state.cached_json(&headers, CacheStrategy::MempoolHash(0), &uri, move |q| q.cpfp(txid)).await
|
|
},
|
|
|op| op
|
|
.id("get_cpfp")
|
|
.transactions_tag()
|
|
.summary("CPFP info")
|
|
.description("Returns ancestors and descendants for a CPFP transaction.\n\n*[Mempool.space docs](https://mempool.space/docs/api/rest#get-children-pay-for-parent)*")
|
|
.json_response::<CpfpInfo>()
|
|
.not_found()
|
|
.server_error(),
|
|
),
|
|
)
|
|
.api_route(
|
|
"/api/tx/{txid}",
|
|
get_with(
|
|
async |
|
|
uri: Uri,
|
|
headers: HeaderMap,
|
|
Path(txid): Path<TxidParam>,
|
|
State(state): State<AppState>
|
|
| {
|
|
state.cached_json(&headers, CacheStrategy::Height, &uri, move |q| q.transaction(txid)).await
|
|
},
|
|
|op| op
|
|
.id("get_tx")
|
|
.transactions_tag()
|
|
.summary("Transaction information")
|
|
.description(
|
|
"Retrieve complete transaction data by transaction ID (txid). Returns inputs, outputs, fee, size, and confirmation status.\n\n*[Mempool.space docs](https://mempool.space/docs/api/rest#get-transaction)*",
|
|
)
|
|
.json_response::<Transaction>()
|
|
.not_modified()
|
|
.bad_request()
|
|
.not_found()
|
|
.server_error(),
|
|
),
|
|
)
|
|
.api_route(
|
|
"/api/tx/{txid}/hex",
|
|
get_with(
|
|
async |
|
|
uri: Uri,
|
|
headers: HeaderMap,
|
|
Path(txid): Path<TxidParam>,
|
|
State(state): State<AppState>
|
|
| {
|
|
state.cached_text(&headers, CacheStrategy::Height, &uri, move |q| q.transaction_hex(txid)).await
|
|
},
|
|
|op| op
|
|
.id("get_tx_hex")
|
|
.transactions_tag()
|
|
.summary("Transaction hex")
|
|
.description(
|
|
"Retrieve the raw transaction as a hex-encoded string. Returns the serialized transaction in hexadecimal format.\n\n*[Mempool.space docs](https://mempool.space/docs/api/rest#get-transaction-hex)*",
|
|
)
|
|
.text_response()
|
|
.not_modified()
|
|
.bad_request()
|
|
.not_found()
|
|
.server_error(),
|
|
),
|
|
)
|
|
.api_route(
|
|
"/api/tx/{txid}/merkleblock-proof",
|
|
get_with(
|
|
async |uri: Uri, headers: HeaderMap, Path(txid): Path<TxidParam>, State(state): State<AppState>| {
|
|
state.cached_text(&headers, CacheStrategy::Height, &uri, move |q| q.merkleblock_proof(txid)).await
|
|
},
|
|
|op| op
|
|
.id("get_tx_merkleblock_proof")
|
|
.transactions_tag()
|
|
.summary("Transaction merkleblock proof")
|
|
.description("Get the merkleblock proof for a transaction (BIP37 format, hex encoded).\n\n*[Mempool.space docs](https://mempool.space/docs/api/rest#get-transaction-merkleblock-proof)*")
|
|
.text_response()
|
|
.not_modified()
|
|
.bad_request()
|
|
.not_found()
|
|
.server_error(),
|
|
),
|
|
)
|
|
.api_route(
|
|
"/api/tx/{txid}/merkle-proof",
|
|
get_with(
|
|
async |uri: Uri, headers: HeaderMap, Path(txid): Path<TxidParam>, State(state): State<AppState>| {
|
|
state.cached_json(&headers, CacheStrategy::Height, &uri, move |q| q.merkle_proof(txid)).await
|
|
},
|
|
|op| op
|
|
.id("get_tx_merkle_proof")
|
|
.transactions_tag()
|
|
.summary("Transaction merkle proof")
|
|
.description("Get the merkle inclusion proof for a transaction.\n\n*[Mempool.space docs](https://mempool.space/docs/api/rest#get-transaction-merkle-proof)*")
|
|
.json_response::<MerkleProof>()
|
|
.not_modified()
|
|
.bad_request()
|
|
.not_found()
|
|
.server_error(),
|
|
),
|
|
)
|
|
.api_route(
|
|
"/api/tx/{txid}/outspend/{vout}",
|
|
get_with(
|
|
async |
|
|
uri: Uri,
|
|
headers: HeaderMap,
|
|
Path(path): Path<TxidVout>,
|
|
State(state): State<AppState>
|
|
| {
|
|
let txid = TxidParam { txid: path.txid };
|
|
state.cached_json(&headers, CacheStrategy::Height, &uri, move |q| q.outspend(txid, path.vout)).await
|
|
},
|
|
|op| op
|
|
.id("get_tx_outspend")
|
|
.transactions_tag()
|
|
.summary("Output spend status")
|
|
.description(
|
|
"Get the spending status of a transaction output. Returns whether the output has been spent and, if so, the spending transaction details.\n\n*[Mempool.space docs](https://mempool.space/docs/api/rest#get-transaction-outspend)*",
|
|
)
|
|
.json_response::<TxOutspend>()
|
|
.not_modified()
|
|
.bad_request()
|
|
.not_found()
|
|
.server_error(),
|
|
),
|
|
)
|
|
.api_route(
|
|
"/api/tx/{txid}/outspends",
|
|
get_with(
|
|
async |
|
|
uri: Uri,
|
|
headers: HeaderMap,
|
|
Path(txid): Path<TxidParam>,
|
|
State(state): State<AppState>
|
|
| {
|
|
state.cached_json(&headers, CacheStrategy::Height, &uri, move |q| q.outspends(txid)).await
|
|
},
|
|
|op| op
|
|
.id("get_tx_outspends")
|
|
.transactions_tag()
|
|
.summary("All output spend statuses")
|
|
.description(
|
|
"Get the spending status of all outputs in a transaction. Returns an array with the spend status for each output.\n\n*[Mempool.space docs](https://mempool.space/docs/api/rest#get-transaction-outspends)*",
|
|
)
|
|
.json_response::<Vec<TxOutspend>>()
|
|
.not_modified()
|
|
.bad_request()
|
|
.not_found()
|
|
.server_error(),
|
|
),
|
|
)
|
|
.api_route(
|
|
"/api/tx/{txid}/raw",
|
|
get_with(
|
|
async |uri: Uri, headers: HeaderMap, Path(txid): Path<TxidParam>, State(state): State<AppState>| {
|
|
state.cached_bytes(&headers, CacheStrategy::Height, &uri, move |q| q.transaction_raw(txid)).await
|
|
},
|
|
|op| op
|
|
.id("get_tx_raw")
|
|
.transactions_tag()
|
|
.summary("Transaction raw")
|
|
.description("Returns a transaction as binary data.\n\n*[Mempool.space docs](https://mempool.space/docs/api/rest#get-transaction-raw)*")
|
|
.json_response::<Vec<u8>>()
|
|
.not_modified()
|
|
.bad_request()
|
|
.not_found()
|
|
.server_error(),
|
|
),
|
|
)
|
|
.api_route(
|
|
"/api/tx/{txid}/status",
|
|
get_with(
|
|
async |
|
|
uri: Uri,
|
|
headers: HeaderMap,
|
|
Path(txid): Path<TxidParam>,
|
|
State(state): State<AppState>
|
|
| {
|
|
state.cached_json(&headers, CacheStrategy::Height, &uri, move |q| q.transaction_status(txid)).await
|
|
},
|
|
|op| op
|
|
.id("get_tx_status")
|
|
.transactions_tag()
|
|
.summary("Transaction status")
|
|
.description(
|
|
"Retrieve the confirmation status of a transaction. Returns whether the transaction is confirmed and, if so, the block height, hash, and timestamp.\n\n*[Mempool.space docs](https://mempool.space/docs/api/rest#get-transaction-status)*",
|
|
)
|
|
.json_response::<TxStatus>()
|
|
.not_modified()
|
|
.bad_request()
|
|
.not_found()
|
|
.server_error(),
|
|
),
|
|
)
|
|
.api_route(
|
|
"/api/v1/transaction-times",
|
|
get_with(
|
|
async |uri: Uri, headers: HeaderMap, State(state): State<AppState>| {
|
|
let params = TxidsParam::from_query(uri.query().unwrap_or(""));
|
|
state.cached_json(&headers, CacheStrategy::MempoolHash(0), &uri, move |q| q.transaction_times(¶ms.txids)).await
|
|
},
|
|
|op| op
|
|
.id("get_transaction_times")
|
|
.transactions_tag()
|
|
.summary("Transaction first-seen times")
|
|
.description("Returns timestamps when transactions were first seen in the mempool. Returns 0 for mined or unknown transactions.\n\n*[Mempool.space docs](https://mempool.space/docs/api/rest#get-transaction-times)*")
|
|
.json_response::<Vec<u64>>()
|
|
.server_error(),
|
|
),
|
|
)
|
|
.api_route(
|
|
"/api/tx",
|
|
post_with(
|
|
async |State(state): State<AppState>, body: String| {
|
|
let hex = body.trim().to_string();
|
|
state.sync(|q| q.broadcast_transaction(&hex))
|
|
.map(|txid| txid.to_string())
|
|
.map_err(crate::Error::from)
|
|
},
|
|
|op| {
|
|
op.id("post_tx")
|
|
.transactions_tag()
|
|
.summary("Broadcast transaction")
|
|
.description("Broadcast a raw transaction to the network. The transaction should be provided as hex in the request body. The txid will be returned on success.\n\n*[Mempool.space docs](https://mempool.space/docs/api/rest#post-transaction)*")
|
|
.json_response::<Txid>()
|
|
.bad_request()
|
|
.server_error()
|
|
},
|
|
),
|
|
)
|
|
}
|
|
}
|