/*
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();
const oldArgs = walletArgs;
walletArgs = { ...oldArgs };
if (oldArgs && oldArgs.persistentStoragePath) {
try {
fs.unlinkSync(oldArgs.persistentStoragePath);
} catch (e) {
console.error("Error while deleting the wallet db:", e);
}
// Prevent further storage!
walletArgs.persistentStoragePath = undefined;
}
maybeWallet = undefined;
const w = await getDefaultNodeWallet(walletArgs);
maybeWallet = w;
w.runLoopScheduledRetries().catch((e) => {
console.error("Error during wallet retry loop", e);
});
wp.resolve(w);
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");
}