/*
 This file is part of GNU Taler
 (C) 2021-2023 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 { useBackendContext } from "../context/backend.js";
import { AccessToken, MerchantBackend } from "../declaration.js";
import {
  useBackendBaseRequest,
  useBackendInstanceRequest,
  useCredentialsChecker,
  useMatchMutate,
} from "./backend.js";
// FIX default import https://github.com/microsoft/TypeScript/issues/49189
import _useSWR, { SWRHook, useSWRConfig } from "swr";
const useSWR = _useSWR as unknown as SWRHook;
interface InstanceAPI {
  updateInstance: (
    data: MerchantBackend.Instances.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: MerchantBackend.Instances.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: MerchantBackend.Instances.InstanceConfigurationMessage,
  ) => Promise;
  deleteInstance: (id: string) => Promise;
  purgeInstance: (id: string) => Promise;
}
export function useManagementAPI(instanceId: string): InstanceAPI {
  const mutateAll = useMatchMutate();
  const { url: backendURL } = useBackendContext()
  const { updateToken } = useBackendContext();
  const { request } = useBackendBaseRequest();
  const { requestNewLoginToken } = useCredentialsChecker()
  const updateInstance = async (
    instance: MerchantBackend.Instances.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) {
      const { token, expiration } = resp
      updateToken({ token, expiration });
    } else {
      updateToken(undefined)
    }
    mutateAll(/\/management\/instances/);
  };
  return { updateInstance, deleteInstance, setNewAccessToken, clearAccessToken };
}
export function useInstanceAPI(): InstanceAPI {
  const { mutate } = useSWRConfig();
  const { url: backendURL, updateToken } = useBackendContext()
  const {
    token: adminToken,
  } = useBackendContext();
  const { request } = useBackendInstanceRequest();
  const { requestNewLoginToken } = useCredentialsChecker()
  const updateInstance = async (
    instance: MerchantBackend.Instances.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) {
      const { token, expiration } = resp
      updateToken({ token, expiration });
    } else {
      updateToken(undefined)
    }
    mutate([`/private/`], null);
  };
  return { updateInstance, deleteInstance, setNewAccessToken, clearAccessToken };
}
export function useInstanceDetails(): HttpResponse<
  MerchantBackend.Instances.QueryInstancesResponse,
  MerchantBackend.ErrorDetail
> {
  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: MerchantBackend.KYC.AccountKycRedirects };
export function useInstanceKYCDetails(): HttpResponse<
  KYCStatus,
  MerchantBackend.ErrorDetail
> {
  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<
  MerchantBackend.Instances.QueryInstancesResponse,
  MerchantBackend.ErrorDetail
> {
  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<
  MerchantBackend.Instances.InstancesResponse,
  MerchantBackend.ErrorDetail
> {
  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 };
}