/*
 This file is part of GNU Taler
 (C) 2019 GNUnet e.V.
 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 
 */
/**
 * Imports.
 */
import { Wallet } from "../wallet";
import {
  getDefaultNodeWallet,
  withdrawTestBalance,
  DefaultNodeWalletArgs,
  NodeHttpLib,
} from "../headless/helpers";
import { openPromise, OpenedPromise } from "../util/promiseUtils";
import fs = require("fs");
import { HttpRequestLibrary, HttpResponse } from "../util/http";
// @ts-ignore: special built-in module
//import akono = require("akono");
export class AndroidHttpLib implements HttpRequestLibrary {
  useNfcTunnel: boolean = false;
  private nodeHttpLib: HttpRequestLibrary = new NodeHttpLib();
  private requestId = 1;
  private requestMap: { [id: number]: OpenedPromise } = {};
  constructor(private sendMessage: (m: string) => void) {}
  get(url: string): Promise {
    if (this.useNfcTunnel) {
      const myId = this.requestId++;
      const p = openPromise();
      this.requestMap[myId] = p;
      const request = {
        method: "get",
        url,
      };
      this.sendMessage(
        JSON.stringify({
          type: "tunnelHttp",
          request,
          id: myId,
        }),
      );
      return p.promise;
    } else {
      return this.nodeHttpLib.get(url);
    }
  }
  postJson(url: string, body: any): Promise {
    if (this.useNfcTunnel) {
      const myId = this.requestId++;
      const p = openPromise();
      this.requestMap[myId] = p;
      const request = {
        method: "postJson",
        url,
        body,
      };
      this.sendMessage(
        JSON.stringify({ type: "tunnelHttp", request, id: myId }),
      );
      return p.promise;
    } else {
      return this.nodeHttpLib.postJson(url, body);
    }
  }
  handleTunnelResponse(msg: any) {
    const myId = msg.id;
    const p = this.requestMap[myId];
    if (!p) {
      console.error(`no matching request for tunneled HTTP response, id=${myId}`);
    }
    if (msg.status == 200) {
      p.resolve({ responseJson: msg.responseJson, status: msg.status });
    } else {
      p.reject(new Error(`unexpected HTTP status code ${msg.status}`));
    }
    delete this.requestMap[myId];
  }
}
export function installAndroidWalletListener() {
  // @ts-ignore
  const sendMessage: (m: string) => void = globalThis.__akono_sendMessage;
  if (typeof sendMessage !== "function") {
    const errMsg =
      "FATAL: cannot install android wallet listener: akono functions missing";
    console.error(errMsg);
    throw new Error(errMsg);
  }
  let maybeWallet: Wallet | undefined;
  let wp = openPromise();
  let httpLib = new AndroidHttpLib(sendMessage);
  let walletArgs: DefaultNodeWalletArgs | undefined;
  const onMessage = async (msgStr: any) => {
    if (typeof msgStr !== "string") {
      console.error("expected string as message");
      return;
    }
    const msg = JSON.parse(msgStr);
    const operation = msg.operation;
    if (typeof operation !== "string") {
      console.error(
        "message to android wallet helper must contain operation of type string",
      );
      return;
    }
    const id = msg.id;
    let result;
    switch (operation) {
      case "init": {
        walletArgs = {
          notifyHandler: async () => {
            sendMessage(JSON.stringify({ type: "notification" }));
          },
          persistentStoragePath: msg.args.persistentStoragePath,
          httpLib: httpLib,
        };
        const w = await getDefaultNodeWallet(walletArgs);
        maybeWallet = w;
        w.runLoopScheduledRetries().catch((e) => {
          console.error("Error during wallet retry loop", e);
        });
        wp.resolve(w);
        result = true;
        break;
      }
      case "getBalances": {
        const wallet = await wp.promise;
        result = await wallet.getBalances();
        break;
      }
      case "getPendingOperations": {
        const wallet = await wp.promise;
        result = await wallet.getPendingOperations();
        break;
      }
      case "withdrawTestkudos": {
        const wallet = await wp.promise;
        await withdrawTestBalance(wallet);
        break;
      }
      case "getHistory": {
        const wallet = await wp.promise;
        result = await wallet.getHistory();
        break;
      }
      case "preparePay": {
        const wallet = await wp.promise;
        result = await wallet.preparePay(msg.args.url);
        break;
      }
      case "confirmPay": {
        const wallet = await wp.promise;
        result = await wallet.confirmPay(msg.args.proposalId, msg.args.sessionId);
        break;
      }
      case "startTunnel": {
        httpLib.useNfcTunnel = true;
        break;
      }
      case "stopTunnel": {
        httpLib.useNfcTunnel = false;
        break;
      }
      case "tunnelResponse": {
        httpLib.handleTunnelResponse(msg.args);
        break;
      }
      case "getWithdrawalInfo": {
        const wallet = await wp.promise;
        result = await wallet.getWithdrawalInfo(msg.args.talerWithdrawUri);
        break;
      }
      case "acceptWithdrawal": {
        const wallet = await wp.promise;
        result = await wallet.acceptWithdrawal(msg.args.talerWithdrawUri, msg.args.selectedExchange);
        break;
      }
      case "reset": {
        const wallet = await wp.promise;
        wallet.stop();
        wp = openPromise();
        if (walletArgs && walletArgs.persistentStoragePath) {
          try {
            fs.unlinkSync(walletArgs.persistentStoragePath);
          } catch (e) {
            console.error("Error while deleting the wallet db:", e);
          }
          // Prevent further storage!
          walletArgs.persistentStoragePath = undefined;
        }
        maybeWallet = undefined;
        break;
      }
      default:
        console.error(`operation "${operation}" not understood`);
        return;
    }
    const respMsg = { result, id, operation, type: "response" };
    sendMessage(JSON.stringify(respMsg));
  };
  // @ts-ignore
  globalThis.__akono_onMessage = onMessage;
  console.log("android wallet listener installed");
}