Add Docker and Gitea Actions CI/CD pipeline
CI / build-and-push (push) Successful in 1m10s

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
This commit is contained in:
2026-04-24 18:27:07 -04:00
parent 938df33789
commit 68540ebcf9
10 changed files with 154 additions and 2 deletions
+7
View File
@@ -0,0 +1,7 @@
node_modules
dist
.turbo
*.tsbuildinfo
.git
.claude
.gitea
+30
View File
@@ -0,0 +1,30 @@
name: CI
on:
push:
branches: [main]
env:
REGISTRY: gitea.thewrightserver.net
IMAGE_PREFIX: gitea.thewrightserver.net/josh/aihostingtycoon
jobs:
build-and-push:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v4
- name: Log in to Gitea registry
run: echo "${{ secrets.REGISTRY_TOKEN }}" | docker login ${{ env.REGISTRY }} -u josh --password-stdin
- name: Build and push web image
run: |
docker build -f apps/web/Dockerfile -t ${{ env.IMAGE_PREFIX }}/web:latest -t ${{ env.IMAGE_PREFIX }}/web:${{ github.sha }} .
docker push ${{ env.IMAGE_PREFIX }}/web:latest
docker push ${{ env.IMAGE_PREFIX }}/web:${{ github.sha }}
- name: Build and push server image
run: |
docker build -f apps/server/Dockerfile -t ${{ env.IMAGE_PREFIX }}/server:latest -t ${{ env.IMAGE_PREFIX }}/server:${{ github.sha }} .
docker push ${{ env.IMAGE_PREFIX }}/server:latest
docker push ${{ env.IMAGE_PREFIX }}/server:${{ github.sha }}
+29
View File
@@ -0,0 +1,29 @@
FROM node:22-slim AS base
RUN corepack enable && corepack prepare pnpm@10.33.0 --activate
WORKDIR /app
FROM base AS deps
COPY pnpm-lock.yaml pnpm-workspace.yaml package.json ./
COPY packages/shared/package.json packages/shared/
COPY packages/tsconfig/package.json packages/tsconfig/
COPY apps/server/package.json apps/server/
RUN pnpm install --frozen-lockfile --prod=false
FROM base AS build
COPY --from=deps /app/node_modules ./node_modules
COPY --from=deps /app/packages/shared/node_modules ./packages/shared/node_modules
COPY --from=deps /app/apps/server/node_modules ./apps/server/node_modules
COPY . .
RUN pnpm --filter @ai-tycoon/shared build && \
pnpm --filter @ai-tycoon/server typecheck
FROM base AS production
COPY --from=deps /app/node_modules ./node_modules
COPY --from=deps /app/packages/shared/node_modules ./packages/shared/node_modules
COPY --from=deps /app/apps/server/node_modules ./apps/server/node_modules
COPY packages/shared ./packages/shared
COPY packages/tsconfig ./packages/tsconfig
COPY apps/server ./apps/server
ENV NODE_ENV=production
EXPOSE 3001
CMD ["node", "--import", "tsx", "apps/server/src/index.ts"]
+2 -1
View File
@@ -5,7 +5,8 @@
"type": "module", "type": "module",
"scripts": { "scripts": {
"dev": "tsx watch src/index.ts", "dev": "tsx watch src/index.ts",
"build": "tsc && tsx src/index.ts", "build": "tsc",
"start": "node --import tsx src/index.ts",
"typecheck": "tsc --noEmit", "typecheck": "tsc --noEmit",
"db:generate": "drizzle-kit generate", "db:generate": "drizzle-kit generate",
"db:migrate": "drizzle-kit migrate", "db:migrate": "drizzle-kit migrate",
+3 -1
View File
@@ -10,7 +10,9 @@ const app = new Hono();
app.use('*', logger()); app.use('*', logger());
app.use('*', cors({ app.use('*', cors({
origin: ['http://localhost:5173', 'http://localhost:5174', 'http://localhost:5175', 'http://localhost:5178'], origin: process.env.CORS_ORIGIN
? process.env.CORS_ORIGIN.split(',')
: ['http://localhost:5173', 'http://localhost:5174', 'http://localhost:5175', 'http://localhost:5178'],
allowMethods: ['GET', 'POST', 'PUT', 'DELETE'], allowMethods: ['GET', 'POST', 'PUT', 'DELETE'],
allowHeaders: ['Content-Type', 'Authorization'], allowHeaders: ['Content-Type', 'Authorization'],
})); }));
+28
View File
@@ -0,0 +1,28 @@
FROM node:22-slim AS base
RUN corepack enable && corepack prepare pnpm@10.33.0 --activate
WORKDIR /app
FROM base AS deps
COPY pnpm-lock.yaml pnpm-workspace.yaml package.json ./
COPY packages/shared/package.json packages/shared/
COPY packages/tsconfig/package.json packages/tsconfig/
COPY packages/game-engine/package.json packages/game-engine/
COPY apps/web/package.json apps/web/
RUN pnpm install --frozen-lockfile
FROM base AS build
COPY --from=deps /app/node_modules ./node_modules
COPY --from=deps /app/packages/shared/node_modules ./packages/shared/node_modules
COPY --from=deps /app/packages/game-engine/node_modules ./packages/game-engine/node_modules
COPY --from=deps /app/apps/web/node_modules ./apps/web/node_modules
COPY . .
ARG VITE_API_URL=/api
ENV VITE_API_URL=$VITE_API_URL
RUN pnpm --filter @ai-tycoon/shared build && \
pnpm --filter @ai-tycoon/game-engine build && \
pnpm --filter @ai-tycoon/web build
FROM nginx:alpine
COPY --from=build /app/apps/web/dist /usr/share/nginx/html
COPY apps/web/nginx.conf /etc/nginx/conf.d/default.conf
EXPOSE 80
+14
View File
@@ -0,0 +1,14 @@
server {
listen 80;
root /usr/share/nginx/html;
index index.html;
location / {
try_files $uri $uri/ /index.html;
}
location /assets/ {
expires 1y;
add_header Cache-Control "public, immutable";
}
}
+39
View File
@@ -0,0 +1,39 @@
services:
web:
image: gitea.thewrightserver.net/josh/aihostingtycoon/web:latest
ports:
- "80:80"
depends_on:
- server
restart: unless-stopped
server:
image: gitea.thewrightserver.net/josh/aihostingtycoon/server:latest
ports:
- "3001:3001"
environment:
- DATABASE_URL=postgresql://aitycoon:aitycoon@db:5432/aitycoon
- PORT=3001
- CORS_ORIGIN=*
depends_on:
db:
condition: service_healthy
restart: unless-stopped
db:
image: postgres:17-alpine
environment:
- POSTGRES_USER=aitycoon
- POSTGRES_PASSWORD=aitycoon
- POSTGRES_DB=aitycoon
volumes:
- pgdata:/var/lib/postgresql/data
healthcheck:
test: ["CMD-SHELL", "pg_isready -U aitycoon"]
interval: 5s
timeout: 5s
retries: 5
restart: unless-stopped
volumes:
pgdata:
+1
View File
@@ -6,6 +6,7 @@
"main": "./src/index.ts", "main": "./src/index.ts",
"types": "./src/index.ts", "types": "./src/index.ts",
"scripts": { "scripts": {
"build": "tsc",
"typecheck": "tsc --noEmit" "typecheck": "tsc --noEmit"
}, },
"dependencies": { "dependencies": {
+1
View File
@@ -6,6 +6,7 @@
"main": "./src/index.ts", "main": "./src/index.ts",
"types": "./src/index.ts", "types": "./src/index.ts",
"scripts": { "scripts": {
"build": "tsc",
"typecheck": "tsc --noEmit" "typecheck": "tsc --noEmit"
}, },
"devDependencies": { "devDependencies": {