/* This file is part of GNU Taler (C) 2021-2024 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 */ import { HttpResponse, HttpResponseOk, RequestError } from "@gnu-taler/web-util/browser"; import { useBackendBaseRequest, useBackendInstanceRequest, useMatchMutate, } from "./backend.js"; // FIX default import https://github.com/microsoft/TypeScript/issues/49189 import { AccessToken, TalerErrorDetail, TalerMerchantApi } from "@gnu-taler/taler-util"; import _useSWR, { SWRHook, useSWRConfig } from "swr"; import { useSessionContext } from "../context/session.js"; const useSWR = _useSWR as unknown as SWRHook; interface InstanceAPI { updateInstance: ( data: TalerMerchantApi.InstanceReconfigurationMessage, ) => Promise; deleteInstance: () => Promise; clearAccessToken: (currentToken: AccessToken | undefined) => Promise; // setNewAccessToken: ( // currentToken: AccessToken | undefined, // token: AccessToken, // ) => Promise; } export function useAdminAPI(): AdminAPI { const { request } = useBackendBaseRequest(); const mutateAll = useMatchMutate(); const createInstance = async ( instance: TalerMerchantApi.InstanceConfigurationMessage, ): Promise => { await request(`/management/instances`, { method: "POST", data: instance, }); mutateAll(/\/management\/instances/); }; const deleteInstance = async (id: string): Promise => { await request(`/management/instances/${id}`, { method: "DELETE", }); mutateAll(/\/management\/instances/); }; const purgeInstance = async (id: string): Promise => { await request(`/management/instances/${id}`, { method: "DELETE", params: { purge: "YES", }, }); mutateAll(/\/management\/instances/); }; return { createInstance, deleteInstance, purgeInstance }; } export interface AdminAPI { createInstance: ( data: TalerMerchantApi.InstanceConfigurationMessage, ) => Promise; deleteInstance: (id: string) => Promise; purgeInstance: (id: string) => Promise; } export function useManagementAPI(instanceId: string): InstanceAPI { const mutateAll = useMatchMutate(); const { state: { backendUrl }, logIn, logOut, } = useSessionContext(); const { request } = useBackendBaseRequest(); const updateInstance = async ( instance: TalerMerchantApi.InstanceReconfigurationMessage, ): Promise => { await request(`/management/instances/${instanceId}`, { method: "PATCH", data: instance, }); mutateAll(/\/management\/instances/); }; const deleteInstance = async (): Promise => { await request(`/management/instances/${instanceId}`, { method: "DELETE", }); mutateAll(/\/management\/instances/); }; const clearAccessToken = async ( currentToken: AccessToken | undefined, ): Promise => { await request(`/management/instances/${instanceId}/auth`, { method: "POST", token: currentToken, data: { method: "external" }, }); mutateAll(/\/management\/instances/); }; // const setNewAccessToken = async ( // currentToken: AccessToken | undefined, // newToken: AccessToken, // ): Promise => { // await request(`/management/instances/${instanceId}/auth`, { // method: "POST", // token: currentToken, // data: { method: "token", token: newToken }, // }); // const resp = await requestNewLoginToken(backendUrl, newToken); // if (resp.valid) { // logIn({ token: resp.token as AccessToken }); // } else { // logOut(); // } // mutateAll(/\/management\/instances/); // }; return { updateInstance, deleteInstance, // setNewAccessToken, clearAccessToken, }; } export function useInstanceAPI(): InstanceAPI { const { mutate } = useSWRConfig(); const { state: { backendUrl }, } = useSessionContext(); const { request } = useBackendInstanceRequest(); const { state, logIn, logOut } = useSessionContext(); const adminToken = state.status === "loggedIn" && state.isAdmin ? state.token : undefined; const updateInstance = async ( instance: TalerMerchantApi.InstanceReconfigurationMessage, ): Promise => { await request(`/private/`, { method: "PATCH", data: instance, }); if (adminToken) { mutate(["/private/instances", adminToken, backendUrl], null); } mutate([`/private/`], null); }; const deleteInstance = async (): Promise => { await request(`/private/`, { method: "DELETE", // token: adminToken, }); if (adminToken) { mutate(["/private/instances", adminToken, backendUrl], null); } mutate([`/private/`], null); }; const clearAccessToken = async ( currentToken: AccessToken | undefined, ): Promise => { await request(`/private/auth`, { method: "POST", token: currentToken, data: { method: "external" }, }); mutate([`/private/`], null); }; // const setNewAccessToken = async ( // currentToken: AccessToken | undefined, // newToken: AccessToken, // ): Promise => { // await request(`/private/auth`, { // method: "POST", // token: currentToken, // data: { method: "token", token: newToken }, // }); // const resp = await requestNewLoginToken(backendUrl, newToken); // if (resp.valid) { // logIn({ token: resp.token as AccessToken }); // } else { // logOut(); // } // mutate([`/private/`], null); // }; return { updateInstance, deleteInstance, // setNewAccessToken, clearAccessToken, }; } export function useInstanceDetails(): HttpResponse< TalerMerchantApi.QueryInstancesResponse, TalerErrorDetail > { const { fetcher } = useBackendInstanceRequest(); const { data, error, isValidating } = useSWR< HttpResponseOk, RequestError >([`/private/`], fetcher, { refreshInterval: 0, refreshWhenHidden: false, revalidateOnFocus: false, revalidateOnReconnect: false, refreshWhenOffline: false, revalidateIfStale: false, errorRetryCount: 0, errorRetryInterval: 1, shouldRetryOnError: false, }); if (isValidating) return { loading: true, data: data?.data }; if (data) return data; if (error) return error.cause; return { loading: true }; } type KYCStatus = | { type: "ok" } | { type: "redirect"; status: TalerMerchantApi.AccountKycRedirects }; export function useInstanceKYCDetails(): HttpResponse< KYCStatus, TalerErrorDetail > { const { fetcher } = useBackendInstanceRequest(); const { data, error } = useSWR< HttpResponseOk, RequestError >([`/private/kyc`], fetcher, { refreshInterval: 60 * 1000, refreshWhenHidden: false, revalidateOnFocus: false, revalidateIfStale: false, revalidateOnMount: false, revalidateOnReconnect: false, refreshWhenOffline: false, errorRetryCount: 0, errorRetryInterval: 1, shouldRetryOnError: false, }); if (data) { if (data.info?.status === 202) return { ok: true, data: { type: "redirect", status: data.data } }; return { ok: true, data: { type: "ok" } }; } if (error) return error.cause; return { loading: true }; } export function useManagedInstanceDetails( instanceId: string, ): HttpResponse< TalerMerchantApi.QueryInstancesResponse, TalerErrorDetail > { const { request } = useBackendBaseRequest(); const { data, error, isValidating } = useSWR< HttpResponseOk, RequestError >([`/management/instances/${instanceId}`], request, { refreshInterval: 0, refreshWhenHidden: false, revalidateOnFocus: false, revalidateOnReconnect: false, refreshWhenOffline: false, errorRetryCount: 0, errorRetryInterval: 1, shouldRetryOnError: false, }); if (isValidating) return { loading: true, data: data?.data }; if (data) return data; if (error) return error.cause; return { loading: true }; } export function useBackendInstances(): HttpResponse< TalerMerchantApi.InstancesResponse, TalerErrorDetail > { const { request } = useBackendBaseRequest(); const { data, error, isValidating } = useSWR< HttpResponseOk, RequestError >(["/management/instances"], request); if (isValidating) return { loading: true, data: data?.data }; if (data) return data; if (error) return error.cause; return { loading: true }; }