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:
parent
e74b169a21
commit
2e169b50fe
43
README.md
43
README.md
@ -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
286
index.html
Normal 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
45
nginx.conf
Normal 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;
|
||||||
|
#}
|
||||||
|
}
|
||||||
|
|
||||||
Loading…
Reference in New Issue
Block a user