/*
 This file is part of TALER
 (C) 2019 GNUnet e.V.
 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.
 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
 TALER; see the file COPYING.  If not, see 
 */
import commander = require("commander");
import os = require("os");
import { getDefaultNodeWallet, withdrawTestBalance } from "./helpers";
import { MerchantBackendConnection } from "./merchant";
import { runIntegrationTest } from "./integrationtest";
import { Wallet } from "../wallet";
import querystring = require("querystring");
import qrcodeGenerator = require("qrcode-generator");
import readline = require("readline");
const program = new commander.Command();
program.version("0.0.1").option("--verbose", "enable verbose output", false);
const walletDbPath = os.homedir + "/" + ".talerwalletdb.json";
function prompt(question: string): Promise {
  const stdinReadline = readline.createInterface({
    input: process.stdin,
    output: process.stdout,
  });
  return new Promise((resolve, reject) => {
    stdinReadline.question(question, res => {
      resolve(res);
      stdinReadline.close();
    });
  });
}
function applyVerbose(verbose: boolean) {
  if (verbose) {
    console.log("enabled verbose logging");
    Wallet.enableTracing = true;
  }
}
program
  .command("test-withdraw")
  .option(
    "-e, --exchange ",
    "exchange base URL",
    "https://exchange.test.taler.net/",
  )
  .option("-a, --amount ", "amount to withdraw", "TESTKUDOS:10")
  .option("-b, --bank ", "bank base URL", "https://bank.test.taler.net/")
  .description("withdraw test currency from the test bank")
  .action(async cmdObj => {
    applyVerbose(program.verbose);
    console.log("test-withdraw command called");
    const wallet = await getDefaultNodeWallet({
      persistentStoragePath: walletDbPath,
    });
    await withdrawTestBalance(wallet, cmdObj.amount, cmdObj.bank, cmdObj.exchange);
    process.exit(0);
  });
program
  .command("balance")
  .description("show wallet balance")
  .action(async () => {
    applyVerbose(program.verbose);
    console.log("balance command called");
    const wallet = await getDefaultNodeWallet({
      persistentStoragePath: walletDbPath,
    });
    console.log("got wallet");
    const balance = await wallet.getBalances();
    console.log(JSON.stringify(balance, undefined, 2));
    process.exit(0);
  });
program
  .command("history")
  .description("show wallet history")
  .action(async () => {
    applyVerbose(program.verbose);
    console.log("history command called");
    const wallet = await getDefaultNodeWallet({
      persistentStoragePath: walletDbPath,
    });
    console.log("got wallet");
    const history = await wallet.getHistory();
    console.log(JSON.stringify(history, undefined, 2));
    process.exit(0);
  });
async function asyncSleep(milliSeconds: number): Promise {
  return new Promise((resolve, reject) => {
    setTimeout(() => resolve(), milliSeconds);
  });
}
program
  .command("test-merchant-qrcode")
  .option("-a, --amount ", "amount to spend", "TESTKUDOS:1")
  .option("-s, --summary ", "contract summary", "Test Payment")
  .action(async cmdObj => {
    applyVerbose(program.verbose);
    console.log("creating order");
    const merchantBackend = new MerchantBackendConnection(
      "https://backend.test.taler.net/",
      "sandbox",
    );
    const orderResp = await merchantBackend.createOrder(
      cmdObj.amount,
      cmdObj.summary,
      "",
    );
    console.log("created new order with order ID", orderResp.orderId);
    const checkPayResp = await merchantBackend.checkPayment(orderResp.orderId);
    const qrcode = qrcodeGenerator(0, "M");
    const talerPayUri = checkPayResp.taler_pay_uri;
    if (!talerPayUri) {
      console.error("fatal: no taler pay URI received from backend");
      process.exit(1);
      return;
    }
    console.log("taler pay URI:", talerPayUri);
    qrcode.addData(talerPayUri);
    qrcode.make();
    console.log(qrcode.createASCII());
    console.log("waiting for payment ...");
    while (1) {
      await asyncSleep(500);
      const checkPayResp2 = await merchantBackend.checkPayment(
        orderResp.orderId,
      );
      if (checkPayResp2.paid) {
        console.log("payment successfully received!");
        break;
      }
    }
  });
