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 { openTalerDb } from "../db";
|
2019-12-09 13:29:11 +01:00
|
|
|
import Axios, { AxiosPromise, AxiosResponse } from "axios";
|
|
|
|
import {
|
|
|
|
HttpRequestLibrary,
|
|
|
|
HttpRequestOptions,
|
|
|
|
} from "../util/http";
|
2019-12-02 00:42:40 +01:00
|
|
|
import * as amounts from "../util/amounts";
|
2019-08-17 01:54:01 +02:00
|
|
|
import { Bank } from "./bank";
|
|
|
|
|
|
|
|
import fs = require("fs");
|
2019-12-02 00:42:40 +01:00
|
|
|
import { Logger } from "../util/logging";
|
2019-12-05 19:38:19 +01:00
|
|
|
import { NodeThreadCryptoWorkerFactory } from "../crypto/workers/nodeThreadWorker";
|
2019-12-06 02:52:16 +01:00
|
|
|
import { NotificationType, WalletNotification } from "../walletTypes";
|
|
|
|
import { SynchronousCryptoWorkerFactory } from "../crypto/workers/synchronousWorker";
|
2019-12-09 13:29:11 +01:00
|
|
|
import { RequestThrottler } from "../util/RequestThrottler";
|
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;
|
|
|
|
};
|
|
|
|
return {
|
|
|
|
headers: resp.headers,
|
|
|
|
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,
|
2019-12-02 00:42:40 +01:00
|
|
|
): 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.
|
|
|
|
*/
|
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;
|
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 openTalerDb(
|
|
|
|
myIdbFactory,
|
|
|
|
myVersionChange,
|
|
|
|
myUnsupportedUpgrade,
|
|
|
|
);
|
|
|
|
|
2019-12-05 19:38:19 +01:00
|
|
|
//const worker = new SynchronousCryptoWorkerFactory();
|
2019-11-28 00:46:34 +01:00
|
|
|
//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);
|
2019-12-06 02:52:16 +01:00
|
|
|
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
|
|
|
}
|