mirror of
https://github.com/mattermost/mattermost.git
synced 2026-04-15 05:57:37 -04:00
Some checks are pending
API / build (push) Waiting to run
Server CI / Compute Go Version (push) Waiting to run
Server CI / Check mocks (push) Blocked by required conditions
Server CI / Check go mod tidy (push) Blocked by required conditions
Server CI / check-style (push) Blocked by required conditions
Server CI / Check serialization methods for hot structs (push) Blocked by required conditions
Server CI / Vet API (push) Blocked by required conditions
Server CI / Check migration files (push) Blocked by required conditions
Server CI / Generate email templates (push) Blocked by required conditions
Server CI / Check store layers (push) Blocked by required conditions
Server CI / Check mmctl docs (push) Blocked by required conditions
Server CI / Postgres with binary parameters (push) Blocked by required conditions
Server CI / Postgres (shard 0) (push) Blocked by required conditions
Server CI / Postgres (shard 1) (push) Blocked by required conditions
Server CI / Postgres (shard 2) (push) Blocked by required conditions
Server CI / Postgres (shard 3) (push) Blocked by required conditions
Server CI / Merge Postgres Test Results (push) Blocked by required conditions
Server CI / Postgres FIPS (shard 0) (push) Blocked by required conditions
Server CI / Postgres FIPS (shard 1) (push) Blocked by required conditions
Server CI / Postgres FIPS (shard 2) (push) Blocked by required conditions
Server CI / Postgres FIPS (shard 3) (push) Blocked by required conditions
Server CI / Merge Postgres FIPS Test Results (push) Blocked by required conditions
Server CI / Coverage (shard 0) (push) Blocked by required conditions
Server CI / Coverage (shard 1) (push) Blocked by required conditions
Server CI / Coverage (shard 2) (push) Blocked by required conditions
Server CI / Coverage (shard 3) (push) Blocked by required conditions
Server CI / Run mmctl tests (push) Blocked by required conditions
Server CI / Run mmctl tests (FIPS) (push) Blocked by required conditions
Server CI / Build mattermost server app (push) Blocked by required conditions
Tools CI / check-style (mattermost-govet) (push) Waiting to run
Tools CI / Test (mattermost-govet) (push) Waiting to run
Web App CI / check-lint (push) Waiting to run
Web App CI / check-i18n (push) Blocked by required conditions
Web App CI / check-external-links (push) Blocked by required conditions
Web App CI / check-types (push) Blocked by required conditions
Web App CI / test (platform) (push) Blocked by required conditions
Web App CI / test (mattermost-redux) (push) Blocked by required conditions
Web App CI / test (channels shard 1/4) (push) Blocked by required conditions
Web App CI / test (channels shard 2/4) (push) Blocked by required conditions
Web App CI / test (channels shard 3/4) (push) Blocked by required conditions
Web App CI / test (channels shard 4/4) (push) Blocked by required conditions
Web App CI / upload-coverage (push) Blocked by required conditions
Web App CI / build (push) Blocked by required conditions
The allow-failure input was defined twice in the workflow_call inputs, causing GitHub Actions to reject the workflow with 0 jobs on master push. Duplicate was introduced in #35743 merge. Release Note NONE Co-authored-by: Claude <claude@anthropic.com>
243 lines
9.5 KiB
YAML
243 lines
9.5 KiB
YAML
name: Server Test Template
|
|
on:
|
|
workflow_call:
|
|
inputs:
|
|
name:
|
|
required: true
|
|
type: string
|
|
datasource:
|
|
required: true
|
|
type: string
|
|
drivername:
|
|
required: true
|
|
type: string
|
|
logsartifact:
|
|
required: true
|
|
type: string
|
|
fullyparallel:
|
|
required: false
|
|
type: boolean
|
|
default: true
|
|
allow-failure:
|
|
required: false
|
|
type: boolean
|
|
default: false
|
|
enablecoverage:
|
|
required: false
|
|
type: boolean
|
|
default: false
|
|
go-version:
|
|
required: true
|
|
type: string
|
|
fips-enabled:
|
|
required: false
|
|
default: false
|
|
type: boolean
|
|
# -- Test sharding inputs (leave defaults for non-sharded callers) --
|
|
shard-index:
|
|
required: false
|
|
type: number
|
|
default: -1 # -1 = no sharding; run all tests
|
|
shard-total:
|
|
required: false
|
|
type: number
|
|
default: 1
|
|
|
|
permissions:
|
|
id-token: write
|
|
contents: read
|
|
|
|
jobs:
|
|
test:
|
|
name: ${{ inputs.name }}
|
|
runs-on: ubuntu-latest-8-cores
|
|
continue-on-error: ${{ inputs.allow-failure }} # Used to avoid blocking PRs in case of flakiness
|
|
env:
|
|
COMPOSE_PROJECT_NAME: ghactions
|
|
steps:
|
|
- name: buildenv/docker-login
|
|
# Only FIPS requires login for private build container. (Forks won't have credentials.)
|
|
if: inputs.fips-enabled
|
|
uses: docker/login-action@b45d80f862d83dbcd57f89517bcf500b2ab88fb2 # v4.0.0
|
|
with:
|
|
username: ${{ secrets.DOCKERHUB_USERNAME }}
|
|
password: ${{ secrets.DOCKERHUB_TOKEN }}
|
|
|
|
- name: Checkout mattermost project
|
|
uses: actions/checkout@de0fac2e4500dabe0009e67214ff5f5447ce83dd # v6.0.2
|
|
|
|
- name: Restore test timing data
|
|
if: inputs.shard-total > 1
|
|
id: timing-cache
|
|
uses: actions/cache/restore@5a3ec84eff668545956fd18022155c47e93e2684 # v4.2.3
|
|
with:
|
|
path: |
|
|
server/prev-report.xml
|
|
server/prev-gotestsum.json
|
|
# Always restore from master — timing is only saved on the default
|
|
# branch and is stable enough for shard balancing.
|
|
key: server-test-timing-master
|
|
restore-keys: |
|
|
server-test-timing-
|
|
|
|
- name: Setup BUILD_IMAGE
|
|
id: build
|
|
run: |
|
|
if [[ ${{ inputs.fips-enabled }} == 'true' ]]; then
|
|
echo "BUILD_IMAGE=mattermost/mattermost-build-server-fips:${{ inputs.go-version }}" >> "${GITHUB_OUTPUT}"
|
|
echo "LOG_ARTIFACT_NAME=${{ inputs.logsartifact }}-fips" >> "${GITHUB_OUTPUT}"
|
|
else
|
|
echo "BUILD_IMAGE=mattermost/mattermost-build-server:${{ inputs.go-version }}" >> "${GITHUB_OUTPUT}"
|
|
echo "LOG_ARTIFACT_NAME=${{ inputs.logsartifact }}" >> "${GITHUB_OUTPUT}"
|
|
fi
|
|
|
|
- name: Store required variables for publishing results
|
|
run: |
|
|
echo "${{ inputs.name }}" > server/test-name
|
|
echo "${{ github.event.pull_request.number }}" > server/pr-number
|
|
|
|
- name: Run docker compose
|
|
env:
|
|
POSTGRES_PASSWORD: ${{ inputs.fips-enabled && 'mostest-fips-test' || 'mostest' }}
|
|
run: |
|
|
cd server/build
|
|
docker compose --ansi never run --rm start_dependencies
|
|
cat ../tests/custom-schema-objectID.ldif | docker compose --ansi never exec -T openldap bash -c 'ldapadd -Y EXTERNAL -H ldapi:/// -w mostest || true';
|
|
cat ../tests/custom-schema-cpa.ldif | docker compose --ansi never exec -T openldap bash -c 'ldapadd -Y EXTERNAL -H ldapi:/// -w mostest || true';
|
|
cat ../tests/test-data.ldif | docker compose --ansi never exec -T openldap bash -c 'ldapadd -x -D "cn=admin,dc=mm,dc=test,dc=com" -w mostest';
|
|
docker compose --ansi never exec -T minio sh -c 'mkdir -p /data/mattermost-test';
|
|
docker compose --ansi never ps
|
|
|
|
# ── Test-level sharding ────────────────────────────────────────────
|
|
# When shard-total > 1, we split tests across N parallel runners.
|
|
#
|
|
# Two-tier splitting strategy:
|
|
# - "Light" packages (< 5 min): assigned whole to a shard
|
|
# - "Heavy" packages (≥ 5 min, e.g. api4, app): individual tests
|
|
# are distributed across shards using -run regex filters
|
|
#
|
|
# See server/scripts/shard-split.js for the full algorithm.
|
|
# ─────────────────────────────────────────────────────────────────────
|
|
|
|
- name: Setup Go for test discovery
|
|
if: inputs.shard-total > 1
|
|
uses: actions/setup-go@4b73464bb391d4059bd26b0524d20df3927bd417 # v6.3.0
|
|
with:
|
|
go-version: ${{ inputs.go-version }}
|
|
|
|
- name: Split tests across shards
|
|
if: inputs.shard-total > 1
|
|
id: test_split
|
|
working-directory: server
|
|
env:
|
|
SHARD_INDEX: ${{ inputs.shard-index }}
|
|
SHARD_TOTAL: ${{ inputs.shard-total }}
|
|
run: |
|
|
set -euo pipefail
|
|
|
|
# ── List all test packages ──
|
|
echo "::group::Listing test packages"
|
|
TE_PKGS=$(find ./public/ ./ -name '*_test.go' -not -path './enterprise/*' -not -path './cmd/mmctl/*' 2>/dev/null \
|
|
| sed 's|/[^/]*$||' | sort -u \
|
|
| sed 's|^\./|github.com/mattermost/mattermost/server/v8/|' \
|
|
| sed 's|github.com/mattermost/mattermost/server/v8/public/|github.com/mattermost/mattermost/server/public/|')
|
|
EE_PKGS=$(find ./enterprise/ -name '*_test.go' 2>/dev/null \
|
|
| sed 's|/[^/]*$||' | sort -u \
|
|
| sed 's|^\./|github.com/mattermost/mattermost/server/v8/|')
|
|
ALL_PKGS=$(printf '%s\n%s' "$TE_PKGS" "$EE_PKGS" | grep -v '^$' | sort -u)
|
|
TOTAL_PKGS=$(echo "$ALL_PKGS" | wc -l)
|
|
echo "Found $TOTAL_PKGS test packages"
|
|
echo "::endgroup::"
|
|
|
|
if [[ "$TOTAL_PKGS" -eq 0 ]]; then
|
|
echo "WARNING: No test packages found"
|
|
echo "has_packages=false" >> "$GITHUB_OUTPUT"
|
|
exit 0
|
|
fi
|
|
|
|
echo "$ALL_PKGS" > all-packages.txt
|
|
|
|
# ── Run shard solver ──
|
|
node scripts/shard-split.js
|
|
|
|
echo "has_packages=true" >> "$GITHUB_OUTPUT"
|
|
|
|
- name: Run Tests
|
|
env:
|
|
BUILD_IMAGE: ${{ steps.build.outputs.BUILD_IMAGE }}
|
|
run: |
|
|
if [[ ${{ github.ref_name }} == 'master' && ${{ inputs.fullyparallel }} != true ]]; then
|
|
export RACE_MODE="-race"
|
|
fi
|
|
|
|
MAKE_ARGS="test-server${RACE_MODE} BUILD_NUMBER=${GITHUB_HEAD_REF}-${GITHUB_RUN_ID}"
|
|
DOCKER_CMD="make ${MAKE_ARGS}"
|
|
|
|
# When sharding is active, use the multi-run wrapper script
|
|
if [[ "${{ inputs.shard-total }}" -gt 1 && -f server/shard-te-packages.txt ]]; then
|
|
SHARD_TE=$(cat server/shard-te-packages.txt)
|
|
SHARD_EE=$(cat server/shard-ee-packages.txt)
|
|
HEAVY_RUNS=""
|
|
if [[ -f server/shard-heavy-runs.txt && -s server/shard-heavy-runs.txt ]]; then
|
|
HEAVY_RUNS=$(cat server/shard-heavy-runs.txt)
|
|
fi
|
|
|
|
if [[ -z "$HEAVY_RUNS" ]]; then
|
|
# No heavy packages — single run via Makefile with package filter.
|
|
# Read packages from files at runtime to avoid interpolating
|
|
# file-system-derived paths into a generated shell script.
|
|
cat > server/run-shard-tests.sh <<SHARD_EOF
|
|
#!/bin/bash
|
|
exec make ${MAKE_ARGS} \\
|
|
TE_PACKAGES="\$(cat shard-te-packages.txt)" \\
|
|
EE_PACKAGES="\$(cat shard-ee-packages.txt)"
|
|
SHARD_EOF
|
|
else
|
|
# Use the multi-run wrapper script
|
|
cp server/scripts/run-shard-tests.sh server/run-shard-tests.sh
|
|
fi
|
|
|
|
chmod +x server/run-shard-tests.sh
|
|
DOCKER_CMD="/mattermost/server/run-shard-tests.sh"
|
|
fi
|
|
|
|
docker run --net ghactions_mm-test \
|
|
--ulimit nofile=8096:8096 \
|
|
--env-file=server/build/dotenv/test.env \
|
|
--env MM_SQLSETTINGS_DRIVERNAME="${{ inputs.drivername }}" \
|
|
--env MM_SQLSETTINGS_DATASOURCE="${{ inputs.datasource }}" \
|
|
--env TEST_DATABASE_POSTGRESQL_DSN="${{ inputs.datasource }}" \
|
|
--env ENABLE_FULLY_PARALLEL_TESTS="${{ inputs.fullyparallel }}" \
|
|
--env ENABLE_COVERAGE="${{ inputs.enablecoverage }}" \
|
|
--env FIPS_ENABLED="${{ inputs.fips-enabled }}" \
|
|
--env RACE_MODE \
|
|
-v $PWD:/mattermost \
|
|
-w /mattermost/server \
|
|
$BUILD_IMAGE \
|
|
$DOCKER_CMD
|
|
- name: Upload coverage to Codecov
|
|
if: ${{ inputs.enablecoverage }}
|
|
uses: codecov/codecov-action@671740ac38dd9b0130fbe1cec585b89eea48d3de # v5.5.2
|
|
with:
|
|
token: ${{ secrets.CODECOV_TOKEN }}
|
|
disable_search: true
|
|
files: server/cover.out
|
|
flags: server
|
|
|
|
- name: Stop docker compose
|
|
run: |
|
|
cd server/build
|
|
docker compose --ansi never stop
|
|
|
|
- name: Archive logs
|
|
if: ${{ always() }}
|
|
uses: actions/upload-artifact@bbbca2ddaa5d8feaa63e36b76fdaad77386f024f # v7.0.0
|
|
with:
|
|
name: ${{ steps.build.outputs.LOG_ARTIFACT_NAME }}
|
|
path: |
|
|
server/gotestsum.json
|
|
server/report.xml
|
|
server/cover.out
|
|
server/test-name
|
|
server/pr-number
|
|
|