program
  .command("withdraw-uri ")
  .action(async (withdrawUrl, cmdObj) => {
    applyVerbose(program.verbose);
    console.log("withdrawing", withdrawUrl);
    const wallet = await getDefaultNodeWallet({
      persistentStoragePath: walletDbPath,
    });
    const withdrawInfo = await wallet.getWithdrawalInfo(withdrawUrl);
    console.log("withdraw info", withdrawInfo);
    const selectedExchange = withdrawInfo.suggestedExchange;
    if (!selectedExchange) {
      console.error("no suggested exchange!");
      process.exit(1);
      return;
    }
    const {
      reservePub,
      confirmTransferUrl,
    } = await wallet.acceptWithdrawal(
      withdrawUrl,
      selectedExchange,
    );
    if (confirmTransferUrl) {
      console.log("please confirm the transfer at", confirmTransferUrl);
    }
    await wallet.processReserve(reservePub);
    console.log("finished withdrawing");
    wallet.stop();
  });
program
  .command("tip-uri ")
  .action(async (tipUri, cmdObj) => {
    applyVerbose(program.verbose);
    console.log("getting tip", tipUri);
    const wallet = await getDefaultNodeWallet({
      persistentStoragePath: walletDbPath,
    });
    const res = await wallet.getTipStatus(tipUri);
    console.log("tip status", res);
    await wallet.acceptTip(tipUri);
    wallet.stop();
  });
  program
  .command("refund-uri ")
  .action(async (refundUri, cmdObj) => {
    applyVerbose(program.verbose);
    console.log("getting refund", refundUri);
    const wallet = await getDefaultNodeWallet({
      persistentStoragePath: walletDbPath,
    });
    await wallet.applyRefund(refundUri);
    wallet.stop();
  });
program
  .command("pay-uri  {
    applyVerbose(program.verbose);
    console.log("paying for", payUrl);
    const wallet = await getDefaultNodeWallet({
      persistentStoragePath: walletDbPath,
    });
    const result = await wallet.preparePay(payUrl);
    if (result.status === "error") {
      console.error("Could not pay:", result.error);
      process.exit(1);
      return;
    }
    if (result.status === "insufficient-balance") {
      console.log("contract", result.contractTerms!);
      console.error("insufficient balance");
      process.exit(1);
      return;
    }
    if (result.status === "paid") {
      console.log("already paid!");
      process.exit(0);
      return;
    }
    if (result.status === "payment-possible") {
      console.log("paying ...");
    } else {
      throw Error("not reached");
    }
    console.log("contract", result.contractTerms!);
    let pay;
    if (cmdObj.yes) {
      pay = true;
    } else {
      while (true) {
        const yesNoResp = (await prompt("Pay? [Y/n]")).toLowerCase();
        if (yesNoResp === "" || yesNoResp === "y" || yesNoResp === "yes") {
          pay = true;
          break;
        } else if (yesNoResp === "n" || yesNoResp === "no") {
          pay = false;
          break;
        } else {
          console.log("please answer y/n");
        }
      }
    }
    if (pay) {
      const payRes = await wallet.confirmPay(result.proposalId!, undefined);
      console.log("paid!");
    } else {
      console.log("not paying");
    }
    wallet.stop();
  });
program
  .command("integrationtest")
  .option(
    "-e, --exchange ",
    "exchange base URL",
    "https://exchange.test.taler.net/",
  )
  .option(
    "-m, --merchant ",
    "merchant base URL",
    "https://backend.test.taler.net/",
  )
  .option(
    "-k, --merchant-api-key ",
    "merchant API key",
    "sandbox",
  )
  .option(
    "-b, --bank ",
    "bank base URL",
    "https://bank.test.taler.net/",
  )
  .option(
    "-w, --withdraw-amount ",
    "amount to withdraw",
    "TESTKUDOS:10",
  )
  .option("-s, --spend-amount ", "amount to spend", "TESTKUDOS:4")
  .description("Run integration test with bank, exchange and merchant.")
  .action(async cmdObj => {
    applyVerbose(program.verbose);
    try {
      await runIntegrationTest({
        amountToSpend: cmdObj.spendAmount,
        amountToWithdraw: cmdObj.withdrawAmount,
        bankBaseUrl: cmdObj.bank,
        exchangeBaseUrl: cmdObj.exchange,
        merchantApiKey: cmdObj.merchantApiKey,
        merchantBaseUrl: cmdObj.merchant,
      }).catch(err => {
        console.error("Failed with exception:");
        console.error(err);
      });
      process.exit(0);
    } catch (e) {
      console.error(e);
      process.exit(1);
    }
  });
// error on unknown commands
program.on("command:*", function() {
  console.error(
    "Invalid command: %s\nSee --help for a list of available commands.",
    program.args.join(" "),
  );
  process.exit(1);
});
program.parse(process.argv);
if (process.argv.length <= 2) {
  console.error("Error: No command given.");
  program.help();
}