4 Commits

Author SHA1 Message Date
tyler 4e51dd4a83 feat/polish (#5)
Build and Deploy / deploy (push) Successful in 51s
Reorder homepage sections (journey above projects), refine component styles, update copy and data across projects, timeline, and homelab.

Reviewed-on: #5
2026-04-20 00:34:50 +00:00
tyler 8e9fcfaeeb Merge pull request 'feat(content): reposition for security engineer, expand services' (#4) from feat/timeline into dev
Build and Deploy / deploy (push) Successful in 1m11s
Reviewed-on: #4
2026-04-17 01:24:22 +00:00
lerko96 2946801517 ci: auto-tag CalVer after deploy if commit untagged 2026-04-16 21:16:01 -04:00
lerko96 49028e7783 feat(content): reposition for security engineer, expand services to 37
Hero subtitle and blurb rewritten to lead with security operations
and homelab credentials over generic "builder" framing.

Projects: archive plAIground, service-monitor, ThoughtSpace (2025);
add open-pact and helm as featured; add risk-ops to archive (2026).
Add statusBadge + externalUrl to Project type; wire golf-book-mobile.

Services: 24 → 37 — split grouped arr/media entries, add mail relay,
gluetun, Home Assistant, Glance, Filebrowser, Prowlarr, Bazarr,
nzbget, qBittorrent, Kavita, Openshelf. Drop Calibre-Web.

Skills: add Go to Languages. Timeline: update monitoring stack.
Homelab ADRs: add Authentik over Authelia.
2026-04-16 21:06:18 -04:00
17 changed files with 478 additions and 205 deletions
+27
View File
@@ -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
-4
View File
@@ -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
+10 -10
View File
@@ -10,9 +10,9 @@ export const metadata: Metadata = {
export default function ArchivePage() { export default function ArchivePage() {
return ( return (
<> <>
<div className="mb-16"> <div className="mb-4lh">
<p className="font-mono text-base font-bold text-[var(--color-text)] mb-3"> <p className="font-mono text-sm font-bold text-[var(--color-text)] mb-1lh">
<span className="text-[var(--color-accent-green)] select-none mr-2" aria-hidden="true"></span> <span className="text-[var(--color-accent-green)] select-none mr-1ch" aria-hidden="true"></span>
tyler/projects/archive tyler/projects/archive
</p> </p>
<p className="font-mono text-sm text-[var(--color-text)] leading-relaxed max-w-xl opacity-80"> <p className="font-mono text-sm text-[var(--color-text)] leading-relaxed max-w-xl opacity-80">
@@ -29,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>
)} )}
@@ -45,16 +45,16 @@ export default function ArchivePage() {
<p className="font-mono 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"
> >
+16 -1
View File
@@ -29,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;
@@ -41,7 +55,8 @@
/* Base */ /* Base */
html { html {
scroll-behavior: smooth; scroll-behavior: smooth;
font-size: 14px; 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);
+111 -51
View File
@@ -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,26 +109,38 @@ 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">
<p className="font-mono text-base font-bold text-[var(--color-text)] mb-3"> <p className="font-mono text-sm font-bold text-[var(--color-text)] mb-1lh">
<span className="text-[var(--color-accent-green)] select-none mr-2" aria-hidden="true"></span> <span
className="text-[var(--color-accent-green)] select-none mr-1ch"
aria-hidden="true"
>
</span>
homelab homelab
</p> </p>
<p className="font-mono text-sm text-[var(--color-text)] leading-relaxed max-w-2xl opacity-80"> <p className="font-mono text-sm text-[var(--color-text)] leading-relaxed max-w-2xl opacity-80">
Personal infrastructure environment for learning, self-hosting, and operational Personal infrastructure environment for learning, self-hosting, and
practice. Running 24/7 on production-grade hardware with real network segmentation, operational practice. Running 24/7 on production-grade hardware with
SSO, monitoring, and IaC-style documentation. real network segmentation, SSO, monitoring, and IaC-style
documentation.
</p> </p>
</div> </div>
@@ -96,14 +148,13 @@ export default function HomelabPage() {
<Widget title="homelab/overview" 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,19 +167,19 @@ export default function HomelabPage() {
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>
@@ -139,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-mono 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>
@@ -155,31 +210,27 @@ export default function HomelabPage() {
</Widget> </Widget>
{/* Services */} {/* Services */}
<Widget <Widget title="homelab/services" badge={services.length} as="section">
title="homelab/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-mono text-sm text-[var(--color-text)] leading-relaxed opacity-75"> <p className="font-mono text-sm text-[var(--color-text)] leading-relaxed opacity-75">
@@ -206,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-mono 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-mono 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>
@@ -222,19 +279,22 @@ export default function HomelabPage() {
</div> </div>
</Widget> </Widget>
{/* GitHub CTA */} {/* Gitea CTA */}
<section className="pt-2"> <section className="pt-qtr-lh">
<p className="font-mono text-sm text-[var(--color-text-dim)] mb-1">homelab/docs github.com/lerko96/homelab-wip</p> <p className="font-mono text-sm text-[var(--color-text-dim)] mb-half-lh">
<p className="font-mono text-sm text-[var(--color-text)] mb-3 opacity-75"> homelab/docs
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-sm 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 -1
View File
@@ -37,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 />
+2 -2
View File
@@ -16,15 +16,15 @@ export default function Home() {
return ( return (
<> <>
<Hero /> <Hero />
<Timeline />
<Widget title="tyler/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 /> <Skills />
<Timeline />
</> </>
); );
} }
+19 -3
View File
@@ -1,11 +1,11 @@
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-sm text-[var(--color-text-dim)]"> <span className="font-mono text-sm text-[var(--color-text-dim)]">
&copy; {new Date().getFullYear()} Tyler Koenig &copy; {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"
@@ -15,6 +15,15 @@ export default function Footer() {
> >
[github] [github]
</a> </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
href="https://www.linkedin.com/in/tyler-koenig" href="https://www.linkedin.com/in/tyler-koenig"
target="_blank" target="_blank"
@@ -24,6 +33,13 @@ export default function Footer() {
> >
[linkedin] [linkedin]
</a> </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>
</footer> </footer>
+31 -12
View File
@@ -1,26 +1,36 @@
export default function Hero() { export default function Hero() {
return ( return (
<section className="mb-16"> <section className="mb-16">
<div className="flex flex-col gap-3"> <div className="flex flex-col gap-1ch">
<div> <div>
<p className="font-mono text-base font-bold text-[var(--color-text)]"> <p className="font-mono text-sm font-bold text-[var(--color-text)]">
<span className="text-[var(--color-accent-green)] select-none mr-2" aria-hidden="true"></span> <span
className="text-[var(--color-accent-green)] select-none mr-1ch"
aria-hidden="true"
>
</span>
tyler koenig tyler koenig
</p> </p>
<p className="font-mono text-sm text-[var(--color-text-label)] mt-0.5"> <p className="font-mono text-sm text-[var(--color-text-label)] mt-0.5">
SOC Helpdesk I · Homelab Operator Security Operations · Self-Hosted Infrastructure
</p> </p>
</div> </div>
<p className="font-mono text-sm text-[var(--color-text)] leading-relaxed max-w-lg opacity-70"> <p className="font-mono text-sm text-[var(--color-text)] leading-relaxed opacity-70">
I write software and run infrastructure that goes well past what my Homelab runs 30+
job title implies. Games, AI tooling, mobile apps, and a homelab services across segmented VLANs pfSense, Authentik SSO, full
running 20+ self-hosted services on segmented VLANs. Continuously observability stack. Write software too: mobile apps, Go backends,
learning by building things that actually work.{' '} open protocols. Daily drivers, all of it.{" "}
<span className="animate-cursor text-[var(--color-accent-green)]" aria-hidden="true"></span> <span
className="animate-cursor text-[var(--color-accent-green)]"
aria-hidden="true"
>
</span>
</p> </p>
<div className="flex flex-wrap items-center gap-x-5 gap-y-1"> <div className="flex flex-wrap items-center gap-x-1ch gap-y-half-lh">
<span className="font-mono text-sm text-[var(--color-accent-green)]"> <span className="font-mono text-sm text-[var(--color-accent-green)]">
available available
</span> </span>
@@ -33,6 +43,15 @@ export default function Hero() {
> >
[github] [github]
</a> </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
href="https://www.linkedin.com/in/tyler-koenig" href="https://www.linkedin.com/in/tyler-koenig"
target="_blank" target="_blank"
@@ -43,7 +62,7 @@ export default function Hero() {
[linkedin] [linkedin]
</a> </a>
<a <a
href="mailto:tylerkoenig96@gmail.com" href="mailto:tyler@lerkolabs.com"
aria-label="Email" aria-label="Email"
className="font-mono text-sm 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)]"
> >
+2 -2
View File
@@ -15,7 +15,7 @@ export default function Nav() {
const { isDark, toggle } = useTheme(); 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)] hover:text-[var(--color-text-label)]" className="font-mono text-sm font-bold text-[var(--color-text)] hover:text-[var(--color-text-label)]"
@@ -23,7 +23,7 @@ export default function Nav() {
~/ ~/
</Link> </Link>
<ul className="flex items-center 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(/\/$/, "");
+22 -5
View File
@@ -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)] flex flex-col gap-4 p-5 hover:bg-[var(--color-surface-raised)]"> <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,12 +16,23 @@ 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-sm 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"
@@ -34,13 +45,19 @@ export default function ProjectCard({ project }: Props) {
</div> </div>
</div> </div>
{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"> <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 mt-1"> <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>
))} ))}
+13 -13
View File
@@ -1,26 +1,26 @@
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);
@@ -32,7 +32,7 @@ export default function Skills() {
{skillGroups.map(({ label, skills }) => ( {skillGroups.map(({ label, skills }) => (
<div <div
key={label} key={label}
className="flex flex-col xs:flex-row gap-1 xs:gap-6 py-3" className="flex flex-col xs:flex-row gap-1ch xs:gap-2ch py-half-lh"
> >
<span className="font-mono text-sm text-[var(--color-text-dim)] w-28 shrink-0"> <span className="font-mono text-sm text-[var(--color-text-dim)] w-28 shrink-0">
{label} {label}
+8 -8
View File
@@ -54,9 +54,9 @@ export default function Timeline() {
return ( return (
<Widget title="tyler/journey"> <Widget title="tyler/journey">
<ol ref={listRef} className="relative border-l border-[var(--color-border)] ml-1.5 flex flex-col gap-0"> <ol ref={listRef} className="relative border-l border-[var(--color-border)] ml-[2px] flex flex-col gap-0">
{timeline.map((entry, i) => ( {timeline.map((entry, i) => (
<li key={i} data-tl-entry className="pl-6 pb-8 last:pb-0 relative"> <li key={i} data-tl-entry className="pl-[3ch] pb-2lh last:pb-0 relative">
{/* Spine dot */} {/* Spine dot */}
<span <span
className="absolute -left-[7px] top-[3px] w-3 h-3 rounded-full border border-[var(--color-bg)] shrink-0" className="absolute -left-[7px] top-[3px] w-3 h-3 rounded-full border border-[var(--color-bg)] shrink-0"
@@ -65,10 +65,10 @@ export default function Timeline() {
/> />
{/* Date + type badge */} {/* Date + type badge */}
<div className="flex items-center gap-2 mb-1"> <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 text-[var(--color-text-dim)]">{entry.date}</span>
<span <span
className="font-mono text-[10px] uppercase tracking-wider px-1 rounded-sm border" className="font-mono text-sm px-1 border"
style={{ style={{
color: typeColor[entry.type], color: typeColor[entry.type],
borderColor: typeColor[entry.type], borderColor: typeColor[entry.type],
@@ -80,20 +80,20 @@ export default function Timeline() {
</div> </div>
{/* Title */} {/* Title */}
<p className="font-mono text-sm font-semibold text-[var(--color-text)] mb-1"> <p className="font-mono text-sm font-semibold text-[var(--color-text)] mb-half-lh">
{entry.title} {entry.title}
</p> </p>
{/* Description */} {/* Description */}
<p className="font-mono text-sm text-[var(--color-text)] opacity-70 leading-relaxed mb-2"> <p className="font-mono text-sm text-[var(--color-text)] opacity-70 leading-relaxed mb-half-lh">
{entry.description} {entry.description}
</p> </p>
{/* Tags */} {/* Tags */}
{entry.tags && entry.tags.length > 0 && ( {entry.tags && entry.tags.length > 0 && (
<div className="flex flex-wrap gap-x-3 gap-y-1"> <div className="flex flex-wrap gap-x-1ch gap-y-half-lh">
{entry.tags.map((tag) => ( {entry.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>
))} ))}
+4 -4
View File
@@ -20,17 +20,17 @@ export default function Widget({
const name = slashIdx >= 0 ? title.slice(slashIdx + 1) : title; 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-2 mb-8"> <div className="flex items-center gap-1ch mb-2lh">
{prefix && ( {prefix && (
<span className="font-mono text-sm text-[var(--color-text-dim)] select-none">{prefix}</span> <span className="font-mono text-sm text-[var(--color-text-dim)] select-none">{prefix}</span>
)} )}
<span className="font-mono text-sm font-semibold text-[var(--color-text)]">{name}</span> <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)]">[{badge}]</span> <span className="font-mono text-sm text-[var(--color-text-dim)]">[{badge}]</span>
)} )}
{meta && ( {meta && (
<span className="font-mono text-xs text-[var(--color-text-dim)] ml-1"> {meta}</span> <span className="font-mono text-sm text-[var(--color-text-dim)]"> {meta}</span>
)} )}
</div> </div>
{children} {children}
+98 -32
View File
@@ -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
View File
@@ -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 — 600900 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"][] = [
+96 -52
View File
@@ -1,75 +1,119 @@
export type TimelineType = 'career' | 'cert' | 'project' | 'homelab' | 'education' export type TimelineType =
| "career"
| "cert"
| "project"
| "homelab"
| "education";
export interface TimelineEntry { export interface TimelineEntry {
date: string date: string;
title: string title: string;
type: TimelineType type: TimelineType;
description: string description: string;
tags?: string[] tags?: string[];
} }
export const timeline: TimelineEntry[] = [ export const timeline: TimelineEntry[] = [
{ {
date: '2026', date: "WIP",
title: 'CompTIA Network+ — in progress', title: "CompTIA Network+ — in progress",
type: 'cert', type: "cert",
description: 'Studying for Network+ to formalize networking knowledge built through the homelab.', description:
tags: ['networking', 'certification'], "Studying for Network+ to formalize networking knowledge built through the homelab.",
tags: ["networking", "certification"],
}, },
{ {
date: '2025', date: "2026-04",
title: 'Portfolio Site v2', title: "Portfolio Site v2",
type: 'project', type: "project",
description: 'Next.js 16 static site, self-hosted in a DMZ LXC behind Nginx, deployed via Gitea Actions CI.', description:
tags: ['next.js', 'tailwind', 'self-hosted'], "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: '2024', date: "2026-04",
title: 'CompTIA A+', title: "lerkolabs.com",
type: 'cert', type: "homelab",
description: 'Earned A+ certification, formalizing hardware and OS fundamentals.', description:
tags: ['certification'], "Self-hosted in a DMZ LXC behind Nginx, deployed via Gitea Actions CI.",
tags: ["LXC", "DMZ", "self-hosted"],
}, },
{ {
date: '2024', date: "2026-03",
title: 'Project Helm', title: "Helm",
type: 'project', type: "project",
description: 'Full-stack task and project management tool built in Go + React.', description:
tags: ['go', 'react', 'typescript'], "Full-stack task and project management tool built in Go + React.",
tags: ["go", "react", "typescript"],
}, },
{ {
date: 'ongoing', date: "2024-08",
title: 'Homelab — Proxmox Cluster', title: "Proxmox Cluster",
type: 'homelab', type: "homelab",
description: '8-VLAN segmented network, Proxmox VMs/LXCs, SSO via Authentik, full monitoring stack (Grafana + Prometheus + Loki).', description:
tags: ['proxmox', 'networking', 'monitoring', 'sso'], "Proxmox VMs/LXCs, SSO via Authentik, full monitoring stack (VictoriaMetrics + Grafana + Beszel + ntfy).",
tags: ["proxmox", "networking", "monitoring", "sso"],
}, },
{ {
date: '2023-10', date: "2024-06",
title: 'SOC Analyst I — Fortress SRM', title: "CompTIA A+",
type: 'career', type: "cert",
description: 'Threat monitoring, incident triage, and client-facing security operations in a managed SOC.', description:
tags: ['soc', 'security'], "Earned A+ certification, formalizing hardware and OS fundamentals.",
tags: ["certification"],
}, },
{ {
date: '2023-03', date: "2024-03",
title: 'Config Tech II — MCPc', title: "pfSense",
type: 'career', type: "homelab",
description: 'Promoted to Config Tech II. Led imaging workflows and expanded into scripting for endpoint provisioning.', description: "Netgate pfSense n100 picked up on ebay.",
tags: ['sysadmin', 'scripting'], tags: ["network", "firewall", "vlan", "dhcp"],
}, },
{ {
date: '2022-07', date: "2023-10",
title: 'Config Tech I — MCPc', title: "SOC Analyst I — Fortress SRM",
type: 'career', type: "career",
description: 'Hardware configuration, OS imaging, and deployment at scale for enterprise clients.', description:
tags: ['sysadmin', 'hardware'], "Threat monitoring, incident triage, and client-facing security operations in a managed SOC.",
tags: ["soc", "security"],
}, },
{ {
date: '2021', date: "2023-03",
title: 'We Can Code ITJava Bootcamp', title: "Config Tech IIMCPc",
type: 'education', type: "career",
description: '9-month intensive bootcamp covering Java, OOP, SQL, REST APIs, and Agile development practices.', description:
tags: ['java', 'sql', 'agile'], "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"],
},
];