Compare commits
9 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 4e51dd4a83 | |||
| 8e9fcfaeeb | |||
| 2946801517 | |||
| 49028e7783 | |||
| c36cc94437 | |||
| 6d0b4e29d8 | |||
| 7d9b300d84 | |||
| 4718d5df31 | |||
| 153bdf502c |
@@ -41,3 +41,30 @@ jobs:
|
|||||||
docker stop portfolio 2>/dev/null || true && \
|
docker stop portfolio 2>/dev/null || true && \
|
||||||
docker rm portfolio 2>/dev/null || true && \
|
docker rm portfolio 2>/dev/null || true && \
|
||||||
docker run -d --name portfolio -p 80:80 --restart unless-stopped portfolio"
|
docker run -d --name portfolio -p 80:80 --restart unless-stopped portfolio"
|
||||||
|
|
||||||
|
- name: Tag release (CalVer)
|
||||||
|
run: |
|
||||||
|
git fetch --tags
|
||||||
|
if git describe --exact-match --tags HEAD 2>/dev/null; then
|
||||||
|
echo "Commit already tagged, skipping."
|
||||||
|
else
|
||||||
|
YEAR=$(date +%Y)
|
||||||
|
MONTH=$(date +%m)
|
||||||
|
LATEST=$(git tag --sort=-v:refname | grep -E '^[0-9]{4}\.[0-9]{2}\.[0-9]+$' | head -1)
|
||||||
|
if [ -n "$LATEST" ]; then
|
||||||
|
LATEST_YEAR=$(echo "$LATEST" | cut -d. -f1)
|
||||||
|
LATEST_MONTH=$(echo "$LATEST" | cut -d. -f2)
|
||||||
|
LATEST_MICRO=$(echo "$LATEST" | cut -d. -f3)
|
||||||
|
if [ "$YEAR" = "$LATEST_YEAR" ] && [ "$MONTH" = "$LATEST_MONTH" ]; then
|
||||||
|
MICRO=$((LATEST_MICRO + 1))
|
||||||
|
else
|
||||||
|
MICRO=1
|
||||||
|
fi
|
||||||
|
else
|
||||||
|
MICRO=1
|
||||||
|
fi
|
||||||
|
NEW_TAG="${YEAR}.$(printf '%02d' $MONTH).${MICRO}"
|
||||||
|
git tag "$NEW_TAG"
|
||||||
|
git push origin "$NEW_TAG"
|
||||||
|
echo "Tagged $NEW_TAG"
|
||||||
|
fi
|
||||||
|
|||||||
@@ -0,0 +1,44 @@
|
|||||||
|
name: Deploy to GitHub Pages
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches:
|
||||||
|
- dev
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
contents: read
|
||||||
|
pages: write
|
||||||
|
id-token: write
|
||||||
|
|
||||||
|
concurrency:
|
||||||
|
group: pages
|
||||||
|
cancel-in-progress: false
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- uses: actions/setup-node@v4
|
||||||
|
with:
|
||||||
|
node-version: 22
|
||||||
|
cache: npm
|
||||||
|
|
||||||
|
- name: Build
|
||||||
|
run: npm ci && npm run build
|
||||||
|
|
||||||
|
- uses: actions/upload-pages-artifact@v3
|
||||||
|
with:
|
||||||
|
path: out/
|
||||||
|
|
||||||
|
deploy:
|
||||||
|
needs: build
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
environment:
|
||||||
|
name: github-pages
|
||||||
|
url: ${{ steps.deployment.outputs.page_url }}
|
||||||
|
steps:
|
||||||
|
- name: Deploy
|
||||||
|
id: deployment
|
||||||
|
uses: actions/deploy-pages@v4
|
||||||
@@ -29,9 +29,5 @@ yarn-error.log*
|
|||||||
*.tsbuildinfo
|
*.tsbuildinfo
|
||||||
next-env.d.ts
|
next-env.d.ts
|
||||||
|
|
||||||
# claude code
|
|
||||||
CLAUDE.md
|
|
||||||
.claude/
|
|
||||||
|
|
||||||
# docs
|
# docs
|
||||||
/docs
|
/docs
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
Personal portfolio site. Live at [lerkolabs.com](https://lerkolabs.com) — self-hosted.
|
Personal portfolio site. Live at [lerkolabs.com](https://lerkolabs.com) — self-hosted.
|
||||||
|
|
||||||
Source: [gitea.lerkolabs.com/lerko/lerkolabs.com](https://gitea.lerkolabs.com/lerko/lerkolabs.com)
|
Source: [gitea.lerkolabs.com/lerko/portfolio](https://gitea.lerkolabs.com/lerko/portfolio)
|
||||||
|
|
||||||
**Stack:** Next.js 16 · React 19 · TypeScript · Tailwind v4
|
**Stack:** Next.js 16 · React 19 · TypeScript · Tailwind v4
|
||||||
|
|
||||||
|
|||||||
+15
-20
@@ -10,23 +10,18 @@ export const metadata: Metadata = {
|
|||||||
export default function ArchivePage() {
|
export default function ArchivePage() {
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<div className="mb-16">
|
<div className="mb-4lh">
|
||||||
<div className="flex items-center gap-3 mb-4">
|
<p className="font-mono text-sm font-bold text-[var(--color-text)] mb-1lh">
|
||||||
<span className="font-mono text-xs text-[var(--color-text-label)] tracking-widest uppercase">
|
<span className="text-[var(--color-accent-green)] select-none mr-1ch" aria-hidden="true">❯</span>
|
||||||
archive
|
tyler/projects/archive
|
||||||
</span>
|
</p>
|
||||||
<div className="flex-1 h-px bg-[var(--color-border)]" aria-hidden="true" />
|
<p className="font-mono text-sm text-[var(--color-text)] leading-relaxed max-w-xl opacity-80">
|
||||||
</div>
|
|
||||||
<h1 className="font-mono text-lg font-bold text-[var(--color-text)] mb-3">
|
|
||||||
Earlier Work
|
|
||||||
</h1>
|
|
||||||
<p className="font-sans text-sm text-[var(--color-text)] leading-relaxed max-w-xl opacity-80">
|
|
||||||
Experiments, browser extensions, and bootcamp projects. Kept here for context — not
|
Experiments, browser extensions, and bootcamp projects. Kept here for context — not
|
||||||
representative of current work.
|
representative of current work.
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<Widget title="projects" badge={archiveProjects.length} as="section">
|
<Widget title="tyler/projects/archive" badge={archiveProjects.length} as="section">
|
||||||
<div className="flex flex-col gap-px bg-[var(--color-border)]">
|
<div className="flex flex-col gap-px bg-[var(--color-border)]">
|
||||||
{archiveProjects.map((project) => (
|
{archiveProjects.map((project) => (
|
||||||
<a
|
<a
|
||||||
@@ -34,12 +29,12 @@ export default function ArchivePage() {
|
|||||||
href={project.githubUrl}
|
href={project.githubUrl}
|
||||||
target="_blank"
|
target="_blank"
|
||||||
rel="noopener noreferrer"
|
rel="noopener noreferrer"
|
||||||
className="bg-[var(--color-surface)] hover:bg-[var(--color-surface-raised)] flex items-start justify-between gap-6 px-4 py-4 group"
|
className="bg-[var(--color-surface)] hover:bg-[var(--color-surface-raised)] flex items-start justify-between gap-2ch px-2ch py-1lh group"
|
||||||
>
|
>
|
||||||
<div className="flex flex-col gap-2 flex-1 min-w-0">
|
<div className="flex flex-col gap-1ch flex-1 min-w-0">
|
||||||
<div className="flex items-center gap-3">
|
<div className="flex items-center gap-1ch">
|
||||||
{project.year && (
|
{project.year && (
|
||||||
<span className="font-mono text-xs text-[var(--color-text-dim)] shrink-0">
|
<span className="font-mono text-sm text-[var(--color-text-dim)] shrink-0">
|
||||||
{project.year}
|
{project.year}
|
||||||
</span>
|
</span>
|
||||||
)}
|
)}
|
||||||
@@ -47,19 +42,19 @@ export default function ArchivePage() {
|
|||||||
{project.title}
|
{project.title}
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
<p className="font-sans text-sm text-[var(--color-text)] leading-relaxed opacity-75">
|
<p className="font-mono text-sm text-[var(--color-text)] leading-relaxed opacity-75">
|
||||||
{project.description}
|
{project.description}
|
||||||
</p>
|
</p>
|
||||||
<div className="flex flex-wrap gap-x-3 gap-y-0.5">
|
<div className="flex flex-wrap gap-x-1ch gap-y-0.5">
|
||||||
{project.tags.map((tag) => (
|
{project.tags.map((tag) => (
|
||||||
<span key={tag} className="font-mono text-xs text-[var(--color-text-dim)]">
|
<span key={tag} className="font-mono text-sm text-[var(--color-text-dim)]">
|
||||||
{tag}
|
{tag}
|
||||||
</span>
|
</span>
|
||||||
))}
|
))}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<span
|
<span
|
||||||
className="font-mono text-xs text-[var(--color-text-label)] group-hover:text-[var(--color-text)] shrink-0 mt-0.5"
|
className="font-mono text-sm text-[var(--color-text-label)] group-hover:text-[var(--color-text)] shrink-0 mt-0.5"
|
||||||
aria-hidden="true"
|
aria-hidden="true"
|
||||||
>
|
>
|
||||||
↗
|
↗
|
||||||
|
|||||||
+57
-11
@@ -3,17 +3,24 @@
|
|||||||
@variant dark (&:where(.dark, .dark *));
|
@variant dark (&:where(.dark, .dark *));
|
||||||
|
|
||||||
@theme {
|
@theme {
|
||||||
/* Terminal-Noir palette */
|
/* macOS Classic Dark (default) */
|
||||||
--color-bg: #0a0a0a;
|
--color-bg: #131313;
|
||||||
--color-surface: #111111;
|
--color-surface: #1e1d1e;
|
||||||
--color-surface-raised: #1a1a1a;
|
--color-surface-raised: #272727;
|
||||||
--color-border: #2a2a2a;
|
--color-border: #3a3a3a;
|
||||||
--color-border-bright: #444444;
|
--color-border-bright: #404040;
|
||||||
--color-text: #e8e8e8;
|
--color-text: #caccca;
|
||||||
--color-text-label: #666666;
|
--color-text-label: #9e9e9e;
|
||||||
--color-text-dim: #444444;
|
--color-text-dim: #8f8f8f;
|
||||||
--color-accent-green: #00cc44;
|
--color-accent-green: #62ba46;
|
||||||
--color-accent-red: #cc2200;
|
--color-accent-red: #c74028;
|
||||||
|
|
||||||
|
/* Timeline type colors — dark */
|
||||||
|
--color-timeline-career: #62ba46;
|
||||||
|
--color-timeline-education: #c28b12;
|
||||||
|
--color-timeline-cert: #c75828;
|
||||||
|
--color-timeline-project: #c72855;
|
||||||
|
--color-timeline-homelab: #e1d797;
|
||||||
|
|
||||||
/* Typography */
|
/* Typography */
|
||||||
--font-mono: "Source Code Pro", ui-monospace, monospace;
|
--font-mono: "Source Code Pro", ui-monospace, monospace;
|
||||||
@@ -22,6 +29,20 @@
|
|||||||
/* Breakpoints */
|
/* Breakpoints */
|
||||||
--breakpoint-xs: 576px;
|
--breakpoint-xs: 576px;
|
||||||
|
|
||||||
|
/* Character-grid spacing — horizontal (ch) */
|
||||||
|
--spacing-1ch: 1ch;
|
||||||
|
--spacing-2ch: 2ch;
|
||||||
|
--spacing-3ch: 3ch;
|
||||||
|
--spacing-4ch: 4ch;
|
||||||
|
|
||||||
|
/* Character-grid spacing — vertical (lh, requires line-height:1.5 on html) */
|
||||||
|
--spacing-qtr-lh: 0.25lh;
|
||||||
|
--spacing-half-lh: 0.5lh;
|
||||||
|
--spacing-1lh: 1lh;
|
||||||
|
--spacing-2lh: 2lh;
|
||||||
|
--spacing-3lh: 3lh;
|
||||||
|
--spacing-4lh: 4lh;
|
||||||
|
|
||||||
/* Animations */
|
/* Animations */
|
||||||
--animate-fade-in: fadeIn 120ms linear forwards;
|
--animate-fade-in: fadeIn 120ms linear forwards;
|
||||||
|
|
||||||
@@ -34,11 +55,16 @@
|
|||||||
/* Base */
|
/* Base */
|
||||||
html {
|
html {
|
||||||
scroll-behavior: smooth;
|
scroll-behavior: smooth;
|
||||||
|
font-size: 15px;
|
||||||
|
line-height: 1.5;
|
||||||
background-color: var(--color-bg);
|
background-color: var(--color-bg);
|
||||||
color: var(--color-text);
|
color: var(--color-text);
|
||||||
font-family: var(--font-mono);
|
font-family: var(--font-mono);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@keyframes blink { 50% { opacity: 0; } }
|
||||||
|
.animate-cursor { animation: blink 1s step-start infinite; }
|
||||||
|
|
||||||
@layer base {
|
@layer base {
|
||||||
* {
|
* {
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
@@ -47,6 +73,26 @@ html {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* macOS Classic Light overrides */
|
||||||
|
:root:not(.dark) {
|
||||||
|
--color-bg: #ffffff;
|
||||||
|
--color-surface: #f9f9f9;
|
||||||
|
--color-surface-raised: #f7f7f7;
|
||||||
|
--color-border: #e0e0e0;
|
||||||
|
--color-border-bright: #d2d2d2;
|
||||||
|
--color-text: #000000;
|
||||||
|
--color-text-label: #505050;
|
||||||
|
--color-text-dim: #929292;
|
||||||
|
--color-accent-green: #036a07;
|
||||||
|
--color-accent-red: #d21f07;
|
||||||
|
|
||||||
|
--color-timeline-career: #036a07;
|
||||||
|
--color-timeline-education: #0433ff;
|
||||||
|
--color-timeline-cert: #957931;
|
||||||
|
--color-timeline-project: #6f42c1;
|
||||||
|
--color-timeline-homelab: #0000a2;
|
||||||
|
}
|
||||||
|
|
||||||
/* Default transitions — linear, fast */
|
/* Default transitions — linear, fast */
|
||||||
a,
|
a,
|
||||||
button {
|
button {
|
||||||
|
|||||||
+119
-64
@@ -21,14 +21,54 @@ const glanceStats = [
|
|||||||
];
|
];
|
||||||
|
|
||||||
const vlans = [
|
const vlans = [
|
||||||
{ id: "1000", name: "MGMT", subnet: "10.0.0.0/24", purpose: "Network equipment only" },
|
{
|
||||||
{ id: "1010", name: "LAN", subnet: "10.1.0.0/24", purpose: "Trusted personal devices" },
|
id: "1000",
|
||||||
{ id: "1020", name: "Homelab", subnet: "10.2.0.0/24", purpose: "All self-hosted services" },
|
name: "MGMT",
|
||||||
{ id: "1030", name: "Guests", subnet: "10.3.0.0/24", purpose: "Internet only, RFC1918 blocked" },
|
subnet: "10.0.0.0/24",
|
||||||
{ id: "1040", name: "IoT", subnet: "10.4.0.0/24", purpose: "Smart home, isolated" },
|
purpose: "Network equipment only",
|
||||||
{ id: "1050", name: "WFH", subnet: "10.5.0.0/24", purpose: "Work devices, no personal access" },
|
},
|
||||||
{ id: "DMZ", name: "DMZ", subnet: "10.99.0.0/24", purpose: "Public-facing, hard-blocked internally" },
|
{
|
||||||
{ id: "VPN", name: "VPN", subnet: "10.200.0.0/24", purpose: "WireGuard clients = LAN access" },
|
id: "1010",
|
||||||
|
name: "LAN",
|
||||||
|
subnet: "10.1.0.0/24",
|
||||||
|
purpose: "Trusted personal devices",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "1020",
|
||||||
|
name: "Homelab",
|
||||||
|
subnet: "10.2.0.0/24",
|
||||||
|
purpose: "All self-hosted services",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "1030",
|
||||||
|
name: "Guests",
|
||||||
|
subnet: "10.3.0.0/24",
|
||||||
|
purpose: "Internet only, RFC1918 blocked",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "1040",
|
||||||
|
name: "IoT",
|
||||||
|
subnet: "10.4.0.0/24",
|
||||||
|
purpose: "Smart home, isolated",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "1050",
|
||||||
|
name: "WFH",
|
||||||
|
subnet: "10.5.0.0/24",
|
||||||
|
purpose: "Work devices, no personal access",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "1099",
|
||||||
|
name: "DMZ",
|
||||||
|
subnet: "10.99.0.0/24",
|
||||||
|
purpose: "Public-facing, hard-blocked internally",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "VPN",
|
||||||
|
name: "VPN",
|
||||||
|
subnet: "10.200.0.0/24",
|
||||||
|
purpose: "WireGuard clients = LAN access",
|
||||||
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
const adrs = [
|
const adrs = [
|
||||||
@@ -69,46 +109,52 @@ const adrs = [
|
|||||||
why: "Avoids 15 separate DB containers. Reduces RAM overhead significantly. All productivity apps share the same LXC (10.2.0.60).",
|
why: "Avoids 15 separate DB containers. Reduces RAM overhead significantly. All productivity apps share the same LXC (10.2.0.60).",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: "Gitea CI/CD: Self-hosted runner with container build + SSH rsync deploy",
|
title:
|
||||||
|
"Gitea CI/CD: Self-hosted runner with container build + SSH rsync deploy",
|
||||||
decision:
|
decision:
|
||||||
"act_runner v0.3.1 on Gitea LXC (10.99.0.22). Push to dev → node:22-alpine container builds Next.js → rsync out/ to Portfolio LXC → SSH docker rebuild.",
|
"act_runner v0.3.1 on Gitea LXC (10.99.0.22). Push to dev → node:22-alpine container builds Next.js → rsync out/ to Portfolio LXC → SSH docker rebuild.",
|
||||||
why: "Keeps the full pipeline internal — no GitHub Actions, no external runners. Build runs in an isolated Alpine container so the Gitea LXC isn't polluted. Portfolio LXC (10.99.0.23) just serves pre-built static files via nginx.",
|
why: "Keeps the full pipeline internal — no GitHub Actions, no external runners. Build runs in an isolated Alpine container so the Gitea LXC isn't polluted. Portfolio LXC (10.99.0.23) just serves pre-built static files via nginx.",
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
title: "Authentik over Authelia",
|
||||||
|
decision: "Authentik as the SSO provider across all self-hosted services.",
|
||||||
|
why: "Full OIDC provider + forward auth in one. Lets services like Outline, Gitea, and Vikunja use real SSO rather than just a login gate. Authelia is forward-auth only — no OIDC provider capability.",
|
||||||
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
export default function HomelabPage() {
|
export default function HomelabPage() {
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
{/* Header */}
|
{/* Header */}
|
||||||
<div className="mb-16">
|
<div className="mb-4lh">
|
||||||
<div className="flex items-center gap-3 mb-4">
|
<p className="font-mono text-sm font-bold text-[var(--color-text)] mb-1lh">
|
||||||
<span className="font-mono text-xs text-[var(--color-text-label)] tracking-widest uppercase">
|
<span
|
||||||
lerkolabs
|
className="text-[var(--color-accent-green)] select-none mr-1ch"
|
||||||
|
aria-hidden="true"
|
||||||
|
>
|
||||||
|
❯
|
||||||
</span>
|
</span>
|
||||||
<div className="flex-1 h-px bg-[var(--color-border)]" aria-hidden="true" />
|
homelab
|
||||||
</div>
|
</p>
|
||||||
<h1 className="font-mono text-lg font-bold text-[var(--color-text)] mb-3">
|
<p className="font-mono text-sm text-[var(--color-text)] leading-relaxed max-w-2xl opacity-80">
|
||||||
Home Infrastructure Lab
|
Personal infrastructure environment for learning, self-hosting, and
|
||||||
</h1>
|
operational practice. Running 24/7 on production-grade hardware with
|
||||||
<p className="font-sans text-sm text-[var(--color-text)] leading-relaxed max-w-2xl opacity-80">
|
real network segmentation, SSO, monitoring, and IaC-style
|
||||||
Personal infrastructure environment for learning, self-hosting, and operational
|
documentation.
|
||||||
practice. Running 24/7 on production-grade hardware with real network segmentation,
|
|
||||||
SSO, monitoring, and IaC-style documentation.
|
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* At a Glance */}
|
{/* At a Glance */}
|
||||||
<Widget title="at a glance" badge={glanceStats.length} as="section">
|
<Widget title="homelab/overview" badge={glanceStats.length} as="section">
|
||||||
<div className="grid grid-cols-1 xs:grid-cols-2 md:grid-cols-3 gap-px bg-[var(--color-border)]">
|
<div className="grid grid-cols-1 xs:grid-cols-2 md:grid-cols-3 gap-px bg-[var(--color-border)]">
|
||||||
{glanceStats.map(({ label, value }) => (
|
{glanceStats.map(({ label, value }) => (
|
||||||
<div
|
<div key={label} className="bg-[var(--color-surface)] px-2ch py-half-lh">
|
||||||
key={label}
|
<p className="font-mono text-sm text-[var(--color-text-dim)] mb-half-lh">
|
||||||
className="bg-[var(--color-surface)] px-4 py-3"
|
|
||||||
>
|
|
||||||
<p className="font-mono text-xs text-[var(--color-text-dim)] uppercase tracking-wider mb-1">
|
|
||||||
{label}
|
{label}
|
||||||
</p>
|
</p>
|
||||||
<p className="font-mono text-sm text-[var(--color-text)]">{value}</p>
|
<p className="font-mono text-sm text-[var(--color-text)]">
|
||||||
|
{value}
|
||||||
|
</p>
|
||||||
</div>
|
</div>
|
||||||
))}
|
))}
|
||||||
</div>
|
</div>
|
||||||
@@ -116,24 +162,24 @@ export default function HomelabPage() {
|
|||||||
|
|
||||||
{/* VLAN table */}
|
{/* VLAN table */}
|
||||||
<Widget
|
<Widget
|
||||||
title="network"
|
title="homelab/network"
|
||||||
meta="8 isolated vlans · default deny inter-vlan"
|
meta="8 isolated vlans · default deny inter-vlan"
|
||||||
as="section"
|
as="section"
|
||||||
>
|
>
|
||||||
<div className="overflow-x-auto">
|
<div className="overflow-x-auto">
|
||||||
<table className="w-full text-xs border-collapse">
|
<table className="w-full text-sm border-collapse">
|
||||||
<thead>
|
<thead>
|
||||||
<tr className="border-b border-[var(--color-border)]">
|
<tr className="border-b border-[var(--color-border)]">
|
||||||
<th className="font-mono text-[var(--color-text-dim)] text-left py-2 pr-6 uppercase tracking-wider">
|
<th className="font-mono text-[var(--color-text-dim)] text-left py-qtr-lh pr-[3ch] uppercase">
|
||||||
VLAN
|
VLAN
|
||||||
</th>
|
</th>
|
||||||
<th className="font-mono text-[var(--color-text-dim)] text-left py-2 pr-6 uppercase tracking-wider">
|
<th className="font-mono text-[var(--color-text-dim)] text-left py-qtr-lh pr-[3ch] uppercase">
|
||||||
Name
|
Name
|
||||||
</th>
|
</th>
|
||||||
<th className="font-mono text-[var(--color-text-dim)] text-left py-2 pr-6 uppercase tracking-wider">
|
<th className="font-mono text-[var(--color-text-dim)] text-left py-qtr-lh pr-[3ch] uppercase">
|
||||||
Subnet
|
Subnet
|
||||||
</th>
|
</th>
|
||||||
<th className="font-mono text-[var(--color-text-dim)] text-left py-2 uppercase tracking-wider">
|
<th className="font-mono text-[var(--color-text-dim)] text-left py-qtr-lh uppercase">
|
||||||
Purpose
|
Purpose
|
||||||
</th>
|
</th>
|
||||||
</tr>
|
</tr>
|
||||||
@@ -144,14 +190,18 @@ export default function HomelabPage() {
|
|||||||
key={v.id}
|
key={v.id}
|
||||||
className="border-b border-[var(--color-border)] hover:bg-[var(--color-surface)]"
|
className="border-b border-[var(--color-border)] hover:bg-[var(--color-surface)]"
|
||||||
>
|
>
|
||||||
<td className="font-mono text-[var(--color-accent-green)] py-2.5 pr-6">
|
<td className="font-mono text-[var(--color-accent-green)] py-half-lh pr-[3ch]">
|
||||||
{v.id}
|
{v.id}
|
||||||
</td>
|
</td>
|
||||||
<td className="font-mono text-[var(--color-text)] py-2.5 pr-6">{v.name}</td>
|
<td className="font-mono text-[var(--color-text)] py-half-lh pr-[3ch]">
|
||||||
<td className="font-mono text-[var(--color-text-label)] py-2.5 pr-6">
|
{v.name}
|
||||||
|
</td>
|
||||||
|
<td className="font-mono text-[var(--color-text-label)] py-half-lh pr-[3ch]">
|
||||||
{v.subnet}
|
{v.subnet}
|
||||||
</td>
|
</td>
|
||||||
<td className="font-sans text-sm text-[var(--color-text)] py-2.5 opacity-80">{v.purpose}</td>
|
<td className="font-mono text-sm text-[var(--color-text)] py-2.5 opacity-80">
|
||||||
|
{v.purpose}
|
||||||
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
))}
|
))}
|
||||||
</tbody>
|
</tbody>
|
||||||
@@ -160,34 +210,30 @@ export default function HomelabPage() {
|
|||||||
</Widget>
|
</Widget>
|
||||||
|
|
||||||
{/* Services */}
|
{/* Services */}
|
||||||
<Widget
|
<Widget title="homelab/services" badge={services.length} as="section">
|
||||||
title="services"
|
<div className="flex flex-col gap-3ch">
|
||||||
badge={services.length}
|
|
||||||
as="section"
|
|
||||||
>
|
|
||||||
<div className="flex flex-col gap-8">
|
|
||||||
{categoryOrder.map((cat) => {
|
{categoryOrder.map((cat) => {
|
||||||
const catServices = services.filter((s) => s.category === cat);
|
const catServices = services.filter((s) => s.category === cat);
|
||||||
return (
|
return (
|
||||||
<div key={cat}>
|
<div key={cat}>
|
||||||
<p className="font-mono text-xs text-[var(--color-text-dim)] uppercase tracking-wider mb-3">
|
<p className="font-mono text-sm text-[var(--color-text-dim)] mb-1lh">
|
||||||
{categoryLabels[cat]}
|
{categoryLabels[cat]}
|
||||||
</p>
|
</p>
|
||||||
<div className="grid grid-cols-1 xs:grid-cols-2 md:grid-cols-3 gap-px bg-[var(--color-border)]">
|
<div className="grid grid-cols-1 xs:grid-cols-2 md:grid-cols-3 gap-px bg-[var(--color-border)]">
|
||||||
{catServices.map((svc) => (
|
{catServices.map((svc) => (
|
||||||
<div
|
<div
|
||||||
key={svc.name}
|
key={svc.name}
|
||||||
className="bg-[var(--color-surface)] hover:bg-[var(--color-surface-raised)] flex items-start gap-3 px-4 py-3"
|
className="bg-[var(--color-surface)] hover:bg-[var(--color-surface-raised)] flex items-start gap-1ch px-2ch py-half-lh"
|
||||||
>
|
>
|
||||||
<i
|
<i
|
||||||
className={`${svc.icon} text-[var(--color-text-label)] text-xs mt-0.5 w-3.5 shrink-0`}
|
className={`${svc.icon} text-[var(--color-text-label)] text-xs mt-0.5 w-3.5 shrink-0`}
|
||||||
aria-hidden="true"
|
aria-hidden="true"
|
||||||
/>
|
/>
|
||||||
<div>
|
<div>
|
||||||
<p className="font-mono text-xs text-[var(--color-text)] mb-0.5">
|
<p className="font-mono text-sm text-[var(--color-text)] mb-0.5">
|
||||||
{svc.name}
|
{svc.name}
|
||||||
</p>
|
</p>
|
||||||
<p className="font-sans text-sm text-[var(--color-text)] leading-relaxed opacity-75">
|
<p className="font-mono text-sm text-[var(--color-text)] leading-relaxed opacity-75">
|
||||||
{svc.description}
|
{svc.description}
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
@@ -202,7 +248,7 @@ export default function HomelabPage() {
|
|||||||
|
|
||||||
{/* ADRs */}
|
{/* ADRs */}
|
||||||
<Widget
|
<Widget
|
||||||
title="architecture decisions"
|
title="homelab/ADRs"
|
||||||
meta="why things are configured the way they are"
|
meta="why things are configured the way they are"
|
||||||
badge={adrs.length}
|
badge={adrs.length}
|
||||||
as="section"
|
as="section"
|
||||||
@@ -211,15 +257,21 @@ export default function HomelabPage() {
|
|||||||
{adrs.map((adr) => (
|
{adrs.map((adr) => (
|
||||||
<div
|
<div
|
||||||
key={adr.title}
|
key={adr.title}
|
||||||
className="bg-[var(--color-surface)] hover:bg-[var(--color-surface-raised)] px-4 py-4"
|
className="bg-[var(--color-surface)] hover:bg-[var(--color-surface-raised)] px-2ch py-1lh"
|
||||||
>
|
>
|
||||||
<p className="font-mono text-sm text-[var(--color-text)] mb-2">{adr.title}</p>
|
<p className="font-mono text-sm text-[var(--color-text)] mb-1lh">
|
||||||
<p className="font-sans text-sm text-[var(--color-text)] leading-relaxed mb-1.5 opacity-75">
|
{adr.title}
|
||||||
<span className="text-[var(--color-text-label)] opacity-100">decision: </span>
|
</p>
|
||||||
|
<p className="font-mono text-sm text-[var(--color-text)] leading-relaxed mb-half-lh opacity-75">
|
||||||
|
<span className="text-[var(--color-text-label)] opacity-100">
|
||||||
|
decision:{" "}
|
||||||
|
</span>
|
||||||
{adr.decision}
|
{adr.decision}
|
||||||
</p>
|
</p>
|
||||||
<p className="font-sans text-sm text-[var(--color-text)] leading-relaxed opacity-75">
|
<p className="font-mono text-sm text-[var(--color-text)] leading-relaxed opacity-75">
|
||||||
<span className="text-[var(--color-text-label)] opacity-100">why: </span>
|
<span className="text-[var(--color-text-label)] opacity-100">
|
||||||
|
why:{" "}
|
||||||
|
</span>
|
||||||
{adr.why}
|
{adr.why}
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
@@ -227,19 +279,22 @@ export default function HomelabPage() {
|
|||||||
</div>
|
</div>
|
||||||
</Widget>
|
</Widget>
|
||||||
|
|
||||||
{/* GitHub CTA */}
|
{/* Gitea CTA */}
|
||||||
<section className="border-t border-[var(--color-border)] pt-6">
|
<section className="pt-qtr-lh">
|
||||||
<p className="font-mono text-sm text-[var(--color-text-label)] mb-1">lerkolabs on GitHub</p>
|
<p className="font-mono text-sm text-[var(--color-text-dim)] mb-half-lh">
|
||||||
<p className="font-sans text-sm text-[var(--color-text)] mb-3 opacity-75">
|
homelab/docs
|
||||||
Full documentation: VLAN maps, runbooks, service registry, config exports, and setup guides.
|
</p>
|
||||||
|
<p className="font-mono text-sm text-[var(--color-text)] mb-1lh opacity-75">
|
||||||
|
VLAN maps, runbooks, service registry, config exports, and setup
|
||||||
|
guides.
|
||||||
</p>
|
</p>
|
||||||
<a
|
<a
|
||||||
href="https://github.com/lerko96/homelab-wip"
|
href="https://gitea.lerkolabs.com/lerko/homelab"
|
||||||
target="_blank"
|
target="_blank"
|
||||||
rel="noopener noreferrer"
|
rel="noopener noreferrer"
|
||||||
className="font-mono text-xs text-[var(--color-text-label)] hover:text-[var(--color-text)]"
|
className="font-mono text-sm text-[var(--color-text-label)] hover:text-[var(--color-text)]"
|
||||||
>
|
>
|
||||||
↗ github.com/lerko96/homelab-wip
|
↗ gitea.lerkolabs.com/lerko/homelab
|
||||||
</a>
|
</a>
|
||||||
</section>
|
</section>
|
||||||
</>
|
</>
|
||||||
|
|||||||
+1
-7
@@ -27,12 +27,6 @@ export default function RootLayout({
|
|||||||
<html lang="en" className="dark">
|
<html lang="en" className="dark">
|
||||||
<head>
|
<head>
|
||||||
<ThemeScript />
|
<ThemeScript />
|
||||||
<link
|
|
||||||
rel="stylesheet"
|
|
||||||
href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/6.5.0/css/all.min.css"
|
|
||||||
crossOrigin="anonymous"
|
|
||||||
referrerPolicy="no-referrer"
|
|
||||||
/>
|
|
||||||
</head>
|
</head>
|
||||||
<body
|
<body
|
||||||
className={`${sourceCodePro.variable} bg-[var(--color-bg)] text-[var(--color-text)] font-mono min-h-screen`}
|
className={`${sourceCodePro.variable} bg-[var(--color-bg)] text-[var(--color-text)] font-mono min-h-screen`}
|
||||||
@@ -43,7 +37,7 @@ export default function RootLayout({
|
|||||||
|
|
||||||
{/* Centered content column — border-l/r makes centering always visible */}
|
{/* Centered content column — border-l/r makes centering always visible */}
|
||||||
<div className="max-w-[740px] mx-auto border-l border-r border-[var(--color-border)]">
|
<div className="max-w-[740px] mx-auto border-l border-r border-[var(--color-border)]">
|
||||||
<main className="px-8 py-14">
|
<main className="px-4ch py-3lh">
|
||||||
{children}
|
{children}
|
||||||
</main>
|
</main>
|
||||||
<Footer />
|
<Footer />
|
||||||
|
|||||||
+5
-3
@@ -1,6 +1,7 @@
|
|||||||
import type { Metadata } from "next";
|
import type { Metadata } from "next";
|
||||||
import Hero from "@/components/Hero";
|
import Hero from "@/components/Hero";
|
||||||
import Skills from "@/components/Skills";
|
import Skills from "@/components/Skills";
|
||||||
|
import Timeline from "@/components/Timeline";
|
||||||
import ProjectCard from "@/components/ProjectCard";
|
import ProjectCard from "@/components/ProjectCard";
|
||||||
import Widget from "@/components/Widget";
|
import Widget from "@/components/Widget";
|
||||||
import { featuredProjects } from "@/data/projects";
|
import { featuredProjects } from "@/data/projects";
|
||||||
@@ -15,14 +16,15 @@ export default function Home() {
|
|||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<Hero />
|
<Hero />
|
||||||
<Skills />
|
<Timeline />
|
||||||
<Widget title="projects" badge={featuredProjects.length}>
|
<Widget title="tyler/projects" badge={featuredProjects.length}>
|
||||||
<div className="grid grid-cols-1 sm:grid-cols-2 gap-3">
|
<div className="grid grid-cols-1 sm:grid-cols-2 gap-1ch">
|
||||||
{featuredProjects.map((project) => (
|
{featuredProjects.map((project) => (
|
||||||
<ProjectCard key={project.slug} project={project} />
|
<ProjectCard key={project.slug} project={project} />
|
||||||
))}
|
))}
|
||||||
</div>
|
</div>
|
||||||
</Widget>
|
</Widget>
|
||||||
|
<Skills />
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
+24
-10
@@ -1,30 +1,44 @@
|
|||||||
export default function Footer() {
|
export default function Footer() {
|
||||||
return (
|
return (
|
||||||
<footer className="border-t border-[var(--color-border)] py-5 mt-8">
|
<footer className="border-t border-[var(--color-border)] py-1lh mt-2lh">
|
||||||
<div className="px-8 flex items-center justify-between">
|
<div className="px-4ch flex items-center justify-between">
|
||||||
<span className="font-mono text-xs text-[var(--color-text-dim)]">
|
<span className="font-mono text-sm text-[var(--color-text-dim)]">
|
||||||
© {new Date().getFullYear()} Tyler Koenig
|
© {new Date().getFullYear()} Tyler Koenig
|
||||||
</span>
|
</span>
|
||||||
<div className="flex items-center gap-5">
|
<div className="flex items-center gap-2ch">
|
||||||
<a
|
<a
|
||||||
href="https://github.com/lerko96"
|
href="https://github.com/lerko96"
|
||||||
target="_blank"
|
target="_blank"
|
||||||
rel="noopener noreferrer"
|
rel="noopener noreferrer"
|
||||||
aria-label="GitHub"
|
aria-label="GitHub"
|
||||||
className="font-mono text-xs text-[var(--color-text-label)] hover:text-[var(--color-text)]"
|
className="font-mono text-sm text-[var(--color-text-label)] hover:text-[var(--color-text)]"
|
||||||
>
|
>
|
||||||
<i className="fab fa-github mr-1.5" aria-hidden="true" />
|
[github]
|
||||||
github
|
</a>
|
||||||
|
<a
|
||||||
|
href="https://gitea.lerkolabs.com/lerko"
|
||||||
|
target="_blank"
|
||||||
|
rel="noopener noreferrer"
|
||||||
|
aria-label="Gitea"
|
||||||
|
className="font-mono text-sm text-[var(--color-text-label)] hover:text-[var(--color-text)]"
|
||||||
|
>
|
||||||
|
[gitea]
|
||||||
</a>
|
</a>
|
||||||
<a
|
<a
|
||||||
href="https://www.linkedin.com/in/tyler-koenig"
|
href="https://www.linkedin.com/in/tyler-koenig"
|
||||||
target="_blank"
|
target="_blank"
|
||||||
rel="noopener noreferrer"
|
rel="noopener noreferrer"
|
||||||
aria-label="LinkedIn"
|
aria-label="LinkedIn"
|
||||||
className="font-mono text-xs text-[var(--color-text-label)] hover:text-[var(--color-text)]"
|
className="font-mono text-sm text-[var(--color-text-label)] hover:text-[var(--color-text)]"
|
||||||
>
|
>
|
||||||
<i className="fab fa-linkedin mr-1.5" aria-hidden="true" />
|
[linkedin]
|
||||||
linkedin
|
</a>
|
||||||
|
<a
|
||||||
|
href="mailto:tyler@lerkolabs.com"
|
||||||
|
aria-label="Email"
|
||||||
|
className="font-mono text-sm text-[var(--color-text-label)] hover:text-[var(--color-text)]"
|
||||||
|
>
|
||||||
|
[email]
|
||||||
</a>
|
</a>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
+66
-68
@@ -1,75 +1,73 @@
|
|||||||
import Image from "next/image";
|
|
||||||
|
|
||||||
export default function Hero() {
|
export default function Hero() {
|
||||||
return (
|
return (
|
||||||
<section className="mb-16">
|
<section className="mb-16">
|
||||||
{/* Section header */}
|
<div className="flex flex-col gap-1ch">
|
||||||
<div className="flex items-center gap-3 mb-6">
|
<div>
|
||||||
<span className="font-mono text-xs text-[var(--color-text-label)] tracking-widest uppercase">
|
<p className="font-mono text-sm font-bold text-[var(--color-text)]">
|
||||||
identity
|
<span
|
||||||
</span>
|
className="text-[var(--color-accent-green)] select-none mr-1ch"
|
||||||
<div className="flex-1 h-px bg-[var(--color-border)]" aria-hidden="true" />
|
aria-hidden="true"
|
||||||
</div>
|
>
|
||||||
|
❯
|
||||||
<div className="flex items-start gap-5">
|
|
||||||
<Image
|
|
||||||
src="/images/headshot-tyler_koenig.png"
|
|
||||||
alt="Tyler Koenig"
|
|
||||||
width={56}
|
|
||||||
height={56}
|
|
||||||
className="shrink-0 object-cover"
|
|
||||||
priority
|
|
||||||
/>
|
|
||||||
|
|
||||||
<div className="flex flex-col gap-3">
|
|
||||||
<div>
|
|
||||||
<p className="font-mono text-base font-bold text-[var(--color-text)]">
|
|
||||||
Tyler Koenig
|
|
||||||
</p>
|
|
||||||
<p className="font-mono text-xs text-[var(--color-text-label)] mt-0.5">
|
|
||||||
SOC Helpdesk I · Homelab Operator
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<p className="font-sans text-sm text-[var(--color-text)] leading-relaxed max-w-lg opacity-80">
|
|
||||||
I write software and run infrastructure that goes well past what my
|
|
||||||
job title implies. Games, AI tooling, mobile apps, and a homelab
|
|
||||||
running 20+ self-hosted services on segmented VLANs. Continuously
|
|
||||||
learning by building things that actually work.
|
|
||||||
</p>
|
|
||||||
|
|
||||||
<div className="flex flex-wrap items-center gap-x-5 gap-y-1">
|
|
||||||
<span className="font-mono text-xs text-[var(--color-accent-green)]">
|
|
||||||
● online
|
|
||||||
</span>
|
</span>
|
||||||
<a
|
tyler koenig
|
||||||
href="https://github.com/lerko96"
|
</p>
|
||||||
target="_blank"
|
<p className="font-mono text-sm text-[var(--color-text-label)] mt-0.5">
|
||||||
rel="noopener noreferrer"
|
Security Operations · Self-Hosted Infrastructure
|
||||||
aria-label="GitHub"
|
</p>
|
||||||
className="font-mono text-xs text-[var(--color-text-label)] hover:text-[var(--color-text)]"
|
</div>
|
||||||
>
|
|
||||||
<i className="fab fa-github mr-1.5" aria-hidden="true" />
|
<p className="font-mono text-sm text-[var(--color-text)] leading-relaxed opacity-70">
|
||||||
github
|
Homelab runs 30+
|
||||||
</a>
|
services across segmented VLANs — pfSense, Authentik SSO, full
|
||||||
<a
|
observability stack. Write software too: mobile apps, Go backends,
|
||||||
href="https://www.linkedin.com/in/tyler-koenig"
|
open protocols. Daily drivers, all of it.{" "}
|
||||||
target="_blank"
|
<span
|
||||||
rel="noopener noreferrer"
|
className="animate-cursor text-[var(--color-accent-green)]"
|
||||||
aria-label="LinkedIn"
|
aria-hidden="true"
|
||||||
className="font-mono text-xs text-[var(--color-text-label)] hover:text-[var(--color-text)]"
|
>
|
||||||
>
|
█
|
||||||
<i className="fab fa-linkedin mr-1.5" aria-hidden="true" />
|
</span>
|
||||||
linkedin
|
</p>
|
||||||
</a>
|
|
||||||
<a
|
<div className="flex flex-wrap items-center gap-x-1ch gap-y-half-lh">
|
||||||
href="mailto:tylerkoenig96@gmail.com"
|
<span className="font-mono text-sm text-[var(--color-accent-green)]">
|
||||||
aria-label="Email"
|
● available
|
||||||
className="font-mono text-xs text-[var(--color-text-label)] hover:text-[var(--color-text)]"
|
</span>
|
||||||
>
|
<a
|
||||||
@ email
|
href="https://github.com/lerko96"
|
||||||
</a>
|
target="_blank"
|
||||||
</div>
|
rel="noopener noreferrer"
|
||||||
|
aria-label="GitHub"
|
||||||
|
className="font-mono text-sm text-[var(--color-text-label)] hover:text-[var(--color-text)]"
|
||||||
|
>
|
||||||
|
[github]
|
||||||
|
</a>
|
||||||
|
<a
|
||||||
|
href="https://gitea.lerkolabs.com/lerko"
|
||||||
|
target="_blank"
|
||||||
|
rel="noopener noreferrer"
|
||||||
|
aria-label="Gitea"
|
||||||
|
className="font-mono text-sm text-[var(--color-text-label)] hover:text-[var(--color-text)]"
|
||||||
|
>
|
||||||
|
[gitea]
|
||||||
|
</a>
|
||||||
|
<a
|
||||||
|
href="https://www.linkedin.com/in/tyler-koenig"
|
||||||
|
target="_blank"
|
||||||
|
rel="noopener noreferrer"
|
||||||
|
aria-label="LinkedIn"
|
||||||
|
className="font-mono text-sm text-[var(--color-text-label)] hover:text-[var(--color-text)]"
|
||||||
|
>
|
||||||
|
[linkedin]
|
||||||
|
</a>
|
||||||
|
<a
|
||||||
|
href="mailto:tyler@lerkolabs.com"
|
||||||
|
aria-label="Email"
|
||||||
|
className="font-mono text-sm text-[var(--color-text-label)] hover:text-[var(--color-text)]"
|
||||||
|
>
|
||||||
|
[email]
|
||||||
|
</a>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</section>
|
</section>
|
||||||
|
|||||||
+17
-6
@@ -2,26 +2,28 @@
|
|||||||
|
|
||||||
import Link from "next/link";
|
import Link from "next/link";
|
||||||
import { usePathname } from "next/navigation";
|
import { usePathname } from "next/navigation";
|
||||||
|
import { useTheme } from "@/context/ThemeContext";
|
||||||
|
|
||||||
const links = [
|
const links = [
|
||||||
{ href: "/", label: "home" },
|
{ href: "/", label: "tyler" },
|
||||||
{ href: "/homelab/", label: "homelab" },
|
{ href: "/homelab/", label: "homelab" },
|
||||||
{ href: "/archive/", label: "archive" },
|
{ href: "/archive/", label: "archive" },
|
||||||
];
|
];
|
||||||
|
|
||||||
export default function Nav() {
|
export default function Nav() {
|
||||||
const pathname = usePathname();
|
const pathname = usePathname();
|
||||||
|
const { isDark, toggle } = useTheme();
|
||||||
return (
|
return (
|
||||||
<header className="sticky top-0 z-50 bg-[var(--color-surface)] border-b border-[var(--color-border)]">
|
<header className="sticky top-0 z-50 bg-[var(--color-surface)] border-b border-[var(--color-border)]">
|
||||||
<nav className="max-w-[740px] mx-auto px-8 h-11 flex items-center justify-between">
|
<nav className="max-w-[740px] mx-auto px-4ch h-11 flex items-center justify-between">
|
||||||
<Link
|
<Link
|
||||||
href="/"
|
href="/"
|
||||||
className="font-mono text-sm font-bold text-[var(--color-text)] tracking-widest hover:text-[var(--color-text-label)]"
|
className="font-mono text-sm font-bold text-[var(--color-text)] hover:text-[var(--color-text-label)]"
|
||||||
>
|
>
|
||||||
tk
|
~/
|
||||||
</Link>
|
</Link>
|
||||||
|
|
||||||
<ul className="flex gap-6">
|
<ul className="flex items-center gap-2ch">
|
||||||
{links.map(({ href, label }) => {
|
{links.map(({ href, label }) => {
|
||||||
const active =
|
const active =
|
||||||
pathname === href || pathname === href.replace(/\/$/, "");
|
pathname === href || pathname === href.replace(/\/$/, "");
|
||||||
@@ -30,7 +32,7 @@ export default function Nav() {
|
|||||||
<Link
|
<Link
|
||||||
href={href}
|
href={href}
|
||||||
aria-current={active ? "page" : undefined}
|
aria-current={active ? "page" : undefined}
|
||||||
className={`font-mono text-xs tracking-widest ${
|
className={`font-mono text-sm ${
|
||||||
active
|
active
|
||||||
? "text-[var(--color-text)]"
|
? "text-[var(--color-text)]"
|
||||||
: "text-[var(--color-text-label)] hover:text-[var(--color-text)]"
|
: "text-[var(--color-text-label)] hover:text-[var(--color-text)]"
|
||||||
@@ -41,6 +43,15 @@ export default function Nav() {
|
|||||||
</li>
|
</li>
|
||||||
);
|
);
|
||||||
})}
|
})}
|
||||||
|
<li>
|
||||||
|
<button
|
||||||
|
onClick={toggle}
|
||||||
|
aria-label={isDark ? "Switch to light mode" : "Switch to dark mode"}
|
||||||
|
className="font-mono text-sm text-[var(--color-text-label)] hover:text-[var(--color-text)] cursor-pointer"
|
||||||
|
>
|
||||||
|
{isDark ? "[light]" : "[dark]"}
|
||||||
|
</button>
|
||||||
|
</li>
|
||||||
</ul>
|
</ul>
|
||||||
</nav>
|
</nav>
|
||||||
</header>
|
</header>
|
||||||
|
|||||||
@@ -6,8 +6,8 @@ type Props = {
|
|||||||
|
|
||||||
export default function ProjectCard({ project }: Props) {
|
export default function ProjectCard({ project }: Props) {
|
||||||
return (
|
return (
|
||||||
<article className="border border-[var(--color-border)] bg-[var(--color-surface)] hover:border-[var(--color-border-bright)] flex flex-col gap-4 p-5">
|
<article className="border border-[var(--color-border)] bg-[var(--color-surface)] flex flex-col gap-1lh p-2ch hover:bg-[var(--color-surface-raised)]">
|
||||||
<div className="flex items-start justify-between gap-3">
|
<div className="flex items-start justify-between gap-1ch">
|
||||||
<a
|
<a
|
||||||
href={project.githubUrl}
|
href={project.githubUrl}
|
||||||
target="_blank"
|
target="_blank"
|
||||||
@@ -16,31 +16,48 @@ export default function ProjectCard({ project }: Props) {
|
|||||||
>
|
>
|
||||||
{project.title}
|
{project.title}
|
||||||
</a>
|
</a>
|
||||||
<div className="flex items-center gap-3 shrink-0">
|
<div className="flex items-center gap-1ch shrink-0">
|
||||||
{project.stats && (
|
{project.stats && (
|
||||||
<span className="font-mono text-xs text-[var(--color-text-dim)]">
|
<span className="font-mono text-sm text-[var(--color-text-dim)]">
|
||||||
{project.stats}
|
{project.stats}
|
||||||
</span>
|
</span>
|
||||||
)}
|
)}
|
||||||
|
{project.externalUrl && (
|
||||||
|
<a
|
||||||
|
href={project.externalUrl}
|
||||||
|
target="_blank"
|
||||||
|
rel="noopener noreferrer"
|
||||||
|
aria-label={`View ${project.title} externally`}
|
||||||
|
className="font-mono text-sm text-[var(--color-text-label)] hover:text-[var(--color-text)]"
|
||||||
|
>
|
||||||
|
↗
|
||||||
|
</a>
|
||||||
|
)}
|
||||||
<a
|
<a
|
||||||
href={project.githubUrl}
|
href={project.githubUrl}
|
||||||
target="_blank"
|
target="_blank"
|
||||||
rel="noopener noreferrer"
|
rel="noopener noreferrer"
|
||||||
aria-label={`View ${project.title} on GitHub`}
|
aria-label={`View ${project.title} on GitHub`}
|
||||||
className="font-mono text-xs text-[var(--color-text-label)] hover:text-[var(--color-text)]"
|
className="font-mono text-sm text-[var(--color-text-label)] hover:text-[var(--color-text)]"
|
||||||
>
|
>
|
||||||
↗
|
↗
|
||||||
</a>
|
</a>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<p className="font-sans text-sm text-[var(--color-text)] leading-relaxed flex-1 opacity-75">
|
{project.statusBadge && (
|
||||||
|
<span className="font-mono text-sm text-[var(--color-accent-amber,#d4a027)] border border-[var(--color-accent-amber,#d4a027)] px-1ch py-0.5 w-fit opacity-80">
|
||||||
|
{project.statusBadge}
|
||||||
|
</span>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<p className="font-mono text-sm text-[var(--color-text)] leading-relaxed flex-1 opacity-70">
|
||||||
{project.description}
|
{project.description}
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
<div className="flex flex-wrap gap-x-3 gap-y-1 pt-2 border-t border-[var(--color-border)]">
|
<div className="flex flex-wrap gap-x-1ch gap-y-half-lh mt-half-lh">
|
||||||
{project.tags.map((tag) => (
|
{project.tags.map((tag) => (
|
||||||
<span key={tag} className="font-mono text-xs text-[var(--color-text-dim)]">
|
<span key={tag} className="font-mono text-sm text-[var(--color-text-dim)]">
|
||||||
{tag}
|
{tag}
|
||||||
</span>
|
</span>
|
||||||
))}
|
))}
|
||||||
|
|||||||
+16
-18
@@ -1,42 +1,40 @@
|
|||||||
import Widget from "@/components/Widget";
|
import Widget from "@/components/Widget";
|
||||||
|
|
||||||
const skillGroups = [
|
const skillGroups = [
|
||||||
{
|
|
||||||
label: "Languages",
|
|
||||||
skills: ["JavaScript", "TypeScript", "HTML", "CSS"],
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: "Frontend",
|
|
||||||
skills: ["React", "React Native", "Expo", "Next.js", "Three.js"],
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: "Desktop & Tools",
|
|
||||||
skills: ["Electron", "Node.js", "REST APIs", "Git", "Docker", "TDD"],
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
label: "Infrastructure",
|
label: "Infrastructure",
|
||||||
skills: ["Proxmox", "pfSense", "VLANs", "WireGuard", "Linux", "Caddy"],
|
skills: ["Proxmox", "pfSense", "VLANs", "WireGuard", "Linux", "Caddy"],
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
label: "Desktop & Tools",
|
||||||
|
skills: ["Git", "Docker", "TDD", "Node.js", "REST APIs", ],
|
||||||
|
},
|
||||||
{
|
{
|
||||||
label: "Practices",
|
label: "Practices",
|
||||||
skills: ["Agile / Scrum", "Relational Databases", "Self-hosting"],
|
skills: ["Agile / Scrum", "Relational Databases", "Self-hosting"],
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
label: "Languages",
|
||||||
|
skills: ["Go", "JavaScript", "TypeScript", "HTML", "CSS"],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: "Frontend",
|
||||||
|
skills: ["React", "React Native", "Expo", "Next.js", "Three.js"],
|
||||||
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
const totalCount = skillGroups.reduce((n, g) => n + g.skills.length, 0);
|
const totalCount = skillGroups.reduce((n, g) => n + g.skills.length, 0);
|
||||||
|
|
||||||
export default function Skills() {
|
export default function Skills() {
|
||||||
return (
|
return (
|
||||||
<Widget title="skills" badge={totalCount} as="section">
|
<Widget title="tyler/skills" badge={totalCount} as="section">
|
||||||
<div className="flex flex-col">
|
<div className="flex flex-col">
|
||||||
{skillGroups.map(({ label, skills }, i) => (
|
{skillGroups.map(({ label, skills }) => (
|
||||||
<div
|
<div
|
||||||
key={label}
|
key={label}
|
||||||
className={`flex flex-col xs:flex-row gap-1 xs:gap-6 py-4 ${
|
className="flex flex-col xs:flex-row gap-1ch xs:gap-2ch py-half-lh"
|
||||||
i < skillGroups.length - 1 ? "border-b border-[var(--color-border)]" : ""
|
|
||||||
}`}
|
|
||||||
>
|
>
|
||||||
<span className="font-mono text-xs text-[var(--color-text-dim)] w-28 shrink-0 uppercase tracking-wider">
|
<span className="font-mono text-sm text-[var(--color-text-dim)] w-28 shrink-0">
|
||||||
{label}
|
{label}
|
||||||
</span>
|
</span>
|
||||||
<span className="font-mono text-sm text-[var(--color-text)]">
|
<span className="font-mono text-sm text-[var(--color-text)]">
|
||||||
|
|||||||
@@ -0,0 +1,107 @@
|
|||||||
|
'use client'
|
||||||
|
|
||||||
|
import { useEffect, useRef } from 'react'
|
||||||
|
import Widget from '@/components/Widget'
|
||||||
|
import { timeline, type TimelineType } from '@/data/timeline'
|
||||||
|
|
||||||
|
const typeColor: Record<TimelineType, string> = {
|
||||||
|
career: 'var(--color-timeline-career)',
|
||||||
|
education: 'var(--color-timeline-education)',
|
||||||
|
cert: 'var(--color-timeline-cert)',
|
||||||
|
project: 'var(--color-timeline-project)',
|
||||||
|
homelab: 'var(--color-timeline-homelab)',
|
||||||
|
}
|
||||||
|
|
||||||
|
const typeLabel: Record<TimelineType, string> = {
|
||||||
|
career: 'career',
|
||||||
|
education: 'education',
|
||||||
|
cert: 'cert',
|
||||||
|
project: 'project',
|
||||||
|
homelab: 'homelab',
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function Timeline() {
|
||||||
|
const listRef = useRef<HTMLOListElement>(null)
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (window.matchMedia('(prefers-reduced-motion: reduce)').matches) return
|
||||||
|
|
||||||
|
const entries = listRef.current?.querySelectorAll<HTMLLIElement>('[data-tl-entry]')
|
||||||
|
if (!entries) return
|
||||||
|
|
||||||
|
const observer = new IntersectionObserver(
|
||||||
|
(observed) => {
|
||||||
|
observed.forEach((entry) => {
|
||||||
|
if (entry.isIntersecting) {
|
||||||
|
;(entry.target as HTMLElement).style.opacity = '1'
|
||||||
|
;(entry.target as HTMLElement).style.transform = 'translateY(0)'
|
||||||
|
observer.unobserve(entry.target)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
},
|
||||||
|
{ threshold: 0.15 },
|
||||||
|
)
|
||||||
|
|
||||||
|
entries.forEach((el) => {
|
||||||
|
el.style.opacity = '0'
|
||||||
|
el.style.transform = 'translateY(8px)'
|
||||||
|
el.style.transition = 'opacity 240ms linear, transform 240ms linear'
|
||||||
|
observer.observe(el)
|
||||||
|
})
|
||||||
|
|
||||||
|
return () => observer.disconnect()
|
||||||
|
}, [])
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Widget title="tyler/journey">
|
||||||
|
<ol ref={listRef} className="relative border-l border-[var(--color-border)] ml-[2px] flex flex-col gap-0">
|
||||||
|
{timeline.map((entry, i) => (
|
||||||
|
<li key={i} data-tl-entry className="pl-[3ch] pb-2lh last:pb-0 relative">
|
||||||
|
{/* Spine dot */}
|
||||||
|
<span
|
||||||
|
className="absolute -left-[7px] top-[3px] w-3 h-3 rounded-full border border-[var(--color-bg)] shrink-0"
|
||||||
|
style={{ backgroundColor: typeColor[entry.type] }}
|
||||||
|
aria-hidden="true"
|
||||||
|
/>
|
||||||
|
|
||||||
|
{/* Date + type badge */}
|
||||||
|
<div className="flex items-center gap-1ch mb-half-lh">
|
||||||
|
<span className="font-mono text-sm text-[var(--color-text-dim)]">{entry.date}</span>
|
||||||
|
<span
|
||||||
|
className="font-mono text-sm px-1 border"
|
||||||
|
style={{
|
||||||
|
color: typeColor[entry.type],
|
||||||
|
borderColor: typeColor[entry.type],
|
||||||
|
opacity: 0.7,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{typeLabel[entry.type]}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Title */}
|
||||||
|
<p className="font-mono text-sm font-semibold text-[var(--color-text)] mb-half-lh">
|
||||||
|
{entry.title}
|
||||||
|
</p>
|
||||||
|
|
||||||
|
{/* Description */}
|
||||||
|
<p className="font-mono text-sm text-[var(--color-text)] opacity-70 leading-relaxed mb-half-lh">
|
||||||
|
{entry.description}
|
||||||
|
</p>
|
||||||
|
|
||||||
|
{/* Tags */}
|
||||||
|
{entry.tags && entry.tags.length > 0 && (
|
||||||
|
<div className="flex flex-wrap gap-x-1ch gap-y-half-lh">
|
||||||
|
{entry.tags.map((tag) => (
|
||||||
|
<span key={tag} className="font-mono text-sm text-[var(--color-text-dim)]">
|
||||||
|
{tag}
|
||||||
|
</span>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</li>
|
||||||
|
))}
|
||||||
|
</ol>
|
||||||
|
</Widget>
|
||||||
|
)
|
||||||
|
}
|
||||||
+13
-13
@@ -15,22 +15,22 @@ export default function Widget({
|
|||||||
className,
|
className,
|
||||||
children,
|
children,
|
||||||
}: WidgetProps) {
|
}: WidgetProps) {
|
||||||
|
const slashIdx = title.lastIndexOf("/");
|
||||||
|
const prefix = slashIdx >= 0 ? title.slice(0, slashIdx + 1) : null;
|
||||||
|
const name = slashIdx >= 0 ? title.slice(slashIdx + 1) : title;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Tag className={`mb-16 ${className ?? ""}`}>
|
<Tag className={`mb-4lh ${className ?? ""}`}>
|
||||||
<div className="flex items-center gap-3 mb-8">
|
<div className="flex items-center gap-1ch mb-2lh">
|
||||||
<span className="font-mono text-xs text-[var(--color-text-label)] tracking-widest uppercase whitespace-nowrap">
|
{prefix && (
|
||||||
{title}
|
<span className="font-mono text-sm text-[var(--color-text-dim)] select-none">{prefix}</span>
|
||||||
</span>
|
|
||||||
{meta && (
|
|
||||||
<span className="font-mono text-xs text-[var(--color-text-dim)] whitespace-nowrap">
|
|
||||||
{meta}
|
|
||||||
</span>
|
|
||||||
)}
|
)}
|
||||||
<div className="flex-1 h-px bg-[var(--color-border)]" aria-hidden="true" />
|
<span className="font-mono text-sm font-semibold text-[var(--color-text)]">{name}</span>
|
||||||
{badge !== undefined && (
|
{badge !== undefined && (
|
||||||
<span className="font-mono text-xs text-[var(--color-text-dim)] whitespace-nowrap">
|
<span className="font-mono text-sm text-[var(--color-text-dim)]">[{badge}]</span>
|
||||||
{badge}
|
)}
|
||||||
</span>
|
{meta && (
|
||||||
|
<span className="font-mono text-sm text-[var(--color-text-dim)]">— {meta}</span>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
{children}
|
{children}
|
||||||
|
|||||||
+98
-32
@@ -7,10 +7,63 @@ export type Project = {
|
|||||||
tier: "featured" | "archive";
|
tier: "featured" | "archive";
|
||||||
stats?: string;
|
stats?: string;
|
||||||
year?: number;
|
year?: number;
|
||||||
|
statusBadge?: string;
|
||||||
|
externalUrl?: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
export const projects: Project[] = [
|
export const projects: Project[] = [
|
||||||
// --- Featured ---
|
// --- Featured ---
|
||||||
|
{
|
||||||
|
slug: "homelab",
|
||||||
|
title: "homelab",
|
||||||
|
description:
|
||||||
|
"8-VLAN segmented network, Proxmox VMs/LXCs, SSO via Authentik, full monitoring stack (VictoriaMetrics + Grafana + Beszel + ntfy).",
|
||||||
|
tags: ["Markdown", "Mermaid", "Proxmox", "Monitor", "Backup"],
|
||||||
|
githubUrl: "https://gitea.lerkolabs.com/lerko/homelab",
|
||||||
|
tier: "featured",
|
||||||
|
year: 2026,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
slug: "open-pact",
|
||||||
|
title: "open-pact",
|
||||||
|
description:
|
||||||
|
"Open protocol for AI agent identity, delegation, and portable memory. Ed25519 keypair identity, signed delegation warrants, portable signed memory facts. No central registry.",
|
||||||
|
tags: ["TypeScript", "Ed25519", "DID", "npm", "CC0"],
|
||||||
|
githubUrl: "https://github.com/lerko96/open-pact",
|
||||||
|
tier: "featured",
|
||||||
|
year: 2026,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
slug: "portfolio",
|
||||||
|
title: "portfolio",
|
||||||
|
description:
|
||||||
|
"Next.js 16 static site, self-hosted in a DMZ LXC behind Nginx, deployed via Gitea Actions CI.",
|
||||||
|
tags: ["Next.js", "Dockerfile", "Tailwind", "nginx", "Caddy"],
|
||||||
|
githubUrl: "https://gitea.lerkolabs.com/lerko/portfolio",
|
||||||
|
tier: "featured",
|
||||||
|
year: 2021,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
slug: "helm",
|
||||||
|
title: "helm",
|
||||||
|
description:
|
||||||
|
"Full-stack personal productivity dashboard. Go backend with chi router and SQLite, React + TypeScript frontend. Notes, todos, calendar (CalDAV), clipboard, bookmarks, memos. Self-hosted, single-user, daily use.",
|
||||||
|
tags: ["Go", "React", "TypeScript", "SQLite", "CalDAV"],
|
||||||
|
githubUrl: "https://github.com/lerko96/helm",
|
||||||
|
tier: "featured",
|
||||||
|
year: 2026,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
slug: "claude-vault",
|
||||||
|
title: "claude-vault",
|
||||||
|
description:
|
||||||
|
"A scaffolding system for maintaining a living project knowledge base alongside a code repo, powered by Claude Code skills.",
|
||||||
|
tags: ["Shell", "Developer-Tools", "Claude", "Knowledge-Management"],
|
||||||
|
githubUrl: "https://github.com/lerko96/claude-vault",
|
||||||
|
tier: "featured",
|
||||||
|
year: 2026,
|
||||||
|
},
|
||||||
|
// --- Archive ---
|
||||||
{
|
{
|
||||||
slug: "golf-book-mobile",
|
slug: "golf-book-mobile",
|
||||||
title: "golf-book-mobile",
|
title: "golf-book-mobile",
|
||||||
@@ -18,38 +71,21 @@ export const projects: Project[] = [
|
|||||||
"Offline-first mobile app for tracking golf rounds, managing your 14-club bag, and getting AI-powered club recommendations from a Smart Caddie. Covers 7 shot types per hole with full scorecard history.",
|
"Offline-first mobile app for tracking golf rounds, managing your 14-club bag, and getting AI-powered club recommendations from a Smart Caddie. Covers 7 shot types per hole with full scorecard history.",
|
||||||
tags: ["React Native", "Expo", "Zustand", "AI", "Mobile"],
|
tags: ["React Native", "Expo", "Zustand", "AI", "Mobile"],
|
||||||
githubUrl: "https://github.com/lerko96/golf-book-mobile",
|
githubUrl: "https://github.com/lerko96/golf-book-mobile",
|
||||||
tier: "featured",
|
tier: "archive",
|
||||||
stats: "211 commits",
|
stats: "200+ commits",
|
||||||
|
statusBadge: "Pending App Store Approval",
|
||||||
|
year: 2025,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
slug: "plaiground",
|
slug: "risk-ops",
|
||||||
title: "plAIground",
|
title: "risk-ops",
|
||||||
description:
|
description:
|
||||||
"Cross-platform desktop AI chat app for developers. Supports OpenAI, Anthropic Claude, and Google Gemini in a single interface with real-time cost tracking, conversation export, and automatic code explanation.",
|
"Browser-based strategy dashboard for Risk: Global Domination (SMG Studio). Open one HTML file — no install needed.",
|
||||||
tags: ["Electron", "Node.js", "OpenAI", "Claude", "Gemini"],
|
tags: ["HTML", "JavaScript"],
|
||||||
githubUrl: "https://github.com/lerko96/plaiground",
|
githubUrl: "#",
|
||||||
tier: "featured",
|
tier: "archive",
|
||||||
|
year: 2026,
|
||||||
},
|
},
|
||||||
{
|
|
||||||
slug: "service-monitor",
|
|
||||||
title: "service-monitor",
|
|
||||||
description:
|
|
||||||
"Web dashboard for tracking uptime across multiple services with 30-second polling, status history visualization, JWT-authenticated API, and Docker + nginx deployment.",
|
|
||||||
tags: ["React 18", "Vite", "Express", "SQLite", "Docker", "JWT"],
|
|
||||||
githubUrl: "https://github.com/lerko96/service-monitor",
|
|
||||||
tier: "featured",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
slug: "tht-1.2",
|
|
||||||
title: "ThoughtSpace",
|
|
||||||
description:
|
|
||||||
"3D visualization platform for exploring and organizing thoughts using a radio-tuning metaphor. Filter ideas by frequency and bandwidth in an instanced Three.js scene with persistent local storage.",
|
|
||||||
tags: ["React", "TypeScript", "Three.js", "React Three Fiber", "Zustand"],
|
|
||||||
githubUrl: "https://github.com/lerko96/tht-1.2",
|
|
||||||
tier: "featured",
|
|
||||||
},
|
|
||||||
|
|
||||||
// --- Archive ---
|
|
||||||
{
|
{
|
||||||
slug: "twitter-thread-ext",
|
slug: "twitter-thread-ext",
|
||||||
title: "twitter-thread-ext",
|
title: "twitter-thread-ext",
|
||||||
@@ -58,7 +94,7 @@ export const projects: Project[] = [
|
|||||||
tags: ["Chrome Extension", "Manifest V3", "JavaScript", "jsPDF"],
|
tags: ["Chrome Extension", "Manifest V3", "JavaScript", "jsPDF"],
|
||||||
githubUrl: "https://github.com/lerko96/twitter-thread-ext",
|
githubUrl: "https://github.com/lerko96/twitter-thread-ext",
|
||||||
tier: "archive",
|
tier: "archive",
|
||||||
year: 2023,
|
year: 2025,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
slug: "notes-app-1.0",
|
slug: "notes-app-1.0",
|
||||||
@@ -68,7 +104,37 @@ export const projects: Project[] = [
|
|||||||
tags: ["HTML5 Canvas", "JavaScript", "CSS"],
|
tags: ["HTML5 Canvas", "JavaScript", "CSS"],
|
||||||
githubUrl: "https://github.com/lerko96/notes-app-1.0",
|
githubUrl: "https://github.com/lerko96/notes-app-1.0",
|
||||||
tier: "archive",
|
tier: "archive",
|
||||||
year: 2022,
|
year: 2025,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
slug: "plaiground",
|
||||||
|
title: "plAIground",
|
||||||
|
description:
|
||||||
|
"Cross-platform desktop AI chat app for developers. Supports OpenAI, Anthropic Claude, and Google Gemini in a single interface with real-time cost tracking, conversation export, and automatic code explanation.",
|
||||||
|
tags: ["Electron", "Node.js", "OpenAI", "Claude", "Gemini"],
|
||||||
|
githubUrl: "https://github.com/lerko96/plaiground",
|
||||||
|
tier: "archive",
|
||||||
|
year: 2025,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
slug: "service-monitor",
|
||||||
|
title: "service-monitor",
|
||||||
|
description:
|
||||||
|
"Web dashboard for tracking uptime across multiple services with 30-second polling, status history visualization, JWT-authenticated API, and Docker + nginx deployment.",
|
||||||
|
tags: ["React 18", "Vite", "Express", "SQLite", "Docker", "JWT"],
|
||||||
|
githubUrl: "https://github.com/lerko96/service-monitor",
|
||||||
|
tier: "archive",
|
||||||
|
year: 2025,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
slug: "tht-1.2",
|
||||||
|
title: "ThoughtSpace",
|
||||||
|
description:
|
||||||
|
"3D visualization platform for exploring and organizing thoughts using a radio-tuning metaphor. Filter ideas by frequency and bandwidth in an instanced Three.js scene with persistent local storage.",
|
||||||
|
tags: ["React", "TypeScript", "Three.js", "React Three Fiber", "Zustand"],
|
||||||
|
githubUrl: "https://github.com/lerko96/tht-1.2",
|
||||||
|
tier: "archive",
|
||||||
|
year: 2025,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
slug: "were-hooked",
|
slug: "were-hooked",
|
||||||
@@ -78,7 +144,7 @@ export const projects: Project[] = [
|
|||||||
tags: ["Java", "Spring", "Thymeleaf", "HTML", "CSS"],
|
tags: ["Java", "Spring", "Thymeleaf", "HTML", "CSS"],
|
||||||
githubUrl: "https://github.com/lerko96/were-hooked-repo",
|
githubUrl: "https://github.com/lerko96/were-hooked-repo",
|
||||||
tier: "archive",
|
tier: "archive",
|
||||||
year: 2022,
|
year: 2021,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
slug: "mystery-educator",
|
slug: "mystery-educator",
|
||||||
@@ -88,7 +154,7 @@ export const projects: Project[] = [
|
|||||||
tags: ["JavaScript", "REST APIs", "HTML", "CSS"],
|
tags: ["JavaScript", "REST APIs", "HTML", "CSS"],
|
||||||
githubUrl: "https://github.com/lerko96/mystery-educator",
|
githubUrl: "https://github.com/lerko96/mystery-educator",
|
||||||
tier: "archive",
|
tier: "archive",
|
||||||
year: 2022,
|
year: 2021,
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
|
|||||||
+18
-5
@@ -7,10 +7,13 @@ export type Service = {
|
|||||||
|
|
||||||
export const services: Service[] = [
|
export const services: Service[] = [
|
||||||
// Infrastructure
|
// Infrastructure
|
||||||
{ name: "pfSense", description: "Firewall, DHCP, routing, WireGuard VPN", category: "infrastructure", icon: "fas fa-shield-halved" },
|
{ name: "pfSense", description: "Firewall, DHCP, routing gateway on N100", category: "infrastructure", icon: "fas fa-shield-halved" },
|
||||||
{ name: "Caddy", description: "Reverse proxy with automatic wildcard TLS via Cloudflare DNS-01", category: "infrastructure", icon: "fas fa-globe" },
|
{ name: "Caddy", description: "Reverse proxy with automatic wildcard TLS via Cloudflare DNS-01", category: "infrastructure", icon: "fas fa-globe" },
|
||||||
{ name: "Pi-hole", description: "Network-wide DNS + ad blocking", category: "infrastructure", icon: "fas fa-filter" },
|
{ name: "Pi-hole", description: "Network-wide DNS + ad blocking", category: "infrastructure", icon: "fas fa-filter" },
|
||||||
{ name: "WireGuard", description: "VPN — 600–900 Mbps on N100, full LAN access for clients", category: "infrastructure", icon: "fas fa-lock" },
|
{ name: "WireGuard", description: "VPN — full LAN access for remote clients", category: "infrastructure", icon: "fas fa-lock" },
|
||||||
|
{ name: "mail relay", description: "Outbound SMTP relay for self-hosted service notifications", category: "infrastructure", icon: "fas fa-envelope" },
|
||||||
|
{ name: "gluetun", description: "VPN container routing download client traffic", category: "infrastructure", icon: "fas fa-shield" },
|
||||||
|
{ name: "Home Assistant", description: "Smart home automation and device management", category: "infrastructure", icon: "fas fa-house" },
|
||||||
|
|
||||||
// Security / Auth
|
// Security / Auth
|
||||||
{ name: "Authentik", description: "SSO provider — OIDC + forward auth across all services", category: "security", icon: "fas fa-id-badge" },
|
{ name: "Authentik", description: "SSO provider — OIDC + forward auth across all services", category: "security", icon: "fas fa-id-badge" },
|
||||||
@@ -34,11 +37,21 @@ export const services: Service[] = [
|
|||||||
{ name: "Traggo", description: "Time tracking", category: "productivity", icon: "fas fa-clock" },
|
{ name: "Traggo", description: "Time tracking", category: "productivity", icon: "fas fa-clock" },
|
||||||
{ name: "Baikal", description: "CalDAV / CardDAV server", category: "productivity", icon: "fas fa-calendar" },
|
{ name: "Baikal", description: "CalDAV / CardDAV server", category: "productivity", icon: "fas fa-calendar" },
|
||||||
{ name: "Grist", description: "Spreadsheets and structured data", category: "productivity", icon: "fas fa-table" },
|
{ name: "Grist", description: "Spreadsheets and structured data", category: "productivity", icon: "fas fa-table" },
|
||||||
|
{ name: "Glance", description: "Self-hosted start page with feeds and service status", category: "productivity", icon: "fas fa-gauge" },
|
||||||
|
{ name: "Filebrowser", description: "Web-based file manager", category: "productivity", icon: "fas fa-folder-open" },
|
||||||
|
|
||||||
// Media
|
// Media
|
||||||
{ name: "Plex + Jellyfin", description: "Media streaming", category: "media", icon: "fas fa-film" },
|
{ name: "Plex", description: "Media streaming — movies, TV, music", category: "media", icon: "fas fa-film" },
|
||||||
{ name: "Sonarr / Radarr / Lidarr", description: "Automated media management", category: "media", icon: "fas fa-download" },
|
{ name: "Jellyfin", description: "Open-source media streaming", category: "media", icon: "fas fa-play" },
|
||||||
{ name: "Calibre-Web", description: "Book library with auto-ingest", category: "media", icon: "fas fa-book-open" },
|
{ name: "Sonarr", description: "Automated TV show management", category: "media", icon: "fas fa-tv" },
|
||||||
|
{ name: "Radarr", description: "Automated movie management", category: "media", icon: "fas fa-video" },
|
||||||
|
{ name: "Lidarr", description: "Automated music management", category: "media", icon: "fas fa-music" },
|
||||||
|
{ name: "Prowlarr", description: "Indexer manager and proxy for the *arr stack", category: "media", icon: "fas fa-magnifying-glass" },
|
||||||
|
{ name: "Bazarr", description: "Automatic subtitle download and management", category: "media", icon: "fas fa-closed-captioning" },
|
||||||
|
{ name: "nzbget", description: "Usenet downloader", category: "media", icon: "fas fa-download" },
|
||||||
|
{ name: "qBittorrent", description: "Torrent client with web UI", category: "media", icon: "fas fa-magnet" },
|
||||||
|
{ name: "Kavita", description: "Self-hosted manga and book reader", category: "media", icon: "fas fa-book-open" },
|
||||||
|
{ name: "Openshelf", description: "Book library with auto-ingest", category: "media", icon: "fas fa-book-open" },
|
||||||
];
|
];
|
||||||
|
|
||||||
export const categoryOrder: Service["category"][] = [
|
export const categoryOrder: Service["category"][] = [
|
||||||
|
|||||||
@@ -0,0 +1,119 @@
|
|||||||
|
export type TimelineType =
|
||||||
|
| "career"
|
||||||
|
| "cert"
|
||||||
|
| "project"
|
||||||
|
| "homelab"
|
||||||
|
| "education";
|
||||||
|
|
||||||
|
export interface TimelineEntry {
|
||||||
|
date: string;
|
||||||
|
title: string;
|
||||||
|
type: TimelineType;
|
||||||
|
description: string;
|
||||||
|
tags?: string[];
|
||||||
|
}
|
||||||
|
|
||||||
|
export const timeline: TimelineEntry[] = [
|
||||||
|
{
|
||||||
|
date: "WIP",
|
||||||
|
title: "CompTIA Network+ — in progress",
|
||||||
|
type: "cert",
|
||||||
|
description:
|
||||||
|
"Studying for Network+ to formalize networking knowledge built through the homelab.",
|
||||||
|
tags: ["networking", "certification"],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
date: "2026-04",
|
||||||
|
title: "Portfolio Site v2",
|
||||||
|
type: "project",
|
||||||
|
description:
|
||||||
|
"Next.js 16 portfolio site, self-hosted in a DMZ LXC behind Nginx, deployed via Gitea Actions CI.",
|
||||||
|
tags: ["next.js", "tailwind", "self-hosted"],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
date: "2026-04",
|
||||||
|
title: "lerkolabs.com",
|
||||||
|
type: "homelab",
|
||||||
|
description:
|
||||||
|
"Self-hosted in a DMZ LXC behind Nginx, deployed via Gitea Actions CI.",
|
||||||
|
tags: ["LXC", "DMZ", "self-hosted"],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
date: "2026-03",
|
||||||
|
title: "Helm",
|
||||||
|
type: "project",
|
||||||
|
description:
|
||||||
|
"Full-stack task and project management tool built in Go + React.",
|
||||||
|
tags: ["go", "react", "typescript"],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
date: "2024-08",
|
||||||
|
title: "Proxmox Cluster",
|
||||||
|
type: "homelab",
|
||||||
|
description:
|
||||||
|
"Proxmox VMs/LXCs, SSO via Authentik, full monitoring stack (VictoriaMetrics + Grafana + Beszel + ntfy).",
|
||||||
|
tags: ["proxmox", "networking", "monitoring", "sso"],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
date: "2024-06",
|
||||||
|
title: "CompTIA A+",
|
||||||
|
type: "cert",
|
||||||
|
description:
|
||||||
|
"Earned A+ certification, formalizing hardware and OS fundamentals.",
|
||||||
|
tags: ["certification"],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
date: "2024-03",
|
||||||
|
title: "pfSense",
|
||||||
|
type: "homelab",
|
||||||
|
description: "Netgate pfSense n100 picked up on ebay.",
|
||||||
|
tags: ["network", "firewall", "vlan", "dhcp"],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
date: "2023-10",
|
||||||
|
title: "SOC Analyst I — Fortress SRM",
|
||||||
|
type: "career",
|
||||||
|
description:
|
||||||
|
"Threat monitoring, incident triage, and client-facing security operations in a managed SOC.",
|
||||||
|
tags: ["soc", "security"],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
date: "2023-03",
|
||||||
|
title: "Config Tech II — MCPc",
|
||||||
|
type: "career",
|
||||||
|
description:
|
||||||
|
"Promoted to Config Tech II. Led imaging workflows and expanded into scripting for endpoint provisioning.",
|
||||||
|
tags: ["sysadmin", "scripting"],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
date: "2022-11",
|
||||||
|
title: "PC Build",
|
||||||
|
type: "homelab",
|
||||||
|
description: "Sourced parts online and built a personal computer.",
|
||||||
|
tags: ["amd", "windows 10", "configure", "desktop"],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
date: "2022-05",
|
||||||
|
title: "Config Tech I — MCPc",
|
||||||
|
type: "career",
|
||||||
|
description:
|
||||||
|
"Hardware configuration, OS imaging, and deployment at scale for enterprise clients.",
|
||||||
|
tags: ["sysadmin", "hardware"],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
date: "2021-10",
|
||||||
|
title: "Portfolio Site v1",
|
||||||
|
type: "project",
|
||||||
|
description:
|
||||||
|
"React portfolio deployed to www.lerko96.github.io using github pages.",
|
||||||
|
tags: ["React", "CSS", "github pages"],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
date: "2021-01",
|
||||||
|
title: "We Can Code IT — Java Bootcamp",
|
||||||
|
type: "education",
|
||||||
|
description:
|
||||||
|
"9-month intensive bootcamp covering Java, OOP, SQL, REST APIs, and Agile development practices.",
|
||||||
|
tags: ["java", "sql", "agile"],
|
||||||
|
},
|
||||||
|
];
|
||||||
Reference in New Issue
Block a user