Compare commits

..

11 Commits

Author SHA1 Message Date
9752501ee5 Pixel URI fix
All checks were successful
ci/woodpecker/push/ci Pipeline was successful
ci/woodpecker/push/build Pipeline was successful
ci/woodpecker/push/deploy Pipeline was successful
2026-02-17 12:50:23 -03:00
75f7f4b09c Tracking and Pixel
All checks were successful
ci/woodpecker/push/ci Pipeline was successful
ci/woodpecker/push/build Pipeline was successful
ci/woodpecker/push/deploy Pipeline was successful
2026-02-17 12:25:02 -03:00
21b28ba376 Umami analytics setup
All checks were successful
ci/woodpecker/push/ci Pipeline was successful
ci/woodpecker/push/build Pipeline was successful
ci/woodpecker/push/deploy Pipeline was successful
2026-02-17 11:02:50 -03:00
bbf1f0dde7 gzip that bitch
All checks were successful
ci/woodpecker/push/ci Pipeline was successful
ci/woodpecker/push/build Pipeline was successful
ci/woodpecker/push/deploy Pipeline was successful
2026-02-16 13:04:23 -03:00
7b63c0c922 WCAG fixes
All checks were successful
ci/woodpecker/push/ci Pipeline was successful
ci/woodpecker/push/build Pipeline was successful
ci/woodpecker/push/deploy Pipeline was successful
2026-02-16 12:57:11 -03:00
93bbfee7f7 Accessibility fix
All checks were successful
ci/woodpecker/push/ci Pipeline was successful
ci/woodpecker/push/build Pipeline was successful
ci/woodpecker/push/deploy Pipeline was successful
2026-02-16 12:28:14 -03:00
c81cc54d91 Added sitemap.xml
All checks were successful
ci/woodpecker/push/ci Pipeline was successful
ci/woodpecker/push/build Pipeline was successful
ci/woodpecker/push/deploy Pipeline was successful
2026-02-16 11:21:10 -03:00
88f0e84537 robots.txt
All checks were successful
ci/woodpecker/push/ci Pipeline was successful
ci/woodpecker/push/build Pipeline was successful
ci/woodpecker/push/deploy Pipeline was successful
2026-02-16 11:07:30 -03:00
502bc1765f JSON-LD and silencing dev tools noise
All checks were successful
ci/woodpecker/push/ci Pipeline was successful
ci/woodpecker/push/build Pipeline was successful
ci/woodpecker/push/deploy Pipeline was successful
2026-02-16 11:03:05 -03:00
4a79266a27 Tweaks and updates
All checks were successful
ci/woodpecker/push/ci Pipeline was successful
ci/woodpecker/push/build Pipeline was successful
ci/woodpecker/push/deploy Pipeline was successful
2026-02-13 18:21:18 -03:00
27808cfd0e Tweaks, fixes, etc
All checks were successful
ci/woodpecker/push/ci Pipeline was successful
ci/woodpecker/push/build Pipeline was successful
ci/woodpecker/push/deploy Pipeline was successful
2026-02-13 18:12:52 -03:00
18 changed files with 842 additions and 259 deletions

View File

@@ -0,0 +1,27 @@
{
"name": "mail-landing",
"image": "mcr.microsoft.com/devcontainers/javascript-node:1-22-bookworm",
"onCreateCommand": "npm install -g pnpm@10",
"postCreateCommand": "pnpm install",
"forwardPorts": [3000],
"portsAttributes": {
"3000": {
"label": "Preview",
"onAutoForward": "notify"
}
},
"customizations": {
"vscode": {
"extensions": [
"esbenp.prettier-vscode",
"stylelint.vscode-stylelint",
"dbaeumer.vscode-eslint"
],
"settings": {
"editor.formatOnSave": true,
"editor.defaultFormatter": "esbenp.prettier-vscode"
}
}
},
"remoteUser": "node"
}

View File

