/* 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 */ /* eslint-disable react-hooks/rules-of-hooks */ import { Amounts, TalerErrorDetail, TalerProtocolTimestamp, } from "@gnu-taler/taler-util"; import { TalerError, WalletApiOperation } from "@gnu-taler/taler-wallet-core"; import { isFuture, parse } from "date-fns"; import { useState } from "preact/hooks"; import { alertFromError } from "../../context/alert.js"; import { useBackendContext } from "../../context/backend.js"; import { useTranslationContext } from "../../context/translation.js"; import { useAsyncAsHook } from "../../hooks/useAsyncAsHook.js"; import { useSelectedExchange } from "../../hooks/useSelectedExchange.js"; import { RecursiveState } from "../../utils/index.js"; import { Props, State } from "./index.js"; export function useComponentState({ amount: amountStr, onClose, onSuccess, }: Props): RecursiveState { const amount = Amounts.parseOrThrow(amountStr); const api = useBackendContext(); const hook = useAsyncAsHook(() => api.wallet.call(WalletApiOperation.ListExchanges, {}), ); const { i18n } = useTranslationContext(); if (!hook) { return { status: "loading", error: undefined, }; } if (hook.hasError) { return { status: "error", error: alertFromError( i18n.str`Could not load the status of the term of service`, hook, ), }; } // if (hook.hasError) { // return { // status: "loading-uri", // error: hook, // }; // } const exchangeList = hook.response.exchanges; return () => { const [subject, setSubject] = useState(); const [timestamp, setTimestamp] = useState(); const selectedExchange = useSelectedExchange({ currency: amount.currency, defaultExchange: undefined, list: exchangeList, }); if (selectedExchange.status !== "ready") { return selectedExchange; } const exchange = selectedExchange.selected; const hook = useAsyncAsHook(async () => { const resp = await api.wallet.call( WalletApiOperation.PreparePeerPullPayment, { amount: amountStr, exchangeBaseUrl: exchange.exchangeBaseUrl, }, ); return resp; }); if (!hook) { return { status: "loading", error: undefined, }; } if (hook.hasError) { return { status: "error", error: alertFromError( i18n.str`Could not load the status of the term of service`, hook, ), }; // return { // status: "loading-uri", // error: hook, // }; } const { amountEffective, amountRaw } = hook.response; const requestAmount = Amounts.parseOrThrow(amountRaw); const toBeReceived = Amounts.parseOrThrow(amountEffective); let purse_expiration: TalerProtocolTimestamp | undefined = undefined; let timestampError: string | undefined = undefined; const t = timestamp === undefined ? undefined : parse(timestamp, "dd/MM/yyyy", new Date()); if (t !== undefined) { if (Number.isNaN(t.getTime())) { timestampError = 'Should have the format "dd/MM/yyyy"'; } else { if (!isFuture(t)) { timestampError = "Should be in the future"; } else { purse_expiration = { t_s: t.getTime() / 1000, }; } } } async function accept(): Promise { if (!subject || !purse_expiration) return; try { const resp = await api.wallet.call( WalletApiOperation.InitiatePeerPullPayment, { exchangeBaseUrl: exchange.exchangeBaseUrl, partialContractTerms: { amount: Amounts.stringify(amount), summary: subject, purse_expiration, }, }, ); onSuccess(resp.transactionId); } catch (e) { if (e instanceof TalerError) { setOperationError(e.errorDetail); } console.error(e); throw Error("error trying to accept"); } } const unableToCreate = !subject || Amounts.isZero(amount) || !purse_expiration; return { status: "ready", subject: { error: subject === undefined ? undefined : !subject ? "Can't be empty" : undefined, value: subject ?? "", onInput: async (e) => setSubject(e), }, expiration: { error: timestampError, value: timestamp === undefined ? "" : timestamp, onInput: async (e) => { setTimestamp(e); }, }, doSelectExchange: selectedExchange.doSelect, exchangeUrl: exchange.exchangeBaseUrl, create: { onClick: unableToCreate ? undefined : accept, }, cancel: { onClick: onClose, }, requestAmount, toBeReceived, error: undefined, }; }; }