mirror of
https://github.com/TecharoHQ/anubis.git
synced 2026-04-09 18:18:49 +00:00
119 lines
3.0 KiB
TypeScript
119 lines
3.0 KiB
TypeScript
import { u } from "../../../lib/xeact";
|
|
import { simd } from "wasm-feature-detect";
|
|
// import { compile } from '@haribala/wasm2js';
|
|
|
|
type ProgressCallback = (nonce: number | string) => void;
|
|
|
|
interface ProcessOptions {
|
|
basePrefix: string;
|
|
version: string;
|
|
algorithm: string;
|
|
}
|
|
|
|
const getHardwareConcurrency = () =>
|
|
navigator.hardwareConcurrency !== undefined ? navigator.hardwareConcurrency : 1;
|
|
|
|
// https://stackoverflow.com/questions/47879864/how-can-i-check-if-a-browser-supports-webassembly
|
|
const isWASMSupported = (() => {
|
|
try {
|
|
if (typeof WebAssembly === "object"
|
|
&& typeof WebAssembly.instantiate === "function") {
|
|
const module = new WebAssembly.Module(Uint8Array.of(0x0, 0x61, 0x73, 0x6d, 0x01, 0x00, 0x00, 0x00));
|
|
if (module instanceof WebAssembly.Module)
|
|
return new WebAssembly.Instance(module) instanceof WebAssembly.Instance;
|
|
}
|
|
} catch (e) {
|
|
return false;
|
|
}
|
|
return false;
|
|
})();
|
|
|
|
export default function process(
|
|
options: ProcessOptions,
|
|
data: string,
|
|
difficulty: number = 5,
|
|
signal: AbortSignal | null = null,
|
|
progressCallback?: ProgressCallback,
|
|
threads: number = Math.trunc(Math.max(getHardwareConcurrency() / 2, 1)),
|
|
): Promise<string> {
|
|
const { basePrefix, version, algorithm } = options;
|
|
|
|
let worker = "wasm";
|
|
|
|
if (!isWASMSupported) {
|
|
worker = "wasm2js";
|
|
}
|
|
|
|
return new Promise(async (resolve, reject) => {
|
|
let wasmFeatures = "baseline";
|
|
|
|
if (await simd()) {
|
|
wasmFeatures = "simd128";
|
|
}
|
|
|
|
let module = undefined;
|
|
|
|
if (isWASMSupported) {
|
|
module = await fetch(u(`${basePrefix}/.within.website/x/cmd/anubis/static/wasm/${wasmFeatures}/${algorithm}.wasm?cacheBuster=${version}`))
|
|
.then(x => WebAssembly.compileStreaming(x));
|
|
}
|
|
|
|
const webWorkerURL = `${basePrefix}/.within.website/x/cmd/anubis/static/js/worker/${worker}.mjs?cacheBuster=${version}`;
|
|
|
|
const workers: Worker[] = [];
|
|
let settled = false;
|
|
|
|
const onAbort = () => {
|
|
console.log("PoW aborted");
|
|
cleanup();
|
|
reject(new DOMException("Aborted", "AbortError"));
|
|
};
|
|
|
|
const cleanup = () => {
|
|
if (settled) {
|
|
return;
|
|
}
|
|
settled = true;
|
|
workers.forEach((w) => w.terminate());
|
|
if (signal != null) {
|
|
signal.removeEventListener("abort", onAbort);
|
|
}
|
|
};
|
|
|
|
if (signal != null) {
|
|
if (signal.aborted) {
|
|
return onAbort();
|
|
}
|
|
signal.addEventListener("abort", onAbort, { once: true });
|
|
}
|
|
|
|
for (let i = 0; i < threads; i++) {
|
|
let worker = new Worker(webWorkerURL);
|
|
|
|
worker.onmessage = (event) => {
|
|
if (typeof event.data === "number") {
|
|
progressCallback?.(event.data);
|
|
} else {
|
|
cleanup();
|
|
resolve(event.data);
|
|
}
|
|
}
|
|
|
|
worker.onerror = (event) => {
|
|
cleanup();
|
|
reject(event);
|
|
}
|
|
|
|
worker.postMessage({
|
|
data,
|
|
difficulty,
|
|
nonce: i,
|
|
threads,
|
|
algorithm,
|
|
module,
|
|
});
|
|
}
|
|
});
|
|
};
|
|
|