Add new Anubis mascot (#204)

* Add new Anubis mascot

Signed-off-by: Xe Iaso <me@xeiaso.net>

* web: add artist credit to footer

Signed-off-by: Xe Iaso <me@xeiaso.net>

---------

Signed-off-by: Xe Iaso <me@xeiaso.net>
This commit is contained in:
Xe Iaso 2025-04-03 07:54:24 -04:00 committed by GitHub
parent 21a9d77788
commit b14aa6a0c3
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
13 changed files with 27 additions and 26 deletions

View File

@ -11,6 +11,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
## [Unreleased] ## [Unreleased]
- The placeholder Anubis mascot has been replaced with a design by [CELPHASE](https://bsky.app/profile/celphase.bsky.social).
- Allow iMessage's link preview fetcher through Anubis by default. - Allow iMessage's link preview fetcher through Anubis by default.
- Added a periodic cleanup routine for the decaymap that removes expired entries, ensuring stale data is properly pruned. - Added a periodic cleanup routine for the decaymap that removes expired entries, ensuring stale data is properly pruned.
- Added a no-store Cache-Control header to the challenge page - Added a no-store Cache-Control header to the challenge page

View File

@ -76,7 +76,7 @@ const config: Config = {
title: 'Anubis', title: 'Anubis',
logo: { logo: {
alt: 'A happy jackal woman with brown hair and red eyes', alt: 'A happy jackal woman with brown hair and red eyes',
src: 'img/happy.webp', src: 'img/favicon.webp',
}, },
items: [ items: [
{ {

Binary file not shown.

Before

Width:  |  Height:  |  Size: 222 KiB

After

Width:  |  Height:  |  Size: 106 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 15 KiB

After

Width:  |  Height:  |  Size: 15 KiB

BIN
docs/static/img/favicon.webp vendored Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 5.9 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 58 KiB

After

Width:  |  Height:  |  Size: 30 KiB

View File

@ -66,6 +66,7 @@ templ base(title string, body templ.Component) {
href="https://techaro.lol" href="https://techaro.lol"
>Techaro</a>. Made with ❤️ in 🇨🇦. >Techaro</a>. Made with ❤️ in 🇨🇦.
</p> </p>
<p>Mascot design by <a href="https://bsky.app/profile/celphase.bsky.social">CELPHASE</a>.</p>
</center> </center>
</footer> </footer>
</main> </main>
@ -114,7 +115,7 @@ templ errorPage(message string) {
<img <img
id="image" id="image"
style="width:100%;max-width:256px;" style="width:100%;max-width:256px;"
src={ "/.within.website/x/cmd/anubis/static/img/sad.webp?cacheBuster=" + anubis.Version } src={ "/.within.website/x/cmd/anubis/static/img/reject.webp?cacheBuster=" + anubis.Version }
/> />
<p>{ message }.</p> <p>{ message }.</p>
<button onClick="window.location.reload();">Try again</button> <button onClick="window.location.reload();">Try again</button>
@ -137,8 +138,7 @@ templ bench() {
<th style="width:4rem">Iters B</th> <th style="width:4rem">Iters B</th>
</tr> </tr>
</thead> </thead>
<tbody id="results" style="padding-top:0.25rem;display:grid;grid-template-columns:subgrid;grid-auto-rows:min-content;grid-column:1/-1;row-gap:0.25rem;overflow-y:auto;font-variant-numeric:tabular-nums"> <tbody id="results" style="padding-top:0.25rem;display:grid;grid-template-columns:subgrid;grid-auto-rows:min-content;grid-column:1/-1;row-gap:0.25rem;overflow-y:auto;font-variant-numeric:tabular-nums"></tbody>
</tbody>
</table> </table>
<div class="centered-div"> <div class="centered-div">
<img <img
@ -167,7 +167,7 @@ templ bench() {
<div style="margin-top:0.25rem;display:flex;justify-content:end"> <div style="margin-top:0.25rem;display:flex;justify-content:end">
<label for="compare-select" style="margin-right:0.5rem">Compare:</label> <label for="compare-select" style="margin-right:0.5rem">Compare:</label>
<select id="compare-select" name="compare"> <select id="compare-select" name="compare">
<option value="NONE">-</option> <option value="NONE">-</option>
</select> </select>
</div> </div>
</form> </form>

14
web/index_templ.go generated
View File

@ -89,7 +89,7 @@ func base(title string, body templ.Component) templ.Component {
if templ_7745c5c3_Err != nil { if templ_7745c5c3_Err != nil {
return templ_7745c5c3_Err return templ_7745c5c3_Err
} }
templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 6, "<footer><center><p>Protected by <a href=\"https://github.com/TecharoHQ/anubis\">Anubis</a> from <a href=\"https://techaro.lol\">Techaro</a>. Made with ❤️ in 🇨🇦.</p></center></footer></main></body></html>") templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 6, "<footer><center><p>Protected by <a href=\"https://github.com/TecharoHQ/anubis\">Anubis</a> from <a href=\"https://techaro.lol\">Techaro</a>. Made with ❤️ in 🇨🇦.</p><p>Mascot design by <a href=\"https://bsky.app/profile/celphase.bsky.social\">CELPHASE</a>.</p></center></footer></main></body></html>")
if templ_7745c5c3_Err != nil { if templ_7745c5c3_Err != nil {
return templ_7745c5c3_Err return templ_7745c5c3_Err
} }
@ -126,7 +126,7 @@ func index() templ.Component {
templ_7745c5c3_Var6, templ_7745c5c3_Err = templ.JoinStringErrs("/.within.website/x/cmd/anubis/static/img/pensive.webp?cacheBuster=" + templ_7745c5c3_Var6, templ_7745c5c3_Err = templ.JoinStringErrs("/.within.website/x/cmd/anubis/static/img/pensive.webp?cacheBuster=" +
anubis.Version) anubis.Version)
if templ_7745c5c3_Err != nil { if templ_7745c5c3_Err != nil {
return templ.Error{Err: templ_7745c5c3_Err, FileName: `index.templ`, Line: 82, Col: 18} return templ.Error{Err: templ_7745c5c3_Err, FileName: `index.templ`, Line: 83, Col: 18}
} }
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var6)) _, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var6))
if templ_7745c5c3_Err != nil { if templ_7745c5c3_Err != nil {
@ -140,7 +140,7 @@ func index() templ.Component {
templ_7745c5c3_Var7, templ_7745c5c3_Err = templ.JoinStringErrs("/.within.website/x/cmd/anubis/static/img/happy.webp?cacheBuster=" + templ_7745c5c3_Var7, templ_7745c5c3_Err = templ.JoinStringErrs("/.within.website/x/cmd/anubis/static/img/happy.webp?cacheBuster=" +
anubis.Version) anubis.Version)
if templ_7745c5c3_Err != nil { if templ_7745c5c3_Err != nil {
return templ.Error{Err: templ_7745c5c3_Err, FileName: `index.templ`, Line: 88, Col: 18} return templ.Error{Err: templ_7745c5c3_Err, FileName: `index.templ`, Line: 89, Col: 18}
} }
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var7)) _, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var7))
if templ_7745c5c3_Err != nil { if templ_7745c5c3_Err != nil {
@ -153,7 +153,7 @@ func index() templ.Component {
var templ_7745c5c3_Var8 string var templ_7745c5c3_Var8 string
templ_7745c5c3_Var8, templ_7745c5c3_Err = templ.JoinStringErrs("/.within.website/x/cmd/anubis/static/js/main.mjs?cacheBuster=" + anubis.Version) templ_7745c5c3_Var8, templ_7745c5c3_Err = templ.JoinStringErrs("/.within.website/x/cmd/anubis/static/js/main.mjs?cacheBuster=" + anubis.Version)
if templ_7745c5c3_Err != nil { if templ_7745c5c3_Err != nil {
return templ.Error{Err: templ_7745c5c3_Err, FileName: `index.templ`, Line: 91, Col: 116} return templ.Error{Err: templ_7745c5c3_Err, FileName: `index.templ`, Line: 92, Col: 116}
} }
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var8)) _, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var8))
if templ_7745c5c3_Err != nil { if templ_7745c5c3_Err != nil {
@ -193,9 +193,9 @@ func errorPage(message string) templ.Component {
return templ_7745c5c3_Err return templ_7745c5c3_Err
} }
var templ_7745c5c3_Var10 string var templ_7745c5c3_Var10 string
templ_7745c5c3_Var10, templ_7745c5c3_Err = templ.JoinStringErrs("/.within.website/x/cmd/anubis/static/img/sad.webp?cacheBuster=" + anubis.Version) templ_7745c5c3_Var10, templ_7745c5c3_Err = templ.JoinStringErrs("/.within.website/x/cmd/anubis/static/img/reject.webp?cacheBuster=" + anubis.Version)
if templ_7745c5c3_Err != nil { if templ_7745c5c3_Err != nil {
return templ.Error{Err: templ_7745c5c3_Err, FileName: `index.templ`, Line: 117, Col: 90} return templ.Error{Err: templ_7745c5c3_Err, FileName: `index.templ`, Line: 118, Col: 93}
} }
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var10)) _, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var10))
if templ_7745c5c3_Err != nil { if templ_7745c5c3_Err != nil {
@ -208,7 +208,7 @@ func errorPage(message string) templ.Component {
var templ_7745c5c3_Var11 string var templ_7745c5c3_Var11 string
templ_7745c5c3_Var11, templ_7745c5c3_Err = templ.JoinStringErrs(message) templ_7745c5c3_Var11, templ_7745c5c3_Err = templ.JoinStringErrs(message)
if templ_7745c5c3_Err != nil { if templ_7745c5c3_Err != nil {
return templ.Error{Err: templ_7745c5c3_Err, FileName: `index.templ`, Line: 119, Col: 14} return templ.Error{Err: templ_7745c5c3_Err, FileName: `index.templ`, Line: 120, Col: 14}
} }
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var11)) _, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var11))
if templ_7745c5c3_Err != nil { if templ_7745c5c3_Err != nil {

View File

@ -91,7 +91,7 @@ function showContinueBar(hash, nonce, t0, t1) {
} }
}); });
} }
const ohNoes = ({ titleMsg, statusMsg, imageSrc }) => { const ohNoes = ({ titleMsg, statusMsg, imageSrc }) => {
title.innerHTML = titleMsg; title.innerHTML = titleMsg;
status.innerHTML = statusMsg; status.innerHTML = statusMsg;
@ -103,7 +103,7 @@ function showContinueBar(hash, nonce, t0, t1) {
ohNoes({ ohNoes({
titleMsg: "Your context is not secure!", 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>.`, 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), imageSrc: imageURL("reject", anubisVersion),
}); });
return; return;
} }
@ -116,7 +116,7 @@ function showContinueBar(hash, nonce, t0, t1) {
// if (!videoWorks) { // if (!videoWorks) {
// title.innerHTML = "Oh no!"; // title.innerHTML = "Oh no!";
// status.innerHTML = "Checks failed. Please check your browser's settings and try again."; // status.innerHTML = "Checks failed. Please check your browser's settings and try again.";
// image.src = imageURL("sad"); // image.src = imageURL("reject");
// progress.style.display = "none"; // progress.style.display = "none";
// return; // return;
// } // }
@ -128,7 +128,7 @@ function showContinueBar(hash, nonce, t0, t1) {
ohNoes({ ohNoes({
titleMsg: `Missing feature ${name}`, titleMsg: `Missing feature ${name}`,
statusMsg: msg, statusMsg: msg,
imageSrc: imageURL("sad", anubisVersion), imageSrc: imageURL("reject", anubisVersion),
}); });
} }
} }
@ -142,7 +142,7 @@ function showContinueBar(hash, nonce, t0, t1) {
ohNoes({ ohNoes({
titleMsg: "Internal error!", titleMsg: "Internal error!",
statusMsg: `Failed to fetch challenge config: ${err.message}`, statusMsg: `Failed to fetch challenge config: ${err.message}`,
imageSrc: imageURL("sad", anubisVersion), imageSrc: imageURL("reject", anubisVersion),
}); });
throw err; throw err;
}); });
@ -152,14 +152,14 @@ function showContinueBar(hash, nonce, t0, t1) {
ohNoes({ ohNoes({
titleMsg: "Challenge error!", titleMsg: "Challenge error!",
statusMsg: `Failed to resolve check algorithm. You may want to reload the page.`, statusMsg: `Failed to resolve check algorithm. You may want to reload the page.`,
imageSrc: imageURL("sad", anubisVersion), imageSrc: imageURL("reject", anubisVersion),
}); });
return; return;
} }
status.innerHTML = `Calculating...<br/>Difficulty: ${rules.report_as}, `; status.innerHTML = `Calculating...<br/>Difficulty: ${rules.report_as}, `;
progress.style.display = "inline-block"; progress.style.display = "inline-block";
// the whole text, including "Speed:", as a single node, because some browsers // the whole text, including "Speed:", as a single node, because some browsers
// (Firefox mobile) present screen readers with each node as a separate piece // (Firefox mobile) present screen readers with each node as a separate piece
// of text. // of text.
@ -188,7 +188,7 @@ function showContinueBar(hash, nonce, t0, t1) {
// apply a polynomial ease-out function to move faster in the beginning // apply a polynomial ease-out function to move faster in the beginning
// and then slow down as things get increasingly unlikely. quadratic felt // and then slow down as things get increasingly unlikely. quadratic felt
// the best in testing, but this may need adjustment in the future. // the best in testing, but this may need adjustment in the future.
const probability = Math.pow(1 - likelihood, iters); const probability = Math.pow(1 - likelihood, iters);
const distance = (1 - Math.pow(probability, 2)) * 100; const distance = (1 - Math.pow(probability, 2)) * 100;
progress["aria-valuenow"] = distance; progress["aria-valuenow"] = distance;
@ -215,7 +215,7 @@ function showContinueBar(hash, nonce, t0, t1) {
if (userReadDetails) { if (userReadDetails) {
const container = document.getElementById("progress"); const container = document.getElementById("progress");
// Style progress bar as a continue button // Style progress bar as a continue button
container.style.display = "flex"; container.style.display = "flex";
container.style.alignItems = "center"; container.style.alignItems = "center";
@ -231,7 +231,7 @@ function showContinueBar(hash, nonce, t0, t1) {
container.style.width = "min(20rem, 90%)"; container.style.width = "min(20rem, 90%)";
container.style.margin = "1rem auto 2rem"; container.style.margin = "1rem auto 2rem";
container.innerHTML = "I've finished reading, continue →"; container.innerHTML = "I've finished reading, continue →";
function onDetailsExpand() { function onDetailsExpand() {
const redir = window.location.href; const redir = window.location.href;
window.location.replace( window.location.replace(
@ -243,10 +243,10 @@ function showContinueBar(hash, nonce, t0, t1) {
}), }),
); );
} }
container.onclick = onDetailsExpand; container.onclick = onDetailsExpand;
setTimeout(onDetailsExpand, 30000); setTimeout(onDetailsExpand, 30000);
} else { } else {
setTimeout(() => { setTimeout(() => {
const redir = window.location.href; const redir = window.location.href;
@ -260,12 +260,12 @@ function showContinueBar(hash, nonce, t0, t1) {
); );
}, 250); }, 250);
} }
} catch (err) { } catch (err) {
ohNoes({ ohNoes({
titleMsg: "Calculation error!", titleMsg: "Calculation error!",
statusMsg: `Failed to calculate challenge: ${err.message}`, statusMsg: `Failed to calculate challenge: ${err.message}`,
imageSrc: imageURL("sad", anubisVersion), imageSrc: imageURL("reject", anubisVersion),
}); });
} }
})(); })();

Binary file not shown.

Before

Width:  |  Height:  |  Size: 58 KiB

After

Width:  |  Height:  |  Size: 30 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 48 KiB

After

Width:  |  Height:  |  Size: 28 KiB

BIN
web/static/img/reject.webp Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 26 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 50 KiB