@@ -1,5 +1,9 @@
# Avoid sending secrets or dev tooling into the build context # Avoid sending secrets or dev tooling into the build context
# config/ and plugins/ are included (no secrets; PHP configs read from ENV at runtime) # config/ and plugins/ are included (no secrets; PHP configs read from ENV at runtime)
.devcontainer
.pnpm-store
.vscode
.woodpecker
node_modules node_modules
scripts scripts
.git .git

2
.gitignore vendored
View File

@@ -1,3 +1,5 @@
.pnpm-store
.env .env
.env.* .env.*
!.env.example !.env.example

View File

@@ -1,3 +1,5 @@
.pnpm-store
node_modules node_modules
pnpm-lock.yaml pnpm-lock.yaml

View File

@@ -1,7 +1,7 @@
{ {
"semi": false, "semi": false,
"singleQuote": true, "singleQuote": true,
"tabWidth": 2, "tabWidth": 4,
"trailingComma": "none", "trailingComma": "none",
"overrides": [ "overrides": [
{ {

View File

@@ -81,7 +81,7 @@ Static landing site for **mifi.holdings** (and www). Plain HTML/CSS/JS source; a
- **Frontend**: Static HTML + CSS + JS in `src/`; production build minifies and inlines critical CSS. - **Frontend**: Static HTML + CSS + JS in `src/`; production build minifies and inlines critical CSS.
- **Server**: nginx (Alpine) in Docker. - **Server**: nginx (Alpine) in Docker.
- **Tooling**: **pnpm**; Prettier (format); ESLint (JS), Stylelint (CSS), yamllint (YAML); **terser**, **clean-css**, **critters** (build). - **Tooling**: **pnpm**; Prettier (format); ESLint (JS), Stylelint (CSS), yamllint (YAML); **terser**, **clean-css**, **beasties** (build).
- **Deployment**: Docker image (from `dist/`) → Gitea registry → Portainer stack redeploy. - **Deployment**: Docker image (from `dist/`) → Gitea registry → Portainer stack redeploy.
--- ---

View File

@@ -2,6 +2,7 @@ services:
mifi-holdings-landing: mifi-holdings-landing:
image: git.mifi.dev/mifi-holdings/landing:latest image: git.mifi.dev/mifi-holdings/landing:latest
container_name: mifi-holdings-landing container_name: mifi-holdings-landing
restart: unless-stopped
networks: networks:
- marina-net - marina-net
labels: labels:
@@ -9,7 +10,7 @@ services:
- 'traefik.docker.network=marina-net' - 'traefik.docker.network=marina-net'
- 'traefik.http.routers.holdings-landing.rule=Host(`mifi.holdings`) || Host(`www.mifi.holdings`)' - 'traefik.http.routers.holdings-landing.rule=Host(`mifi.holdings`) || Host(`www.mifi.holdings`)'
- 'traefik.http.routers.holdings-landing.entrypoints=websecure' - 'traefik.http.routers.holdings-landing.entrypoints=websecure'
- 'traefik.http.routers.holdings-landing.middlewares=security-supermax-with-analytics@file,redirect-www-to-non-www@file' - 'traefik.http.routers.holdings-landing.middlewares=gzip@file,security-supermax-with-analytics@file,redirect-www-to-non-www@file'
- 'traefik.http.routers.holdings-landing.tls=true' - 'traefik.http.routers.holdings-landing.tls=true'
- 'traefik.http.routers.holdings-landing.tls.certresolver=letsencrypt' - 'traefik.http.routers.holdings-landing.tls.certresolver=letsencrypt'
- 'traefik.http.services.holdings-landing.loadbalancer.server.port=80' - 'traefik.http.services.holdings-landing.loadbalancer.server.port=80'

View File

@@ -1,6 +1,6 @@
{ {
"type": "module",
"name": "mifi-holdings-landing", "name": "mifi-holdings-landing",
"type": "module",
"version": "1.0.0", "version": "1.0.0",
"packageManager": "pnpm@10.29.3", "packageManager": "pnpm@10.29.3",
"scripts": { "scripts": {
@@ -16,14 +16,17 @@
"lint:fix": "pnpm run lint:fix:js && pnpm run lint:fix:css && pnpm run lint:fix:yaml", "lint:fix": "pnpm run lint:fix:js && pnpm run lint:fix:css && pnpm run lint:fix:yaml",
"lint:fix:js": "eslint src/ --fix", "lint:fix:js": "eslint src/ --fix",
"lint:fix:css": "stylelint \"src/**/*.css\" --fix", "lint:fix:css": "stylelint \"src/**/*.css\" --fix",
"lint:fix:yaml": "yamllint .woodpecker/ci.yml .woodpecker/build.yml .woodpecker/deploy.yml docker-compose.yml --fix" "lint:fix:yaml": "yamllint .woodpecker/ci.yml .woodpecker/build.yml .woodpecker/deploy.yml docker-compose.yml --fix",
"preview": "serve src -l 3000",
"preview:prod": "pnpm build && serve dist -l 3000"
}, },
"devDependencies": { "devDependencies": {
"clean-css": "^5.3.3", "clean-css": "^5.3.3",
"critters": "^0.0.25", "beasties": "^0.4.1",
"eslint": "^10.0.0", "eslint": "^10.0.0",
"eslint-config-prettier": "^10.1.8", "eslint-config-prettier": "^10.1.8",
"prettier": "^3.4.2", "prettier": "^3.4.2",
"serve": "^14.2.4",
"stylelint": "^17.3.0", "stylelint": "^17.3.0",
"stylelint-config-standard": "^40.0.0", "stylelint-config-standard": "^40.0.0",
"terser": "^5.46.0", "terser": "^5.46.0",

522
pnpm-lock.yaml generated

File diff suppressed because it is too large Load Diff

View File

@@ -1,5 +1,5 @@
/** /**
* Build script: copy src → dist, minify JS/CSS, inline critical CSS (Critters). * Build script: copy src → dist, minify JS/CSS, inline critical CSS (Beasties).
* Run with: pnpm build * Run with: pnpm build
*/ */
import { import {
@@ -12,7 +12,7 @@ import {
} from 'fs' } from 'fs'
import { join, dirname, extname } from 'path' import { join, dirname, extname } from 'path'
import { fileURLToPath } from 'url' import { fileURLToPath } from 'url'
import Critters from 'critters' import Beasties from 'beasties'
import { minify as minifyJs } from 'terser' import { minify as minifyJs } from 'terser'
import CleanCSS from 'clean-css' import CleanCSS from 'clean-css'
@@ -56,16 +56,18 @@ async function main() {
if (!result.errors.length) writeFileSync(f, result.styles) if (!result.errors.length) writeFileSync(f, result.styles)
} }
// 4. Inline critical CSS with Critters (no browser; works in CI) // 4. Inline critical CSS with Beasties for all HTML files (no browser; works in CI)
const critters = new Critters({ const htmlFiles = distFiles.filter((f) => extname(f) === '.html')
const beasties = new Beasties({
path: distDir, path: distDir,
preload: 'default', preload: 'default',
logLevel: 'warn' logLevel: 'warn'
}) })
const indexPath = join(distDir, 'index.html') for (const htmlFile of htmlFiles) {
const html = readFileSync(indexPath, 'utf8') const html = readFileSync(htmlFile, 'utf8')
const inlined = await critters.process(html) const inlined = await beasties.process(html)
writeFileSync(indexPath, inlined) writeFileSync(htmlFile, inlined)
}
console.log('Build complete: dist/') console.log('Build complete: dist/')
} }

View File

@@ -0,0 +1 @@
{}

View File

@@ -1,3 +1,23 @@
.skip-link {
position: absolute;
top: -100%;
left: 0;
z-index: 100;
padding: 0.75rem 1.25rem;
background: #70ffd7;
color: #1b1e22;
font-weight: 700;
text-decoration: none;
border-radius: 0 0 0.25rem;
transition: top 0.2s;
}
.skip-link:focus {
top: 0;
outline: 2px solid #70ffd7;
outline-offset: 2px;
}
html, html,
body { body {
height: 100%; height: 100%;

View File

@@ -11,6 +11,11 @@
src="/assets/js/ga-init.js" src="/assets/js/ga-init.js"
data-ga-id="G-4000VNMXLK" data-ga-id="G-4000VNMXLK"
></script> ></script>
<script
defer
src="https://analytics.mifi.holdings/script.js"
data-website-id="ce2a7f8a-95e9-4bc5-93cb-2e6075d836b8"
></script>
<meta charset="UTF-8" /> <meta charset="UTF-8" />
<meta name="viewport" content="width=device-width,initial-scale=1.0" /> <meta name="viewport" content="width=device-width,initial-scale=1.0" />
@@ -26,26 +31,64 @@
<link rel="stylesheet" href="assets/css/style.css" /> <link rel="stylesheet" href="assets/css/style.css" />
<link rel="icon" type="image/svg+xml" href="/assets/images/favicon.svg" /> <link
<link rel="icon" type="image/x-icon" href="/assets/images/favicon.ico" /> rel="icon"
type="image/svg+xml"
href="/assets/images/favicon.svg"
/>
<link
rel="icon"
type="image/x-icon"
href="/assets/images/favicon.ico"
/>
<link <link
rel="apple-touch-icon" rel="apple-touch-icon"
sizes="180x180" sizes="180x180"
href="/assets/images/apple-touch-icon.png" href="/assets/images/apple-touch-icon.png"
/> />
<script type="application/ld+json">
{
"@context": "https://schema.org",
"@type": "WebSite",
"name": "mifi Holdings",
"url": "https://mifi.holdings",
"description": "This is just a landing page so something exists at the root domain of all the digital holdings of mifi.",
"publisher": {
"@type": "Organization",
"name": "mifi Ventures",
"url": "https://mifi.ventures",
"email": "postmaster@mifi.holdings"
}
}
</script>
</head> </head>
<body> <body>
<div class="container"> <a
href="#main-content"
class="skip-link"
data-umami-event="skip to main content"
>Skip to main content</a
>
<main id="main-content" class="container" tabindex="-1">
<div class="emoji">🛸</div> <div class="emoji">🛸</div>
<h1>Nothing to See Here</h1> <h1>Nothing to See Here</h1>
<p> <p>
You&apos;ve stumbled onto <b>mifi.holdings</b> — the legendary vault of You&apos;ve stumbled onto <b>mifi.holdings</b> — the legendary
digital oddities, curios, and coffee-fueled experiments belonging to a vault of digital oddities, curios, and coffee-fueled experiments
possibly-human, definitely-mysterious entity named belonging to a possibly-human, definitely-mysterious entity
<b>mifi</b>.<br /><br /> named <b>mifi</b>.<br /><br />
There&apos;s nothing here for you.<br /> There&apos;s nothing here for you.<br />
Go on. Shoo. Scram. Or just keep wondering. Go on. Shoo. Scram. Or just keep wondering.
</p> </p>
</div> </main>
<img
src="https://analytics.mifi.holdings/p/wQ9GYnLIg"
alt=""
width="1"
height="1"
role="presentation"
loading="eager"
/>
</body> </body>
</html> </html>

4
src/robots.txt Normal file
View File

@@ -0,0 +1,4 @@
User-agent: *
Allow: /
Sitemap: https://mifi.holdings/sitemap.xml

8
src/sitemap.xml Normal file
View File

@@ -0,0 +1,8 @@
<?xml version="1.0" encoding="UTF-8"?>
<urlset xmlns="http://www.sitemaps.org/schemas/sitemap/0.9">
<url>
<loc>https://mifi.holdings/</loc>
<changefreq>monthly</changefreq>
<priority>1.0</priority>
</url>
</urlset>