wallet-core/packages/taler-wallet-core/src/host-impl.node.ts

160 lines
5.0 KiB
TypeScript
Raw Normal View History

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.
*/
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";
import { AccessStats } from "@gnu-taler/idb-bridge";
2023-02-15 23:32:42 +01:00
import { Logger } from "@gnu-taler/taler-util";
import * as fs from "fs";
2023-02-15 23:32:42 +01:00
import { NodeThreadCryptoWorkerFactory } from "./crypto/workers/nodeThreadWorker.js";
import { SynchronousCryptoWorkerFactoryPlain } from "./crypto/workers/synchronousWorkerFactoryPlain.js";
import { openTalerDatabase } from "./index.js";
import {
createPlatformHttpLib,
} from "@gnu-taler/taler-util/http";
import { SetTimeoutTimerAPI } from "./util/timer.js";
import { Wallet } from "./wallet.js";
import { DefaultNodeWalletArgs, makeTempfileId } from "./host-common.js";
2019-08-17 01:54:01 +02:00
2023-02-15 23:32:42 +01:00
const logger = new Logger("host-impl.node.ts");
2020-09-06 12:56:27 +02:00
2022-01-11 21:00:12 +01:00
/**
* Get a wallet instance with default settings for node.
*
* Extended version that allows getting DB stats.
*/
2023-02-15 23:32:42 +01:00
export async function createNativeWalletHost2(
2022-01-11 21:00:12 +01:00
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 {
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);
} 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;
}
2023-02-15 23:32:42 +01:00
const tmpPath = `${args.persistentStoragePath}-${makeTempfileId(5)}.tmp`;
logger.trace("exported DB dump");
2019-08-17 01:54:01 +02:00
const dbContent = myBackend.exportDump();
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.
fs.renameSync(tmpPath, args.persistentStoragePath);
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 {
2023-02-15 23:32:42 +01:00
myHttpLib = createPlatformHttpLib({
enableThrottling: true,
});
2019-08-22 23:36:36 +02:00
}
2019-08-17 01:54:01 +02:00
2020-04-07 10:07:32 +02:00
const myVersionChange = (): Promise<void> => {
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
let workerFactory;
const cryptoWorkerType = args.cryptoWorkerType ?? "node-worker-thread";
if (cryptoWorkerType === "sync") {
logger.info("using synchronous crypto worker");
2023-02-15 23:32:42 +01:00
workerFactory = new SynchronousCryptoWorkerFactoryPlain();
} else if (cryptoWorkerType === "node-worker-thread") {
try {
// Try if we have worker threads available, fails in older node versions.
const _r = "require";
// eslint-disable-next-line no-unused-vars
const worker_threads = module[_r]("worker_threads");
// require("worker_threads");
workerFactory = new NodeThreadCryptoWorkerFactory();
logger.info("using node thread crypto worker");
} catch (e) {
logger.warn(
"worker threads not available, falling back to synchronous workers",
);
2023-02-15 23:32:42 +01:00
workerFactory = new SynchronousCryptoWorkerFactoryPlain();
}
} else {
throw Error(`unsupported crypto worker type '${cryptoWorkerType}'`);
}
const timer = new SetTimeoutTimerAPI();
const w = await Wallet.create(myDb, myHttpLib, timer, workerFactory);
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
}