2019-08-17 01:54:01 +02:00
|
|
|
/*
|
|
|
|
This file is part of GNU Taler
|
2019-12-15 17:48:22 +01:00
|
|
|
(C) 2019 Taler Systems S.A.
|
2019-08-17 01:54:01 +02:00
|
|
|
|
|
|
|
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/>
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Helpers to create headless wallets.
|
2019-12-15 17:48:22 +01:00
|
|
|
* @author Florian Dold <dold@taler.net>
|
2019-08-17 01:54:01 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Imports.
|
|
|
|
*/
|
2022-01-16 21:47:43 +01:00
|
|
|
import type { IDBFactory } from "@gnu-taler/idb-bridge";
|
|
|
|
// eslint-disable-next-line no-duplicate-imports
|
2021-03-17 17:56:37 +01:00
|
|
|
import {
|
2022-03-23 21:24:23 +01:00
|
|
|
BridgeIDBFactory,
|
|
|
|
MemoryBackend,
|
|
|
|
shimIndexedDB,
|
2021-03-17 17:56:37 +01:00
|
|
|
} from "@gnu-taler/idb-bridge";
|
2022-01-16 21:47:43 +01:00
|
|
|
import { AccessStats } from "@gnu-taler/idb-bridge/src/MemoryBackend";
|
|
|
|
import { Logger, WalletNotification } from "@gnu-taler/taler-util";
|
|
|
|
import * as fs from "fs";
|
|
|
|
import { NodeThreadCryptoWorkerFactory } from "../crypto/workers/nodeThreadWorker.js";
|
|
|
|
import { SynchronousCryptoWorkerFactory } from "../crypto/workers/synchronousWorkerFactory.js";
|
2021-08-19 13:48:36 +02:00
|
|
|
import { openTalerDatabase } from "../db-utils.js";
|
2021-06-17 17:40:42 +02:00
|
|
|
import { HttpRequestLibrary } from "../util/http.js";
|
2022-04-11 20:10:16 +02:00
|
|
|
import { SetTimeoutTimerAPI } from "../util/timer.js";
|
2021-06-17 21:06:45 +02:00
|
|
|
import { Wallet } from "../wallet.js";
|
2022-01-16 21:47:43 +01:00
|
|
|
import { NodeHttpLib } from "./NodeHttpLib.js";
|
2019-11-30 00:36:20 +01:00
|
|
|
|
2020-08-01 10:22:08 +02:00
|
|
|
const logger = new Logger("headless/helpers.ts");
|
2019-08-17 01:54:01 +02:00
|
|
|
|
2019-08-20 23:36:56 +02:00
|
|
|
export interface DefaultNodeWalletArgs {
|
2019-08-17 01:54:01 +02:00
|
|
|
/**
|
|
|
|
* Location of the wallet database.
|
|
|
|
*
|
|
|
|
* If not specified, the wallet starts out with an empty database and
|
|
|
|
* the wallet database is stored only in memory.
|
|
|
|
*/
|
|
|
|
persistentStoragePath?: string;
|
2019-08-19 20:24:29 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Handler for asynchronous notifications from the wallet.
|
|
|
|
*/
|
2019-12-06 02:52:16 +01:00
|
|
|
notifyHandler?: (n: WalletNotification) => void;
|
2019-08-22 23:36:36 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* If specified, use this as HTTP request library instead
|
|
|
|
* of the default one.
|
|
|
|
*/
|
|
|
|
httpLib?: HttpRequestLibrary;
|
2022-10-05 15:45:10 +02:00
|
|
|
|
|
|
|
cryptoWorkerType?: "sync" | "node-worker-thread";
|
2019-08-17 01:54:01 +02:00
|
|
|
}
|
|
|
|
|
2020-09-06 12:56:27 +02:00
|
|
|
/**
|
|
|
|
* Generate a random alphanumeric ID. Does *not* use cryptographically
|
|
|
|
* secure randomness.
|
|
|
|
*/
|
|
|
|
function makeId(length: number): string {
|
|
|
|
let result = "";
|
2020-12-14 16:45:15 +01:00
|
|
|
const characters = "abcdefghijklmnopqrstuvwxyz0123456789";
|
2020-09-06 12:56:27 +02:00
|
|
|
for (let i = 0; i < length; i++) {
|
|
|
|
result += characters.charAt(Math.floor(Math.random() * characters.length));
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2019-08-17 01:54:01 +02:00
|
|
|
/**
|
|
|
|
* Get a wallet instance with default settings for node.
|
|
|
|
*/
|
|
|
|
export async function getDefaultNodeWallet(
|
|
|
|
args: DefaultNodeWalletArgs = {},
|
2021-06-17 21:06:45 +02:00
|
|
|
): Promise<Wallet> {
|
2022-01-11 21:00:12 +01:00
|
|
|
const res = await getDefaultNodeWallet2(args);
|
|
|
|
return res.wallet;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get a wallet instance with default settings for node.
|
|
|
|
*
|
|
|
|
* Extended version that allows getting DB stats.
|
|
|
|
*/
|
|
|
|
export async function getDefaultNodeWallet2(
|
|
|
|
args: DefaultNodeWalletArgs = {},
|
|
|
|
): Promise<{
|
|
|
|
wallet: Wallet;
|
|
|
|
getDbStats: () => AccessStats;
|
|
|
|
}> {
|
2019-09-01 22:59:48 +02:00
|
|
|
BridgeIDBFactory.enableTracing = false;
|
2019-08-17 01:54:01 +02:00
|
|
|
const myBackend = new MemoryBackend();
|
2019-09-01 22:59:48 +02:00
|
|
|
myBackend.enableTracing = false;
|
2019-08-17 01:54:01 +02:00
|
|
|
|
|
|
|
const storagePath = args.persistentStoragePath;
|
|
|
|
if (storagePath) {
|
|
|
|
try {
|
2021-08-20 13:01:35 +02:00
|
|
|
const dbContentStr: string = fs.readFileSync(storagePath, {
|
2019-12-09 13:29:11 +01:00
|
|
|
encoding: "utf-8",
|
|
|
|
});
|
2019-08-17 01:54:01 +02:00
|
|
|
const dbContent = JSON.parse(dbContentStr);
|
|
|
|
myBackend.importDump(dbContent);
|
2021-10-07 12:01:40 +02:00
|
|
|
} catch (e: any) {
|
2020-09-06 12:56:27 +02:00
|
|
|
const code: string = e.code;
|
|
|
|
if (code === "ENOENT") {
|
|
|
|
logger.trace("wallet file doesn't exist yet");
|
|
|
|
} else {
|
|
|
|
logger.error("could not open wallet database file");
|
|
|
|
throw e;
|
|
|
|
}
|
2019-08-17 01:54:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
myBackend.afterCommitCallback = async () => {
|
2021-08-06 16:27:18 +02:00
|
|
|
logger.trace("committing database");
|
2019-08-20 23:36:56 +02:00
|
|
|
// Allow caller to stop persisting the wallet.
|
|
|
|
if (args.persistentStoragePath === undefined) {
|
|
|
|
return;
|
|
|
|
}
|
2020-09-06 12:56:27 +02:00
|
|
|
const tmpPath = `${args.persistentStoragePath}-${makeId(5)}.tmp`;
|
2019-08-17 01:54:01 +02:00
|
|
|
const dbContent = myBackend.exportDump();
|
2021-08-20 13:01:35 +02:00
|
|
|
fs.writeFileSync(tmpPath, JSON.stringify(dbContent, undefined, 2), {
|
|
|
|
encoding: "utf-8",
|
|
|
|
});
|
2020-09-06 12:56:27 +02:00
|
|
|
// Atomically move the temporary file onto the DB path.
|
2021-08-20 13:01:35 +02:00
|
|
|
fs.renameSync(tmpPath, args.persistentStoragePath);
|
2022-08-24 11:11:02 +02:00
|
|
|
logger.trace("committing database done");
|
2019-08-17 01:54:01 +02:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
BridgeIDBFactory.enableTracing = false;
|
|
|
|
|
|
|
|
const myBridgeIdbFactory = new BridgeIDBFactory(myBackend);
|
2022-01-11 21:00:12 +01:00
|
|
|
const myIdbFactory: IDBFactory = myBridgeIdbFactory as any as IDBFactory;
|
2019-08-17 01:54:01 +02:00
|
|
|
|
2019-08-22 23:36:36 +02:00
|
|
|
let myHttpLib;
|
|
|
|
if (args.httpLib) {
|
|
|
|
myHttpLib = args.httpLib;
|
|
|
|
} else {
|
|
|
|
myHttpLib = new NodeHttpLib();
|
|
|
|
}
|
2019-08-17 01:54:01 +02:00
|
|
|
|
2020-04-07 10:07:32 +02:00
|
|
|
const myVersionChange = (): Promise<void> => {
|
2020-08-14 12:23:50 +02:00
|
|
|
logger.error("version change requested, should not happen");
|
2020-12-14 16:45:15 +01:00
|
|
|
throw Error(
|
|
|
|
"BUG: wallet DB version change event can't happen with memory IDB",
|
|
|
|
);
|
2019-08-17 01:54:01 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
shimIndexedDB(myBridgeIdbFactory);
|
|
|
|
|
2019-12-19 13:48:37 +01:00
|
|
|
const myDb = await openTalerDatabase(myIdbFactory, myVersionChange);
|
2019-08-17 01:54:01 +02:00
|
|
|
|
2020-02-24 18:08:07 +01:00
|
|
|
let workerFactory;
|
2022-10-05 15:45:10 +02:00
|
|
|
const cryptoWorkerType = args.cryptoWorkerType ?? "node-worker-thread";
|
|
|
|
if (cryptoWorkerType === "sync") {
|
2021-11-16 17:20:36 +01:00
|
|
|
logger.info("using synchronous crypto worker");
|
2020-02-24 18:08:07 +01:00
|
|
|
workerFactory = new SynchronousCryptoWorkerFactory();
|
2022-10-05 15:45:10 +02:00
|
|
|
} else if (cryptoWorkerType === "node-worker-thread") {
|
2021-11-16 17:20:36 +01:00
|
|
|
try {
|
|
|
|
// Try if we have worker threads available, fails in older node versions.
|
|
|
|
const _r = "require";
|
2022-01-16 21:47:43 +01:00
|
|
|
// eslint-disable-next-line no-unused-vars
|
2021-11-16 17:20:36 +01:00
|
|
|
const worker_threads = module[_r]("worker_threads");
|
|
|
|
// require("worker_threads");
|
|
|
|
workerFactory = new NodeThreadCryptoWorkerFactory();
|
|
|
|
} catch (e) {
|
|
|
|
logger.warn(
|
|
|
|
"worker threads not available, falling back to synchronous workers",
|
|
|
|
);
|
|
|
|
workerFactory = new SynchronousCryptoWorkerFactory();
|
|
|
|
}
|
2022-10-05 15:45:10 +02:00
|
|
|
} else {
|
|
|
|
throw Error(`unsupported crypto worker type '${cryptoWorkerType}'`);
|
2020-02-24 18:08:07 +01:00
|
|
|
}
|
2022-04-11 20:10:16 +02:00
|
|
|
|
2022-08-24 11:11:02 +02:00
|
|
|
const timer = new SetTimeoutTimerAPI();
|
2022-04-11 20:10:16 +02:00
|
|
|
|
|
|
|
const w = await Wallet.create(myDb, myHttpLib, timer, workerFactory);
|
2021-06-15 18:52:43 +02:00
|
|
|
|
2019-12-06 02:52:16 +01:00
|
|
|
if (args.notifyHandler) {
|
|
|
|
w.addNotificationListener(args.notifyHandler);
|
|
|
|
}
|
2022-01-11 21:00:12 +01:00
|
|
|
return {
|
|
|
|
wallet: w,
|
|
|
|
getDbStats: () => myBackend.accessStats,
|
|
|
|
};
|
2019-08-17 01:54:01 +02:00
|
|
|
}
|