diff --git a/.github/workflows/rust.yml b/.github/workflows/rust.yml index f95916de..9773f08d 100644 --- a/.github/workflows/rust.yml +++ b/.github/workflows/rust.yml @@ -36,6 +36,8 @@ jobs: run: | cargo test --package payjoin --verbose --features=send,receive,v2 --test integration v2_to_v2 cargo test --package payjoin --verbose --features=send,receive,v2 --test integration v1_to_v2 + - name: integration test payjoin lib + run: cargo test --package payjoin --test integration --features send --features receive -- --nocapture - name: test payjoin-cli bin run: cargo test --package payjoin-cli --verbose --features=danger-local-https diff --git a/payjoin/Cargo.toml b/payjoin/Cargo.toml index fb593748..81e122f7 100644 --- a/payjoin/Cargo.toml +++ b/payjoin/Cargo.toml @@ -30,9 +30,9 @@ bhttp = { version = "0.4.0", optional = true } rand = { version = "0.8.4", optional = true } serde = { version = "1.0.186", default-features = false } url = "2.2.2" +bitcoind = { version = "0.31.1", features = ["0_21_2"] } [dev-dependencies] -bitcoind = { version = "0.31.1", features = ["0_21_2"] } env_logger = "0.9.0" rustls = "0.21.9" testcontainers = "0.15.0" @@ -42,3 +42,8 @@ ureq = "2.8.0" [package.metadata.docs.rs] features = ["send", "receive", "base64"] +env_logger = "0.9.0" + + +[lib] +doctest = false diff --git a/payjoin/src/receive/error.rs b/payjoin/src/receive/error.rs index 5de611bd..5dc61999 100644 --- a/payjoin/src/receive/error.rs +++ b/payjoin/src/receive/error.rs @@ -5,7 +5,7 @@ use std::fmt::{self, Display}; pub enum Error { /// To be returned as HTTP 400 BadRequest(RequestError), - // To be returned as HTTP 500 + /// To be returned as HTTP 500 Server(Box), // V2 d/encapsulation failed #[cfg(feature = "v2")] diff --git a/payjoin/src/send/error.rs b/payjoin/src/send/error.rs index 299c3d3e..d052bf38 100644 --- a/payjoin/src/send/error.rs +++ b/payjoin/src/send/error.rs @@ -3,6 +3,7 @@ use std::str::FromStr; use bitcoin::locktime::absolute::LockTime; use bitcoin::Sequence; +use bitcoind::bitcoincore_rpc::jsonrpc::serde_json::Value; use crate::input_type::{InputType, InputTypeError}; @@ -238,18 +239,19 @@ impl From for CreateRequestError { } pub enum ResponseError { - /// `WellKnown` errors following the BIP78 spec - /// https://github.com/bitcoin/bips/blob/master/bip-0078.mediawiki#user-content-Receivers_well_known_errors + /// `WellKnown` errors following the BIP78 spec. + /// /// These errors are displayed to end users. /// - /// The `WellKnownError` represents `errorCode` and `message`. - /// The `String` is a custom message that can be used for debug logs. + /// `WellKnownError` represents `errorCode` and `message`. + /// `String` is a custom message that can be used for debug logs. WellKnown(WellKnownError, Option), - /// `Unrecognized` errors are errors that are not well known and are only displayed in debug logs. + /// `Unrecognized` errors are errors that are not + /// mentioned in the spec and are only displayed in debug logs. /// They are not displayed to end users. /// - /// The first `String` is `errorCode` - /// The second `String` is `message`. + /// First `String` is `errorCode` + /// Second `String` is `message`. Unrecognized(String, String), /// `Validation` errors are errors that are caused by malformed responses. /// They are only displayed in debug logs. @@ -275,7 +277,7 @@ impl Display for ResponseError { write!(f, "The receiver doesn't support this version of the protocol.") } WellKnownError::OriginalPsbtRejected => { - write!(f, "The receiver rejected the original PSBT.") + write!(f, "{}", WellKnownError::OriginalPsbtRejected.message()) } }, // Don't display unknowns to end users, only debug logs @@ -324,7 +326,9 @@ impl Debug for ResponseError { } } -#[derive(Debug)] +impl std::error::Error for ResponseError {} + +#[derive(Debug, Eq, PartialEq)] pub enum WellKnownError { Unavailable, NotEnoughMoney, @@ -367,4 +371,54 @@ impl WellKnownError { Self::OriginalPsbtRejected => "The receiver rejected the original PSBT.", } } + + pub fn error_code_from_json(json: Value) -> Result { + let error_code = json + .as_object() + .and_then(|v| v.get("errorCode")) + .and_then(|v| v.as_str()) + .ok_or(InternalValidationError::Parse)?; + WellKnownError::from_str(error_code).map_err(|_| { + let message = match json.get("message") { + Some(v) => v.to_string(), + None => { + log::debug!("Unrecognized Error detected, {}", json); + "Unrecognized Error detected".to_string() + } + }; + ResponseError::Unrecognized(error_code.to_string(), message) + }) + } + + pub fn error_code_from_str(error_code: &str) -> Result { + Self::error_code_from_json(Value::from_str(&error_code).map_err(|e| { + log::debug!("Invalid json detected, {}", e); + InternalValidationError::Parse + })?) + } +} + +#[cfg(test)] +mod tests { + use bitcoind::bitcoincore_rpc::jsonrpc::serde_json::json; + + use super::*; + + #[test] + fn test_parse_json() { + let json_error = json!({ + "errorCode": "version-unsupported", + "message": "This version of payjoin is not supported." + }); + assert_eq!( + WellKnownError::error_code_from_json(json_error).unwrap().to_string(), + WellKnownError::VersionUnsupported.to_string() + ); + let str_error = "{\"errorCode\":\"version-unsupported\", + \"message\":\"This version of payjoin is not supported.\"}"; + assert_eq!( + WellKnownError::error_code_from_str(str_error).unwrap().to_string(), + WellKnownError::VersionUnsupported.to_string() + ); + } } diff --git a/payjoin/src/send/mod.rs b/payjoin/src/send/mod.rs index 5b7d6089..c5fd2d1c 100644 --- a/payjoin/src/send/mod.rs +++ b/payjoin/src/send/mod.rs @@ -155,11 +155,10 @@ use crate::PjUri; #[cfg(not(any(target_pointer_width = "32", target_pointer_width = "64")))] compile_error!("This crate currently only supports 32 bit and 64 bit architectures"); -mod error; +pub mod error; type InternalResult = Result; -#[derive(Debug, Clone)] pub struct RequestBuilder<'a> { psbt: Psbt, uri: PjUri<'a>, @@ -167,7 +166,6 @@ pub struct RequestBuilder<'a> { fee_contribution: Option<(bitcoin::Amount, Option)>, clamp_fee_contribution: bool, min_fee_rate: FeeRate, - version: usize, } impl<'a> RequestBuilder<'a> { @@ -179,7 +177,6 @@ impl<'a> RequestBuilder<'a> { pub fn from_psbt_and_uri( psbt: Psbt, uri: crate::Uri<'a, NetworkChecked>, - version: usize, ) -> Result { let uri = uri .check_pj_supported() @@ -191,7 +188,6 @@ impl<'a> RequestBuilder<'a> { uri, // Sender's optional parameters disable_output_substitution, - version, fee_contribution: None, clamp_fee_contribution: false, min_fee_rate: FeeRate::ZERO, @@ -366,7 +362,7 @@ impl<'a> RequestContext<'a> { self.disable_output_substitution, self.fee_contribution, self.min_fee_rate, - self.version, + 1, ) .map_err(InternalCreateRequestError::Url)?; let body = self.psbt.to_string().as_bytes().to_vec(); @@ -543,7 +539,7 @@ impl ContextV1 { let proposal = Psbt::from_str(&res_str).or_else(|_| { // That wasn't a valid PSBT. Maybe it's a valid error response? match WellKnownError::error_code_from_str(&res_str) { - Ok(well_known) => return Err(ResponseError::WellKnown(well_known, "".to_string())), + Ok(well_known) => return Err(ResponseError::WellKnown(well_known, None)), Err(e) => Err(e), } })?; @@ -914,6 +910,7 @@ fn serialize_v2_body( disable_output_substitution, fee_contribution, min_feerate, + 2, ) .map_err(InternalCreateRequestError::Url)?; let query_params = placeholder_url.query().unwrap_or_default(); @@ -946,6 +943,11 @@ fn serialize_url( Ok(url) } +pub fn serialize_psbt(psbt: &Psbt) -> Vec { + let bytes = psbt.serialize(); + bitcoin::base64::encode(bytes).into_bytes() +} + #[cfg(test)] mod tests { #[test]