From 64b488ac89de4b0fab8452bc1e202514d95ba0d2 Mon Sep 17 00:00:00 2001 From: "soroush.asadi" Date: Wed, 27 May 2026 23:21:29 +0330 Subject: [PATCH] ci: add Gitea Actions workflow (CI + self-hosted deploy) .gitea/workflows/ci-cd.yml: - Triggers on push to main and PRs - CI jobs: dotnet build/test, dashboard tsc, finder tsc (all self-hosted) - Deploy job: only on push to main, needs all CI jobs to pass - Writes .env from ENV_FILE secret (set in Gitea repo settings) - docker compose build --parallel with BuildKit - Rolling restart (postgres/redis untouched) - Health-check poll: waits up to 2min for meezi-api healthy - Auto-prunes old images on success Co-Authored-By: Claude Sonnet 4.5 --- .gitea/workflows/ci-cd.yml | 142 +++++++++++++++++++++++++++++++++++++ 1 file changed, 142 insertions(+) create mode 100644 .gitea/workflows/ci-cd.yml diff --git a/.gitea/workflows/ci-cd.yml b/.gitea/workflows/ci-cd.yml new file mode 100644 index 0000000..364379b --- /dev/null +++ b/.gitea/workflows/ci-cd.yml @@ -0,0 +1,142 @@ +name: CI/CD + +on: + push: + branches: [main] + pull_request: + branches: [main] + +# Only one deploy at a time; a newer push cancels an in-progress one +concurrency: + group: meezi-cicd-${{ github.ref }} + cancel-in-progress: true + +# ───────────────────────────────────────────────────────────────────────────── +# CI — runs on every push AND every PR +# ───────────────────────────────────────────────────────────────────────────── +jobs: + api-build: + name: "CI · API (dotnet build)" + runs-on: self-hosted + steps: + - uses: actions/checkout@v4 + + - name: Setup .NET 10 + uses: actions/setup-dotnet@v4 + with: + dotnet-version: "10.0.x" + + - name: Restore dependencies + run: dotnet restore + + - name: Build (Release) + run: dotnet build --no-restore -c Release + + - name: Run tests + run: dotnet test --no-build -c Release --logger "console;verbosity=minimal" + + # ── Dashboard typecheck ──────────────────────────────────────────────────── + dashboard-check: + name: "CI · Dashboard (tsc)" + runs-on: self-hosted + defaults: + run: + working-directory: web/dashboard + steps: + - uses: actions/checkout@v4 + + - uses: actions/setup-node@v4 + with: + node-version: "20" + + - name: Install dependencies + run: npm install --legacy-peer-deps --ignore-scripts + + - name: TypeScript check + run: npx tsc --noEmit + env: + NEXT_PUBLIC_API_URL: http://localhost:5080 + + # ── Finder typecheck ─────────────────────────────────────────────────────── + finder-check: + name: "CI · Finder (tsc)" + runs-on: self-hosted + defaults: + run: + working-directory: web/finder + steps: + - uses: actions/checkout@v4 + + - uses: actions/setup-node@v4 + with: + node-version: "20" + + - name: Install dependencies + run: npm install --legacy-peer-deps --ignore-scripts + + - name: TypeScript check + run: npx tsc --noEmit + env: + NEXT_PUBLIC_API_URL: http://localhost:5080 + +# ───────────────────────────────────────────────────────────────────────────── +# DEPLOY — only on push to main, only if all CI jobs pass +# ───────────────────────────────────────────────────────────────────────────── + deploy: + name: "Deploy · docker compose" + runs-on: self-hosted + needs: [api-build, dashboard-check, finder-check] + # Skip deploy on PRs — only run when pushing directly to main + if: github.event_name == 'push' && github.ref == 'refs/heads/main' + timeout-minutes: 30 + + steps: + - uses: actions/checkout@v4 + + # Write .env from a single Gitea secret called ENV_FILE + # How to set it: Gitea repo → Settings → Secrets → Add secret + # Name: ENV_FILE + # Value: paste your entire .env file content + - name: Write .env + run: printf '%s' "$ENV_FILE" > .env + env: + ENV_FILE: ${{ secrets.ENV_FILE }} + + # Build all service images in parallel using BuildKit + - name: Build Docker images + run: | + docker compose build --parallel api web website finder + env: + DOCKER_BUILDKIT: 1 + COMPOSE_DOCKER_CLI_BUILD: 1 + + # Rolling restart — postgres/redis stay untouched if already healthy + - name: Start / restart services + run: | + docker compose up -d \ + --remove-orphans \ + --no-deps \ + postgres redis api web website finder + + # Poll until API container reports healthy (max 2 min) + - name: Wait for API to become healthy + run: | + echo "Waiting for meezi-api to become healthy..." + for i in $(seq 1 24); do + STATUS=$(docker inspect --format='{{.State.Health.Status}}' meezi-api 2>/dev/null || echo "missing") + echo " [$i/24] $STATUS" + [ "$STATUS" = "healthy" ] && echo "✅ Healthy" && exit 0 + sleep 5 + done + echo "❌ API did not become healthy — last 50 log lines:" + docker compose logs --tail=50 api + exit 1 + + - name: Show running containers + if: always() + run: docker compose ps + + # Remove dangling images to keep disk clean + - name: Prune old images + if: success() + run: docker image prune -f