wallet-core/src/android/index.ts

378 lines
11 KiB
TypeScript
Raw Normal View History

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";
import { openPromise, OpenedPromise } from "../util/promiseUtils";
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";
import {
WALLET_EXCHANGE_PROTOCOL_VERSION,
WALLET_MERCHANT_PROTOCOL_VERSION,
} from "../operations/versions";
2020-07-16 19:22:56 +02:00
import { Amounts } from "../util/amounts";
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-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 = {
// FIXME: pass through this URL
requestUrl: "",
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,
},
2020-05-12 12:21:40 +02:00
};
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
}
case "listExchanges": {
const wallet = await this.wp.promise;
return await wallet.getExchanges();
}
case "addExchange": {
const wallet = await this.wp.promise;
await wallet.updateExchangeFromUrl(args.exchangeBaseUrl);
return {};
}
case "getWithdrawalDetailsForAmount": {
const wallet = await this.wp.promise;
return await wallet.getWithdrawalDetailsForAmount(
args.exchangeBaseUrl,
args.amount,
);
}
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
}
case "getExchangeTos": {
const wallet = await this.wp.promise;
const exchangeBaseUrl = args.exchangeBaseUrl;
return wallet.getExchangeTos(exchangeBaseUrl);
}
case "setExchangeTosAccepted": {
const wallet = await this.wp.promise;
await wallet.acceptExchangeTermsOfService(
args.exchangeBaseUrl,
args.acceptedEtag,
);
return {};
}
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
}
2020-07-16 19:22:56 +02:00
case "acceptManualWithdrawal": {
2020-06-21 15:41:50 +02:00
const wallet = await this.wp.promise;
const res = await wallet.acceptManualWithdrawal(
args.exchangeBaseUrl,
Amounts.parseOrThrow(args.amount),
);
2020-06-21 15:41:50 +02:00
return res;
}
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
}