wallet-core/packages/taler-wallet-webextension/src/cta/Withdraw/state.ts

448 lines
12 KiB
TypeScript
Raw Normal View History

2022-07-21 15:36:15 +02:00
/*
This file is part of GNU Taler
(C) 2022 Taler Systems S.A.
GNU Taler is free software; you can redistribute it and/or modify it under the
terms of the GNU General Public License as published by the Free Software
Foundation; either version 3, or (at your option) any later version.
GNU Taler is distributed in the hope that it will be useful, but WITHOUT ANY
WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
A PARTICULAR PURPOSE. See the GNU General Public License for more details.
You should have received a copy of the GNU General Public License along with
GNU Taler; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
*/
2022-08-29 16:32:07 +02:00
import { Amounts, parsePaytoUri } from "@gnu-taler/taler-util";
2022-07-21 15:36:15 +02:00
import { TalerError } from "@gnu-taler/taler-wallet-core";
2022-08-10 16:50:46 +02:00
import { useState } from "preact/hooks";
2022-07-21 15:36:15 +02:00
import { useAsyncAsHook } from "../../hooks/useAsyncAsHook.js";
import { buildTermsOfServiceState } from "../../utils/index.js";
import * as wxApi from "../../wxApi.js";
2022-08-29 16:32:07 +02:00
import { PropsFromURI, PropsFromParams, State } from "./index.js";
2022-07-21 15:36:15 +02:00
2022-08-29 16:32:07 +02:00
export function useComponentStateFromParams(
2022-09-16 21:03:58 +02:00
{ amount, cancel, onSuccess }: PropsFromParams,
2022-08-29 16:32:07 +02:00
api: typeof wxApi,
): State {
const [ageRestricted, setAgeRestricted] = useState(0);
const exchangeHook = useAsyncAsHook(api.listExchanges);
2022-08-29 16:32:07 +02:00
const exchangeHookDep =
!exchangeHook || exchangeHook.hasError || !exchangeHook.response
? undefined
: exchangeHook.response;
const chosenAmount = Amounts.parseOrThrow(amount);
// get the first exchange with the currency as the default one
2022-09-16 19:29:35 +02:00
const exchange = exchangeHookDep
? exchangeHookDep.exchanges.find(
2022-09-16 21:03:58 +02:00
(e) => e.currency === chosenAmount.currency,
)
2022-09-16 19:29:35 +02:00
: undefined;
2022-08-29 16:32:07 +02:00
/**
* For the exchange selected, bring the status of the terms of service
*/
const terms = useAsyncAsHook(async () => {
2022-09-16 19:29:35 +02:00
if (!exchange) return undefined;
2022-08-29 16:32:07 +02:00
2022-09-16 19:29:35 +02:00
const exchangeTos = await api.getExchangeTos(exchange.exchangeBaseUrl, [
"text/xml",
]);
2022-08-29 16:32:07 +02:00
const state = buildTermsOfServiceState(exchangeTos);
return { state };
}, [exchangeHookDep]);
/**
* With the exchange and amount, ask the wallet the information
* about the withdrawal
*/
const amountHook = useAsyncAsHook(async () => {
2022-09-16 19:29:35 +02:00
if (!exchange) return undefined;
2022-08-29 16:32:07 +02:00
const info = await api.getExchangeWithdrawalInfo({
exchangeBaseUrl: exchange.exchangeBaseUrl,
amount: chosenAmount,
tosAcceptedFormat: ["text/xml"],
ageRestricted,
2022-08-29 16:32:07 +02:00
});
const withdrawAmount = {
raw: Amounts.parseOrThrow(info.withdrawalAmountRaw),
effective: Amounts.parseOrThrow(info.withdrawalAmountEffective),
2022-09-16 19:29:35 +02:00
};
2022-08-29 16:32:07 +02:00
2022-09-16 19:29:35 +02:00
return {
amount: withdrawAmount,
ageRestrictionOptions: info.ageRestrictionOptions,
};
2022-08-29 16:32:07 +02:00
}, [exchangeHookDep]);
const [reviewing, setReviewing] = useState<boolean>(false);
const [reviewed, setReviewed] = useState<boolean>(false);
const [withdrawError, setWithdrawError] = useState<TalerError | undefined>(
undefined,
);
const [doingWithdraw, setDoingWithdraw] = useState<boolean>(false);
2022-09-16 19:29:35 +02:00
if (!exchangeHook) return { status: "loading", error: undefined };
2022-08-29 16:32:07 +02:00
if (exchangeHook.hasError) {
return {
status: "loading-uri",
error: exchangeHook,
};
}
if (!exchange) {
return {
status: "loading-exchange",
error: {
hasError: true,
operational: false,
message: "ERROR_NO-DEFAULT-EXCHANGE",
},
};
}
async function doWithdrawAndCheckError(): Promise<void> {
if (!exchange) return;
try {
setDoingWithdraw(true);
const response = await wxApi.acceptManualWithdrawal(
exchange.exchangeBaseUrl,
Amounts.stringify(amount),
);
2022-09-16 21:03:58 +02:00
onSuccess(response.transactionId);
2022-08-29 16:32:07 +02:00
} catch (e) {
if (e instanceof TalerError) {
setWithdrawError(e);
}
}
setDoingWithdraw(false);
}
if (!amountHook) {
2022-09-16 19:29:35 +02:00
return { status: "loading", error: undefined };
2022-08-29 16:32:07 +02:00
}
if (amountHook.hasError) {
return {
status: "loading-info",
error: amountHook,
};
}
if (!amountHook.response) {
return { status: "loading", error: undefined };
}
const withdrawalFee = Amounts.sub(
amountHook.response.amount.raw,
amountHook.response.amount.effective,
).amount;
const toBeReceived = amountHook.response.amount.effective;
const { state: termsState } = (!terms
? undefined
: terms.hasError
2022-09-16 21:03:58 +02:00
? undefined
: terms.response) || { state: undefined };
2022-08-29 16:32:07 +02:00
async function onAccept(accepted: boolean): Promise<void> {
if (!termsState || !exchange) return;
try {
await api.setExchangeTosAccepted(
exchange.exchangeBaseUrl,
accepted ? termsState.version : undefined,
);
setReviewed(accepted);
} catch (e) {
if (e instanceof Error) {
//FIXME: uncomment this and display error
// setErrorAccepting(e.message);
}
}
}
const mustAcceptFirst =
termsState !== undefined &&
(termsState.status === "changed" || termsState.status === "new");
2022-09-16 19:29:35 +02:00
const ageRestrictionOptions =
amountHook.response.ageRestrictionOptions?.reduce(
(p, c) => ({ ...p, [c]: `under ${c}` }),
{} as Record<string, string>,
);
2022-08-29 16:32:07 +02:00
2022-09-16 19:29:35 +02:00
const ageRestrictionEnabled = ageRestrictionOptions !== undefined;
if (ageRestrictionEnabled) {
2022-08-29 16:32:07 +02:00
ageRestrictionOptions["0"] = "Not restricted";
}
//TODO: calculate based on exchange info
2022-09-16 19:29:35 +02:00
const ageRestriction = ageRestrictionEnabled
? {
2022-09-16 21:03:58 +02:00
list: ageRestrictionOptions,
value: String(ageRestricted),
onChange: async (v: string) => setAgeRestricted(parseInt(v, 10)),
}
2022-09-16 19:29:35 +02:00
: undefined;
2022-08-29 16:32:07 +02:00
return {
status: "success",
error: undefined,
exchangeUrl: exchange.exchangeBaseUrl,
toBeReceived,
withdrawalFee,
chosenAmount,
ageRestriction,
doWithdrawal: {
onClick:
doingWithdraw || (mustAcceptFirst && !reviewed)
? undefined
: doWithdrawAndCheckError,
error: withdrawError,
},
tosProps: !termsState
? undefined
: {
2022-09-16 21:03:58 +02:00
onAccept,
onReview: setReviewing,
reviewed: reviewed,
reviewing: reviewing,
terms: termsState,
},
2022-08-29 16:32:07 +02:00
mustAcceptFirst,
cancel,
};
}
export function useComponentStateFromURI(
2022-09-16 21:03:58 +02:00
{ talerWithdrawUri, cancel, onSuccess }: PropsFromURI,
2022-07-21 15:36:15 +02:00
api: typeof wxApi,
): State {
const [ageRestricted, setAgeRestricted] = useState(0);
/**
* Ask the wallet about the withdraw URI
*/
const uriInfoHook = useAsyncAsHook(async () => {
if (!talerWithdrawUri) throw Error("ERROR_NO-URI-FOR-WITHDRAWAL");
const uriInfo = await api.getWithdrawalDetailsForUri({
talerWithdrawUri,
});
2022-09-16 19:29:35 +02:00
const { amount, defaultExchangeBaseUrl } = uriInfo;
2022-08-10 16:50:46 +02:00
return { amount, thisExchange: defaultExchangeBaseUrl };
2022-07-21 15:36:15 +02:00
});
/**
* Get the amount and select one exchange
*/
const uriHookDep =
!uriInfoHook || uriInfoHook.hasError || !uriInfoHook.response
? undefined
: uriInfoHook.response;
/**
* For the exchange selected, bring the status of the terms of service
*/
const terms = useAsyncAsHook(async () => {
2022-08-10 16:50:46 +02:00
if (!uriHookDep?.thisExchange) return false;
2022-07-21 15:36:15 +02:00
2022-09-16 19:29:35 +02:00
const exchangeTos = await api.getExchangeTos(uriHookDep.thisExchange, [
"text/xml",
]);
2022-07-21 15:36:15 +02:00
const state = buildTermsOfServiceState(exchangeTos);
return { state };
2022-08-10 16:50:46 +02:00
}, [uriHookDep]);
2022-07-21 15:36:15 +02:00
/**
* With the exchange and amount, ask the wallet the information
* about the withdrawal
*/
2022-08-10 16:50:46 +02:00
const amountHook = useAsyncAsHook(async () => {
if (!uriHookDep?.thisExchange) return false;
2022-07-21 15:36:15 +02:00
const info = await api.getExchangeWithdrawalInfo({
2022-08-10 16:50:46 +02:00
exchangeBaseUrl: uriHookDep?.thisExchange,
amount: Amounts.parseOrThrow(uriHookDep.amount),
2022-07-21 15:36:15 +02:00
tosAcceptedFormat: ["text/xml"],
ageRestricted,
2022-07-21 15:36:15 +02:00
});
2022-08-10 16:50:46 +02:00
const withdrawAmount = {
raw: Amounts.parseOrThrow(info.withdrawalAmountRaw),
effective: Amounts.parseOrThrow(info.withdrawalAmountEffective),
2022-09-16 19:29:35 +02:00
};
2022-07-21 15:36:15 +02:00
2022-09-16 19:29:35 +02:00
return {
amount: withdrawAmount,
ageRestrictionOptions: info.ageRestrictionOptions,
};
2022-08-10 16:50:46 +02:00
}, [uriHookDep]);
2022-07-21 15:36:15 +02:00
const [reviewing, setReviewing] = useState<boolean>(false);
const [reviewed, setReviewed] = useState<boolean>(false);
const [withdrawError, setWithdrawError] = useState<TalerError | undefined>(
undefined,
);
const [doingWithdraw, setDoingWithdraw] = useState<boolean>(false);
2022-09-16 19:29:35 +02:00
if (!uriInfoHook) return { status: "loading", error: undefined };
2022-07-31 01:55:41 +02:00
if (uriInfoHook.hasError) {
2022-07-21 15:36:15 +02:00
return {
status: "loading-uri",
2022-07-31 01:55:41 +02:00
error: uriInfoHook,
2022-07-21 15:36:15 +02:00
};
}
2022-09-16 19:29:35 +02:00
const { amount, thisExchange } = uriInfoHook.response;
2022-08-29 16:32:07 +02:00
2022-08-10 16:50:46 +02:00
const chosenAmount = Amounts.parseOrThrow(amount);
if (!thisExchange) {
2022-07-21 15:36:15 +02:00
return {
status: "loading-exchange",
2022-07-31 01:55:41 +02:00
error: {
2022-07-21 15:36:15 +02:00
hasError: true,
operational: false,
message: "ERROR_NO-DEFAULT-EXCHANGE",
},
};
}
2022-08-10 16:50:46 +02:00
// const selectedExchange = thisExchange;
2022-07-21 15:36:15 +02:00
async function doWithdrawAndCheckError(): Promise<void> {
2022-08-10 16:50:46 +02:00
if (!thisExchange) return;
2022-07-21 15:36:15 +02:00
try {
setDoingWithdraw(true);
if (!talerWithdrawUri) return;
const res = await api.acceptWithdrawal(
talerWithdrawUri,
2022-08-10 16:50:46 +02:00
thisExchange,
2022-07-21 15:36:15 +02:00
!ageRestricted ? undefined : ageRestricted,
);
if (res.confirmTransferUrl) {
document.location.href = res.confirmTransferUrl;
2022-09-16 21:03:58 +02:00
} else {
onSuccess(res.transactionId)
2022-07-21 15:36:15 +02:00
}
2022-09-16 21:03:58 +02:00
2022-07-21 15:36:15 +02:00
} catch (e) {
if (e instanceof TalerError) {
setWithdrawError(e);
}
}
setDoingWithdraw(false);
}
2022-08-10 16:50:46 +02:00
if (!amountHook) {
2022-09-16 19:29:35 +02:00
return { status: "loading", error: undefined };
2022-07-31 01:55:41 +02:00
}
2022-08-10 16:50:46 +02:00
if (amountHook.hasError) {
2022-07-21 15:36:15 +02:00
return {
status: "loading-info",
2022-08-10 16:50:46 +02:00
error: amountHook,
2022-07-21 15:36:15 +02:00
};
}
2022-08-10 16:50:46 +02:00
if (!amountHook.response) {
2022-07-31 01:55:41 +02:00
return { status: "loading", error: undefined };
2022-07-21 15:36:15 +02:00
}
2022-08-10 16:50:46 +02:00
const withdrawalFee = Amounts.sub(
amountHook.response.amount.raw,
amountHook.response.amount.effective,
).amount;
const toBeReceived = amountHook.response.amount.effective;
2022-07-21 15:36:15 +02:00
const { state: termsState } = (!terms
? undefined
: terms.hasError
2022-09-16 21:03:58 +02:00
? undefined
: terms.response) || { state: undefined };
2022-07-21 15:36:15 +02:00
async function onAccept(accepted: boolean): Promise<void> {
2022-08-10 16:50:46 +02:00
if (!termsState || !thisExchange) return;
2022-07-21 15:36:15 +02:00
try {
await api.setExchangeTosAccepted(
2022-08-10 16:50:46 +02:00
thisExchange,
2022-07-21 15:36:15 +02:00
accepted ? termsState.version : undefined,
);
setReviewed(accepted);
} catch (e) {
if (e instanceof Error) {
//FIXME: uncomment this and display error
// setErrorAccepting(e.message);
}
}
}
const mustAcceptFirst =
termsState !== undefined &&
(termsState.status === "changed" || termsState.status === "new");
2022-09-16 19:29:35 +02:00
const ageRestrictionOptions =
amountHook.response.ageRestrictionOptions?.reduce(
(p, c) => ({ ...p, [c]: `under ${c}` }),
{} as Record<string, string>,
);
2022-07-21 15:36:15 +02:00
2022-09-16 19:29:35 +02:00
const ageRestrictionEnabled = ageRestrictionOptions !== undefined;
if (ageRestrictionEnabled) {
2022-07-21 15:36:15 +02:00
ageRestrictionOptions["0"] = "Not restricted";
}
2022-08-10 16:50:46 +02:00
//TODO: calculate based on exchange info
2022-09-16 19:29:35 +02:00
const ageRestriction = ageRestrictionEnabled
? {
2022-09-16 21:03:58 +02:00
list: ageRestrictionOptions,
value: String(ageRestricted),
onChange: async (v: string) => setAgeRestricted(parseInt(v, 10)),
}
2022-09-16 19:29:35 +02:00
: undefined;
2022-08-10 16:50:46 +02:00
2022-07-21 15:36:15 +02:00
return {
status: "success",
2022-07-31 01:55:41 +02:00
error: undefined,
2022-08-10 16:50:46 +02:00
exchangeUrl: thisExchange,
2022-07-21 15:36:15 +02:00
toBeReceived,
withdrawalFee,
2022-08-10 16:50:46 +02:00
chosenAmount,
talerWithdrawUri,
2022-08-10 16:50:46 +02:00
ageRestriction,
2022-07-21 15:36:15 +02:00
doWithdrawal: {
onClick:
doingWithdraw || (mustAcceptFirst && !reviewed)
? undefined
: doWithdrawAndCheckError,
error: withdrawError,
},
tosProps: !termsState
? undefined
: {
2022-09-16 21:03:58 +02:00
onAccept,
onReview: setReviewing,
reviewed: reviewed,
reviewing: reviewing,
terms: termsState,
},
2022-07-21 15:36:15 +02:00
mustAcceptFirst,
2022-08-10 16:50:46 +02:00
cancel,
2022-07-21 15:36:15 +02:00
};
}