Some checks failed
Docker image builds / build (push) Waiting to run
Asset Build Verification / asset_verification (push) Has been cancelled
Docs deploy / build (push) Has been cancelled
Go Mod Tidy Check / go_mod_tidy_check (push) Has been cancelled
Go / go_tests (push) Has been cancelled
Package builds (unstable) / package_builds (push) Has been cancelled
Smoke tests / smoke-test (default-config-macro) (push) Has been cancelled
Smoke tests / smoke-test (docker-registry) (push) Has been cancelled
Smoke tests / smoke-test (double_slash) (push) Has been cancelled
Smoke tests / smoke-test (forced-language) (push) Has been cancelled
Smoke tests / smoke-test (git-clone) (push) Has been cancelled
Smoke tests / smoke-test (git-push) (push) Has been cancelled
Smoke tests / smoke-test (healthcheck) (push) Has been cancelled
Smoke tests / smoke-test (i18n) (push) Has been cancelled
Smoke tests / smoke-test (log-file) (push) Has been cancelled
Smoke tests / smoke-test (nginx) (push) Has been cancelled
Smoke tests / smoke-test (palemoon/amd64) (push) Has been cancelled
Smoke tests / smoke-test (robots_txt) (push) Has been cancelled
Check Spelling / Check Spelling (push) Has been cancelled
SSH CI / ssh (aarch64-16k) (push) Has been cancelled
SSH CI / ssh (aarch64-4k) (push) Has been cancelled
SSH CI / ssh (ppc64le) (push) Has been cancelled
SSH CI / ssh (riscv64) (push) Has been cancelled
zizmor / zizmor latest via PyPI (push) Has been cancelled
95 lines
2.3 KiB
TypeScript
95 lines
2.3 KiB
TypeScript
type ProgressCallback = (nonce: number) => void;
|
||
|
||
interface ProcessOptions {
|
||
basePrefix: string;
|
||
version: string;
|
||
}
|
||
|
||
const getHardwareConcurrency = () =>
|
||
navigator.hardwareConcurrency !== undefined
|
||
? navigator.hardwareConcurrency
|
||
: 1;
|
||
|
||
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> {
|
||
console.debug("fast algo");
|
||
|
||
// Choose worker based on secure context.
|
||
// Use the WebCrypto worker if the page is a secure context; otherwise fall back to pure‑JS.
|
||
let workerMethod: "webcrypto" | "purejs" = "purejs";
|
||
if (window.isSecureContext) {
|
||
workerMethod = "webcrypto";
|
||
}
|
||
|
||
if (
|
||
navigator.userAgent.includes("Firefox") ||
|
||
navigator.userAgent.includes("Goanna")
|
||
) {
|
||
console.log("Firefox detected, using pure-JS fallback");
|
||
workerMethod = "purejs";
|
||
}
|
||
|
||
return new Promise((resolve, reject) => {
|
||
let webWorkerURL = `${options.basePrefix}/.within.website/x/cmd/nuke/static/js/worker/sha256-${workerMethod}.mjs?cacheBuster=${options.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,
|
||
});
|
||
|
||
workers.push(worker);
|
||
}
|
||
});
|
||
}
|