From fbb074496014516818a0a70989f6e5112325aff4 Mon Sep 17 00:00:00 2001 From: Mariusz Klochowicz Date: Wed, 3 Nov 2021 13:46:04 +1030 Subject: [PATCH 1/2] Provide feedback about maker failed actions in HTTP response Invoke user actions (cfd actions & posting a new sell order) synchronously in order to be able to communicate the results. Use HttpApiProblem to send error details to the frontend in a standard way. --- Cargo.lock | 25 +++++++++++++++ daemon/Cargo.toml | 2 ++ daemon/src/maker_cfd.rs | 14 +++++---- daemon/src/routes_maker.rs | 61 +++++++++++++++++++++---------------- daemon/src/to_sse_event.rs | 2 +- daemon/tests/happy_path.rs | 6 ++-- daemon/tests/harness/mod.rs | 8 +++-- 7 files changed, 80 insertions(+), 38 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index d9ce6d7..a66b25d 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -637,9 +637,11 @@ dependencies = [ "cfd_protocol", "chrono", "clap", + "derive_more", "futures", "hex", "hkdf", + "http-api-problem", "itertools", "mockall", "mockall_derive", @@ -712,6 +714,17 @@ dependencies = [ "syn", ] +[[package]] +name = "derive_more" +version = "0.99.16" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "40eebddd2156ce1bb37b20bbe5151340a31828b1f2d22ba4141f3531710e38df" +dependencies = [ + "proc-macro2", + "quote", + "syn", +] + [[package]] name = "devise" version = "0.3.1" @@ -1211,6 +1224,18 @@ dependencies = [ "itoa", ] +[[package]] +name = "http-api-problem" +version = "0.51.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6d43e8970113f6e4a9138e6cd57b20de3ed99585cab427eb99d27a440827dbe2" +dependencies = [ + "http", + "rocket", + "serde", + "serde_json", +] + [[package]] name = "http-body" version = "0.4.3" diff --git a/daemon/Cargo.toml b/daemon/Cargo.toml index af4922d..ff3be62 100644 --- a/daemon/Cargo.toml +++ b/daemon/Cargo.toml @@ -12,9 +12,11 @@ bytes = "1" cfd_protocol = { path = "../cfd_protocol" } chrono = { version = "0.4", features = ["serde"] } clap = "3.0.0-beta.5" +derive_more = { version = "0.99.16", default-features = false, features = ["display"] } futures = { version = "0.3", default-features = false } hex = "0.4" hkdf = "0.11" +http-api-problem = { version = "0.51.0", features = ["rocket"] } itertools = "0.10" nalgebra = { version = "0.29", default-features = false, features = ["std"] } ndarray = "0.15.3" diff --git a/daemon/src/maker_cfd.rs b/daemon/src/maker_cfd.rs index 767718b..62d0929 100644 --- a/daemon/src/maker_cfd.rs +++ b/daemon/src/maker_cfd.rs @@ -925,7 +925,7 @@ where + xtra::Handler + xtra::Handler, { - async fn handle(&mut self, msg: CfdAction, ctx: &mut Context) { + async fn handle(&mut self, msg: CfdAction, ctx: &mut Context) -> Result<()> { use CfdAction::*; if let Err(e) = match msg { AcceptOrder { order_id } => self.handle_accept_order(order_id, ctx).await, @@ -936,8 +936,9 @@ where RejectRollOver { order_id } => self.handle_reject_roll_over(order_id).await, Commit { order_id } => self.handle_commit(order_id).await, } { - tracing::error!("Message handler failed: {:#}", e); + anyhow::bail!("Message handler failed: {:#}", e); } + Ok(()) } } @@ -946,8 +947,9 @@ impl Handler for Actor where T: xtra::Handler, { - async fn handle(&mut self, msg: NewOrder, _ctx: &mut Context) { - log_error!(self.handle_new_order(msg.price, msg.min_quantity, msg.max_quantity)); + async fn handle(&mut self, msg: NewOrder, _ctx: &mut Context) -> Result<()> { + self.handle_new_order(msg.price, msg.min_quantity, msg.max_quantity) + .await } } @@ -1067,7 +1069,7 @@ where } impl Message for NewOrder { - type Result = (); + type Result = Result<()>; } impl Message for NewTakerOnline { @@ -1083,7 +1085,7 @@ impl Message for CfdRollOverCompleted { } impl Message for CfdAction { - type Result = (); + type Result = Result<()>; } impl Message for FromTaker { diff --git a/daemon/src/routes_maker.rs b/daemon/src/routes_maker.rs index 761dcc4..ed68ae3 100644 --- a/daemon/src/routes_maker.rs +++ b/daemon/src/routes_maker.rs @@ -6,6 +6,7 @@ use daemon::model::{Price, Usd, WalletInfo}; use daemon::routes::EmbeddedFileExt; use daemon::to_sse_event::{CfdAction, CfdsWithAuxData, ToSseEvent}; use daemon::{bitmex_price_feed, maker_cfd}; +use http_api_problem::{HttpApiProblem, StatusCode}; use rocket::http::{ContentType, Header, Status}; use rocket::response::stream::EventStream; use rocket::response::{status, Responder}; @@ -17,7 +18,7 @@ use std::borrow::Cow; use std::path::PathBuf; use tokio::select; use tokio::sync::watch; -use xtra::prelude::MessageChannel; +use xtra::prelude::*; #[rocket::get("/feed")] pub async fn maker_feed( @@ -109,18 +110,24 @@ pub struct CfdNewOrderRequest { } #[rocket::post("/order/sell", data = "")] -pub fn post_sell_order( +pub async fn post_sell_order( order: Json, new_order_channel: &State>>, _auth: Authenticated, -) -> Result, Status> { +) -> Result, HttpApiProblem> { new_order_channel - .do_send(maker_cfd::NewOrder { + .send(maker_cfd::NewOrder { price: order.price, min_quantity: order.min_quantity, max_quantity: order.max_quantity, }) - .map_err(|_| Status::new(500))?; + .await + .unwrap_or_else(|_| anyhow::bail!("actor disconnected")) // TODO: is there a better way? + .map_err(|_| { + HttpApiProblem::new(StatusCode::INTERNAL_SERVER_ERROR) + .title("Action failed") + .detail("failed to post a sell order") + })?; Ok(status::Accepted(None)) } @@ -143,40 +150,42 @@ pub struct PromptAuthentication { } #[rocket::post("/cfd//")] -pub fn post_cfd_action( +pub async fn post_cfd_action( id: OrderId, action: CfdAction, cfd_action_channel: &State>>, _auth: Authenticated, -) -> Result, status::Custom<()>> { +) -> Result, HttpApiProblem> { use maker_cfd::CfdAction::*; let result = match action { - CfdAction::AcceptOrder => cfd_action_channel.do_send(AcceptOrder { order_id: id }), - CfdAction::RejectOrder => cfd_action_channel.do_send(RejectOrder { order_id: id }), - CfdAction::AcceptSettlement => { - cfd_action_channel.do_send(AcceptSettlement { order_id: id }) - } - CfdAction::RejectSettlement => { - cfd_action_channel.do_send(RejectSettlement { order_id: id }) - } - CfdAction::AcceptRollOver => cfd_action_channel.do_send(AcceptRollOver { order_id: id }), - CfdAction::RejectRollOver => cfd_action_channel.do_send(RejectRollOver { order_id: id }), - CfdAction::Commit => cfd_action_channel.do_send(Commit { order_id: id }), + CfdAction::AcceptOrder => cfd_action_channel.send(AcceptOrder { order_id: id }), + CfdAction::RejectOrder => cfd_action_channel.send(RejectOrder { order_id: id }), + CfdAction::AcceptSettlement => cfd_action_channel.send(AcceptSettlement { order_id: id }), + CfdAction::RejectSettlement => cfd_action_channel.send(RejectSettlement { order_id: id }), + CfdAction::AcceptRollOver => cfd_action_channel.send(AcceptRollOver { order_id: id }), + CfdAction::RejectRollOver => cfd_action_channel.send(RejectRollOver { order_id: id }), + CfdAction::Commit => cfd_action_channel.send(Commit { order_id: id }), CfdAction::Settle => { - tracing::error!("Collaborative settlement can only be triggered by taker"); - - return Err(status::Custom(Status::BadRequest, ())); + let msg = "Collaborative settlement can only be triggered by taker"; + tracing::error!(msg); + return Err(HttpApiProblem::new(StatusCode::BAD_REQUEST).detail(msg)); } CfdAction::RollOver => { - tracing::error!("RollOver proposal can only be triggered by taker"); - - return Err(status::Custom(Status::BadRequest, ())); + let msg = "RollOver proposal can only be triggered by taker"; + tracing::error!(msg); + return Err(HttpApiProblem::new(StatusCode::BAD_REQUEST).detail(msg)); } }; result - .map(|()| status::Accepted(None)) - .map_err(|_| status::Custom(Status::InternalServerError, ())) + .await + .unwrap_or_else(|_| anyhow::bail!("actor disconnected")) // TODO: is there a better way? + .map_err(|_| { + HttpApiProblem::new(StatusCode::INTERNAL_SERVER_ERROR) + .title(action.to_string() + " failed") + })?; + + Ok(status::Accepted(None)) } #[rocket::get("/alive")] diff --git a/daemon/src/to_sse_event.rs b/daemon/src/to_sse_event.rs index 049e7e8..0acd0ed 100644 --- a/daemon/src/to_sse_event.rs +++ b/daemon/src/to_sse_event.rs @@ -167,7 +167,7 @@ pub enum TxLabel { Collaborative, } -#[derive(Debug, Clone, Serialize, Deserialize, PartialEq)] +#[derive(Debug, derive_more::Display, Clone, Serialize, Deserialize, PartialEq)] #[serde(rename_all = "camelCase")] pub enum CfdAction { AcceptOrder, diff --git a/daemon/tests/happy_path.rs b/daemon/tests/happy_path.rs index 2829f3d..7295bd4 100644 --- a/daemon/tests/happy_path.rs +++ b/daemon/tests/happy_path.rs @@ -25,7 +25,7 @@ async fn taker_receives_order_from_maker_on_publication() { assert!(is_next_none(&mut taker.order_feed).await); - maker.publish_order(dummy_new_order()); + maker.publish_order(dummy_new_order()).await; let (published, received) = tokio::join!( next_some(&mut maker.order_feed), @@ -43,7 +43,7 @@ async fn taker_takes_order_and_maker_rejects() { // TODO: Why is this needed? For the cfd stream it is not needed is_next_none(&mut taker.order_feed).await; - maker.publish_order(dummy_new_order()); + maker.publish_order(dummy_new_order()).await; let (_, received) = next_order(&mut maker.order_feed, &mut taker.order_feed).await; @@ -94,7 +94,7 @@ async fn taker_takes_order_and_maker_accepts_and_contract_setup() { is_next_none(&mut taker.order_feed).await; - maker.publish_order(dummy_new_order()); + maker.publish_order(dummy_new_order()).await; let (_, received) = next_order(&mut maker.order_feed, &mut taker.order_feed).await; diff --git a/daemon/tests/harness/mod.rs b/daemon/tests/harness/mod.rs index 5225cd3..3f81d09 100644 --- a/daemon/tests/harness/mod.rs +++ b/daemon/tests/harness/mod.rs @@ -109,8 +109,12 @@ impl Maker { } } - pub fn publish_order(&mut self, new_order_params: maker_cfd::NewOrder) { - self.cfd_actor_addr.do_send(new_order_params).unwrap(); + pub async fn publish_order(&mut self, new_order_params: maker_cfd::NewOrder) { + self.cfd_actor_addr + .send(new_order_params) + .await + .unwrap() + .unwrap(); } pub fn reject_take_request(&self, order: Order) { From 922ecc1f56c25671af4bd2f158ca3b7b21c9574a Mon Sep 17 00:00:00 2001 From: Mariusz Klochowicz Date: Wed, 3 Nov 2021 13:49:38 +1030 Subject: [PATCH 2/2] Display error toasts from failed actions in the maker --- frontend/src/MakerApp.tsx | 12 +++-------- frontend/src/MakerClient.tsx | 5 ++++- frontend/src/components/ErrorToast.tsx | 28 ++++++++++++++++++++++++++ frontend/src/components/HttpError.tsx | 18 +++++++++++++++++ 4 files changed, 53 insertions(+), 10 deletions(-) create mode 100644 frontend/src/components/ErrorToast.tsx create mode 100644 frontend/src/components/HttpError.tsx diff --git a/frontend/src/MakerApp.tsx b/frontend/src/MakerApp.tsx index ac9ef8d..768562d 100644 --- a/frontend/src/MakerApp.tsx +++ b/frontend/src/MakerApp.tsx @@ -22,6 +22,7 @@ import { useEventSource } from "react-sse-hooks"; import { CfdTable } from "./components/cfdtables/CfdTable"; import CurrencyInputField from "./components/CurrencyInputField"; import CurrentPrice from "./components/CurrentPrice"; +import createErrorToast from "./components/ErrorToast"; import useLatestEvent from "./components/Hooks"; import OrderTile from "./components/OrderTile"; import { Cfd, intoCfd, intoOrder, Order, PriceInfo, StateGroupKey, WalletInfo } from "./components/Types"; @@ -42,6 +43,7 @@ export default function App() { const priceInfo = useLatestEvent(source, "quote"); const toast = useToast(); + let [minQuantity, setMinQuantity] = useState("10"); let [maxQuantity, setMaxQuantity] = useState("100"); let [orderPrice, setOrderPrice] = useState("0"); @@ -58,15 +60,7 @@ export default function App() { try { await postCfdSellOrderRequest(payload as CfdSellOrderPayload); } catch (e) { - const description = typeof e === "string" ? e : JSON.stringify(e); - - toast({ - title: "Error", - description, - status: "error", - duration: 9000, - isClosable: true, - }); + createErrorToast(toast, e); } }, }); diff --git a/frontend/src/MakerClient.tsx b/frontend/src/MakerClient.tsx index d98b6da..a98d0a4 100644 --- a/frontend/src/MakerClient.tsx +++ b/frontend/src/MakerClient.tsx @@ -1,3 +1,5 @@ +import { HttpError } from "./components/HttpError"; + export interface CfdSellOrderPayload { price: number; min_quantity: number; @@ -16,6 +18,7 @@ export async function postCfdSellOrderRequest(payload: CfdSellOrderPayload) { if (!res.status.toString().startsWith("2")) { console.log("Status: " + res.status + ", " + res.statusText); - throw new Error("failed to publish new order"); + const resp = await res.json(); + throw new HttpError(resp); } } diff --git a/frontend/src/components/ErrorToast.tsx b/frontend/src/components/ErrorToast.tsx new file mode 100644 index 0000000..48fd9f7 --- /dev/null +++ b/frontend/src/components/ErrorToast.tsx @@ -0,0 +1,28 @@ +import { HttpError } from "./HttpError"; + +// A generic way of creating an error toast +// TODO: Don't use any (`toast: typeof useToast` did not work :( ) +export default function createErrorToast(toast: any, e: any) { + if (e instanceof HttpError) { + const description = e.detail ? e.detail : ""; + + toast({ + title: "Error: " + e.title, + description, + status: "error", + duration: 10000, + isClosable: true, + }); + } else { + console.log(e); + const description = typeof e === "string" ? e : JSON.stringify(e); + + toast({ + title: "Error", + description, + status: "error", + duration: 10000, + isClosable: true, + }); + } +} diff --git a/frontend/src/components/HttpError.tsx b/frontend/src/components/HttpError.tsx new file mode 100644 index 0000000..b80ece6 --- /dev/null +++ b/frontend/src/components/HttpError.tsx @@ -0,0 +1,18 @@ +// A wrapper to parse RFC 7807 +// Pass result of `await response.json()` into the constructor. +export class HttpError extends Error { + title: string; + detail?: string; + + // FIXME: Constructor can't be async, so we can't pass `Response` here + constructor(json_resp: any) { + let title = json_resp.title; + super(title); + this.title = title; + if (json_resp.detail) { + this.detail = json_resp.detail; + } + + Object.setPrototypeOf(this, HttpError.prototype); + } +}