* cmd/anubis: add a debug option for benchmarking hashrate Having the ability to benchmark different proof-of-work implementations is useful for extending Anubis. This adds a flag `--debug-benchmark-js` (and its associated environment variable `DEBUG_BENCHMARK_JS`) for serving a tool to do so. Internally, a there is a new policy action, "DEBUG_BENCHMARK", which serves the benchmarking tool instead of a challenge. The flag then replaces all bot rules with a special rule matching every request to that action. The benchmark page makes heavy use of inline styles, because currently all global styles are shared across all pages. This could be fixed, but I wanted to avoid major changes to the templates. * web/js: add signal for aborting an active proof-of-work algorithm Both proof-of-work algorithms now take an optional `AbortSignal`, which immediately terminates all workers and returns `false` if aborted before the challenge is complete. * web/js: add algorithm comparison to the benchmark page "Compare:" is added to the benchmark page for testing the relative performance between two algorithms. Since benchmark runs generally have high variance, it may take a while for the averages to converge on a stable difference. --------- Signed-off-by: Xe Iaso <me@xeiaso.net> Co-authored-by: Xe Iaso <me@xeiaso.net>
187 lines
No EOL
5.8 KiB
JavaScript
187 lines
No EOL
5.8 KiB
JavaScript
import processFast from "./proof-of-work.mjs";
|
|
import processSlow from "./proof-of-work-slow.mjs";
|
|
import { testVideo } from "./video.mjs";
|
|
|
|
const algorithms = {
|
|
"fast": processFast,
|
|
"slow": processSlow,
|
|
};
|
|
|
|
// from Xeact
|
|
const u = (url = "", params = {}) => {
|
|
let result = new URL(url, window.location.href);
|
|
Object.entries(params).forEach((kv) => {
|
|
let [k, v] = kv;
|
|
result.searchParams.set(k, v);
|
|
});
|
|
return result.toString();
|
|
};
|
|
|
|
const imageURL = (mood, cacheBuster) =>
|
|
u(`/.within.website/x/cmd/anubis/static/img/${mood}.webp`, { cacheBuster });
|
|
|
|
const dependencies = [
|
|
{
|
|
name: "WebCrypto",
|
|
msg: "Your browser doesn't have a functioning web.crypto element. Are you viewing this over a secure context?",
|
|
value: window.crypto,
|
|
},
|
|
{
|
|
name: "Web Workers",
|
|
msg: "Your browser doesn't support web workers (Anubis uses this to avoid freezing your browser). Do you have a plugin like JShelter installed?",
|
|
value: window.Worker,
|
|
},
|
|
];
|
|
|
|
(async () => {
|
|
const status = document.getElementById('status');
|
|
const image = document.getElementById('image');
|
|
const title = document.getElementById('title');
|
|
const progress = document.getElementById('progress');
|
|
const anubisVersion = JSON.parse(document.getElementById('anubis_version').textContent);
|
|
|
|
const ohNoes = ({
|
|
titleMsg, statusMsg, imageSrc,
|
|
}) => {
|
|
title.innerHTML = titleMsg;
|
|
status.innerHTML = statusMsg;
|
|
image.src = imageSrc;
|
|
progress.style.display = "none";
|
|
};
|
|
|
|
if (!window.isSecureContext) {
|
|
ohNoes({
|
|
titleMsg: "Your context is not secure!",
|
|
statusMsg: `Try connecting over HTTPS or let the admin know to set up HTTPS. For more information, see <a href="https://developer.mozilla.org/en-US/docs/Web/Security/Secure_Contexts#when_is_a_context_considered_secure">MDN</a>.`,
|
|
imageSrc: imageURL("sad", anubisVersion),
|
|
});
|
|
return;
|
|
}
|
|
|
|
// const testarea = document.getElementById('testarea');
|
|
|
|
// const videoWorks = await testVideo(testarea);
|
|
// console.log(`videoWorks: ${videoWorks}`);
|
|
|
|
// if (!videoWorks) {
|
|
// title.innerHTML = "Oh no!";
|
|
// status.innerHTML = "Checks failed. Please check your browser's settings and try again.";
|
|
// image.src = imageURL("sad");
|
|
// progress.style.display = "none";
|
|
// return;
|
|
// }
|
|
|
|
status.innerHTML = 'Calculating...';
|
|
|
|
for (const val of dependencies) {
|
|
const { value, name, msg } = val;
|
|
if (!value) {
|
|
ohNoes({
|
|
titleMsg: `Missing feature ${name}`,
|
|
statusMsg: msg,
|
|
imageSrc: imageURL("sad", anubisVersion),
|
|
})
|
|
}
|
|
}
|
|
|
|
const { challenge, rules } = await fetch("/.within.website/x/cmd/anubis/api/make-challenge", { method: "POST" })
|
|
.then(r => {
|
|
if (!r.ok) {
|
|
throw new Error("Failed to fetch config");
|
|
}
|
|
return r.json();
|
|
})
|
|
.catch(err => {
|
|
ohNoes({
|
|
titleMsg: "Internal error!",
|
|
statusMsg: `Failed to fetch challenge config: ${err.message}`,
|
|
imageSrc: imageURL("sad", anubisVersion),
|
|
});
|
|
throw err;
|
|
});
|
|
|
|
const process = algorithms[rules.algorithm];
|
|
if (!process) {
|
|
ohNoes({
|
|
titleMsg: "Challenge error!",
|
|
statusMsg: `Failed to resolve check algorithm. You may want to reload the page.`,
|
|
imageSrc: imageURL("sad", anubisVersion),
|
|
});
|
|
return;
|
|
}
|
|
|
|
status.innerHTML = `Calculating...<br/>Difficulty: ${rules.report_as}, `;
|
|
progress.style.display = "inline-block";
|
|
|
|
// the whole text, including "Speed:", as a single node, because some browsers
|
|
// (Firefox mobile) present screen readers with each node as a separate piece
|
|
// of text.
|
|
const rateText = document.createTextNode("Speed: 0kH/s");
|
|
status.appendChild(rateText);
|
|
|
|
let lastSpeedUpdate = 0;
|
|
let showingApology = false;
|
|
const likelihood = Math.pow(16, -rules.report_as);
|
|
try {
|
|
const t0 = Date.now();
|
|
const { hash, nonce } = await process(
|
|
challenge,
|
|
rules.difficulty,
|
|
null,
|
|
(iters) => {
|
|
const delta = Date.now() - t0;
|
|
// only update the speed every second so it's less visually distracting
|
|
if (delta - lastSpeedUpdate > 1000) {
|
|
lastSpeedUpdate = delta;
|
|
rateText.data = `Speed: ${(iters / delta).toFixed(3)}kH/s`;
|
|
}
|
|
|
|
// the probability of still being on the page is (1 - likelihood) ^ iters.
|
|
// by definition, half of the time the progress bar only gets to half, so
|
|
// apply a polynomial ease-out function to move faster in the beginning
|
|
// and then slow down as things get increasingly unlikely. quadratic felt
|
|
// the best in testing, but this may need adjustment in the future.
|
|
const probability = Math.pow(1 - likelihood, iters);
|
|
const distance = (1 - Math.pow(probability, 2)) * 100;
|
|
progress["aria-valuenow"] = distance;
|
|
progress.firstElementChild.style.width = `${distance}%`;
|
|
|
|
if (probability < 0.1 && !showingApology) {
|
|
status.append(
|
|
document.createElement("br"),
|
|
document.createTextNode(
|
|
"Verification is taking longer than expected. Please do not refresh the page.",
|
|
),
|
|
);
|
|
showingApology = true;
|
|
}
|
|
},
|
|
);
|
|
const t1 = Date.now();
|
|
console.log({ hash, nonce });
|
|
|
|
title.innerHTML = "Success!";
|
|
status.innerHTML = `Done! Took ${t1 - t0}ms, ${nonce} iterations`;
|
|
image.src = imageURL("happy", anubisVersion);
|
|
progress.style.display = "none";
|
|
|
|
setTimeout(() => {
|
|
const redir = window.location.href;
|
|
|
|
window.location.replace(
|
|
u("/.within.website/x/cmd/anubis/api/pass-challenge", {
|
|
response: hash,
|
|
nonce,
|
|
redir,
|
|
elapsedTime: t1 - t0
|
|
}),
|
|
);
|
|
}, 250);
|
|
} catch (err) {
|
|
ohNoes({
|
|
titleMsg: "Calculation error!",
|
|
statusMsg: `Failed to calculate challenge: ${err.message}`,
|
|
imageSrc: imageURL("sad", anubisVersion),
|
|
});
|
|
}
|
|
})(); |