wallet-core/packages/taler-wallet-embedded/src/wallet-qjs.ts

265 lines
7.7 KiB
TypeScript
Raw Normal View History

/*
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/>
*/
/**
* Entry-point for the wallet under qtart, the QuckJS-based GNU Taler runtime.
*/
/**
* Imports.
*/
import {
2023-01-25 18:49:00 +01:00
CoreApiMessageEnvelope,
CoreApiResponse,
CoreApiResponseSuccess,
2023-02-15 23:32:42 +01:00
getErrorDetailFromException,
InitRequest,
Logger,
2022-11-11 20:52:45 +01:00
setGlobalLogLevelFromString,
2022-12-06 14:53:35 +01:00
setPRNG,
WalletNotification,
} from "@gnu-taler/taler-util";
2023-02-16 01:01:18 +01:00
import { createPlatformHttpLib } from "@gnu-taler/taler-util/http";
2023-02-15 23:32:42 +01:00
import { qjsOs } from "@gnu-taler/taler-util/qtart";
import {
createNativeWalletHost2,
DefaultNodeWalletArgs,
openPromise,
Wallet,
WalletApiOperation,
} from "@gnu-taler/taler-wallet-core";
2022-11-11 20:52:45 +01:00
2022-12-06 14:53:35 +01:00
setGlobalLogLevelFromString("trace");
setPRNG(function (x: Uint8Array, n: number) {
// @ts-ignore
const va = globalThis._tart.randomBytes(n);
2022-12-06 14:53:35 +01:00
const v = new Uint8Array(va);
for (let i = 0; i < n; i++) x[i] = v[i];
for (let i = 0; i < v.length; i++) v[i] = 0;
});
const logger = new Logger("taler-wallet-embedded/index.ts");
/**
* Sends JSON to the host application, i.e. the process that
* runs the JavaScript interpreter (quickjs / qtart) to run
* the embedded wallet.
*/
2023-01-25 18:49:00 +01:00
function sendNativeMessage(ev: CoreApiMessageEnvelope): void {
const m = JSON.stringify(ev);
2023-01-03 10:51:05 +01:00
qjsOs.postMessageToHost(m);
}
class NativeWalletMessageHandler {
walletArgs: DefaultNodeWalletArgs | undefined;
initRequest: InitRequest = {};
maybeWallet: Wallet | undefined;
wp = openPromise<Wallet>();
2023-02-15 23:32:42 +01:00
httpLib = createPlatformHttpLib();
/**
* Handle a request from the native wallet.
*/
async handleMessage(
operation: string,
id: string,
args: any,
): Promise<CoreApiResponse> {
const wrapSuccessResponse = (result: unknown): CoreApiResponseSuccess => {
return {
type: "response",
id,
operation,
result,
};
};
let initResponse: any = {};
const reinit = async () => {
logger.info("in reinit");
2023-02-15 23:32:42 +01:00
const wR = await createNativeWalletHost2(this.walletArgs);
const w = wR.wallet;
this.maybeWallet = w;
const resp = await w.handleCoreApiRequest("initWallet", "native-init", {
...this.initRequest,
});
initResponse = resp.type == "response" ? resp.result : resp.error;
w.runTaskLoop().catch((e) => {
logger.error(
`Error during wallet retry loop: ${e.stack ?? e.toString()}`,
);
});
this.wp.resolve(w);
};
switch (operation) {
case "init": {
this.initRequest = {
2023-01-03 10:51:05 +01:00
...args,
};
this.walletArgs = {
notifyHandler: async (notification: WalletNotification) => {
sendNativeMessage({ type: "notification", payload: notification });
},
persistentStoragePath: args.persistentStoragePath,
httpLib: this.httpLib,
cryptoWorkerType: args.cryptoWorkerType,
};
const logLevel = args.logLevel;
if (logLevel) {
setGlobalLogLevelFromString(logLevel);
}
await reinit();
return wrapSuccessResponse({
...initResponse,
});
}
case "startTunnel": {
// this.httpLib.useNfcTunnel = true;
throw Error("not implemented");
}
case "stopTunnel": {
// this.httpLib.useNfcTunnel = false;
throw Error("not implemented");
}
case "tunnelResponse": {
// httpLib.handleTunnelResponse(msg.args);
throw Error("not implemented");
}
case "reset": {
logger.info("resetting wallet");
const oldArgs = this.walletArgs;
this.walletArgs = { ...oldArgs };
if (oldArgs && oldArgs.persistentStoragePath) {
const ret = qjsOs.remove(oldArgs.persistentStoragePath);
if (ret != 0) {
logger.error("removing DB file failed");
}
// Prevent further storage!
this.walletArgs.persistentStoragePath = undefined;
}
const wallet = await this.wp.promise;
wallet.stop();
this.wp = openPromise<Wallet>();
this.maybeWallet = undefined;
await reinit();
logger.info("wallet re-initialized after reset");
return wrapSuccessResponse({});
}
default: {
const wallet = await this.wp.promise;
return await wallet.handleCoreApiRequest(operation, id, args);
}
}
}
}
export function installNativeWalletListener(): void {
2022-11-11 20:52:45 +01:00
setGlobalLogLevelFromString("trace");
const handler = new NativeWalletMessageHandler();
const onMessage = async (msgStr: any): Promise<void> => {
if (typeof msgStr !== "string") {
logger.error("expected string as message");
return;
}
const msg = JSON.parse(msgStr);
const operation = msg.operation;
if (typeof operation !== "string") {
logger.error(
"message to native wallet helper must contain operation of type string",
);
return;
}
const id = msg.id;
logger.info(`native listener: got request for ${operation} (${id})`);
if (operation === "anastasisReduce") {
sendNativeMessage(respMsg);
} else {
let respMsg: CoreApiResponse;
try {
respMsg = await handler.handleMessage(operation, id, msg.args ?? {});
} catch (e) {
respMsg = {
type: "error",
id,
operation,
error: getErrorDetailFromException(e),
};
}
logger.info(
`native listener: sending back ${respMsg.type} message for operation ${operation} (${id})`,
);
sendNativeMessage(respMsg);
}
};
qjsOs.setMessageFromHostHandler((m) => onMessage(m));
logger.info("native wallet listener installed");
}
2022-11-10 14:24:02 +01:00
// @ts-ignore
globalThis.installNativeWalletListener = installNativeWalletListener;
2022-11-11 20:52:45 +01:00
export async function testWithGv() {
2023-02-15 23:32:42 +01:00
const w = await createNativeWalletHost2();
2022-11-11 20:52:45 +01:00
await w.wallet.client.call(WalletApiOperation.InitWallet, {});
await w.wallet.client.call(WalletApiOperation.RunIntegrationTest, {
amountToSpend: "KUDOS:1",
amountToWithdraw: "KUDOS:3",
bankAccessApiBaseUrl:
"https://bank.demo.taler.net/demobanks/default/access-api/",
2022-11-11 20:52:45 +01:00
exchangeBaseUrl: "https://exchange.demo.taler.net/",
merchantBaseUrl: "https://backend.demo.taler.net/",
});
await w.wallet.runTaskLoop({
stopWhenDone: true,
});
}
export async function testWithLocal() {
2023-01-03 10:51:05 +01:00
console.log("running local test");
2023-02-15 23:32:42 +01:00
const w = await createNativeWalletHost2({
2023-01-03 10:51:05 +01:00
persistentStoragePath: "walletdb.json",
});
console.log("created wallet");
2023-01-02 21:00:43 +01:00
await w.wallet.client.call(WalletApiOperation.InitWallet, {
skipDefaults: true,
});
2023-01-03 10:51:05 +01:00
console.log("initialized wallet");
await w.wallet.client.call(WalletApiOperation.RunIntegrationTest, {
amountToSpend: "TESTKUDOS:1",
amountToWithdraw: "TESTKUDOS:3",
bankAccessApiBaseUrl: "http://localhost:8082/taler-bank-access/",
exchangeBaseUrl: "http://localhost:8081/",
2022-12-08 15:01:59 +01:00
merchantBaseUrl: "http://localhost:8083/",
});
2023-01-03 10:51:05 +01:00
console.log("started integration test");
await w.wallet.runTaskLoop({
stopWhenDone: true,
});
2023-01-03 10:51:05 +01:00
console.log("done with task loop");
2022-12-08 15:01:59 +01:00
w.wallet.stop();
}
2022-11-11 20:52:45 +01:00
// @ts-ignore
globalThis.testWithGv = testWithGv;
// @ts-ignore
globalThis.testWithLocal = testWithLocal;