new server setup

This commit is contained in:
2026-04-01 15:46:28 +00:00
parent fdd30692f1
commit c96cde7b7b
8 changed files with 126 additions and 4 deletions

5
.dockerignore Normal file
View File

@@ -0,0 +1,5 @@
node_modules/
server/node_modules/
server/dist/
.git/
.claude/

3
.gitignore vendored
View File

@@ -33,3 +33,6 @@ yarn-debug.log*
yarn-error.log*
package-lock.json
.claude/
uploads/

17
Dockerfile.backend Normal file
View File

@@ -0,0 +1,17 @@
# ---- Build stage ----
FROM node:22-alpine AS builder
WORKDIR /app
COPY server/package.json server/package-lock.json ./
RUN npm ci
COPY server/src ./src
COPY server/tsconfig.json ./
RUN npm run build
# ---- Runtime stage ----
FROM node:22-alpine
WORKDIR /app
COPY --from=builder /app/dist ./dist
COPY server/package.json server/package-lock.json ./
RUN npm ci --omit=dev
EXPOSE 3001
CMD ["node", "dist/index.js"]

13
Dockerfile.frontend Normal file
View File

@@ -0,0 +1,13 @@
# ---- Build stage ----
FROM node:22-alpine AS builder
WORKDIR /app
COPY package.json package-lock.json ./
RUN npm ci
COPY . .
RUN npx vite build
# ---- Serve stage ----
FROM nginx:1.27-alpine
COPY --from=builder /app/dist /usr/share/nginx/html
COPY nginx.conf /etc/nginx/conf.d/default.conf
EXPOSE 5173

57
docker-compose.yml Normal file
View File

@@ -0,0 +1,57 @@
services:
postgres:
image: postgres:16-alpine
restart: unless-stopped
environment:
POSTGRES_USER: labwise
POSTGRES_PASSWORD: ${POSTGRES_PASSWORD}
POSTGRES_DB: labwise_db
volumes:
- postgres_data:/var/lib/postgresql/data
healthcheck:
test: ["CMD-SHELL", "pg_isready -U labwise -d labwise_db"]
interval: 5s
timeout: 5s
retries: 10
deploy:
resources:
limits:
memory: 256M
backend:
build:
context: .
dockerfile: Dockerfile.backend
restart: unless-stopped
# Reads all vars from server/.env, then overrides the two that differ in Docker
env_file: ./server/.env
environment:
DATABASE_URL: postgresql://labwise:${POSTGRES_PASSWORD}@postgres:5432/labwise_db
UPLOADS_DIR: /app/uploads
volumes:
- uploads_data:/app/uploads
depends_on:
postgres:
condition: service_healthy
deploy:
resources:
limits:
memory: 192M
frontend:
build:
context: .
dockerfile: Dockerfile.frontend
restart: unless-stopped
ports:
- "5173:5173"
depends_on:
- backend
deploy:
resources:
limits:
memory: 48M
volumes:
postgres_data:
uploads_data:

26
nginx.conf Normal file
View File

@@ -0,0 +1,26 @@
server {
listen 5173;
root /usr/share/nginx/html;
index index.html;
# Proxy API and uploads to Express backend
location /api/ {
proxy_pass http://backend:3001;
proxy_set_header Host $host;
proxy_set_header X-Real-IP $remote_addr;
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
proxy_set_header X-Forwarded-Proto $scheme;
}
location /uploads/ {
proxy_pass http://backend:3001;
proxy_set_header Host $host;
proxy_set_header X-Real-IP $remote_addr;
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
}
# SPA fallback — all unmatched routes serve index.html
location / {
try_files $uri $uri/ /index.html;
}
}

View File

@@ -12,15 +12,16 @@
"@aws-sdk/client-ses": "^3.1013.0",
"better-auth": "^1.5.5",
"cors": "^2.8.5",
"dotenv": "^17.3.1",
"express": "^4.21.2",
"express-rate-limit": "^7.5.0",
"multer": "^1.4.5-lts.1",
"multer": "^2.0.0",
"pg": "^8.13.3"
},
"devDependencies": {
"@types/cors": "^2.8.17",
"@types/express": "^5.0.1",
"@types/multer": "^1.4.12",
"@types/multer": "^2.1.0",
"@types/node": "^22.13.10",
"@types/pg": "^8.11.11",
"tsx": "^4.19.3",

View File

@@ -18,5 +18,5 @@
}
},
"include": ["**/*.ts", "**/*.tsx"],
"exclude": ["node_modules", "dist"]
"exclude": ["node_modules", "dist", "server"]
}