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.
|
|
|
|
*/
|
2019-08-22 23:36:36 +02:00
|
|
|
import {
|
|
|
|
DefaultNodeWalletArgs,
|
2022-10-19 15:44:28 +02:00
|
|
|
getDefaultNodeWallet,
|
|
|
|
getErrorDetailFromException,
|
2020-08-12 09:11:00 +02:00
|
|
|
handleWorkerError,
|
|
|
|
handleWorkerMessage,
|
2022-10-19 15:44:28 +02:00
|
|
|
Headers,
|
2020-08-12 09:11:00 +02:00
|
|
|
HttpRequestLibrary,
|
|
|
|
HttpRequestOptions,
|
2022-10-19 15:44:28 +02:00
|
|
|
HttpResponse,
|
|
|
|
NodeHttpLib,
|
|
|
|
OpenedPromise,
|
2020-08-12 09:11:00 +02:00
|
|
|
openPromise,
|
2022-10-19 15:44:28 +02:00
|
|
|
Wallet,
|
2020-08-12 09:11:00 +02:00
|
|
|
WALLET_EXCHANGE_PROTOCOL_VERSION,
|
|
|
|
WALLET_MERCHANT_PROTOCOL_VERSION,
|
2021-01-30 16:35:55 +01:00
|
|
|
} from "@gnu-taler/taler-wallet-core";
|
2019-08-22 23:36:36 +02:00
|
|
|
|
2021-03-27 14:35:58 +01:00
|
|
|
import {
|
|
|
|
CoreApiEnvelope,
|
|
|
|
CoreApiResponse,
|
|
|
|
CoreApiResponseSuccess,
|
2022-11-07 11:49:34 +01:00
|
|
|
Logger,
|
2021-08-20 13:06:59 +02:00
|
|
|
WalletNotification,
|
|
|
|
} from "@gnu-taler/taler-util";
|
2022-10-19 15:44:28 +02:00
|
|
|
import fs from "fs";
|
2019-11-14 18:51:54 +01:00
|
|
|
|
2020-08-12 09:11:00 +02:00
|
|
|
export { handleWorkerError, handleWorkerMessage };
|
2019-12-06 02:52:16 +01:00
|
|
|
|
2022-11-07 11:49:34 +01:00
|
|
|
const logger = new Logger("taler-wallet-embedded/index.ts");
|
|
|
|
|
2021-07-07 08:42:55 +02:00
|
|
|
export class NativeHttpLib implements HttpRequestLibrary {
|
2020-04-06 17:45:41 +02:00
|
|
|
useNfcTunnel = false;
|
2019-08-22 23:36:36 +02:00
|
|
|
|
2020-08-12 09:11:00 +02:00
|
|
|
private nodeHttpLib: HttpRequestLibrary = new NodeHttpLib();
|
2019-08-22 23:36:36 +02:00
|
|
|
|
|
|
|
private requestId = 1;
|
|
|
|
|
2020-08-03 09:30:48 +02:00
|
|
|
private requestMap: {
|
2020-08-12 09:11:00 +02:00
|
|
|
[id: number]: OpenedPromise<HttpResponse>;
|
2020-08-03 09:30:48 +02:00
|
|
|
} = {};
|
2019-08-22 23:36:36 +02:00
|
|
|
|
|
|
|
constructor(private sendMessage: (m: string) => void) {}
|
|
|
|
|
2020-12-02 14:55:04 +01:00
|
|
|
fetch(url: string, opt?: HttpRequestOptions): Promise<HttpResponse> {
|
|
|
|
return this.nodeHttpLib.fetch(url, opt);
|
|
|
|
}
|
|
|
|
|
2020-08-12 09:11:00 +02:00
|
|
|
get(url: string, opt?: HttpRequestOptions): Promise<HttpResponse> {
|
2019-08-22 23:36:36 +02:00
|
|
|
if (this.useNfcTunnel) {
|
|
|
|
const myId = this.requestId++;
|
2020-08-12 09:11:00 +02:00
|
|
|
const p = openPromise<HttpResponse>();
|
2019-08-22 23:36:36 +02:00
|
|
|
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,
|
2020-08-12 09:11:00 +02:00
|
|
|
opt?: HttpRequestOptions,
|
|
|
|
): Promise<HttpResponse> {
|
2019-08-22 23:36:36 +02:00
|
|
|
if (this.useNfcTunnel) {
|
|
|
|
const myId = this.requestId++;
|
2020-08-12 09:11:00 +02:00
|
|
|
const p = openPromise<HttpResponse>();
|
2019-08-22 23:36:36 +02:00
|
|
|
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) {
|
2022-11-07 11:49:34 +01:00
|
|
|
logger.error(
|
2019-12-09 19:59:08 +01:00
|
|
|
`no matching request for tunneled HTTP response, id=${myId}`,
|
|
|
|
);
|
2019-08-22 23:36:36 +02:00
|
|
|
}
|
2020-08-12 09:11:00 +02:00
|
|
|
const headers = new Headers();
|
2019-12-09 13:29:11 +01:00
|
|
|
if (msg.status != 0) {
|
2020-08-12 09:11:00 +02:00
|
|
|
const resp: HttpResponse = {
|
2020-07-22 10:52:03 +02:00
|
|
|
// 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,
|
2020-08-05 21:00:36 +02:00
|
|
|
requestMethod: "FIXME",
|
2019-12-09 13:29:11 +01:00
|
|
|
json: async () => JSON.parse(msg.responseText),
|
|
|
|
text: async () => msg.responseText,
|
2020-12-14 16:45:15 +01:00
|
|
|
bytes: async () => {
|
|
|
|
throw Error("bytes() not supported for tunnel response");
|
|
|
|
},
|
2019-12-09 13:29:11 +01:00
|
|
|
};
|
|
|
|
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
|
|
|
|
2021-07-07 08:42:55 +02:00
|
|
|
function sendNativeMessage(ev: CoreApiEnvelope): void {
|
2019-08-19 13:09:11 +02:00
|
|
|
// @ts-ignore
|
2021-07-07 08:42:55 +02:00
|
|
|
const sendMessage = globalThis.__native_sendMessage;
|
2020-07-31 19:16:23 +02:00
|
|
|
if (typeof sendMessage !== "function") {
|
|
|
|
const errMsg =
|
2021-07-07 08:42:55 +02:00
|
|
|
"FATAL: cannot install native wallet listener: native functions missing";
|
2022-11-07 11:49:34 +01:00
|
|
|
logger.error(errMsg);
|
2020-07-31 19:16:23 +02:00
|
|
|
throw new Error(errMsg);
|
|
|
|
}
|
|
|
|
const m = JSON.stringify(ev);
|
|
|
|
// @ts-ignore
|
|
|
|
sendMessage(m);
|
2019-12-17 18:42:14 +01:00
|
|
|
}
|
|
|
|
|
2021-07-07 08:42:55 +02:00
|
|
|
class NativeWalletMessageHandler {
|
2019-12-17 18:42:14 +01:00
|
|
|
walletArgs: DefaultNodeWalletArgs | undefined;
|
2021-06-17 21:06:45 +02:00
|
|
|
maybeWallet: Wallet | undefined;
|
|
|
|
wp = openPromise<Wallet>();
|
2019-12-17 18:42:14 +01:00
|
|
|
httpLib = new NodeHttpLib();
|
|
|
|
|
|
|
|
/**
|
2021-07-07 08:42:55 +02:00
|
|
|
* Handle a request from the native wallet.
|
2019-12-17 18:42:14 +01:00
|
|
|
*/
|
2020-07-29 19:23:17 +02:00
|
|
|
async handleMessage(
|
|
|
|
operation: string,
|
|
|
|
id: string,
|
|
|
|
args: any,
|
2020-08-12 09:11:00 +02:00
|
|
|
): Promise<CoreApiResponse> {
|
|
|
|
const wrapResponse = (result: unknown): CoreApiResponseSuccess => {
|
2020-07-29 19:23:17 +02:00
|
|
|
return {
|
|
|
|
type: "response",
|
|
|
|
id,
|
|
|
|
operation,
|
|
|
|
result,
|
|
|
|
};
|
|
|
|
};
|
2021-06-15 18:52:43 +02:00
|
|
|
|
2022-10-19 15:44:28 +02:00
|
|
|
let initResponse: any = {};
|
|
|
|
|
2021-06-15 18:52:43 +02:00
|
|
|
const reinit = async () => {
|
2022-11-07 11:49:34 +01:00
|
|
|
logger.info("in reinit");
|
2021-06-15 18:52:43 +02:00
|
|
|
const w = await getDefaultNodeWallet(this.walletArgs);
|
|
|
|
this.maybeWallet = w;
|
2022-10-19 15:44:28 +02:00
|
|
|
const resp = await w.handleCoreApiRequest(
|
|
|
|
"initWallet",
|
|
|
|
"native-init",
|
|
|
|
{},
|
|
|
|
);
|
|
|
|
initResponse = resp.type == "response" ? resp.result : resp.error;
|
2021-08-19 15:12:33 +02:00
|
|
|
w.runTaskLoop().catch((e) => {
|
2022-11-07 11:49:34 +01:00
|
|
|
logger.error(
|
|
|
|
`Error during wallet retry loop: ${e.stack ?? e.toString()}`,
|
|
|
|
);
|
2021-06-15 18:52:43 +02:00
|
|
|
});
|
|
|
|
this.wp.resolve(w);
|
|
|
|
};
|
|
|
|
|
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-08-12 09:11:00 +02:00
|
|
|
notifyHandler: async (notification: WalletNotification) => {
|
2021-07-07 08:42:55 +02:00
|
|
|
sendNativeMessage({ 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,
|
2022-10-19 15:44:28 +02:00
|
|
|
cryptoWorkerType: args.cryptoWorkerType,
|
2019-08-20 23:36:56 +02:00
|
|
|
};
|
2021-06-15 18:52:43 +02:00
|
|
|
await reinit();
|
2020-07-29 19:23:17 +02:00
|
|
|
return wrapResponse({
|
2022-10-19 15:44:28 +02:00
|
|
|
...initResponse,
|
2020-07-29 19:23:17 +02:00
|
|
|
});
|
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-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) {
|
2022-11-07 11:49:34 +01:00
|
|
|
logger.error("Error while deleting the wallet db:", e);
|
2019-08-20 23:36:56 +02:00
|
|
|
}
|
|
|
|
// 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();
|
2021-06-17 21:06:45 +02:00
|
|
|
this.wp = openPromise<Wallet>();
|
2019-12-17 18:42:14 +01:00
|
|
|
this.maybeWallet = undefined;
|
2021-06-15 18:52:43 +02:00
|
|
|
await reinit();
|
2020-07-29 19:23:17 +02:00
|
|
|
return wrapResponse({});
|
2019-08-20 23:36:56 +02:00
|
|
|
}
|
2020-07-23 15:54:00 +02:00
|
|
|
default: {
|
|
|
|
const wallet = await this.wp.promise;
|
2021-06-17 21:06:45 +02:00
|
|
|
return await wallet.handleCoreApiRequest(operation, id, args);
|
2020-07-23 15:54:00 +02:00
|
|
|
}
|
2019-08-19 13:09:11 +02:00
|
|
|
}
|
2019-12-17 18:42:14 +01:00
|
|
|
}
|
|
|
|
}
|
2019-08-19 14:08:14 +02:00
|
|
|
|
2021-07-07 08:42:55 +02:00
|
|
|
export function installNativeWalletListener(): void {
|
|
|
|
const handler = new NativeWalletMessageHandler();
|
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") {
|
2022-11-07 11:49:34 +01:00
|
|
|
logger.error("expected string as message");
|
2019-12-17 18:42:14 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
const msg = JSON.parse(msgStr);
|
|
|
|
const operation = msg.operation;
|
|
|
|
if (typeof operation !== "string") {
|
2022-11-07 11:49:34 +01:00
|
|
|
logger.error(
|
2021-07-07 08:42:55 +02:00
|
|
|
"message to native wallet helper must contain operation of type string",
|
2019-12-17 18:42:14 +01:00
|
|
|
);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const id = msg.id;
|
2022-11-07 11:49:34 +01:00
|
|
|
logger.info(`native listener: got request for ${operation} (${id})`);
|
2019-12-05 19:38:19 +01:00
|
|
|
|
2019-12-17 18:42:14 +01:00
|
|
|
try {
|
2020-07-29 20:48:42 +02:00
|
|
|
const respMsg = await handler.handleMessage(operation, id, msg.args);
|
2022-11-07 11:49:34 +01:00
|
|
|
logger.info(
|
2021-07-07 08:42:55 +02:00
|
|
|
`native listener: sending success response for ${operation} (${id})`,
|
2019-12-17 18:42:14 +01:00
|
|
|
);
|
2021-07-07 08:42:55 +02:00
|
|
|
sendNativeMessage(respMsg);
|
2019-12-17 18:42:14 +01:00
|
|
|
} catch (e) {
|
2020-08-12 09:11:00 +02:00
|
|
|
const respMsg: CoreApiResponse = {
|
2020-07-31 19:16:23 +02:00
|
|
|
type: "error",
|
2019-12-17 18:42:14 +01:00
|
|
|
id,
|
|
|
|
operation,
|
2022-03-22 21:16:38 +01:00
|
|
|
error: getErrorDetailFromException(e),
|
2019-12-17 18:42:14 +01:00
|
|
|
};
|
2021-07-07 08:42:55 +02:00
|
|
|
sendNativeMessage(respMsg);
|
2019-12-17 18:42:14 +01:00
|
|
|
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
|
2021-07-07 08:42:55 +02:00
|
|
|
globalThis.__native_onMessage = onMessage;
|
2019-08-19 20:24:29 +02:00
|
|
|
|
2022-11-07 11:49:34 +01:00
|
|
|
logger.info("native wallet listener installed");
|
2019-08-19 14:08:14 +02:00
|
|
|
}
|