name: CI Server on: pull_request: merge_group: permissions: contents: read concurrency: group: ${{ github.workflow }}-${{ github.ref }} cancel-in-progress: ${{ github.ref != 'refs/heads/main' }} env: SERVER_BUILD_CACHE_KEY: server-build jobs: changed-files-check: if: github.event_name != 'merge_group' uses: ./.github/workflows/changed-files.yaml with: files: | package.json yarn.lock packages/twenty-server/** packages/twenty-front/src/generated/** packages/twenty-front/src/generated-metadata/** packages/twenty-client-sdk/** packages/twenty-emails/** packages/twenty-shared/** server-build: needs: changed-files-check if: needs.changed-files-check.outputs.any_changed == 'true' timeout-minutes: 30 runs-on: ubuntu-latest steps: - name: Fetch custom Github Actions and base branch history uses: actions/checkout@v4 with: fetch-depth: 10 - name: Install dependencies uses: ./.github/actions/yarn-install - name: Restore server build cache id: restore-server-build-cache uses: ./.github/actions/restore-cache with: key: ${{ env.SERVER_BUILD_CACHE_KEY }} - name: Build twenty-shared run: npx nx build twenty-shared - name: Server / Write .env run: npx nx reset:env twenty-server - name: Server / Build run: npx nx build twenty-server - name: Save server build cache uses: ./.github/actions/save-cache with: key: ${{ steps.restore-server-build-cache.outputs.cache-primary-key }} server-lint-typecheck: needs: changed-files-check if: needs.changed-files-check.outputs.any_changed == 'true' timeout-minutes: 30 runs-on: ubuntu-latest steps: - name: Fetch custom Github Actions and base branch history uses: actions/checkout@v4 with: fetch-depth: 10 - name: Install dependencies uses: ./.github/actions/yarn-install - name: Build twenty-shared run: npx nx build twenty-shared - name: Server / Run lint & typecheck uses: ./.github/actions/nx-affected with: tag: scope:backend tasks: lint,typecheck server-validation: needs: server-build timeout-minutes: 30 runs-on: ubuntu-latest services: postgres: image: postgres:18 env: POSTGRES_USER: postgres POSTGRES_PASSWORD: postgres ports: - 5432:5432 options: >- --health-cmd pg_isready --health-interval 10s --health-timeout 5s --health-retries 5 redis: image: redis ports: - 6379:6379 steps: - name: Fetch custom Github Actions and base branch history uses: actions/checkout@v4 with: fetch-depth: 10 - name: Install dependencies uses: ./.github/actions/yarn-install - name: Restore server build cache uses: ./.github/actions/restore-cache with: key: ${{ env.SERVER_BUILD_CACHE_KEY }} - name: Build twenty-shared run: npx nx build twenty-shared - name: Server / Write .env run: npx nx reset:env twenty-server - name: Server / Build run: npx nx build twenty-server - name: Server / Create DB run: | PGPASSWORD=postgres psql -h localhost -p 5432 -U postgres -d postgres -c 'CREATE DATABASE "default";' PGPASSWORD=postgres psql -h localhost -p 5432 -U postgres -d postgres -c 'CREATE DATABASE "test";' npx nx run twenty-server:database:init:prod - name: Worker / Run run: | timeout 30s npx nx run twenty-server:worker || exit_code=$? if [ $exit_code -eq 124 ]; then exit 0 elif [ $exit_code -ne 0 ]; then exit $exit_code fi - name: Server / Start run: npx nx start:ci twenty-server & - name: Waiting for server starting... run: | for i in {1..10}; do if curl -f http://localhost:3000/healthz; then echo "Server ready!" exit 0 fi echo "Waiting..." sleep 2 done echo "Server did not become healthy in time" >&2 exit 1 - name: Server / Check for Pending Migrations run: | npx nx database:migrate:generate twenty-server -- --name pending-migration-check || true if ! git diff --quiet; then echo "::error::Unexpected migration files were generated. Please run 'npx nx database:migrate:generate twenty-server -- --name ' and commit the result." echo "" echo "The following migration changes were detected:" echo "===================================================" git diff echo "===================================================" echo "" git checkout -- . exit 1 fi - name: Check for Pending Code Generation run: | HAS_ERRORS=false npx nx run twenty-front:graphql:generate npx nx run twenty-front:graphql:generate --configuration=metadata if ! git diff --quiet -- packages/twenty-front/src/generated packages/twenty-front/src/generated-metadata; then echo "::error::GraphQL schema changes detected. Please run 'npx nx run twenty-front:graphql:generate' and 'npx nx run twenty-front:graphql:generate --configuration=metadata' and commit the changes." echo "" echo "The following GraphQL schema changes were detected:" echo "===================================================" git diff -- packages/twenty-front/src/generated packages/twenty-front/src/generated-metadata echo "===================================================" echo "" HAS_ERRORS=true fi npx nx run twenty-client-sdk:generate-metadata-client if ! git diff --quiet -- packages/twenty-client-sdk/src/metadata/generated; then echo "::error::SDK metadata client changes detected. Please run 'npx nx run twenty-client-sdk:generate-metadata-client' and commit the changes." echo "" echo "The following SDK metadata client changes were detected:" echo "===================================================" git diff -- packages/twenty-client-sdk/src/metadata/generated echo "===================================================" echo "" HAS_ERRORS=true fi if [ "$HAS_ERRORS" = true ]; then exit 1 fi server-test: needs: server-build timeout-minutes: 30 runs-on: ubuntu-latest steps: - name: Fetch custom Github Actions and base branch history uses: actions/checkout@v4 with: fetch-depth: 10 - name: Install dependencies uses: ./.github/actions/yarn-install - name: Restore server build cache uses: ./.github/actions/restore-cache with: key: ${{ env.SERVER_BUILD_CACHE_KEY }} - name: Build twenty-shared run: npx nx build twenty-shared - name: Server / Run Tests uses: ./.github/actions/nx-affected with: tag: scope:backend tasks: test server-integration-test: timeout-minutes: 30 runs-on: ubuntu-latest needs: server-build strategy: fail-fast: false matrix: shard: [1, 2, 3, 4, 5, 6, 7, 8, 9, 10] services: postgres: image: postgres:18 env: POSTGRES_USER: postgres POSTGRES_PASSWORD: postgres ports: - 5432:5432 options: >- --health-cmd pg_isready --health-interval 10s --health-timeout 5s --health-retries 5 redis: image: redis ports: - 6379:6379 clickhouse: image: clickhouse/clickhouse-server:25.8.8 env: CLICKHOUSE_PASSWORD: clickhousePassword CLICKHOUSE_URL: "http://default:clickhousePassword@localhost:8123/twenty" ports: - 8123:8123 - 9000:9000 options: >- --health-cmd "clickhouse-client --host=localhost --port=9000 --user=default --password=clickhousePassword --query='SELECT 1'" --health-interval 10s --health-timeout 5s --health-retries 5 env: NODE_ENV: test ANALYTICS_ENABLED: true CLICKHOUSE_URL: "http://default:clickhousePassword@localhost:8123/twenty" CLICKHOUSE_PASSWORD: clickhousePassword SHARD_COUNTER: 10 steps: - name: Fetch custom Github Actions and base branch history uses: actions/checkout@v4 with: fetch-depth: 10 - name: Install dependencies uses: ./.github/actions/yarn-install - name: Update .env.test for integrations tests run: | echo "" >> .env.test echo "IS_BILLING_ENABLED=true" >> .env.test echo "BILLING_STRIPE_API_KEY=test-api-key" >> .env.test echo "BILLING_STRIPE_BASE_PLAN_PRODUCT_ID=test-base-plan-product-id" >> .env.test echo "BILLING_STRIPE_WEBHOOK_SECRET=test-webhook-secret" >> .env.test echo "BILLING_PLAN_REQUIRED_LINK=http://localhost:3001/stripe-redirection" >> .env.test - name: Restore server build cache uses: ./.github/actions/restore-cache with: key: ${{ env.SERVER_BUILD_CACHE_KEY }} - name: Server / Build run: npx nx build twenty-server - name: Build dependencies run: | npx nx build twenty-shared npx nx build twenty-emails - name: Server / Create Test DB run: | PGPASSWORD=postgres psql -h localhost -p 5432 -U postgres -d postgres -c 'CREATE DATABASE "test";' - name: Run ClickHouse migrations run: npx nx clickhouse:migrate twenty-server - name: Run ClickHouse seeds run: npx nx clickhouse:seed twenty-server - name: Server / Run Integration Tests uses: ./.github/actions/nx-affected with: tag: scope:backend tasks: 'test:integration' configuration: 'with-db-reset' args: --shard=${{ matrix.shard }}/${{ env.SHARD_COUNTER }} ci-server-status-check: if: always() && !cancelled() timeout-minutes: 5 runs-on: ubuntu-latest needs: [ changed-files-check, server-build, server-lint-typecheck, server-validation, server-test, server-integration-test, ] steps: - name: Fail job if any needs failed if: contains(needs.*.result, 'failure') run: exit 1