Browse Source

Refactor multiple uses of `useAsync` into a single `usePostRequest` hook

debug-collab-settlement
Thomas Eizinger 3 years ago
parent
commit
622a8ea696
No known key found for this signature in database GPG Key ID: 651AC83A6C6C8B96
  1. 58
      taker-frontend/src/App.tsx
  2. 30
      taker-frontend/src/components/History.tsx
  3. 35
      taker-frontend/src/components/Wallet.tsx
  4. 62
      taker-frontend/src/usePostRequest.ts

58
taker-frontend/src/App.tsx

@ -12,15 +12,12 @@ import {
} from "@chakra-ui/react"; } from "@chakra-ui/react";
import * as React from "react"; import * as React from "react";
import { useEffect, useState } from "react"; import { useEffect, useState } from "react";
import { useAsync } from "react-async";
import { Route, Routes } from "react-router-dom"; import { Route, Routes } from "react-router-dom";
import { useEventSource } from "react-sse-hooks"; import { useEventSource } from "react-sse-hooks";
import useWebSocket from "react-use-websocket"; import useWebSocket from "react-use-websocket";
import { useBackendMonitor } from "./components/BackendMonitor"; import { useBackendMonitor } from "./components/BackendMonitor";
import createErrorToast from "./components/ErrorToast";
import Footer from "./components/Footer"; import Footer from "./components/Footer";
import History from "./components/History"; import History from "./components/History";
import { HttpError } from "./components/HttpError";
import Nav from "./components/NavBar"; import Nav from "./components/NavBar";
import Trade from "./components/Trade"; import Trade from "./components/Trade";
import { import {
@ -34,38 +31,10 @@ import {
Order, Order,
StateGroupKey, StateGroupKey,
WalletInfo, WalletInfo,
WithdrawRequest,
} from "./components/Types"; } from "./components/Types";
import { Wallet, WalletInfoBar } from "./components/Wallet"; import { Wallet, WalletInfoBar } from "./components/Wallet";
import useLatestEvent from "./useLatestEvent"; import useLatestEvent from "./useLatestEvent";
import usePostRequest from "./usePostRequest";
async function getMargin(payload: MarginRequestPayload): Promise<MarginResponse> {
let res = await fetch(`/api/calculate/margin`, { method: "POST", body: JSON.stringify(payload) });
if (!res.status.toString().startsWith("2")) {
const resp = await res.json();
throw new HttpError(resp);
}
return res.json();
}
async function postCfdOrderRequest(payload: CfdOrderRequestPayload) {
let res = await fetch(`/api/cfd/order`, { method: "POST", body: JSON.stringify(payload) });
if (!res.status.toString().startsWith("2")) {
const resp = await res.json();
throw new HttpError(resp);
}
}
export async function postWithdraw(payload: WithdrawRequest) {
let res = await fetch(`/api/withdraw`, { method: "POST", body: JSON.stringify(payload) });
if (!res.status.toString().startsWith("2")) {
const resp = await res.json();
throw new HttpError(resp);
}
return res.text();
}
export const App = () => { export const App = () => {
const toast = useToast(); const toast = useToast();
@ -104,26 +73,13 @@ export const App = () => {
let effectiveQuantity = userHasEdited ? quantity : (min_quantity?.toString() || "0"); let effectiveQuantity = userHasEdited ? quantity : (min_quantity?.toString() || "0");
let { run: calculateMargin } = useAsync({ let [calculateMargin] = usePostRequest<MarginRequestPayload, MarginResponse>(
deferFn: async ([payload]: any[]) => { "/api/calculate/margin",
try { (response) => {
let res = await getMargin(payload as MarginRequestPayload); setMargin(response.margin.toString());
setMargin(res.margin.toString());
} catch (e) {
createErrorToast(toast, e);
}
}, },
}); );
let [makeNewOrderRequest, isCreatingNewOrderRequest] = usePostRequest<CfdOrderRequestPayload>("/api/cfd/order");
let { run: makeNewOrderRequest, isLoading: isCreatingNewOrderRequest } = useAsync({
deferFn: async ([payload]: any[]) => {
try {
await postCfdOrderRequest(payload as CfdOrderRequestPayload);
} catch (e) {
createErrorToast(toast, e);
}
},
});
useEffect(() => { useEffect(() => {
if (!order) { if (!order) {

30
taker-frontend/src/components/History.tsx

@ -25,12 +25,10 @@ import {
Text, Text,
Tr, Tr,
useColorModeValue, useColorModeValue,
useToast,
VStack, VStack,
} from "@chakra-ui/react"; } from "@chakra-ui/react";
import * as React from "react"; import * as React from "react";
import { useAsync } from "react-async"; import usePostRequest from "../usePostRequest";
import createErrorToast from "./ErrorToast";
import { Cfd, StateGroupKey, StateKey, Tx, TxLabel } from "./Types"; import { Cfd, StateGroupKey, StateKey, Tx, TxLabel } from "./Types";
interface HistoryProps { interface HistoryProps {
@ -63,15 +61,7 @@ interface CfdDetailsProps {
cfd: Cfd; cfd: Cfd;
} }
async function doPostAction(id: string, action: string) {
await fetch(
`/api/cfd/${id}/${action}`,
{ method: "POST", credentials: "include" },
);
}
const CfdDetails = ({ cfd }: CfdDetailsProps) => { const CfdDetails = ({ cfd }: CfdDetailsProps) => {
const toast = useToast();
const initialPrice = `$${cfd.initial_price.toLocaleString()}`; const initialPrice = `$${cfd.initial_price.toLocaleString()}`;
const quantity = `$${cfd.quantity_usd}`; const quantity = `$${cfd.quantity_usd}`;
const margin = `${Math.round((cfd.margin) * 1_000_000) / 1_000_000}`; const margin = `${Math.round((cfd.margin) * 1_000_000) / 1_000_000}`;
@ -86,16 +76,7 @@ const CfdDetails = ({ cfd }: CfdDetailsProps) => {
const txCet = cfd.details.tx_url_list.find((tx) => tx.label === TxLabel.Cet); const txCet = cfd.details.tx_url_list.find((tx) => tx.label === TxLabel.Cet);
const txSettled = cfd.details.tx_url_list.find((tx) => tx.label === TxLabel.Collaborative); const txSettled = cfd.details.tx_url_list.find((tx) => tx.label === TxLabel.Collaborative);
let { run: postAction, isLoading: isActioning } = useAsync({ let [settle, isSettling] = usePostRequest(`/api/cfd/${cfd.order_id}/settle`);
deferFn: async ([orderId, action]: any[]) => {
try {
console.log(`Closing: ${orderId} ${action}`);
await doPostAction(orderId, action);
} catch (e) {
createErrorToast(toast, e);
}
},
});
const disableCloseButton = cfd.state.getGroup() === StateGroupKey.CLOSED const disableCloseButton = cfd.state.getGroup() === StateGroupKey.CLOSED
|| !(cfd.state.key === StateKey.OPEN); || !(cfd.state.key === StateKey.OPEN);
@ -206,11 +187,12 @@ const CfdDetails = ({ cfd }: CfdDetailsProps) => {
<Button <Button
size="sm" size="sm"
colorScheme="red" colorScheme="red"
onClick={async () => { onClick={() => {
await postAction(cfd.order_id, "settle"); console.log(`Settling CFD ${cfd.order_id}`);
settle({});
onClose(); onClose();
}} }}
isLoading={isActioning} isLoading={isSettling}
> >
Close Close
</Button> </Button>

35
taker-frontend/src/components/Wallet.tsx

@ -24,13 +24,11 @@ import {
VStack, VStack,
} from "@chakra-ui/react"; } from "@chakra-ui/react";
import * as React from "react"; import * as React from "react";
import { FormEvent, useState } from "react"; import { useState } from "react";
import { useAsync } from "react-async";
import { useNavigate } from "react-router-dom"; import { useNavigate } from "react-router-dom";
import { postWithdraw } from "../App"; import usePostRequest from "../usePostRequest";
import createErrorToast from "./ErrorToast";
import Timestamp from "./Timestamp"; import Timestamp from "./Timestamp";
import { WalletInfo } from "./Types"; import { WalletInfo, WithdrawRequest } from "./Types";
interface WalletProps { interface WalletProps {
walletInfo: WalletInfo | null; walletInfo: WalletInfo | null;
@ -48,16 +46,7 @@ export default function Wallet(
const [withdrawAmount, setWithdrawAmount] = useState(0); const [withdrawAmount, setWithdrawAmount] = useState(0);
const [fee, setFee] = useState(1); const [fee, setFee] = useState(1);
const [withdrawAddress, setWithdrawAddress] = useState(""); const [withdrawAddress, setWithdrawAddress] = useState("");
const [runWithdraw, isWithdrawing] = usePostRequest<WithdrawRequest, string>("/api/withdraw", (url) => {
let { run: runWithdraw, isLoading: isWithdrawing } = useAsync({
deferFn: async ([event]: FormEvent<HTMLFormElement>[]) => {
event.preventDefault();
try {
const url = await postWithdraw({
amount: withdrawAmount,
fee,
address: withdrawAddress,
});
window.open(url, "_blank"); window.open(url, "_blank");
toast({ toast({
title: "Withdraw successful", title: "Withdraw successful",
@ -68,10 +57,6 @@ export default function Wallet(
duration: 10000, duration: 10000,
isClosable: true, isClosable: true,
}); });
} catch (e) {
createErrorToast(toast, e);
}
},
}); });
return ( return (
@ -108,7 +93,17 @@ export default function Wallet(
<Divider marginTop={2} marginBottom={2} /> <Divider marginTop={2} marginBottom={2} />
<VStack padding={2}> <VStack padding={2}>
<form onSubmit={runWithdraw}> <form
onSubmit={(event) => {
event.preventDefault();
runWithdraw({
amount: withdrawAmount,
fee,
address: withdrawAddress,
});
}}
>
<Heading as="h3" size="sm">Withdraw</Heading> <Heading as="h3" size="sm">Withdraw</Heading>
<FormControl id="address"> <FormControl id="address">
<FormLabel>Address</FormLabel> <FormLabel>Address</FormLabel>

62
taker-frontend/src/usePostRequest.ts

@ -0,0 +1,62 @@
import { useToast } from "@chakra-ui/react";
import { useAsync } from "react-async";
/**
* A React hook for sending a POST request to a certain endpoint.
*
* You can pass a callback (`onSuccess`) to process the response. By default, we extract the HTTP body as JSON
*/
export default function usePostRequest<Req = any, Res = any>(
url: string,
onSuccess: (response: Res) => void = () => {},
): [(req: Req) => void, boolean] {
const toast = useToast();
let { run, isLoading } = useAsync({
deferFn: async ([payload]: any[]) => {
let res = await fetch(url, {
method: "POST",
body: JSON.stringify(payload),
headers: {
"Content-type": "application/json",
},
});
if (!res.status.toString().startsWith("2")) {
let problem = await res.json() as Problem;
toast({
title: "Error: " + problem.title,
description: problem.detail,
status: "error",
duration: 10000,
isClosable: true,
});
return;
}
let responseType = res.headers.get("Content-type");
if (responseType && responseType.startsWith("application/json")) {
onSuccess(await res.json() as Res);
return;
}
if (responseType && responseType.startsWith("text/plain")) {
onSuccess(await res.text() as unknown as Res); // `unknown` cast is not ideal because we known that `.text()` gives us string.
return;
}
// if none of the above content types match, pass bytes to the caller
onSuccess(await res.blob() as unknown as Res); // `unknown` cast is not ideal because we known that `.blob()` gives us as blob.
},
});
return [run as (req: Req) => void, isLoading];
}
interface Problem {
title: string;
detail: string;
}
Loading…
Cancel
Save