wallet-core/src/headless/helpers.ts

252 lines
6.7 KiB
TypeScript
Raw Normal View History

2019-08-17 01:54:01 +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/>
*/
/**
* Helpers to create headless wallets.
*/
/**
* Imports.
*/
2019-12-05 19:38:19 +01:00
import { Wallet } from "../wallet";
2019-08-17 01:54:01 +02:00
import { MemoryBackend, BridgeIDBFactory, shimIndexedDB } from "idb-bridge";
import { openDatabase } from "../db";
2019-12-09 13:29:11 +01:00
import Axios, { AxiosPromise, AxiosResponse } from "axios";
import {
HttpRequestLibrary,
HttpRequestOptions,
2019-12-09 19:59:08 +01:00
Headers,
2019-12-09 13:29:11 +01:00
} from "../util/http";
import * as amounts from "../util/amounts";
2019-08-17 01:54:01 +02:00
import { Bank } from "./bank";
import fs = require("fs");
import { Logger } from "../util/logging";
2019-12-05 19:38:19 +01:00
import { NodeThreadCryptoWorkerFactory } from "../crypto/workers/nodeThreadWorker";
import { SynchronousCryptoWorkerFactory } from "../crypto/workers/synchronousWorker";
2019-12-09 13:29:11 +01:00
import { RequestThrottler } from "../util/RequestThrottler";
import { WalletNotification, NotificationType } from "../types/notifications";
2019-11-30 00:36:20 +01:00
const logger = new Logger("helpers.ts");
2019-08-17 01:54:01 +02:00
export class NodeHttpLib implements HttpRequestLibrary {
2019-12-09 13:29:11 +01:00
private throttle = new RequestThrottler();
private async req(
method: "post" | "get",
url: string,
body: any,
opt?: HttpRequestOptions,
) {
if (this.throttle.applyThrottle(url)) {
throw Error("request throttled");
}
let resp: AxiosResponse;
2019-08-28 02:49:27 +02:00
try {
2019-12-09 13:29:11 +01:00
resp = await Axios({
method,
2019-08-28 02:49:27 +02:00
url: url,
2019-12-09 13:29:11 +01:00
responseType: "text",
headers: opt?.headers,
validateStatus: () => true,
transformResponse: (x) => x,
data: body,
2019-08-28 02:49:27 +02:00
});
} catch (e) {
throw e;
}
2019-12-09 13:29:11 +01:00
const respText = resp.data;
if (typeof respText !== "string") {
throw Error("unexpected response type");
}
const makeJson = async () => {
let responseJson;
try {
responseJson = JSON.parse(respText);
} catch (e) {
throw Error("Invalid JSON from HTTP response");
}
if (responseJson === null || typeof responseJson !== "object") {
throw Error("Invalid JSON from HTTP response");
}
return responseJson;
};
2019-12-09 19:59:08 +01:00
const headers = new Headers();
for (const hn of Object.keys(resp.headers)) {
headers.set(hn, resp.headers[hn]);
}
2019-12-09 13:29:11 +01:00
return {
2019-12-09 19:59:08 +01:00
headers,
2019-12-09 13:29:11 +01:00
status: resp.status,
text: async () => resp.data,
json: makeJson,
};
}
async get(
url: string,
opt?: HttpRequestOptions,
): Promise<import("../util/http").HttpResponse> {
return this.req("get", url, undefined, opt);
2019-08-17 01:54:01 +02:00
}
async postJson(
url: string,
body: any,
2019-12-09 13:29:11 +01:00
opt?: HttpRequestOptions,
): Promise<import("../util/http").HttpResponse> {
2019-12-09 13:29:11 +01:00
return this.req("post", url, body, opt);
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.
*/
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;
2019-08-17 01:54:01 +02:00
}
/**
* Get a wallet instance with default settings for node.
*/
export async function getDefaultNodeWallet(
args: DefaultNodeWalletArgs = {},
): Promise<Wallet> {
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 {
2019-12-09 13:29:11 +01:00
const dbContentStr: string = fs.readFileSync(storagePath, {
encoding: "utf-8",
});
2019-08-17 01:54:01 +02:00
const dbContent = JSON.parse(dbContentStr);
myBackend.importDump(dbContent);
} catch (e) {
2019-11-21 23:09:43 +01:00
console.error("could not read wallet file");
2019-08-17 01:54:01 +02:00
}
myBackend.afterCommitCallback = async () => {
2019-08-20 23:36:56 +02:00
// Allow caller to stop persisting the wallet.
if (args.persistentStoragePath === undefined) {
return;
}
2019-08-17 01:54:01 +02:00
const dbContent = myBackend.exportDump();
2019-12-09 13:29:11 +01:00
fs.writeFileSync(storagePath, JSON.stringify(dbContent, undefined, 2), {
encoding: "utf-8",
});
2019-08-17 01:54:01 +02:00
};
}
BridgeIDBFactory.enableTracing = false;
const myBridgeIdbFactory = new BridgeIDBFactory(myBackend);
const myIdbFactory: IDBFactory = (myBridgeIdbFactory as any) as IDBFactory;
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
const myVersionChange = () => {
console.error("version change requested, should not happen");
throw Error();
};
const myUnsupportedUpgrade = () => {
console.error("unsupported database migration");
throw Error();
};
shimIndexedDB(myBridgeIdbFactory);
const myDb = await openDatabase(
2019-08-17 01:54:01 +02:00
myIdbFactory,
myVersionChange,
myUnsupportedUpgrade,
);
2019-12-05 19:38:19 +01:00
//const worker = new SynchronousCryptoWorkerFactory();
//const worker = new NodeCryptoWorkerFactory();
2019-12-05 19:38:19 +01:00
const worker = new NodeThreadCryptoWorkerFactory();
2019-12-09 13:29:11 +01:00
const w = new Wallet(myDb, myHttpLib, worker);
if (args.notifyHandler) {
w.addNotificationListener(args.notifyHandler);
}
return w;
2019-08-17 01:54:01 +02:00
}
export async function withdrawTestBalance(
myWallet: Wallet,
2019-08-18 15:08:10 +02:00
amount: string = "TESTKUDOS:10",
2019-08-17 01:54:01 +02:00
bankBaseUrl: string = "https://bank.test.taler.net/",
exchangeBaseUrl: string = "https://exchange.test.taler.net/",
) {
const reserveResponse = await myWallet.createReserve({
2019-08-17 18:54:09 +02:00
amount: amounts.parseOrThrow(amount),
2019-08-17 01:54:01 +02:00
exchange: exchangeBaseUrl,
2019-08-28 02:49:27 +02:00
exchangeWire: "payto://unknown",
2019-08-17 01:54:01 +02:00
});
2019-11-21 23:09:43 +01:00
const reservePub = reserveResponse.reservePub;
2019-08-17 01:54:01 +02:00
const bank = new Bank(bankBaseUrl);
const bankUser = await bank.registerRandomUser();
2019-12-09 13:29:11 +01:00
logger.trace(`Registered bank user ${JSON.stringify(bankUser)}`);
2019-08-17 01:54:01 +02:00
2019-12-09 13:29:11 +01:00
const exchangePaytoUri = await myWallet.getExchangePaytoUri(exchangeBaseUrl, [
"x-taler-bank",
]);
2019-08-17 01:54:01 +02:00
2019-12-05 19:38:19 +01:00
const donePromise = new Promise((resolve, reject) => {
2019-12-09 13:29:11 +01:00
myWallet.addNotificationListener(n => {
if (
n.type === NotificationType.ReserveDepleted &&
n.reservePub === reservePub
) {
2019-12-05 19:38:19 +01:00
resolve();
}
});
});
2019-12-09 13:29:11 +01:00
await bank.createReserve(bankUser, amount, reservePub, exchangePaytoUri);
2019-08-17 01:54:01 +02:00
await myWallet.confirmReserve({ reservePub: reserveResponse.reservePub });
2019-12-05 19:38:19 +01:00
await donePromise;
2019-08-17 01:54:01 +02:00
}