/*
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 };
}