Add Docker deployment with Traefik integration

- docker-compose.yml for container orchestration
- Dockerfile with multi-stage build (node builder + nginx)
- nginx.conf for SPA serving
This commit is contained in:
Erwin
2026-03-27 19:20:43 +00:00
parent 1acf1c4ff5
commit 9c20b0ed31
5 changed files with 64 additions and 1 deletions

1
.gitignore vendored
View File

@@ -10,6 +10,7 @@ node_modules/
# Build info # Build info
*.tsbuildinfo *.tsbuildinfo
node_modules/.tmp/
# OS # OS
.DS_Store .DS_Store

20
Dockerfile Normal file
View File

@@ -0,0 +1,20 @@
# Build stage
FROM node:20-alpine AS builder
WORKDIR /app
COPY package*.json ./
RUN npm ci
COPY . .
RUN npm run build
# Production stage
FROM nginx:alpine
COPY --from=builder /app/dist /usr/share/nginx/html
COPY nginx.conf /etc/nginx/conf.d/default.conf
EXPOSE 80
CMD ["nginx", "-g", "daemon off;"]

19
docker-compose.yml Normal file
View File

@@ -0,0 +1,19 @@
services:
mission-control-ui:
build:
context: .
dockerfile: Dockerfile
container_name: mission-control-ui
restart: unless-stopped
labels:
- "traefik.enable=true"
- "traefik.http.routers.mission-control-ui.rule=Host(`mission-control.danielarroyo.cl`)"
- "traefik.http.routers.mission-control-ui.entrypoints=websecure"
- "traefik.http.routers.mission-control-ui.tls.certresolver=letsencrypt"
- "traefik.http.services.mission-control-ui.loadbalancer.server.port=80"
networks:
- traefik
networks:
traefik:
external: true

23
nginx.conf Normal file
View File

@@ -0,0 +1,23 @@
server {
listen 80;
server_name localhost;
root /usr/share/nginx/html;
index index.html;
# Serve static files
location / {
try_files $uri $uri/ /index.html;
}
# Cache static assets
location ~* \.(js|css|png|jpg|jpeg|gif|ico|svg)$ {
expires 1y;
add_header Cache-Control "public, immutable";
}
# Don't cache index.html
location = /index.html {
expires -1;
add_header Cache-Control "no-store, no-cache, must-revalidate";
}
}

View File

@@ -1 +1 @@
{"root":["../../src/App.tsx","../../src/main.tsx","../../src/components/AgentDetailPanel.tsx","../../src/components/AgentNode.tsx","../../src/components/ConnectionsLayer.tsx","../../src/components/MissionControlDashboard.tsx","../../src/components/SearchAndFilters.tsx","../../src/components/StatusBar.tsx","../../src/components/Timeline.tsx","../../src/data/agents.ts","../../src/lib/utils.ts","../../src/types/agent.ts"],"version":"5.9.3"} {"root":["../../src/App.tsx","../../src/main.tsx","../../src/components/AgentDetailPanel.tsx","../../src/components/AgentNode.tsx","../../src/components/ConnectionsLayer.tsx","../../src/components/MissionControlDashboard.tsx","../../src/components/SearchAndFilters.tsx","../../src/components/StatusBar.tsx","../../src/components/Timeline.tsx","../../src/data/agents.ts","../../src/hooks/useOpenClaw.ts","../../src/lib/openclaw-api.ts","../../src/lib/utils.ts","../../src/types/agent.ts"],"version":"5.9.3"}