2019-08-19 13:09:11 +02:00
|
|
|
/*
|
|
|
|
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 <http://www.gnu.org/licenses/>
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Imports.
|
|
|
|
*/
|
|
|
|
import { Wallet } from "../wallet";
|
2019-08-22 23:36:36 +02:00
|
|
|
import {
|
|
|
|
getDefaultNodeWallet,
|
|
|
|
withdrawTestBalance,
|
|
|
|
DefaultNodeWalletArgs,
|
|
|
|
} from "../headless/helpers";
|
2019-12-02 00:42:40 +01:00
|
|
|
import { openPromise, OpenedPromise } from "../util/promiseUtils";
|
2020-04-06 17:35:51 +02:00
|
|
|
import fs from "fs";
|
2019-12-09 19:59:08 +01:00
|
|
|
import {
|
|
|
|
HttpRequestLibrary,
|
|
|
|
HttpResponse,
|
|
|
|
HttpRequestOptions,
|
|
|
|
Headers,
|
|
|
|
} from "../util/http";
|
2019-12-15 17:48:22 +01:00
|
|
|
import { NodeHttpLib } from "../headless/NodeHttpLib";
|
2020-03-06 19:01:35 +01:00
|
|
|
import { WalletNotification } from "../types/notifications";
|
2020-05-12 12:21:40 +02:00
|
|
|
import { WALLET_EXCHANGE_PROTOCOL_VERSION, WALLET_MERCHANT_PROTOCOL_VERSION } from "../operations/versions";
|
2019-08-22 23:36:36 +02:00
|
|
|
|
2019-11-14 18:51:54 +01:00
|
|
|
// @ts-ignore: special built-in module
|
2019-12-02 17:35:47 +01:00
|
|
|
//import akono = require("akono");
|
2019-11-14 18:51:54 +01:00
|
|
|
|
2019-12-09 19:59:08 +01:00
|
|
|
export {
|
|
|
|
handleWorkerError,
|
|
|
|
handleWorkerMessage,
|
|
|
|
} from "../crypto/workers/nodeThreadWorker";
|
2019-12-06 02:52:16 +01:00
|
|
|
|
2019-08-22 23:36:36 +02:00
|
|
|
export class AndroidHttpLib implements HttpRequestLibrary {
|
2020-04-06 17:45:41 +02:00
|
|
|
useNfcTunnel = false;
|
2019-08-22 23:36:36 +02:00
|
|
|
|
|
|
|
private nodeHttpLib: HttpRequestLibrary = new NodeHttpLib();
|
|
|
|
|
|
|
|
private requestId = 1;
|
|
|
|
|
|
|
|
private requestMap: { [id: number]: OpenedPromise<HttpResponse> } = {};
|
|
|
|
|
|
|
|
constructor(private sendMessage: (m: string) => void) {}
|
|
|
|
|
2019-12-09 13:29:11 +01:00
|
|
|
get(url: string, opt?: HttpRequestOptions): Promise<HttpResponse> {
|
2019-08-22 23:36:36 +02:00
|
|
|
if (this.useNfcTunnel) {
|
|
|
|
const myId = this.requestId++;
|
|
|
|
const p = openPromise<HttpResponse>();
|
|
|
|
this.requestMap[myId] = p;
|
|
|
|
const request = {
|
|
|
|
method: "get",
|
|
|
|
url,
|
|
|
|
};
|
|
|
|
this.sendMessage(
|
|
|
|
JSON.stringify({
|
|
|
|
type: "tunnelHttp",
|
|
|
|
request,
|
|
|
|
id: myId,
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
return p.promise;
|
|
|
|
} else {
|
2019-12-09 13:29:11 +01:00
|
|
|
return this.nodeHttpLib.get(url, opt);
|
2019-08-22 23:36:36 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-09 19:59:08 +01:00
|
|
|
postJson(
|
|
|
|
url: string,
|
|
|
|
body: any,
|
|
|
|
opt?: HttpRequestOptions,
|
|
|
|
): Promise<import("../util/http").HttpResponse> {
|
2019-08-22 23:36:36 +02:00
|
|
|
if (this.useNfcTunnel) {
|
|
|
|
const myId = this.requestId++;
|
|
|
|
const p = openPromise<HttpResponse>();
|
|
|
|
this.requestMap[myId] = p;
|
|
|
|
const request = {
|
|
|
|
method: "postJson",
|
|
|
|
url,
|
|
|
|
body,
|
|
|
|
};
|
|
|
|
this.sendMessage(
|
|
|
|
JSON.stringify({ type: "tunnelHttp", request, id: myId }),
|
|
|
|
);
|
|
|
|
return p.promise;
|
|
|
|
} else {
|
2019-12-09 13:29:11 +01:00
|
|
|
return this.nodeHttpLib.postJson(url, body, opt);
|
2019-08-22 23:36:36 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-07 10:07:32 +02:00
|
|
|
handleTunnelResponse(msg: any): void {
|
2019-08-22 23:36:36 +02:00
|
|
|
const myId = msg.id;
|
|
|
|
const p = this.requestMap[myId];
|
|
|
|
if (!p) {
|
2019-12-09 19:59:08 +01:00
|
|
|
console.error(
|
|
|
|
`no matching request for tunneled HTTP response, id=${myId}`,
|
|
|
|
);
|
2019-08-22 23:36:36 +02:00
|
|
|
}
|
2019-12-09 19:59:08 +01:00
|
|
|
const headers = new Headers();
|
2019-12-09 13:29:11 +01:00
|
|
|
if (msg.status != 0) {
|
|
|
|
const resp: HttpResponse = {
|
2019-12-09 19:59:08 +01:00
|
|
|
headers,
|
2019-12-09 13:29:11 +01:00
|
|
|
status: msg.status,
|
|
|
|
json: async () => JSON.parse(msg.responseText),
|
|
|
|
text: async () => msg.responseText,
|
|
|
|
};
|
|
|
|
p.resolve(resp);
|
2019-08-22 23:36:36 +02:00
|
|
|
} else {
|
|
|
|
p.reject(new Error(`unexpected HTTP status code ${msg.status}`));
|
|
|
|
}
|
|
|
|
delete this.requestMap[myId];
|
|
|
|
}
|
|
|
|
}
|
2019-08-19 13:09:11 +02:00
|
|
|
|
2020-04-07 10:07:32 +02:00
|
|
|
function sendAkonoMessage(m: string): void {
|
2019-08-19 13:09:11 +02:00
|
|
|
// @ts-ignore
|
2019-12-17 18:42:14 +01:00
|
|
|
globalThis.__akono_sendMessage(m);
|
|
|
|
}
|
|
|
|
|
|
|
|
class AndroidWalletMessageHandler {
|
|
|
|
walletArgs: DefaultNodeWalletArgs | undefined;
|
|
|
|
maybeWallet: Wallet | undefined;
|
|
|
|
wp = openPromise<Wallet>();
|
|
|
|
httpLib = new NodeHttpLib();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Handle a request from the Android wallet.
|
|
|
|
*/
|
|
|
|
async handleMessage(operation: string, id: string, args: any): Promise<any> {
|
2019-08-19 13:09:11 +02:00
|
|
|
switch (operation) {
|
2019-08-20 23:36:56 +02:00
|
|
|
case "init": {
|
2019-12-17 18:42:14 +01:00
|
|
|
this.walletArgs = {
|
2020-03-06 19:01:35 +01:00
|
|
|
notifyHandler: async (notification: WalletNotification) => {
|
|
|
|
sendAkonoMessage(
|
|
|
|
JSON.stringify({ type: "notification", payload: notification }),
|
|
|
|
);
|
2019-08-20 23:36:56 +02:00
|
|
|
},
|
2019-12-17 18:42:14 +01:00
|
|
|
persistentStoragePath: args.persistentStoragePath,
|
|
|
|
httpLib: this.httpLib,
|
2019-08-20 23:36:56 +02:00
|
|
|
};
|
2019-12-17 18:42:14 +01:00
|
|
|
const w = await getDefaultNodeWallet(this.walletArgs);
|
|
|
|
this.maybeWallet = w;
|
2020-03-30 12:39:32 +02:00
|
|
|
w.runRetryLoop().catch((e) => {
|
2019-12-02 17:35:47 +01:00
|
|
|
console.error("Error during wallet retry loop", e);
|
|
|
|
});
|
2019-12-17 18:42:14 +01:00
|
|
|
this.wp.resolve(w);
|
2020-05-12 12:21:40 +02:00
|
|
|
return {
|
|
|
|
supported_protocol_versions: {
|
|
|
|
exchange: WALLET_EXCHANGE_PROTOCOL_VERSION,
|
|
|
|
merchant: WALLET_MERCHANT_PROTOCOL_VERSION,
|
|
|
|
}
|
|
|
|
};
|
2019-08-20 23:36:56 +02:00
|
|
|
}
|
2020-05-13 18:13:03 +02:00
|
|
|
case "getTransactions": {
|
|
|
|
const wallet = await this.wp.promise;
|
|
|
|
return await wallet.getTransactions(args);
|
|
|
|
}
|
2019-12-20 01:25:22 +01:00
|
|
|
case "abortProposal": {
|
|
|
|
const wallet = await this.wp.promise;
|
|
|
|
if (typeof args.proposalId !== "string") {
|
|
|
|
throw Error("propsalId must be a string");
|
|
|
|
}
|
|
|
|
return await wallet.refuseProposal(args.proposalId);
|
|
|
|
}
|
2019-08-20 23:36:56 +02:00
|
|
|
case "getBalances": {
|
2019-12-17 18:42:14 +01:00
|
|
|
const wallet = await this.wp.promise;
|
|
|
|
return await wallet.getBalances();
|
2019-08-20 23:36:56 +02:00
|
|
|
}
|
2019-12-02 17:35:47 +01:00
|
|
|
case "getPendingOperations": {
|
2019-12-17 18:42:14 +01:00
|
|
|
const wallet = await this.wp.promise;
|
|
|
|
return await wallet.getPendingOperations();
|
2019-12-02 17:35:47 +01:00
|
|
|
}
|
2019-08-20 23:36:56 +02:00
|
|
|
case "withdrawTestkudos": {
|
2019-12-17 18:42:14 +01:00
|
|
|
const wallet = await this.wp.promise;
|
2019-12-05 19:38:19 +01:00
|
|
|
try {
|
|
|
|
await withdrawTestBalance(wallet);
|
|
|
|
} catch (e) {
|
|
|
|
console.log("error during withdrawTestBalance", e);
|
|
|
|
}
|
2019-12-17 18:42:14 +01:00
|
|
|
return {};
|
2019-12-02 17:35:47 +01:00
|
|
|
}
|
|
|
|
case "getHistory": {
|
2019-12-17 18:42:14 +01:00
|
|
|
const wallet = await this.wp.promise;
|
|
|
|
return await wallet.getHistory();
|
2019-08-20 23:36:56 +02:00
|
|
|
}
|
2019-12-03 14:40:05 +01:00
|
|
|
case "retryPendingNow": {
|
2019-12-17 18:42:14 +01:00
|
|
|
const wallet = await this.wp.promise;
|
2019-12-03 14:40:05 +01:00
|
|
|
await wallet.runPending(true);
|
2019-12-17 18:42:14 +01:00
|
|
|
return {};
|
2019-12-03 14:40:05 +01:00
|
|
|
}
|
2019-08-20 23:36:56 +02:00
|
|
|
case "preparePay": {
|
2019-12-17 18:42:14 +01:00
|
|
|
const wallet = await this.wp.promise;
|
2019-12-20 01:25:22 +01:00
|
|
|
return await wallet.preparePayForUri(args.url);
|
2019-08-20 23:36:56 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case "confirmPay": {
|
2019-12-17 18:42:14 +01:00
|
|
|
const wallet = await this.wp.promise;
|
|
|
|
return await wallet.confirmPay(args.proposalId, args.sessionId);
|
2019-08-20 23:36:56 +02:00
|
|
|
}
|
2019-08-22 23:36:36 +02:00
|
|
|
case "startTunnel": {
|
2019-12-17 18:42:14 +01:00
|
|
|
// this.httpLib.useNfcTunnel = true;
|
|
|
|
throw Error("not implemented");
|
2019-08-22 23:36:36 +02:00
|
|
|
}
|
|
|
|
case "stopTunnel": {
|
2019-12-17 18:42:14 +01:00
|
|
|
// this.httpLib.useNfcTunnel = false;
|
|
|
|
throw Error("not implemented");
|
2019-08-22 23:36:36 +02:00
|
|
|
}
|
|
|
|
case "tunnelResponse": {
|
2019-12-17 18:42:14 +01:00
|
|
|
// httpLib.handleTunnelResponse(msg.args);
|
|
|
|
throw Error("not implemented");
|
2019-08-22 23:36:36 +02:00
|
|
|
}
|
2019-12-09 19:59:08 +01:00
|
|
|
case "getWithdrawDetailsForUri": {
|
2019-12-17 18:42:14 +01:00
|
|
|
const wallet = await this.wp.promise;
|
|
|
|
return await wallet.getWithdrawDetailsForUri(
|
|
|
|
args.talerWithdrawUri,
|
|
|
|
args.selectedExchange,
|
2019-12-09 19:59:08 +01:00
|
|
|
);
|
|
|
|
}
|
2020-03-30 12:42:28 +02:00
|
|
|
case "applyRefund": {
|
|
|
|
const wallet = await this.wp.promise;
|
|
|
|
return await wallet.applyRefund(args.talerRefundUri);
|
|
|
|
}
|
2019-12-09 19:59:08 +01:00
|
|
|
case "acceptExchangeTermsOfService": {
|
2019-12-17 18:42:14 +01:00
|
|
|
const wallet = await this.wp.promise;
|
|
|
|
return await wallet.acceptExchangeTermsOfService(
|
|
|
|
args.exchangeBaseUrl,
|
|
|
|
args.etag,
|
2019-12-09 19:59:08 +01:00
|
|
|
);
|
2019-09-01 01:05:38 +02:00
|
|
|
}
|
|
|
|
case "acceptWithdrawal": {
|
2019-12-17 18:42:14 +01:00
|
|
|
const wallet = await this.wp.promise;
|
|
|
|
return await wallet.acceptWithdrawal(
|
|
|
|
args.talerWithdrawUri,
|
|
|
|
args.selectedExchange,
|
2019-12-09 19:59:08 +01:00
|
|
|
);
|
2019-09-01 01:05:38 +02:00
|
|
|
}
|
2019-08-20 23:36:56 +02:00
|
|
|
case "reset": {
|
2019-12-17 18:42:14 +01:00
|
|
|
const oldArgs = this.walletArgs;
|
|
|
|
this.walletArgs = { ...oldArgs };
|
2019-12-03 00:52:15 +01:00
|
|
|
if (oldArgs && oldArgs.persistentStoragePath) {
|
2019-08-20 23:36:56 +02:00
|
|
|
try {
|
2019-12-03 00:52:15 +01:00
|
|
|
fs.unlinkSync(oldArgs.persistentStoragePath);
|
2019-08-20 23:36:56 +02:00
|
|
|
} catch (e) {
|
|
|
|
console.error("Error while deleting the wallet db:", e);
|
|
|
|
}
|
|
|
|
// Prevent further storage!
|
2019-12-17 18:42:14 +01:00
|
|
|
this.walletArgs.persistentStoragePath = undefined;
|
2019-08-19 20:24:29 +02:00
|
|
|
}
|
2019-12-17 18:42:14 +01:00
|
|
|
const wallet = await this.wp.promise;
|
2019-12-03 14:40:05 +01:00
|
|
|
wallet.stop();
|
2019-12-17 18:42:14 +01:00
|
|
|
this.wp = openPromise<Wallet>();
|
|
|
|
this.maybeWallet = undefined;
|
|
|
|
const w = await getDefaultNodeWallet(this.walletArgs);
|
|
|
|
this.maybeWallet = w;
|
2020-03-30 12:39:32 +02:00
|
|
|
w.runRetryLoop().catch((e) => {
|
2019-12-03 00:52:15 +01:00
|
|
|
console.error("Error during wallet retry loop", e);
|
|
|
|
});
|
2019-12-17 18:42:14 +01:00
|
|
|
this.wp.resolve(w);
|
|
|
|
return {};
|
2019-08-20 23:36:56 +02:00
|
|
|
}
|
2019-08-19 13:09:11 +02:00
|
|
|
default:
|
2019-12-17 18:42:14 +01:00
|
|
|
throw Error(`operation "${operation}" not understood`);
|
2019-08-19 13:09:11 +02:00
|
|
|
}
|
2019-12-17 18:42:14 +01:00
|
|
|
}
|
|
|
|
}
|
2019-08-19 14:08:14 +02:00
|
|
|
|
2020-04-07 10:07:32 +02:00
|
|
|
export function installAndroidWalletListener(): void {
|
2019-12-17 18:42:14 +01:00
|
|
|
// @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);
|
|
|
|
}
|
|
|
|
const handler = new AndroidWalletMessageHandler();
|
2020-04-07 10:07:32 +02:00
|
|
|
const onMessage = async (msgStr: any): Promise<void> => {
|
2019-12-17 18:42:14 +01:00
|
|
|
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;
|
|
|
|
console.log(`android listener: got request for ${operation} (${id})`);
|
2019-12-05 19:38:19 +01:00
|
|
|
|
2019-12-17 18:42:14 +01:00
|
|
|
try {
|
|
|
|
const result = await handler.handleMessage(operation, id, msg.args);
|
|
|
|
console.log(
|
|
|
|
`android listener: sending success response for ${operation} (${id})`,
|
|
|
|
);
|
2020-03-06 19:01:35 +01:00
|
|
|
const respMsg = {
|
|
|
|
type: "response",
|
|
|
|
id,
|
|
|
|
operation,
|
|
|
|
isError: false,
|
|
|
|
result,
|
|
|
|
};
|
2019-12-17 18:42:14 +01:00
|
|
|
sendMessage(JSON.stringify(respMsg));
|
|
|
|
} catch (e) {
|
|
|
|
const respMsg = {
|
|
|
|
type: "response",
|
|
|
|
id,
|
|
|
|
operation,
|
|
|
|
isError: true,
|
|
|
|
result: { message: e.toString() },
|
|
|
|
};
|
|
|
|
sendMessage(JSON.stringify(respMsg));
|
|
|
|
return;
|
|
|
|
}
|
2019-08-19 13:09:11 +02:00
|
|
|
};
|
2019-12-17 18:42:14 +01:00
|
|
|
|
2019-08-19 13:09:11 +02:00
|
|
|
// @ts-ignore
|
|
|
|
globalThis.__akono_onMessage = onMessage;
|
2019-08-19 20:24:29 +02:00
|
|
|
|
|
|
|
console.log("android wallet listener installed");
|
2019-08-19 14:08:14 +02:00
|
|
|
}
|