use crate::asset::AssetId;
use crate::tx_graph::TxGraphChunk;
use crate::ArkAddress;
use crate::Error;
use crate::ErrorContext;
use bitcoin::hex::DisplayHex;
use bitcoin::secp256k1::PublicKey;
use bitcoin::taproot::Signature;
use bitcoin::Amount;
use bitcoin::OutPoint;
use bitcoin::Psbt;
use bitcoin::ScriptBuf;
use bitcoin::Transaction;
use bitcoin::Txid;
use bitcoin::XOnlyPublicKey;
use musig::musig;
use std::collections::BTreeMap;
use std::collections::HashMap;
use std::str::FromStr;
#[derive(Debug, Clone)]
pub struct NoncePks(HashMap<Txid, musig::PublicNonce>);
impl NoncePks {
pub fn new(nonce_pks: HashMap<Txid, musig::PublicNonce>) -> Self {
Self(nonce_pks)
}
pub fn get(&self, txid: &Txid) -> Option<musig::PublicNonce> {
self.0.get(txid).copied()
}
pub fn encode(&self) -> HashMap<String, String> {
self.0
.iter()
.map(|(k, v)| (k.to_string(), v.serialize().to_lower_hex_string()))
.collect()
}
pub fn decode(map: HashMap<String, String>) -> Result<Self, Error> {
let map = map
.into_iter()
.map(|(k, v)| {
let key = k
.parse()
.map_err(Error::ad_hoc)
.context("failed to parse TXID")?;
let value = {
let nonce_bytes = bitcoin::hex::FromHex::from_hex(&v)
.map_err(Error::ad_hoc)
.context("failed to decode public nonce from hex")?;
musig::PublicNonce::from_byte_array(&nonce_bytes)
.map_err(Error::ad_hoc)
.context("failed to decode public nonce from bytes")?
};
Ok((key, value))
})
.collect::<Result<HashMap<Txid, musig::PublicNonce>, Error>>()?;
Ok(Self(map))
}
}
#[derive(Debug, Clone)]
pub struct TreeTxNoncePks(pub HashMap<XOnlyPublicKey, musig::PublicNonce>);
impl TreeTxNoncePks {
pub fn new(tree_nonce_pks: HashMap<XOnlyPublicKey, musig::PublicNonce>) -> Self {
Self(tree_nonce_pks)
}
pub fn to_pks(&self) -> Vec<musig::PublicNonce> {
self.0.values().copied().collect()
}
pub fn encode(&self) -> HashMap<String, String> {
self.0
.iter()
.map(|(k, v)| (k.to_string(), v.serialize().to_lower_hex_string()))
.collect()
}
pub fn decode(map: HashMap<String, String>) -> Result<Self, Error> {
let map = map
.into_iter()
.map(|(k, v)| {
let key = k
.parse()
.map_err(Error::ad_hoc)
.context("failed to parse PK")?;
let value = {
let nonce_bytes = bitcoin::hex::FromHex::from_hex(&v)
.map_err(Error::ad_hoc)
.context("failed to decode public nonce from hex")?;
musig::PublicNonce::from_byte_array(&nonce_bytes)
.map_err(Error::ad_hoc)
.context("failed to decode public nonce from bytes")?
};
Ok((key, value))
})
.collect::<Result<HashMap<XOnlyPublicKey, musig::PublicNonce>, Error>>()?;
Ok(Self(map))
}
}
#[derive(Debug, Clone, Default)]
pub struct PartialSigTree(pub HashMap<Txid, musig::PartialSignature>);
impl PartialSigTree {
pub fn encode(&self) -> HashMap<String, String> {
self.0
.iter()
.map(|(k, v)| (k.to_string(), v.serialize().to_lower_hex_string()))
.collect()
}
pub fn decode(map: HashMap<String, String>) -> Result<Self, Error> {
let map = map
.into_iter()
.map(|(k, v)| {
let key = k
.parse()
.map_err(Error::ad_hoc)
.context("failed to parse TXID")?;
let value = {
let sig_bytes = bitcoin::hex::FromHex::from_hex(&v)
.map_err(Error::ad_hoc)
.context("failed to decode partial signature from hex")?;
musig::PartialSignature::from_byte_array(&sig_bytes)
.map_err(Error::ad_hoc)
.context("failed to decode partial signature from bytes")?
};
Ok((key, value))
})
.collect::<Result<HashMap<Txid, musig::PartialSignature>, Error>>()?;
Ok(Self(map))
}
}
#[derive(Debug, Clone, Default)]
pub struct TxTree {
pub nodes: BTreeMap<(usize, usize), TxTreeNode>,
}
impl TxTree {
pub fn new() -> Self {
Self {
nodes: BTreeMap::new(),
}
}
pub fn get_mut(&mut self, level: usize, index: usize) -> Result<&mut TxTreeNode, Error> {
self.nodes
.get_mut(&(level, index))
.ok_or_else(|| Error::ad_hoc("TxTreeNode not found at ({level}, {index})"))
}
pub fn insert(&mut self, node: TxTreeNode, level: usize, index: usize) {
self.nodes.insert((level, index), node);
}
pub fn txs(&self) -> impl Iterator<Item = &Transaction> {
self.nodes.values().map(|node| &node.tx.unsigned_tx)
}
pub fn get_level(&self, level: usize) -> Vec<&TxTreeNode> {
self.nodes
.range((level, 0)..(level + 1, 0))
.map(|(_, node)| node)
.collect()
}
pub fn iter_levels(&self) -> impl Iterator<Item = (usize, Vec<&TxTreeNode>)> {
let max_level = self
.nodes
.keys()
.map(|(level, _)| *level)
.max()
.unwrap_or(0);
(0..=max_level).map(move |level| {
let nodes = self.get_level(level);
(level, nodes)
})
}
}
#[derive(Debug, Clone)]
pub struct TxTreeNode {
pub txid: Txid,
pub tx: Psbt,
pub parent_txid: Txid,
pub level: i32,
pub level_index: i32,
pub leaf: bool,
}
#[derive(Clone)]
pub struct GetVtxosRequest {
reference: GetVtxosRequestReference,
filter: Option<GetVtxosRequestFilter>,
page: Option<PageRequest>,
before: Option<u64>,
after: Option<u64>,
}
#[derive(Debug, Clone, Copy)]
pub struct PageRequest {
pub size: i32,
pub index: i32,
}
impl GetVtxosRequest {
pub fn new_for_addresses(addresses: impl Iterator<Item = ArkAddress>) -> Self {
let scripts = addresses
.flat_map(|a| [a.to_p2tr_script_pubkey()])
.collect();
Self {
reference: GetVtxosRequestReference::Scripts(scripts),
filter: None,
page: None,
before: None,
after: None,
}
}
pub fn new_for_outpoints(outpoints: &[OutPoint]) -> Self {
Self {
reference: GetVtxosRequestReference::OutPoints(outpoints.to_vec()),
filter: None,
page: None,
before: None,
after: None,
}
}
pub fn spendable_only(self) -> Result<Self, Error> {
if self.filter.is_some() {
return Err(Error::ad_hoc("GetVtxosRequest filter already set"));
}
Ok(Self {
filter: Some(GetVtxosRequestFilter::Spendable),
..self
})
}
pub fn spent_only(self) -> Result<Self, Error> {
if self.filter.is_some() {
return Err(Error::ad_hoc("GetVtxosRequest filter already set"));
}
Ok(Self {
filter: Some(GetVtxosRequestFilter::Spent),
..self
})
}
pub fn recoverable_only(self) -> Result<Self, Error> {
if self.filter.is_some() {
return Err(Error::ad_hoc("GetVtxosRequest filter already set"));
}
Ok(Self {
filter: Some(GetVtxosRequestFilter::Recoverable),
..self
})
}
pub fn pending_only(self) -> Result<Self, Error> {
if self.filter.is_some() {
return Err(Error::ad_hoc("GetVtxosRequest filter already set"));
}
Ok(Self {
filter: Some(GetVtxosRequestFilter::PendingOnly),
..self
})
}
pub fn reference(&self) -> &GetVtxosRequestReference {
&self.reference
}
pub fn filter(&self) -> Option<&GetVtxosRequestFilter> {
self.filter.as_ref()
}
pub fn with_page(self, size: i32, index: i32) -> Self {
Self {
page: Some(PageRequest { size, index }),
..self
}
}
pub fn page(&self) -> Option<PageRequest> {
self.page
}
pub fn with_before(self, before: u64) -> Self {
Self {
before: Some(before),
..self
}
}
pub fn with_after(self, after: u64) -> Self {
Self {
after: Some(after),
..self
}
}
pub fn before(&self) -> Option<u64> {
self.before
}
pub fn after(&self) -> Option<u64> {
self.after
}
}
#[derive(Clone)]
pub enum GetVtxosRequestReference {
Scripts(Vec<ScriptBuf>),
OutPoints(Vec<OutPoint>),
}
impl GetVtxosRequestReference {
pub fn is_empty(&self) -> bool {
match self {
GetVtxosRequestReference::Scripts(script_bufs) => script_bufs.is_empty(),
GetVtxosRequestReference::OutPoints(outpoints) => outpoints.is_empty(),
}
}
}
#[derive(Clone, Copy)]
pub enum GetVtxosRequestFilter {
Spendable,
Spent,
Recoverable,
PendingOnly,
}
#[derive(Clone, Debug, PartialEq)]
pub struct VirtualTxOutPoint {
pub outpoint: OutPoint,
pub created_at: i64,
pub expires_at: i64,
pub amount: Amount,
pub script: ScriptBuf,
pub is_preconfirmed: bool,
pub is_swept: bool,
pub is_unrolled: bool,
pub is_spent: bool,
pub spent_by: Option<Txid>,
pub commitment_txids: Vec<Txid>,
pub settled_by: Option<Txid>,
pub ark_txid: Option<Txid>,
pub assets: Vec<Asset>,
}
impl VirtualTxOutPoint {
pub fn is_recoverable(&self, dust: Amount) -> bool {
if self.is_spent {
return false;
}
self.amount < dust || self.is_swept || self.is_expired()
}
pub fn is_expired(&self) -> bool {
#[cfg(not(all(target_arch = "wasm32", target_os = "unknown")))]
let current_timestamp = std::time::SystemTime::now()
.duration_since(std::time::UNIX_EPOCH)
.expect("valid duration")
.as_secs() as i64;
#[cfg(all(target_arch = "wasm32", target_os = "unknown"))]
let current_timestamp = (js_sys::Date::now() / 1000.0) as i64;
current_timestamp > self.expires_at && !self.is_swept && !self.is_spent
}
}
#[derive(Clone, Debug)]
pub struct Info {
pub version: String,
pub signer_pk: PublicKey,
pub forfeit_pk: PublicKey,
pub forfeit_address: bitcoin::Address,
pub checkpoint_tapscript: ScriptBuf,
pub network: bitcoin::Network,
pub session_duration: u64,
pub unilateral_exit_delay: bitcoin::Sequence,
pub boarding_exit_delay: bitcoin::Sequence,
pub utxo_min_amount: Option<Amount>,
pub utxo_max_amount: Option<Amount>,
pub vtxo_min_amount: Option<Amount>,
pub vtxo_max_amount: Option<Amount>,
pub dust: Amount,
pub fees: Option<FeeInfo>,
pub scheduled_session: Option<ScheduledSession>,
pub deprecated_signers: Vec<DeprecatedSigner>,
pub service_status: HashMap<String, String>,
pub digest: String,
pub max_tx_weight: i64,
pub max_op_return_outputs: i64,
}
#[derive(Clone, Debug)]
pub struct FeeInfo {
pub intent_fee: IntentFeeInfo,
pub tx_fee_rate: String,
}
#[derive(Clone, Debug, Default)]
pub struct IntentFeeInfo {
pub offchain_input: Option<String>,
pub offchain_output: Option<String>,
pub onchain_input: Option<String>,
pub onchain_output: Option<String>,
}
#[derive(Clone, Debug)]
pub struct ScheduledSession {
pub next_start_time: i64,
pub next_end_time: i64,
pub period: i64,
pub duration: i64,
pub fees: Option<FeeInfo>,
}
#[derive(Clone, Debug)]
pub struct DeprecatedSigner {
pub pk: PublicKey,
pub cutoff_date: i64,
}
#[derive(Debug, Clone)]
pub struct StreamStartedEvent {
pub id: String,
}
#[derive(Debug, Clone)]
pub struct BatchStartedEvent {
pub id: String,
pub intent_id_hashes: Vec<String>,
pub batch_expiry: bitcoin::Sequence,
}
#[derive(Debug, Clone)]
pub struct BatchFinalizationEvent {
pub id: String,
pub commitment_tx: Psbt,
}
#[derive(Debug, Clone)]
pub struct BatchFinalizedEvent {
pub id: String,
pub commitment_txid: Txid,
}
#[derive(Debug, Clone)]
pub struct BatchFailed {
pub id: String,
pub reason: String,
}
#[derive(Debug, Clone)]
pub struct TreeSigningStartedEvent {
pub id: String,
pub cosigners_pubkeys: Vec<PublicKey>,
pub unsigned_commitment_tx: Psbt,
}
#[derive(Debug, Clone)]
pub struct TreeNoncesAggregatedEvent {
pub id: String,
pub tree_nonces: NoncePks,
}
#[derive(Debug, Clone)]
pub struct TreeTxEvent {
pub id: String,
pub topic: Vec<String>,
pub batch_tree_event_type: BatchTreeEventType,
pub tx_graph_chunk: TxGraphChunk,
}
#[derive(Debug, Clone)]
pub struct TreeSignatureEvent {
pub id: String,
pub topic: Vec<String>,
pub batch_tree_event_type: BatchTreeEventType,
pub txid: Txid,
pub signature: Signature,
}
#[derive(Debug, Clone)]
pub struct TreeNoncesEvent {
pub id: String,
pub topic: Vec<String>,
pub txid: Txid,
pub nonces: TreeTxNoncePks,
}
#[derive(Debug, Clone)]
pub enum BatchTreeEventType {
Vtxo,
Connector,
}
#[derive(Debug, Clone)]
pub enum StreamEvent {
StreamStarted(StreamStartedEvent),
BatchStarted(BatchStartedEvent),
BatchFinalization(BatchFinalizationEvent),
BatchFinalized(BatchFinalizedEvent),
BatchFailed(BatchFailed),
TreeSigningStarted(TreeSigningStartedEvent),
TreeNoncesAggregated(TreeNoncesAggregatedEvent),
TreeTx(TreeTxEvent),
TreeSignature(TreeSignatureEvent),
TreeNonces(TreeNoncesEvent),
Heartbeat,
}
impl StreamEvent {
pub fn name(&self) -> String {
let s = match self {
StreamEvent::StreamStarted(_) => "StreamStarted",
StreamEvent::BatchStarted(_) => "BatchStarted",
StreamEvent::BatchFinalization(_) => "BatchFinalization",
StreamEvent::BatchFinalized(_) => "BatchFinalized",
StreamEvent::BatchFailed(_) => "BatchFailed",
StreamEvent::TreeSigningStarted(_) => "TreeSigningStarted",
StreamEvent::TreeNoncesAggregated(_) => "TreeNoncesAggregated",
StreamEvent::TreeTx(_) => "TreeTx",
StreamEvent::TreeSignature(_) => "TreeSignature",
StreamEvent::TreeNonces(_) => "TreeNoncesEvent",
StreamEvent::Heartbeat => "Heartbeat",
};
s.to_string()
}
}
pub enum StreamTransactionData {
Commitment(CommitmentTransaction),
Ark(ArkTransaction),
Heartbeat,
}
pub struct ArkTransaction {
pub txid: Txid,
pub tx: Option<Psbt>,
pub spent_vtxos: Vec<VirtualTxOutPoint>,
pub unspent_vtxos: Vec<VirtualTxOutPoint>,
pub checkpoint_txs: HashMap<OutPoint, Txid>,
pub swept_vtxos: Vec<OutPoint>,
}
pub struct CommitmentTransaction {
pub txid: Txid,
pub spent_vtxos: Vec<VirtualTxOutPoint>,
pub unspent_vtxos: Vec<VirtualTxOutPoint>,
}
#[derive(Clone, Debug)]
pub enum SubscriptionResponse {
Event(Box<SubscriptionEvent>),
Heartbeat,
}
#[derive(Clone, Debug)]
pub struct SubscriptionEvent {
pub txid: Txid,
pub scripts: Vec<ScriptBuf>,
pub new_vtxos: Vec<VirtualTxOutPoint>,
pub spent_vtxos: Vec<VirtualTxOutPoint>,
pub tx: Option<Transaction>,
pub checkpoint_txs: HashMap<OutPoint, Txid>,
}
pub struct VtxoChains {
pub inner: Vec<VtxoChain>,
}
pub struct VtxoChain {
pub txid: Txid,
pub tx_type: ChainedTxType,
pub spends: Vec<Txid>,
pub expires_at: i64,
}
#[derive(Debug)]
pub enum ChainedTxType {
Commitment,
Tree,
Checkpoint,
Ark,
Unspecified,
}
pub struct SubmitOffchainTxResponse {
pub signed_ark_tx: Psbt,
pub signed_checkpoint_txs: Vec<Psbt>,
}
#[derive(Debug, Clone)]
pub struct PendingTx {
pub ark_txid: Txid,
pub signed_ark_tx: Psbt,
pub signed_checkpoint_txs: Vec<Psbt>,
}
#[derive(Debug, Clone)]
pub struct FinalizeOffchainTxResponse {}
#[derive(Debug)]
pub struct VirtualTxsResponse {
pub txs: Vec<Psbt>,
pub page: Option<IndexerPage>,
}
#[derive(Debug)]
pub struct IndexerPage {
pub current: i32,
pub next: i32,
pub total: i32,
}
#[derive(Clone, Debug)]
pub enum Network {
Bitcoin,
Testnet,
Testnet4,
Signet,
Regtest,
Mutinynet,
}
#[derive(Clone, Debug, PartialEq, Eq, Hash)]
pub struct Asset {
pub asset_id: AssetId,
pub amount: u64,
}
#[derive(Clone, Debug, PartialEq, Eq)]
pub struct AssetInfo {
pub asset_id: AssetId,
pub control_asset_id: Option<AssetId>,
pub supply: u64,
pub metadata: String,
}
impl AssetInfo {
pub fn can_be_reissued(&self) -> bool {
self.control_asset_id.is_some()
}
}
impl From<Network> for bitcoin::Network {
fn from(value: Network) -> Self {
match value {
Network::Bitcoin => bitcoin::Network::Bitcoin,
Network::Testnet => bitcoin::Network::Testnet,
Network::Testnet4 => bitcoin::Network::Testnet4,
Network::Signet => bitcoin::Network::Signet,
Network::Regtest => bitcoin::Network::Regtest,
Network::Mutinynet => bitcoin::Network::Signet,
}
}
}
impl FromStr for Network {
type Err = String;
#[inline]
fn from_str(s: &str) -> Result<Self, Self::Err> {
match s {
"bitcoin" => Ok(Network::Bitcoin),
"testnet" => Ok(Network::Testnet),
"testnet4" => Ok(Network::Testnet4),
"signet" => Ok(Network::Signet),
"regtest" => Ok(Network::Regtest),
"mutinynet" => Ok(Network::Mutinynet),
_ => Err(format!("Unsupported network {}", s.to_owned())),
}
}
}
pub fn parse_sequence_number(value: i64) -> Result<bitcoin::Sequence, Error> {
const ARBITRARY_SEQUENCE_THRESHOLD: i64 = 512;
let sequence = if value.is_negative() {
return Err(Error::ad_hoc(format!("invalid sequence number: {value}")));
} else if value < ARBITRARY_SEQUENCE_THRESHOLD {
bitcoin::Sequence::from_height(value as u16)
} else {
let secs = u32::try_from(value)
.map_err(|_| Error::ad_hoc(format!("sequence seconds overflow: {value}")))?;
bitcoin::Sequence::from_seconds_ceil(secs).map_err(Error::ad_hoc)?
};
Ok(sequence)
}
pub fn parse_fee_amount(amount_str: Option<String>) -> Amount {
amount_str
.and_then(|s| {
if s.is_empty() {
None
} else {
s.parse::<u64>().ok()
}
})
.map(Amount::from_sat)
.unwrap_or(Amount::ZERO)
}