2022-11-10 13:54:39 +01: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 {
|
|
|
|
AccessStats,
|
|
|
|
DefaultNodeWalletArgs,
|
|
|
|
getErrorDetailFromException,
|
2022-11-11 20:52:45 +01:00
|
|
|
Headers,
|
2022-11-10 13:54:39 +01:00
|
|
|
HttpRequestLibrary,
|
|
|
|
HttpRequestOptions,
|
|
|
|
HttpResponse,
|
|
|
|
openPromise,
|
|
|
|
openTalerDatabase,
|
|
|
|
SetTimeoutTimerAPI,
|
|
|
|
SynchronousCryptoWorkerFactoryPlain,
|
|
|
|
Wallet,
|
2022-11-11 20:52:45 +01:00
|
|
|
WalletApiOperation,
|
2022-11-10 13:54:39 +01:00
|
|
|
} from "@gnu-taler/taler-wallet-core";
|
|
|
|
|
|
|
|
import {
|
|
|
|
CoreApiEnvelope,
|
|
|
|
CoreApiResponse,
|
|
|
|
CoreApiResponseSuccess,
|
2022-11-11 20:52:45 +01:00
|
|
|
j2s,
|
2022-11-10 13:54:39 +01:00
|
|
|
Logger,
|
2022-11-11 20:52:45 +01:00
|
|
|
setGlobalLogLevelFromString,
|
2022-12-06 14:53:35 +01:00
|
|
|
setPRNG,
|
2022-11-10 13:54:39 +01:00
|
|
|
WalletNotification,
|
|
|
|
} from "@gnu-taler/taler-util";
|
|
|
|
import { BridgeIDBFactory } from "@gnu-taler/idb-bridge";
|
|
|
|
import { MemoryBackend } from "@gnu-taler/idb-bridge";
|
|
|
|
import { shimIndexedDB } from "@gnu-taler/idb-bridge";
|
|
|
|
import { IDBFactory } from "@gnu-taler/idb-bridge";
|
|
|
|
|
2022-11-11 20:52:45 +01:00
|
|
|
import * as _qjsOsImp from "os";
|
2023-01-03 10:51:05 +01:00
|
|
|
// @ts-ignore
|
|
|
|
import * as _qjsStdImp from "std";
|
2022-11-11 20:52:45 +01:00
|
|
|
|
|
|
|
const textDecoder = new TextDecoder();
|
|
|
|
const textEncoder = new TextEncoder();
|
|
|
|
|
2022-12-06 14:53:35 +01:00
|
|
|
setGlobalLogLevelFromString("trace");
|
|
|
|
|
|
|
|
setPRNG(function (x: Uint8Array, n: number) {
|
|
|
|
// @ts-ignore
|
2022-12-21 18:50:15 +01:00
|
|
|
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;
|
|
|
|
});
|
|
|
|
|
2022-11-11 20:52:45 +01:00
|
|
|
export interface QjsHttpResp {
|
|
|
|
status: number;
|
|
|
|
data: ArrayBuffer;
|
|
|
|
}
|
|
|
|
|
|
|
|
export interface QjsHttpOptions {
|
|
|
|
method: string;
|
|
|
|
debug?: boolean;
|
|
|
|
data?: ArrayBuffer;
|
|
|
|
headers?: string[];
|
|
|
|
}
|
|
|
|
|
|
|
|
export interface QjsOsLib {
|
|
|
|
// Not async!
|
|
|
|
fetchHttp(url: string, options?: QjsHttpOptions): QjsHttpResp;
|
2023-01-03 10:51:05 +01:00
|
|
|
postMessageToHost(s: string): void;
|
|
|
|
setMessageFromHostHandler(h: (s: string) => void): void;
|
|
|
|
rename(oldPath: string, newPath: string): number;
|
|
|
|
}
|
|
|
|
|
|
|
|
export interface QjsStdLib {
|
|
|
|
writeFile(filename: string, contents: string): void;
|
|
|
|
loadFile(filename: string): string;
|
2022-11-11 20:52:45 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// This is not the nodejs "os" module, but the qjs "os" module.
|
|
|
|
const qjsOs: QjsOsLib = _qjsOsImp as any;
|
|
|
|
|
2023-01-03 10:51:05 +01:00
|
|
|
const qjsStd: QjsStdLib = _qjsStdImp as any;
|
2022-11-10 13:54:39 +01:00
|
|
|
|
|
|
|
const logger = new Logger("taler-wallet-embedded/index.ts");
|
|
|
|
|
|
|
|
export class NativeHttpLib implements HttpRequestLibrary {
|
|
|
|
get(
|
|
|
|
url: string,
|
|
|
|
opt?: HttpRequestOptions | undefined,
|
|
|
|
): Promise<HttpResponse> {
|
2022-11-11 20:52:45 +01:00
|
|
|
return this.fetch(url, {
|
|
|
|
method: "GET",
|
|
|
|
...opt,
|
|
|
|
});
|
2022-11-10 13:54:39 +01:00
|
|
|
}
|
|
|
|
postJson(
|
|
|
|
url: string,
|
|
|
|
body: any,
|
|
|
|
opt?: HttpRequestOptions | undefined,
|
|
|
|
): Promise<HttpResponse> {
|
2022-11-11 20:52:45 +01:00
|
|
|
return this.fetch(url, {
|
|
|
|
method: "POST",
|
|
|
|
body,
|
|
|
|
...opt,
|
|
|
|
});
|
2022-11-10 13:54:39 +01:00
|
|
|
}
|
2022-11-11 20:52:45 +01:00
|
|
|
async fetch(
|
2022-11-10 13:54:39 +01:00
|
|
|
url: string,
|
|
|
|
opt?: HttpRequestOptions | undefined,
|
|
|
|
): Promise<HttpResponse> {
|
2022-11-11 20:52:45 +01:00
|
|
|
const method = opt?.method ?? "GET";
|
|
|
|
let data: ArrayBuffer | undefined = undefined;
|
|
|
|
let headers: string[] = [];
|
|
|
|
if (opt?.headers) {
|
|
|
|
for (let headerName of Object.keys(opt.headers)) {
|
|
|
|
headers.push(`${headerName}: ${opt.headers[headerName]}`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (method.toUpperCase() === "POST") {
|
|
|
|
if (opt?.body) {
|
|
|
|
if (typeof opt.body === "string") {
|
|
|
|
data = textEncoder.encode(opt.body).buffer;
|
|
|
|
} else if (ArrayBuffer.isView(opt.body)) {
|
|
|
|
data = opt.body.buffer;
|
|
|
|
} else if (opt.body instanceof ArrayBuffer) {
|
|
|
|
data = opt.body;
|
|
|
|
} else if (typeof opt.body === "object") {
|
|
|
|
data = textEncoder.encode(JSON.stringify(opt.body)).buffer;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
data = new ArrayBuffer(0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
const res = qjsOs.fetchHttp(url, {
|
|
|
|
method,
|
|
|
|
data,
|
|
|
|
headers,
|
|
|
|
});
|
|
|
|
return {
|
|
|
|
requestMethod: method,
|
|
|
|
headers: new Headers(),
|
|
|
|
async bytes() {
|
|
|
|
return res.data;
|
|
|
|
},
|
|
|
|
json() {
|
|
|
|
const text = textDecoder.decode(res.data);
|
|
|
|
return JSON.parse(text);
|
|
|
|
},
|
|
|
|
async text() {
|
|
|
|
const text = textDecoder.decode(res.data);
|
|
|
|
return text;
|
|
|
|
},
|
|
|
|
requestUrl: url,
|
|
|
|
status: res.status,
|
|
|
|
};
|
2022-11-10 13:54:39 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function sendNativeMessage(ev: CoreApiEnvelope): void {
|
|
|
|
const m = JSON.stringify(ev);
|
2023-01-03 10:51:05 +01:00
|
|
|
qjsOs.postMessageToHost(m);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Generate a random alphanumeric ID. Does *not* use cryptographically
|
|
|
|
* secure randomness.
|
|
|
|
*/
|
|
|
|
function makeId(length: number): string {
|
|
|
|
let result = "";
|
|
|
|
const characters =
|
|
|
|
"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789";
|
|
|
|
for (let i = 0; i < length; i++) {
|
|
|
|
result += characters.charAt(Math.floor(Math.random() * characters.length));
|
|
|
|
}
|
|
|
|
return result;
|
2022-11-10 13:54:39 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
export async function getWallet(args: DefaultNodeWalletArgs = {}): Promise<{
|
|
|
|
wallet: Wallet;
|
|
|
|
getDbStats: () => AccessStats;
|
|
|
|
}> {
|
|
|
|
BridgeIDBFactory.enableTracing = false;
|
|
|
|
const myBackend = new MemoryBackend();
|
|
|
|
myBackend.enableTracing = false;
|
|
|
|
|
|
|
|
const storagePath = args.persistentStoragePath;
|
|
|
|
if (storagePath) {
|
2023-01-03 10:51:05 +01:00
|
|
|
const dbContentStr = qjsStd.loadFile(storagePath);
|
|
|
|
if (dbContentStr != null) {
|
|
|
|
const dbContent = JSON.parse(dbContentStr);
|
|
|
|
myBackend.importDump(dbContent);
|
|
|
|
}
|
2022-11-10 13:54:39 +01:00
|
|
|
|
|
|
|
myBackend.afterCommitCallback = async () => {
|
2023-01-03 10:51:05 +01:00
|
|
|
logger.trace("committing database");
|
|
|
|
// Allow caller to stop persisting the wallet.
|
|
|
|
if (args.persistentStoragePath === undefined) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const tmpPath = `${args.persistentStoragePath}-${makeId(5)}.tmp`;
|
|
|
|
const dbContent = myBackend.exportDump();
|
|
|
|
qjsStd.writeFile(tmpPath, JSON.stringify(dbContent, undefined, 2));
|
|
|
|
// Atomically move the temporary file onto the DB path.
|
|
|
|
const res = qjsOs.rename(tmpPath, args.persistentStoragePath);
|
|
|
|
if (res != 0) {
|
|
|
|
throw Error("db commit failed at rename");
|
|
|
|
}
|
|
|
|
logger.trace("committing database done");
|
2022-11-10 13:54:39 +01:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2023-01-03 10:51:05 +01:00
|
|
|
console.log("done processing storage path");
|
|
|
|
|
2022-11-10 13:54:39 +01:00
|
|
|
BridgeIDBFactory.enableTracing = false;
|
|
|
|
|
|
|
|
const myBridgeIdbFactory = new BridgeIDBFactory(myBackend);
|
|
|
|
const myIdbFactory: IDBFactory = myBridgeIdbFactory as any as IDBFactory;
|
|
|
|
|
|
|
|
let myHttpLib;
|
|
|
|
if (args.httpLib) {
|
|
|
|
myHttpLib = args.httpLib;
|
|
|
|
} else {
|
|
|
|
myHttpLib = new NativeHttpLib();
|
|
|
|
}
|
|
|
|
|
|
|
|
const myVersionChange = (): Promise<void> => {
|
|
|
|
logger.error("version change requested, should not happen");
|
|
|
|
throw Error(
|
|
|
|
"BUG: wallet DB version change event can't happen with memory IDB",
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
shimIndexedDB(myBridgeIdbFactory);
|
|
|
|
|
|
|
|
const myDb = await openTalerDatabase(myIdbFactory, myVersionChange);
|
|
|
|
|
|
|
|
let workerFactory;
|
|
|
|
workerFactory = new SynchronousCryptoWorkerFactoryPlain();
|
|
|
|
|
|
|
|
const timer = new SetTimeoutTimerAPI();
|
|
|
|
|
|
|
|
const w = await Wallet.create(myDb, myHttpLib, timer, workerFactory);
|
|
|
|
|
|
|
|
if (args.notifyHandler) {
|
|
|
|
w.addNotificationListener(args.notifyHandler);
|
|
|
|
}
|
|
|
|
return {
|
|
|
|
wallet: w,
|
|
|
|
getDbStats: () => myBackend.accessStats,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
class NativeWalletMessageHandler {
|
|
|
|
walletArgs: DefaultNodeWalletArgs | undefined;
|
|
|
|
maybeWallet: Wallet | undefined;
|
|
|
|
wp = openPromise<Wallet>();
|
|
|
|
httpLib = new NativeHttpLib();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Handle a request from the native wallet.
|
|
|
|
*/
|
|
|
|
async handleMessage(
|
|
|
|
operation: string,
|
|
|
|
id: string,
|
|
|
|
args: any,
|
|
|
|
): Promise<CoreApiResponse> {
|
|
|
|
const wrapResponse = (result: unknown): CoreApiResponseSuccess => {
|
|
|
|
return {
|
|
|
|
type: "response",
|
|
|
|
id,
|
|
|
|
operation,
|
|
|
|
result,
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
let initResponse: any = {};
|
|
|
|
|
|
|
|
const reinit = async () => {
|
|
|
|
logger.info("in reinit");
|
|
|
|
const wR = await getWallet(this.walletArgs);
|
|
|
|
const w = wR.wallet;
|
|
|
|
this.maybeWallet = w;
|
|
|
|
const resp = await w.handleCoreApiRequest(
|
|
|
|
"initWallet",
|
|
|
|
"native-init",
|
2023-01-03 10:51:05 +01:00
|
|
|
{...this.walletArgs},
|
2022-11-10 13:54:39 +01:00
|
|
|
);
|
|
|
|
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.walletArgs = {
|
2023-01-03 10:51:05 +01:00
|
|
|
...args,
|
2022-11-10 13:54:39 +01:00
|
|
|
notifyHandler: async (notification: WalletNotification) => {
|
|
|
|
sendNativeMessage({ type: "notification", payload: notification });
|
|
|
|
},
|
|
|
|
persistentStoragePath: args.persistentStoragePath,
|
|
|
|
httpLib: this.httpLib,
|
|
|
|
cryptoWorkerType: args.cryptoWorkerType,
|
|
|
|
};
|
|
|
|
await reinit();
|
|
|
|
return wrapResponse({
|
|
|
|
...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": {
|
|
|
|
const oldArgs = this.walletArgs;
|
|
|
|
this.walletArgs = { ...oldArgs };
|
|
|
|
if (oldArgs && oldArgs.persistentStoragePath) {
|
|
|
|
try {
|
|
|
|
logger.error("FIXME: reset not implemented");
|
|
|
|
// fs.unlinkSync(oldArgs.persistentStoragePath);
|
|
|
|
} catch (e) {
|
|
|
|
logger.error("Error while deleting the wallet db:", e);
|
|
|
|
}
|
|
|
|
// Prevent further storage!
|
|
|
|
this.walletArgs.persistentStoragePath = undefined;
|
|
|
|
}
|
|
|
|
const wallet = await this.wp.promise;
|
|
|
|
wallet.stop();
|
|
|
|
this.wp = openPromise<Wallet>();
|
|
|
|
this.maybeWallet = undefined;
|
|
|
|
await reinit();
|
|
|
|
return wrapResponse({});
|
|
|
|
}
|
|
|
|
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");
|
2022-11-10 13:54:39 +01:00
|
|
|
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})`);
|
|
|
|
|
|
|
|
try {
|
2023-01-03 10:51:05 +01:00
|
|
|
const respMsg = await handler.handleMessage(operation, id, msg.payload ?? {});
|
2022-11-10 13:54:39 +01:00
|
|
|
logger.info(
|
|
|
|
`native listener: sending success response for ${operation} (${id})`,
|
|
|
|
);
|
|
|
|
sendNativeMessage(respMsg);
|
|
|
|
} catch (e) {
|
|
|
|
const respMsg: CoreApiResponse = {
|
|
|
|
type: "error",
|
|
|
|
id,
|
|
|
|
operation,
|
|
|
|
error: getErrorDetailFromException(e),
|
|
|
|
};
|
|
|
|
sendNativeMessage(respMsg);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2023-01-03 10:51:05 +01:00
|
|
|
qjsOs.setMessageFromHostHandler((m) => onMessage(m))
|
2022-11-10 13:54:39 +01:00
|
|
|
|
|
|
|
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
|
|
|
|
|
|
|
// @ts-ignore
|
|
|
|
globalThis.makeWallet = getWallet;
|
|
|
|
|
|
|
|
export async function testWithGv() {
|
|
|
|
const w = await getWallet();
|
|
|
|
await w.wallet.client.call(WalletApiOperation.InitWallet, {});
|
|
|
|
await w.wallet.client.call(WalletApiOperation.RunIntegrationTest, {
|
|
|
|
amountToSpend: "KUDOS:1",
|
|
|
|
amountToWithdraw: "KUDOS:3",
|
|
|
|
bankBaseUrl: "https://bank.demo.taler.net/demobanks/default/access-api/",
|
|
|
|
exchangeBaseUrl: "https://exchange.demo.taler.net/",
|
|
|
|
merchantBaseUrl: "https://backend.demo.taler.net/",
|
|
|
|
});
|
|
|
|
await w.wallet.runTaskLoop({
|
|
|
|
stopWhenDone: true,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-12-07 15:36:56 +01:00
|
|
|
export async function testWithLocal() {
|
2023-01-03 10:51:05 +01:00
|
|
|
console.log("running local test");
|
|
|
|
const w = await getWallet({
|
|
|
|
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");
|
2022-12-07 15:36:56 +01:00
|
|
|
await w.wallet.client.call(WalletApiOperation.RunIntegrationTest, {
|
|
|
|
amountToSpend: "TESTKUDOS:1",
|
|
|
|
amountToWithdraw: "TESTKUDOS:3",
|
|
|
|
bankBaseUrl: "http://localhost:8082/",
|
|
|
|
bankAccessApiBaseUrl: "http://localhost:8082/taler-bank-access/",
|
|
|
|
exchangeBaseUrl: "http://localhost:8081/",
|
2022-12-08 15:01:59 +01:00
|
|
|
merchantBaseUrl: "http://localhost:8083/",
|
2022-12-07 15:36:56 +01:00
|
|
|
});
|
2023-01-03 10:51:05 +01:00
|
|
|
console.log("started integration test");
|
2022-12-07 15:36:56 +01:00
|
|
|
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-12-07 15:36:56 +01:00
|
|
|
}
|
|
|
|
|
2022-11-11 20:52:45 +01:00
|
|
|
// @ts-ignore
|
|
|
|
globalThis.testWithGv = testWithGv;
|
2022-12-07 15:36:56 +01:00
|
|
|
// @ts-ignore
|
|
|
|
globalThis.testWithLocal = testWithLocal;
|