"
+author = { name = "Bad Manners", url = "https://badmanners.xyz", email = "me@badmanners.xyz>" }
source = "https://git.badmanners.xyz/badmanners/badmanners.xyz/src/branch/main/src/components/icons/brands"
license = { name = "CC0 1.0 Universal", url = "https://creativecommons.org/publicdomain/zero/1.0/" }
items = [
"Cohost",
"Eka's Portal",
"Fur Affinity",
+ "GitGud",
"Inkbunny",
"Itaku",
"Keyoxide",
@@ -97,7 +98,7 @@ All third-party copyrights and trademarks belong to their respective owners, \
and I'm not affiliated with any of them."""
[[attributions]]
-title = "Font Awesome"
+author = "Font Awesome"
description = "Generic icons."
type = "icons"
source = "https://fontawesome.com"
diff --git a/src/layouts/BaseLayout.astro b/src/layouts/BaseLayout.astro
index 4240fa3..9c592d2 100644
--- a/src/layouts/BaseLayout.astro
+++ b/src/layouts/BaseLayout.astro
@@ -73,7 +73,7 @@ const title = pageTitle ? `${pageTitle} | Bad Manners` : "Bad Manners";
>
- Toggle dark mode
+ Toggle dark mode
diff --git a/src/pages/[...config].ts b/src/pages/[...config].ts
deleted file mode 100644
index a8481cc..0000000
--- a/src/pages/[...config].ts
+++ /dev/null
@@ -1,54 +0,0 @@
-import type { APIRoute, GetStaticPaths } from "astro";
-import { APACHE_CONFIG } from "astro:env/server";
-
-const htaccess = String.raw`
-ErrorDocument 404 /404.html
-RedirectMatch 301 ^/tos(/(index.html)?)?$ /terms_of_service/
-RedirectMatch 301 ^/contact(/(index.html)?)?$ /
-RedirectMatch 301 ^/@/(aryion|ekas?(portal)?)\b https://aryion.com/g4/user/BadManners
-RedirectMatch 301 ^/@/(blue[_-]?sky|bsky)\b https://bsky.app/profile/badmanners.xyz
-RedirectMatch 301 ^/@/(buymeacoffee|buy_me_a_coffee|buy-me-a-coffee|bmac)\b https://www.buymeacoffee.com/BadMannersXYZ
-RedirectMatch 301 ^/@/carrd\b https://badmanners.carrd.co
-RedirectMatch 301 ^/@/code[_-]?berg\b https://codeberg.org/BadManners
-RedirectMatch 301 ^/@/co[_-]?host\b https://cohost.org/BadManners
-RedirectMatch 301 ^/@/discord\b /#discord
-RedirectMatch 301 ^/@/e[_-]?mail\b /#e-mail
-RedirectMatch 301 ^/@/(fur[_-]?affinity|fa)\b https://www.furaffinity.net/user/badmanners
-RedirectMatch 301 ^/@/(gallery|stor(y|ies)|games?)\b https://gallery.badmanners.xyz
-RedirectMatch 301 ^/@/(git[_-]?hub|gh)\b https://github.com/BadMannersXYZ
-RedirectMatch 301 ^/@/git[_-]?lab\b https://gitlab.com/Bad_Manners
-RedirectMatch 301 ^/@/(google|g[_-]?mail)\b /#gmail
-RedirectMatch 301 ^/@/gum[_-]?road\b https://badmanners.gumroad.com
-RedirectMatch 301 ^/@/(ink[_-]?bunny|ib)\b https://inkbunny.net/BadManners
-RedirectMatch 301 ^/@/itaku\b https://itaku.ee/profile/badmanners
-RedirectMatch 301 ^/@/itch\b https://bad-manners.itch.io
-RedirectMatch 301 ^/@/keybase\b https://keybase.io/badmanners
-RedirectMatch 301 ^/@/keyoxide\b https://keyoxide.org/aspe:keyoxide.org:UWYBVFCBFXTVUF2U6FS6AYJHLU
-RedirectMatch 301 ^/@/ko[._-]?fi\b https://ko-fi.com/badmanners
-RedirectMatch 301 ^/@/(mastodon|meow[._-]?social|gulp[._-]?cafe)\b https://meow.social/@BadManners
-RedirectMatch 301 ^/@/neo[_-]?cities\b https://badmanners.neocities.org
-RedirectMatch 301 ^/@/picarto\b https://www.picarto.tv/BadManners
-RedirectMatch 301 ^/@/pillow[_-]?fort\b https://www.pillowfort.social/BadManners
-RedirectMatch 301 ^/@/pronouns?\b https://pronouns.cc/@BadManners
-RedirectMatch 301 ^/@/redd\.?it\b https://www.reddit.com/user/BadManners_
-RedirectMatch 301 ^/@/signal\b https://signal.me/#eu/ytt_rk0fFmAB2JAW-x2PbUiJyc_H3kYmfL_Pq4QNh5QIDsiFtjdFHaqFRs1D36tB
-RedirectMatch 301 ^/@/(so[_-]?furry|sf)\b https://bad-manners.sofurry.com
-RedirectMatch 301 ^/@/ssh\b /ssh.pub
-RedirectMatch 301 ^/@/steam(community|powered)?\b https://steamcommunity.com/id/badmanners_
-RedirectMatch 301 ^/@/subscribe[_-]?star\b https://subscribestar.adult/bad-manners
-RedirectMatch 301 ^/@/(telegram|t\.me)\b https://t.me/bad_manners
-RedirectMatch 301 ^/@/tumblr\b https://www.tumblr.com/badmannersxyz
-RedirectMatch 301 ^/@/twitch\b https://www.twitch.tv/bad__manners
-RedirectMatch 301 ^/@/weasyl\b https://www.weasyl.com/~badmanners
-RedirectMatch 301 ^/@/(x|twitter)\b https://x.com/BadManners__
-RedirectMatch 301 ^/@/(you[_-]?tube|youtu\.be|yt)\b https://www.youtube.com/@BadMannersXYZ
-`.trim();
-
-export const getStaticPaths: GetStaticPaths = async () => {
- if (APACHE_CONFIG) {
- return [{ params: { config: ".htaccess" }, props: { body: htaccess } }];
- }
- return [];
-};
-
-export const GET: APIRoute = ({ props: { body } }) => new Response(body);
diff --git a/src/pages/index.astro b/src/pages/index.astro
index aed18c8..16ad22e 100644
--- a/src/pages/index.astro
+++ b/src/pages/index.astro
@@ -8,6 +8,7 @@ import {
IconDiscord,
IconEkasPortal,
IconFurAffinity,
+ IconGitGud,
IconGithub,
IconGitlab,
IconGoogle,
@@ -79,7 +80,7 @@ import {
width="1.75rem"
class="inline transition-transform group-hover:scale-150 group-focus:scale-150 motion-reduce:transition-none motion-reduce:group-hover:scale-100 motion-reduce:group-focus:scale-100"
/>
- https://badmanners.xyz
+ https://badmanners.xyz
) : null
@@ -98,7 +99,7 @@ import {
width="1.75rem"
class="inline transition-transform group-hover:scale-150 group-focus:scale-150 motion-reduce:transition-none motion-reduce:group-hover:scale-100 motion-reduce:group-focus:scale-100"
/>
- Gallery on https://gallery.badmanners.xyz
+ Gallery on https://gallery.badmanners.xyz
@@ -114,7 +115,7 @@ import {
width="1.75rem"
class="inline transition-transform group-hover:scale-150 group-focus:scale-150 motion-reduce:transition-none motion-reduce:group-hover:scale-100 motion-reduce:group-focus:scale-100"
/>
- @BadManners on pronouns.cc
+ @BadManners on pronouns.cc
he/him/his/his/himself
they/them/their/theirs/themself
@@ -134,7 +135,7 @@ import {
width="1.75rem"
class="inline transition-transform group-hover:scale-150 group-focus:scale-150 motion-reduce:transition-none motion-reduce:group-hover:scale-100 motion-reduce:group-focus:scale-100"
/>
- me@badmanners.xyz
+ me@badmanners.xyz
@@ -150,7 +151,7 @@ import {
width="1.75rem"
class="inline transition-transform group-hover:scale-150 group-focus:scale-150 motion-reduce:transition-none motion-reduce:group-hover:scale-100 motion-reduce:group-focus:scale-100"
/>
- @badmanners.xyz on Bluesky
+ @badmanners.xyz on Bluesky
@@ -166,7 +167,7 @@ import {
width="1.75rem"
class="inline transition-transform group-hover:scale-150 group-focus:scale-150 motion-reduce:transition-none motion-reduce:group-hover:scale-100 motion-reduce:group-focus:scale-100"
/>
- BadManners on Codeberg
+ BadManners on Codeberg
@@ -182,7 +183,7 @@ import {
width="1.75rem"
class="inline transition-transform group-hover:scale-150 group-focus:scale-150 motion-reduce:transition-none motion-reduce:group-hover:scale-100 motion-reduce:group-focus:scale-100"
/>
- BadManners on Cohost
+ BadManners on Cohost
@@ -199,7 +200,7 @@ import {
width="1.75rem"
class="inline transition-transform group-hover:scale-150 group-focus:scale-150 motion-reduce:transition-none motion-reduce:group-hover:scale-100 motion-reduce:group-focus:scale-100"
/>
- badmanners on Discord
+ badmanners on Discord
@@ -215,7 +216,7 @@ import {
width="1.75rem"
class="inline transition-transform group-hover:scale-150 group-focus:scale-150 motion-reduce:transition-none motion-reduce:group-hover:scale-100 motion-reduce:group-focus:scale-100"
/>
- BadManners on Eka's Portal
+ BadManners on Eka's Portal
@@ -231,7 +232,23 @@ import {
width="1.75rem"
class="inline transition-transform group-hover:scale-150 group-focus:scale-150 motion-reduce:transition-none motion-reduce:group-hover:scale-100 motion-reduce:group-focus:scale-100"
/>
- BadManners on Fur Affinity
+ BadManners on Fur Affinity
+
+
+
+
+
+ BadMannersXYZ on GitGud
@@ -247,7 +264,7 @@ import {
width="1.75rem"
class="inline transition-transform group-hover:scale-150 group-focus:scale-150 motion-reduce:transition-none motion-reduce:group-hover:scale-100 motion-reduce:group-focus:scale-100"
/>
- BadMannersXYZ on GitHub
+ BadMannersXYZ on GitHub
@@ -263,7 +280,7 @@ import {
width="1.75rem"
class="inline transition-transform group-hover:scale-150 group-focus:scale-150 motion-reduce:transition-none motion-reduce:group-hover:scale-100 motion-reduce:group-focus:scale-100"
/>
- Bad_Manners on GitLab
+ Bad_Manners on GitLab
@@ -281,7 +298,7 @@ import {
width="1.75rem"
class="inline transition-transform group-hover:scale-150 group-focus:scale-150 motion-reduce:transition-none motion-reduce:group-hover:scale-100 motion-reduce:group-focus:scale-100"
/>
- badmanners.vore@gmail.com
+ badmanners.vore@gmail.com
@@ -297,7 +314,7 @@ import {
width="1.75rem"
class="inline transition-transform group-hover:scale-150 group-focus:scale-150 motion-reduce:transition-none motion-reduce:group-hover:scale-100 motion-reduce:group-focus:scale-100"
/>
- BadManners on Inkbunny
+ BadManners on Inkbunny
@@ -313,7 +330,7 @@ import {
width="1.75rem"
class="inline transition-transform group-hover:scale-150 group-focus:scale-150 motion-reduce:transition-none motion-reduce:group-hover:scale-100 motion-reduce:group-focus:scale-100"
/>
- badmanners on Itaku
+ badmanners on Itaku
@@ -329,7 +346,7 @@ import {
width="1.75rem"
class="inline transition-transform group-hover:scale-150 group-focus:scale-150 motion-reduce:transition-none motion-reduce:group-hover:scale-100 motion-reduce:group-focus:scale-100"
/>
- Bad Manners on Itch.io
+ Bad Manners on Itch.io
@@ -345,7 +362,7 @@ import {
width="1.75rem"
class="inline transition-transform group-hover:scale-150 group-focus:scale-150 motion-reduce:transition-none motion-reduce:group-hover:scale-100 motion-reduce:group-focus:scale-100"
/>
- badmanners on Keybase
+ badmanners on Keybase
@@ -361,7 +378,7 @@ import {
width="1.75rem"
class="inline transition-transform group-hover:scale-150 group-focus:scale-150 motion-reduce:transition-none motion-reduce:group-hover:scale-100 motion-reduce:group-focus:scale-100"
/>
- aspe:keyoxide.org:UWYBVFCBFXTVUF2U6FS6AYJHLU on Keyoxide
+ aspe:keyoxide.org:UWYBVFCBFXTVUF2U6FS6AYJHLU on Keyoxide
@@ -377,7 +394,7 @@ import {
width="1.75rem"
class="inline transition-transform group-hover:scale-150 group-focus:scale-150 motion-reduce:transition-none motion-reduce:group-hover:scale-100 motion-reduce:group-focus:scale-100"
/>
- badmanners on Ko-fi
+ badmanners on Ko-fi
@@ -393,7 +410,7 @@ import {
width="1.75rem"
class="inline transition-transform group-hover:scale-150 group-focus:scale-150 motion-reduce:transition-none motion-reduce:group-hover:scale-100 motion-reduce:group-focus:scale-100"
/>
- @BadManners@meow.social on Mastodon
+ @BadManners@meow.social on Mastodon
@@ -409,7 +426,7 @@ import {
width="1.75rem"
class="inline transition-transform group-hover:scale-150 group-focus:scale-150 motion-reduce:transition-none motion-reduce:group-hover:scale-100 motion-reduce:group-focus:scale-100"
/>
- badmanners.neocities.org on Neocities
+ badmanners.neocities.org on Neocities
@@ -425,7 +442,7 @@ import {
width="1.75rem"
class="inline transition-transform group-hover:scale-150 group-focus:scale-150 motion-reduce:transition-none motion-reduce:group-hover:scale-100 motion-reduce:group-focus:scale-100"
/>
- BadManners on Picarto
+ BadManners on Picarto
@@ -441,7 +458,7 @@ import {
width="1.75rem"
class="inline transition-transform group-hover:scale-150 group-focus:scale-150 motion-reduce:transition-none motion-reduce:group-hover:scale-100 motion-reduce:group-focus:scale-100"
/>
- /u/BadManners_ on Reddit
+ /u/BadManners_ on Reddit
@@ -457,7 +474,7 @@ import {
width="1.75rem"
class="inline transition-transform group-hover:scale-150 group-focus:scale-150 motion-reduce:transition-none motion-reduce:group-hover:scale-100 motion-reduce:group-focus:scale-100"
/>
- badmanners.10 on Signal
+ badmanners.10 on Signal
@@ -473,7 +490,7 @@ import {
width="1.75rem"
class="inline transition-transform group-hover:scale-150 group-focus:scale-150 motion-reduce:transition-none motion-reduce:group-hover:scale-100 motion-reduce:group-focus:scale-100"
/>
- Bad Manners on SoFurry
+ Bad Manners on SoFurry
@@ -490,7 +507,7 @@ import {
width="1.75rem"
class="inline transition-transform group-hover:scale-150 group-focus:scale-150 motion-reduce:transition-none motion-reduce:group-hover:scale-100 motion-reduce:group-focus:scale-100"
/>
- SSH public key
+ SSH public key
@@ -506,7 +523,7 @@ import {
width="1.75rem"
class="inline transition-transform group-hover:scale-150 group-focus:scale-150 motion-reduce:transition-none motion-reduce:group-hover:scale-100 motion-reduce:group-focus:scale-100"
/>
- badmanners_ on Steam
+ badmanners_ on Steam
@@ -522,7 +539,7 @@ import {
width="1.75rem"
class="inline transition-transform group-hover:scale-150 group-focus:scale-150 motion-reduce:transition-none motion-reduce:group-hover:scale-100 motion-reduce:group-focus:scale-100"
/>
- Bad Manners on SubscribeStar
+ Bad Manners on SubscribeStar
@@ -538,7 +555,7 @@ import {
width="1.75rem"
class="inline transition-transform group-hover:scale-150 group-focus:scale-150 motion-reduce:transition-none motion-reduce:group-hover:scale-100 motion-reduce:group-focus:scale-100"
/>
- @bad_manners on Telegram
+ @bad_manners on Telegram
@@ -554,7 +571,7 @@ import {
width="1.75rem"
class="inline transition-transform group-hover:scale-150 group-focus:scale-150 motion-reduce:transition-none motion-reduce:group-hover:scale-100 motion-reduce:group-focus:scale-100"
/>
- badmannersxyz on Tumblr
+ badmannersxyz on Tumblr
@@ -570,7 +587,7 @@ import {
width="1.75rem"
class="inline transition-transform group-hover:scale-150 group-focus:scale-150 motion-reduce:transition-none motion-reduce:group-hover:scale-100 motion-reduce:group-focus:scale-100"
/>
- bad__manners on Twitch
+ bad__manners on Twitch
@@ -586,7 +603,7 @@ import {
width="1.75rem"
class="inline transition-transform group-hover:scale-150 group-focus:scale-150 motion-reduce:transition-none motion-reduce:group-hover:scale-100 motion-reduce:group-focus:scale-100"
/>
- BadManners on Weasyl
+ BadManners on Weasyl
@@ -602,7 +619,7 @@ import {
width="1.75rem"
class="inline transition-transform group-hover:scale-150 group-focus:scale-150 motion-reduce:transition-none motion-reduce:group-hover:scale-100 motion-reduce:group-focus:scale-100"
/>
- @BadManners__ on X
+ @BadManners__ on X
@@ -618,7 +635,7 @@ import {
width="1.75rem"
class="inline transition-transform group-hover:scale-150 group-focus:scale-150 motion-reduce:transition-none motion-reduce:group-hover:scale-100 motion-reduce:group-focus:scale-100"
/>
- @BadMannersXYZ on YouTube
+ @BadMannersXYZ on YouTube
diff --git a/src/pages/licenses.toml.ts b/src/pages/licenses.toml.ts
new file mode 100644
index 0000000..242fa52
--- /dev/null
+++ b/src/pages/licenses.toml.ts
@@ -0,0 +1,123 @@
+import type { APIRoute } from "astro";
+import { readFile } from "node:fs/promises";
+import { parse } from "toml";
+
+/**
+ * Verify attributions and copyright according to the [Creative Commons recommended practices](https://wiki.creativecommons.org/wiki/Recommended_practices_for_attribution)
+ * @param copyright Unparsed TOML copyright information.
+ */
+function verifyAttributions(licenses: string) {
+ const { copyright, attributions } = parse(licenses);
+ // Make sure each copyright and attribution follows the TASL format.
+ // - T: title (or description)
+ // - A: author
+ // - S: source
+ // - L: license
+ // - other fields that have custom validation: type, notes, items
+ [copyright, attributions].flat().forEach((value) => {
+ // Title or description must be a valid string
+ const title = value.title ?? value.description;
+ if (typeof title !== "string" || !title) {
+ throw new Error(`Missing "title" and/or "description" for attribution (${JSON.stringify(value)})`);
+ }
+ // Author must be a valid string or object or list
+ const authors = [value.author].flat();
+ if (authors.length === 0) {
+ throw new Error(`Missing "author" for attribution "${title}" (${JSON.stringify(value)})`);
+ }
+ authors.forEach((author) => {
+ if (!author) {
+ throw new Error(`Missing "author" for attribution "${title}" (${JSON.stringify(value)})`);
+ }
+ if (typeof author !== "object" && typeof author !== "string") {
+ throw new Error(`Invalid "${typeof author}" type for "author"${JSON.stringify(author)} for attribution "${title}" (${JSON.stringify(value)})`);
+ }
+ if (typeof author === "object" && !(author.name || author.url)) {
+ throw new Error(`Missing both name and URL for "author" ${JSON.stringify(author)} for attribution "${title}" (${JSON.stringify(value)})`);
+ }
+ });
+ // Source must be a valid string or list of strings
+ const sources = [value.source].flat();
+ if (sources.length === 0) {
+ throw new Error(`Missing "source" for attribution "${title}" (${JSON.stringify(value)})`);
+ }
+ sources.forEach((source) => {
+ if (!source) {
+ throw new Error(`Missing "source" for attribution "${title}" (${JSON.stringify(value)})`);
+ }
+ if (typeof source !== "object" && typeof source !== "string") {
+ throw new Error(`Invalid "${typeof source}" type for "source"${JSON.stringify(source)} for attribution "${title}" (${JSON.stringify(value)})`);
+ }
+ if (typeof source === "object" && !(source.url)) {
+ throw new Error(`Missing URL for "source" ${JSON.stringify(source)} for attribution "${title}" (${JSON.stringify(value)})`);
+ }
+ });
+ // License must be a valid string or object or list
+ const licenses = [value.license].flat();
+ if (licenses.length === 0) {
+ throw new Error(`Missing "license" for attribution "${title}" (${JSON.stringify(value)})`);
+ }
+ licenses.forEach((license) => {
+ if (!license) {
+ throw new Error(`Missing "license" for attribution "${title}" (${JSON.stringify(value)})`);
+ }
+ if (typeof license !== "object" && typeof license !== "string") {
+ throw new Error(`Invalid "${typeof license}" type for "license"${JSON.stringify(license)} for attribution "${title}" (${JSON.stringify(value)})`);
+ }
+ if (typeof license === "object" && !(license.name || license.url)) {
+ throw new Error(`Missing both name and URL for "license" ${JSON.stringify(license)} for attribution "${title}" (${JSON.stringify(value)})`);
+ }
+ });
+ // Validate extra optional fields
+ // 1. Type must be a valid string
+ if (typeof value.type !== "string") {
+ throw new Error(`Invalid "type" for attribution "${title}" (${JSON.stringify(value)})`);
+ }
+ if (!value.type) {
+ throw new Error(`Missing "type" for attribution "${title}" (${JSON.stringify(value)})`);
+ }
+ // 2. Items must be a valid list of strings/objects
+ if ("items" in value) {
+ const items = value.items;
+ if (!Array.isArray(items)) {
+ throw new Error(`Invalid non-array "items" ${JSON.stringify(items)} for attribution "${title}" (${JSON.stringify(value)})`);
+ }
+ items.forEach((item) => {
+ if (!item) {
+ throw new Error(`Invalid item ${JSON.stringify} in "items" for attribution "${title}" (${JSON.stringify(value)})`);
+ }
+ })
+ }
+ // 3. Type must be a valid string
+ if ("notes" in value) {
+ if (typeof value.notes !== "string" || !value.notes) {
+ throw new Error(`Invalid "notes" for attribution "${title}" (${JSON.stringify(value)})`);
+ }
+ }
+ });
+ // Check date for copyrights
+ if (typeof copyright.date !== "string") {
+ throw new Error(`Invalid "date" for copyright (${JSON.stringify(copyright)})`);
+ }
+ if (!copyright.date) {
+ throw new Error(`Missing "date" for copyright (${JSON.stringify(copyright)})`);
+ }
+ // Check notes for additional copyrights
+ if ("additional" in copyright) {
+ const additionals = copyright.additional;
+ if (!Array.isArray(additionals)) {
+ throw new Error(`Invalid non-array "additional" ${JSON.stringify(additionals)} for copyright (${JSON.stringify(copyright)})`);
+ }
+ additionals.forEach((additional) => {
+ if (typeof additional.notes !== "string" || !additional.notes) {
+ throw new Error(`Invalid "notes" for additional copyright (${JSON.stringify(additional)})`);
+ }
+ })
+ }
+}
+
+export const GET: APIRoute = async () => {
+ const licenses = await readFile("./src/data/licenses.toml", { encoding: "utf-8" });
+ verifyAttributions(licenses);
+ return new Response(licenses, { headers: { "Content-Type": "application/toml; charset=utf-8" } })
+};