feat: initial lab portal

Static nginx page serving home.lab.corestory.ai.
Lists all internal lab services with links and descriptions.
This commit is contained in:
Neo 2026-04-26 11:48:28 +00:00
parent e74b169a21
commit 2e169b50fe
3 changed files with 372 additions and 2 deletions

View File

@ -1,3 +1,42 @@
# lab-portal # corestory Lab Portal
corestory Lab internal services portal — home.lab.corestory.ai Internal services portal at [home.lab.corestory.ai](https://home.lab.corestory.ai).
## What it is
A static nginx page listing all internal lab services with links. Served by the `lab-portal` Docker container on the `plane-app_default` network (IP `172.18.0.14:80`), routed through Traefik via `home.lab.corestory.ai`.
## Deployment
The container is managed outside Nomad (Docker Compose / manual `docker run`). The Traefik route is defined in the Nomad `traefik` job template (`local/home.yml`).
To update the portal page:
```bash
# Edit index.html, then:
docker cp index.html lab-portal:/usr/share/nginx/html/index.html
```
To rebuild from scratch:
```bash
docker run -d --name lab-portal \
--network plane-app_default \
-v /path/to/index.html:/usr/share/nginx/html/index.html:ro \
nginx:alpine
```
## Services Listed
| Service | URL |
|---------|-----|
| Plane | https://plane.lab.corestory.ai |
| n8n | https://n8n.lab.corestory.ai |
| Ghost CMS | https://cms.lab.corestory.ai |
| Matrix | https://matrix.lab.corestory.ai |
| Metabase | https://metabase.lab.corestory.ai |
| PostHog | https://posthog.lab.corestory.ai |
| dbt | https://dbt.lab.corestory.ai |
| Gitea | https://git.lab.corestory.ai |
| Grafana | https://grafana.lab.corestory.ai |
| Keycloak | https://keycloak.lab.corestory.ai |
| Traefik | https://traefik.lab.corestory.ai |
| PKI / OpenBao | https://pki.lab.corestory.ai |

286
index.html Normal file
View File

@ -0,0 +1,286 @@
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>corestory Lab</title>
<style>
*, *::before, *::after { box-sizing: border-box; margin: 0; padding: 0; }
body {
font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, sans-serif;
background: #0a0a0f;
color: #e2e8f0;
min-height: 100vh;
padding: 48px 24px;
}
header {
text-align: center;
margin-bottom: 56px;
}
header h1 {
font-size: 2rem;
font-weight: 700;
letter-spacing: -0.02em;
color: #fff;
}
header h1 span {
color: #6366f1;
}
header p {
margin-top: 8px;
font-size: 0.9rem;
color: #64748b;
letter-spacing: 0.08em;
text-transform: uppercase;
}
.grid {
display: grid;
grid-template-columns: repeat(auto-fill, minmax(280px, 1fr));
gap: 16px;
max-width: 1100px;
margin: 0 auto;
}
.section-label {
grid-column: 1 / -1;
font-size: 0.72rem;
font-weight: 600;
letter-spacing: 0.12em;
text-transform: uppercase;
color: #475569;
padding: 8px 0 4px;
border-bottom: 1px solid #1e293b;
margin-top: 8px;
}
a.card {
display: flex;
align-items: center;
gap: 16px;
background: #0f1117;
border: 1px solid #1e293b;
border-radius: 12px;
padding: 18px 20px;
text-decoration: none;
color: inherit;
transition: border-color 0.15s, background 0.15s, transform 0.1s;
}
a.card:hover {
border-color: #6366f1;
background: #13141f;
transform: translateY(-1px);
}
.card-icon {
width: 40px;
height: 40px;
border-radius: 10px;
display: flex;
align-items: center;
justify-content: center;
font-size: 1.3rem;
flex-shrink: 0;
}
.card-body { flex: 1; min-width: 0; }
.card-name {
font-size: 0.95rem;
font-weight: 600;
color: #f1f5f9;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
}
.card-desc {
font-size: 0.78rem;
color: #64748b;
margin-top: 2px;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
}
.card-url {
font-size: 0.7rem;
color: #334155;
margin-top: 4px;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
}
.arrow {
color: #334155;
font-size: 1rem;
flex-shrink: 0;
transition: color 0.15s;
}
a.card:hover .arrow { color: #6366f1; }
footer {
text-align: center;
margin-top: 64px;
font-size: 0.75rem;
color: #1e293b;
}
</style>
</head>
<body>
<header>
<h1>corestory <span>Lab</span></h1>
<p>Internal Services Portal</p>
</header>
<div class="grid">
<!-- WORK -->
<div class="section-label">Work</div>
<a class="card" href="https://plane.lab.corestory.ai" target="_blank">
<div class="card-icon" style="background:#1a1040;">✈️</div>
<div class="card-body">
<div class="card-name">Plane</div>
<div class="card-desc">Project management</div>
<div class="card-url">plane.lab.corestory.ai</div>
</div>
<span class="arrow"></span>
</a>
<a class="card" href="https://n8n.lab.corestory.ai" target="_blank">
<div class="card-icon" style="background:#0f1f0f;"></div>
<div class="card-body">
<div class="card-name">n8n</div>
<div class="card-desc">Workflow automation</div>
<div class="card-url">n8n.lab.corestory.ai</div>
</div>
<span class="arrow"></span>
</a>
<a class="card" href="https://cms.lab.corestory.ai" target="_blank">
<div class="card-icon" style="background:#1a0f0a;">👻</div>
<div class="card-body">
<div class="card-name">Ghost CMS</div>
<div class="card-desc">Content management</div>
<div class="card-url">cms.lab.corestory.ai</div>
</div>
<span class="arrow"></span>
</a>
<!-- COMMS -->
<div class="section-label">Comms</div>
<a class="card" href="https://matrix.lab.corestory.ai" target="_blank">
<div class="card-icon" style="background:#0f1320;">💬</div>
<div class="card-body">
<div class="card-name">Matrix</div>
<div class="card-desc">Bard team chat (Element)</div>
<div class="card-url">matrix.lab.corestory.ai</div>
</div>
<span class="arrow"></span>
</a>
<!-- DATA & ANALYTICS -->
<div class="section-label">Data & Analytics</div>
<a class="card" href="https://metabase.lab.corestory.ai" target="_blank">
<div class="card-icon" style="background:#0a1520;">📊</div>
<div class="card-body">
<div class="card-name">Metabase</div>
<div class="card-desc">Business intelligence</div>
<div class="card-url">metabase.lab.corestory.ai</div>
</div>
<span class="arrow"></span>
</a>
<a class="card" href="https://posthog.lab.corestory.ai" target="_blank">
<div class="card-icon" style="background:#1a0f0f;">🦔</div>
<div class="card-body">
<div class="card-name">PostHog</div>
<div class="card-desc">Product analytics</div>
<div class="card-url">posthog.lab.corestory.ai</div>
</div>
<span class="arrow"></span>
</a>
<a class="card" href="https://dbt.lab.corestory.ai" target="_blank">
<div class="card-icon" style="background:#0f1a10;">🔄</div>
<div class="card-body">
<div class="card-name">dbt</div>
<div class="card-desc">Data transformation</div>
<div class="card-url">dbt.lab.corestory.ai</div>
</div>
<span class="arrow"></span>
</a>
<!-- DEV -->
<div class="section-label">Developer</div>
<a class="card" href="https://git.lab.corestory.ai" target="_blank">
<div class="card-icon" style="background:#100f1a;">🐙</div>
<div class="card-body">
<div class="card-name">Gitea</div>
<div class="card-desc">Git hosting</div>
<div class="card-url">git.lab.corestory.ai</div>
</div>
<span class="arrow"></span>
</a>
<!-- INFRA -->
<div class="section-label">Infrastructure</div>
<a class="card" href="https://grafana.lab.corestory.ai" target="_blank">
<div class="card-icon" style="background:#1a0f00;">📈</div>
<div class="card-body">
<div class="card-name">Grafana</div>
<div class="card-desc">Metrics & monitoring</div>
<div class="card-url">grafana.lab.corestory.ai</div>
</div>
<span class="arrow"></span>
</a>
<a class="card" href="https://keycloak.lab.corestory.ai" target="_blank">
<div class="card-icon" style="background:#001a10;">🔑</div>
<div class="card-body">
<div class="card-name">Keycloak</div>
<div class="card-desc">Identity & SSO</div>
<div class="card-url">keycloak.lab.corestory.ai</div>
</div>
<span class="arrow"></span>
</a>
<a class="card" href="https://traefik.lab.corestory.ai" target="_blank">
<div class="card-icon" style="background:#001520;">🔀</div>
<div class="card-body">
<div class="card-name">Traefik</div>
<div class="card-desc">Reverse proxy dashboard</div>
<div class="card-url">traefik.lab.corestory.ai</div>
</div>
<span class="arrow"></span>
</a>
<a class="card" href="https://pki.lab.corestory.ai" target="_blank">
<div class="card-icon" style="background:#1a1500;">🔒</div>
<div class="card-body">
<div class="card-name">PKI / OpenBao</div>
<div class="card-desc">Secrets & certificates</div>
<div class="card-url">pki.lab.corestory.ai</div>
</div>
<span class="arrow"></span>
</a>
</div>
<footer>corestory lab · *.lab.corestory.ai · 3-node Nomad cluster</footer>
</body>
</html>

45
nginx.conf Normal file
View File

@ -0,0 +1,45 @@
server {
listen 80;
listen [::]:80;
server_name localhost;
#access_log /var/log/nginx/host.access.log main;
location / {
root /usr/share/nginx/html;
index index.html index.htm;
}
#error_page 404 /404.html;
# redirect server error pages to the static page /50x.html
#
error_page 500 502 503 504 /50x.html;
location = /50x.html {
root /usr/share/nginx/html;
}
# proxy the PHP scripts to Apache listening on 127.0.0.1:80
#
#location ~ \.php$ {
# proxy_pass http://127.0.0.1;
#}
# pass the PHP scripts to FastCGI server listening on 127.0.0.1:9000
#
#location ~ \.php$ {
# root html;
# fastcgi_pass 127.0.0.1:9000;
# fastcgi_index index.php;
# fastcgi_param SCRIPT_FILENAME /scripts$fastcgi_script_name;
# include fastcgi_params;
#}
# deny access to .htaccess files, if Apache's document root
# concurs with nginx's one
#
#location ~ /\.ht {
# deny all;
#}
}