name: Build & Deploy on: push: branches: - main tags: - 'v*' workflow_dispatch: concurrency: group: ${{ github.workflow }} cancel-in-progress: false jobs: prepare: name: ๐Ÿ” Prepare Environment runs-on: docker container: image: catthehacker/ubuntu:act-latest outputs: target: ${{ steps.determine.outputs.target }} image_tag: ${{ steps.determine.outputs.image_tag }} env_file: ${{ steps.determine.outputs.env_file }} traefik_host: ${{ steps.determine.outputs.traefik_host }} next_public_base_url: ${{ steps.determine.outputs.next_public_base_url }} directus_url: ${{ steps.determine.outputs.directus_url }} directus_host: ${{ steps.determine.outputs.directus_host }} project_name: ${{ steps.determine.outputs.project_name }} steps: - name: ๐Ÿ” Debug Info shell: bash run: | echo "ref_name: ${{ github.ref_name }}" echo "ref_type: ${{ github.ref_type }}" echo "tag: ${{ github.ref_name }}" - name: ๐Ÿงน Maintenance (Runner Cleanup) continue-on-error: true shell: bash run: | docker image prune -f || true docker builder prune -f --filter "until=24h" || true - name: Checkout repository uses: actions/checkout@v4 with: fetch-depth: 2 - name: ๐Ÿ” Determine Environment id: determine shell: bash run: | TAG="${{ github.ref_name }}" SHORT_SHA=$(echo "${{ github.sha }}" | cut -c1-7) DOMAIN_BASE="mb-grid-solutions.com" PRJ_ID="mb-grid-solutions" if [[ "${{ github.ref_type }}" == "branch" && "$TAG" == "main" ]]; then TARGET="testing" IMAGE_TAG="testing-${SHORT_SHA}" ENV_FILE=".env.testing" TRAEFIK_HOST="testing.${DOMAIN_BASE}" NEXT_PUBLIC_BASE_URL="https://testing.${DOMAIN_BASE}" DIRECTUS_URL="https://cms.testing.${DOMAIN_BASE}" DIRECTUS_HOST="cms.testing.${DOMAIN_BASE}" elif [[ "${{ github.ref_type }}" == "tag" ]]; then if [[ "$TAG" =~ ^v[0-9]+\.[0-9]+\.[0-9]+$ ]]; then TARGET="production" IMAGE_TAG="$TAG" ENV_FILE=".env.prod" TRAEFIK_HOST="${DOMAIN_BASE}, www.${DOMAIN_BASE}" NEXT_PUBLIC_BASE_URL="https://${DOMAIN_BASE}" DIRECTUS_URL="https://cms.${DOMAIN_BASE}" DIRECTUS_HOST="cms.${DOMAIN_BASE}" elif [[ "$TAG" =~ -rc || "$TAG" =~ -beta || "$TAG" =~ -alpha ]]; then TARGET="staging" IMAGE_TAG="$TAG" ENV_FILE=".env.staging" TRAEFIK_HOST="staging.${DOMAIN_BASE}" NEXT_PUBLIC_BASE_URL="https://staging.${DOMAIN_BASE}" DIRECTUS_URL="https://cms.staging.${DOMAIN_BASE}" DIRECTUS_HOST="cms.staging.${DOMAIN_BASE}" else TARGET="skip" fi else TARGET="skip" fi echo "Target determined: $TARGET" echo "target=$TARGET" >> "$GITHUB_OUTPUT" echo "image_tag=$IMAGE_TAG" >> "$GITHUB_OUTPUT" echo "env_file=$ENV_FILE" >> "$GITHUB_OUTPUT" echo "traefik_host=$TRAEFIK_HOST" >> "$GITHUB_OUTPUT" echo "next_public_base_url=$NEXT_PUBLIC_BASE_URL" >> "$GITHUB_OUTPUT" echo "directus_url=$DIRECTUS_URL" >> "$GITHUB_OUTPUT" echo "directus_host=$DIRECTUS_HOST" >> "$GITHUB_OUTPUT" echo "project_name=$PRJ_ID-$TARGET" >> "$GITHUB_OUTPUT" qa: name: ๐Ÿงช QA needs: prepare if: needs.prepare.outputs.target != 'skip' runs-on: docker container: image: catthehacker/ubuntu:act-latest steps: - name: Checkout repository uses: actions/checkout@v4 - name: Setup Node.js uses: actions/setup-node@v4 with: node-version: 20 - name: Install dependencies shell: bash run: | corepack enable pnpm install --frozen-lockfile env: NPM_TOKEN: ${{ secrets.NPM_TOKEN }} - name: ๐Ÿงช Lint shell: bash run: pnpm lint env: NPM_TOKEN: ${{ secrets.NPM_TOKEN }} - name: ๐Ÿ—๏ธ Build Test shell: bash run: pnpm build env: NPM_TOKEN: ${{ secrets.NPM_TOKEN }} NEXT_PUBLIC_BASE_URL: https://dummy.test build: name: ๐Ÿ—๏ธ Build needs: prepare if: needs.prepare.outputs.target != 'skip' runs-on: docker container: image: catthehacker/ubuntu:act-latest steps: - name: Checkout repository uses: actions/checkout@v4 - name: ๐Ÿณ Set up Docker Buildx uses: docker/setup-buildx-action@v3 - name: ๐Ÿ” Registry Login run: echo "${{ secrets.REGISTRY_PASS }}" | docker login registry.infra.mintel.me -u "${{ secrets.REGISTRY_USER }}" --password-stdin - name: ๐Ÿ—๏ธ Build and Push shell: bash run: | docker buildx build \ --pull \ --platform linux/arm64 \ --build-arg NPM_TOKEN=${{ secrets.NPM_TOKEN }} \ --build-arg NEXT_PUBLIC_BASE_URL=${{ needs.prepare.outputs.next_public_base_url }} \ --build-arg NEXT_PUBLIC_TARGET=${{ needs.prepare.outputs.target }} \ --build-arg DIRECTUS_URL=${{ needs.prepare.outputs.directus_url }} \ -t registry.infra.mintel.me/mintel/mb-grid-solutions:${{ needs.prepare.outputs.image_tag }} \ --push . deploy: name: ๐Ÿš€ Deploy needs: [prepare, build, qa] if: needs.prepare.outputs.target != 'skip' runs-on: docker container: image: catthehacker/ubuntu:act-latest steps: - name: Checkout repository uses: actions/checkout@v4 with: fetch-depth: 1 - name: ๐Ÿš€ Deploy via SSH shell: bash run: | echo "Deploying to alpha.mintel.me" mkdir -p ~/.ssh echo "${{ secrets.ALPHA_SSH_KEY }}" > ~/.ssh/id_ed25519 chmod 600 ~/.ssh/id_ed25519 ssh-keyscan -H alpha.mintel.me >> ~/.ssh/known_hosts 2>/dev/null # Generate Environment File cat > .env.deploy << 'EOF' ENV_FILE=${{ needs.prepare.outputs.env_file }} IMAGE_TAG=${{ needs.prepare.outputs.image_tag }} TRAEFIK_HOST=${{ needs.prepare.outputs.traefik_host }} PROJECT_NAME=${{ needs.prepare.outputs.project_name }} NEXT_PUBLIC_BASE_URL=${{ needs.prepare.outputs.next_public_base_url }} # Directus DIRECTUS_URL=${{ needs.prepare.outputs.directus_url }} DIRECTUS_HOST=${{ needs.prepare.outputs.directus_host }} INTERNAL_DIRECTUS_URL=http://directus:8055 DIRECTUS_API_TOKEN=${{ secrets.DIRECTUS_API_TOKEN || vars.DIRECTUS_API_TOKEN }} DIRECTUS_ADMIN_EMAIL=${{ secrets.DIRECTUS_ADMIN_EMAIL || vars.DIRECTUS_ADMIN_EMAIL || 'admin@mintel.me' }} DIRECTUS_ADMIN_PASSWORD=${{ secrets.DIRECTUS_ADMIN_PASSWORD || vars.DIRECTUS_ADMIN_PASSWORD }} DIRECTUS_DB_NAME=${{ secrets.DIRECTUS_DB_NAME || vars.DIRECTUS_DB_NAME || 'directus' }} DIRECTUS_DB_USER=${{ secrets.DIRECTUS_DB_USER || vars.DIRECTUS_DB_USER || 'directus' }} DIRECTUS_DB_PASSWORD=${{ secrets.DIRECTUS_DB_PASSWORD || vars.DIRECTUS_DB_PASSWORD }} DIRECTUS_KEY=${{ secrets.DIRECTUS_KEY || vars.DIRECTUS_KEY }} DIRECTUS_SECRET=${{ secrets.DIRECTUS_SECRET || vars.DIRECTUS_SECRET }} # SMTP Config SMTP_HOST=${{ secrets.SMTP_HOST || vars.SMTP_HOST }} SMTP_PORT=${{ secrets.SMTP_PORT || vars.SMTP_PORT || '587' }} SMTP_SECURE=${{ secrets.SMTP_SECURE || vars.SMTP_SECURE || 'false' }} SMTP_USER=${{ secrets.SMTP_USER || vars.SMTP_USER }} SMTP_PASS=${{ secrets.SMTP_PASS || vars.SMTP_PASS }} SMTP_FROM=${{ secrets.SMTP_FROM || vars.SMTP_FROM }} CONTACT_RECIPIENT=${{ secrets.CONTACT_RECIPIENT || vars.CONTACT_RECIPIENT }} # Authentication GATEKEEPER_PASSWORD=${{ secrets.GATEKEEPER_PASSWORD || vars.GATEKEEPER_PASSWORD }} AUTH_COOKIE_NAME=${{ secrets.AUTH_COOKIE_NAME || vars.AUTH_COOKIE_NAME || 'mintel_gatekeeper_session' }} COOKIE_DOMAIN=${{ secrets.COOKIE_DOMAIN || vars.COOKIE_DOMAIN || '.mb-grid-solutions.com' }} # External Services SENTRY_DSN=${{ secrets.SENTRY_DSN || vars.SENTRY_DSN }} GOTIFY_URL=${{ secrets.GOTIFY_URL || vars.GOTIFY_URL }} GOTIFY_TOKEN=${{ secrets.GOTIFY_TOKEN || vars.GOTIFY_TOKEN }} NEXT_PUBLIC_UMAMI_WEBSITE_ID=${{ secrets.NEXT_PUBLIC_UMAMI_WEBSITE_ID || vars.NEXT_PUBLIC_UMAMI_WEBSITE_ID }} NEXT_PUBLIC_UMAMI_SCRIPT_URL=${{ secrets.NEXT_PUBLIC_UMAMI_SCRIPT_URL || vars.NEXT_PUBLIC_UMAMI_SCRIPT_URL }} # Project PROJECT_COLOR=${{ secrets.PROJECT_COLOR || vars.PROJECT_COLOR || '#82ed20' }} EOF APP_DIR="/home/deploy/sites/mb-grid-solutions.com" ssh -o StrictHostKeyChecking=accept-new root@alpha.mintel.me "mkdir -p $APP_DIR" scp -o StrictHostKeyChecking=accept-new .env.deploy root@alpha.mintel.me:$APP_DIR/${{ needs.prepare.outputs.env_file }} scp -o StrictHostKeyChecking=accept-new docker-compose.yaml root@alpha.mintel.me:$APP_DIR/docker-compose.yaml ssh -o StrictHostKeyChecking=accept-new root@alpha.mintel.me bash << 'EOF' set -e APP_DIR="/home/deploy/sites/mb-grid-solutions.com" cd $APP_DIR echo "${{ secrets.REGISTRY_PASS }}" | docker login registry.infra.mintel.me -u "${{ secrets.REGISTRY_USER }}" --password-stdin docker compose -p "${{ needs.prepare.outputs.project_name }}" --env-file ${{ needs.prepare.outputs.env_file }} pull docker compose -p "${{ needs.prepare.outputs.project_name }}" --env-file ${{ needs.prepare.outputs.env_file }} up -d --remove-orphans docker system prune -f --filter "until=24h" EOF notifications: name: ๐Ÿ”” Notifications needs: [prepare, deploy] if: always() runs-on: docker container: image: catthehacker/ubuntu:act-latest steps: - name: Notify Gotify shell: bash run: | STATUS="${{ needs.deploy.result }}" COLOR="info" [[ "$STATUS" == "success" ]] && PRIORITY=5 || PRIORITY=8 curl -X POST "${{ secrets.GOTIFY_URL }}/message?token=${{ secrets.GOTIFY_TOKEN }}" \ -F "title=mb-grid-solutions Deployment" \ -F "message=Status: $STATUS for ${{ needs.prepare.outputs.target }} (${{ needs.prepare.outputs.image_tag }})" \ -F "priority=$PRIORITY"