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 MDN.`, 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...
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), }); } })();