diff --git a/src/crypto/nodeWorker.ts b/src/crypto/nodeWorker.ts new file mode 100644 index 000000000..afa2930a8 --- /dev/null +++ b/src/crypto/nodeWorker.ts @@ -0,0 +1,73 @@ +/* + This file is part of TALER + (C) 2016 GNUnet e.V. + + 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. + + 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 + TALER; see the file COPYING. If not, see + */ + +const path = require("path"); +const fork = require("child_process").fork; + +const nodeWorkerEntry = path.join(__dirname, "nodeWorkerEntry.js"); + +export class Worker { + child: any; + onmessage: undefined | ((m: any) => void); + onerror: undefined | ((m: any) => void); + constructor(scriptFilename: string) { + this.child = fork(nodeWorkerEntry); + this.onerror = undefined; + this.onmessage = undefined; + + this.child.on("error", (e: any) => { + if (this.onerror) { + this.onerror(e); + } + }); + + this.child.on("message", (msg: any) => { + const message = JSON.parse(msg); + + if (!message.error && this.onmessage) { + this.onmessage(message); + } + + if (message.error && this.onerror) { + const error = new Error(message.error); + error.stack = message.stack; + + this.onerror(error); + } + }); + + this.child.send({scriptFilename,cwd: process.cwd()}); + } + + addEventListener(event: "message" | "error", fn: (x: any) => void): void { + switch (event) { + case "message": + this.onmessage = fn; + break; + case "error": + this.onerror = fn; + break; + } + } + + postMessage (msg: any) { + this.child.send(JSON.stringify({data: msg})); + } + + terminate () { + this.child.kill("SIGINT"); + } +} diff --git a/src/crypto/nodeWorkerEntry.ts b/src/crypto/nodeWorkerEntry.ts new file mode 100644 index 000000000..aa6f57599 --- /dev/null +++ b/src/crypto/nodeWorkerEntry.ts @@ -0,0 +1,79 @@ +/* + This file is part of TALER + (C) 2016 GNUnet e.V. + + 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. + + 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 + TALER; see the file COPYING. If not, see + */ + + +const fs = require("fs"); +const vm = require("vm"); + +process.once("message", (obj: any) => { + const g: any = global as any; + + (g as any).self = { + close: () => { + process.exit(0); + }, + postMessage: (msg: any) => { + const str: string = JSON.stringify({data: msg}); + if (process.send) { + process.send(str); + } + }, + onmessage: undefined, + onerror: (err: any) => { + const str: string = JSON.stringify({error: err.message, stack: err.stack}); + if (process.send) { + process.send(str); + } + }, + addEventListener: (event: "error" | "message", fn: (x: any) => void) => { + if (event == "error") { + g.onerror = fn; + } else if (event == "message") { + g.onmessage = fn; + } + }, + }; + + g.__dirname = obj.cwd; + g.__filename = __filename; + //g.require = require; + //g.module = module; + //g.exports = module.exports; + + g.importScripts = (...files: string[]) => { + if (files.length > 0) { + vm.createScript(files.map(file => fs.readFileSync(file, "utf8")).join("\n")).runInThisContext(); + } + }; + + Object.keys(g.self).forEach(key => { + g[key] = g.self[key]; + }); + + process.on("message", (msg: any) => { + try { + (g.onmessage || g.self.onmessage || (() => {}))(JSON.parse(msg)); + } catch (err) { + (g.onerror || g.self.onerror || (() => {}))(err); + } + }); + + process.on("error", (err: any) => { + (g.onerror || g.self.onerror || (() => {}))(err); + }); + + require(obj.scriptFilename); +}); diff --git a/src/crypto/startWorker.js b/src/crypto/startWorker.js new file mode 100644 index 000000000..a64152c64 --- /dev/null +++ b/src/crypto/startWorker.js @@ -0,0 +1,41 @@ +/* + This file is part of TALER + (C) 2017 Inria and GNUnet e.V. + + 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. + + 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 + TALER; see the file COPYING. If not, see + */ + + +// @ts-nocheck + + +/** + * Start a crypto worker, using different worker + * mechanisms depending on the environment. + * + * @returns {Worker} + */ +export function startWorker() { + let workerCtor; + let workerPath; + if (typeof Worker !== "undefined") { + // we're in the browser + workerCtor = Worker; + workerPath = "/dist/cryptoWorker-bundle.js"; + } else if (typeof "require" !== "undefined") { + workerCtor = require("./nodeWorker").Worker; + workerPath = __dirname + "/cryptoWorker.js"; + } else { + throw Error("Can't create worker, unknown environment"); + } + return new workerCtor(workerPath); +} diff --git a/src/timer.ts b/src/timer.ts new file mode 100644 index 000000000..105fc8c7a --- /dev/null +++ b/src/timer.ts @@ -0,0 +1,77 @@ +/* + This file is part of TALER + (C) 2017 GNUnet e.V. + + 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. + + 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 + TALER; see the file COPYING. If not, see + */ + +/** + * Cross-platform timers. + * + * NodeJS and the browser use slightly different timer API, + * this abstracts over these differences. + */ + +/** + * Cancelable timer. + */ +export interface TimerHandle { + clear(): void; +} + +class IntervalHandle { + constructor(public h: any) { + } + + clear() { + clearTimeout(this.h); + } +} + +class TimeoutHandle { + constructor(public h: any) { + } + + clear() { + clearTimeout(this.h); + } +} + +/** + * Get a performance counter in milliseconds. + */ +export let performanceNow = (() => { + if (typeof "process" !== "undefined") { + return () => { + const t = process.hrtime(); + return t[0] * 1e9 + t[1]; + } + } else if (typeof "performance" !== "undefined") { + return () => performance.now(); + } else { + return () => 0; + } +})(); + +/** + * Call a function every time the delay given in milliseconds passes. + */ +export function every(delayMs: number, callback: () => void): TimerHandle { + return new IntervalHandle(setInterval(callback, delayMs)); +} + +/** + * Call a function after the delay given in milliseconds passes. + */ +export function after(delayMs: number, callback: () => void): TimerHandle { + return new TimeoutHandle(setInterval(callback, delayMs)); +}