Skip to content

Generated projects test sharding

REQUIREMENTS

Test sharding for generated projects uses tuist test for both the build and test phases.

How it works

Test sharding follows a two-phase workflow:

  1. Build phase: Tuist enumerates your tests and creates a shard plan on the server. The server uses historical test timing data from the last 30 days to distribute tests across shards so each shard takes roughly the same amount of time. The build phase outputs a shard matrix that your CI system uses to spawn parallel runners.
  2. Test phase: Each CI runner receives a shard index and executes only the tests assigned to that shard.

Build phase

Generate your project, build your tests, and create a shard plan:

sh
tuist test --shard-total 5

This command:

  1. Generates the Xcode project from your manifests
  2. Builds your tests
  3. Creates a shard plan on the Tuist server using historical timing data
  4. Uploads the .xctestproducts bundle for use by shard runners
  5. Outputs a shard matrix for your CI system
  6. Persists the selective testing graph (if applicable) so shard runners don't need to regenerate the project

Build options

FlagEnvironment variableDescription
--shard-max <N>TUIST_TEST_SHARD_MAXMaximum number of shards. Used with --shard-max-duration to cap the shard count
--shard-min <N>TUIST_TEST_SHARD_MINMinimum number of shards
--shard-total <N>TUIST_TEST_SHARD_TOTALExact number of shards (mutually exclusive with --shard-min/--shard-max)
--shard-max-duration <MS>TUIST_TEST_SHARD_MAX_DURATIONTarget maximum duration per shard in milliseconds
--shard-granularity <LEVEL>TUIST_TEST_SHARD_GRANULARITYmodule (default) distributes entire test modules across shards; suite distributes individual test classes for finer-grained balancing
--shard-reference <REF>TUIST_SHARD_REFERENCEUnique identifier for the shard plan (auto-derived on supported CI providers)

Test phase

Each shard runner executes its assigned tests:

sh
tuist test

Test options

FlagEnvironment variableDescription
--shard-index <N>TUIST_SHARD_INDEXZero-based index of the shard to execute
--shard-reference <REF>TUIST_SHARD_REFERENCEUnique identifier for the shard plan (auto-derived on supported CI providers)

Tuist downloads the .xctestproducts bundle and filters it to include only the tests assigned to that shard.

SELECTIVE TESTING

Test sharding works seamlessly with selective testing. The selective testing graph is persisted during the build phase and restored for each shard, so runners don't need to regenerate the project.

Continuous integration

Test sharding currently supports the following CI providers:

  • GitHub Actions

GitHub Actions

Use a matrix strategy to run shards in parallel:

yaml
name: Tests
on: [pull_request]

jobs:
  build:
    name: Build test shards
    runs-on: macos-latest
    outputs:
      matrix: ${{ steps.build.outputs.matrix }}
    steps:
      - uses: actions/checkout@v4
      - uses: jdx/mise-action@v2
      - run: tuist auth login
      - id: build
        run: tuist test --shard-total 5

  test:
    name: "Shard #${{ matrix.shard }}"
    needs: build
    runs-on: macos-latest
    strategy:
      fail-fast: false
      matrix:
        shard: ${{ fromJson(needs.build.outputs.matrix).shard }}
    env:
      TUIST_SHARD_INDEX: ${{ matrix.shard }}
    steps:
      - uses: actions/checkout@v4
      - uses: jdx/mise-action@v2
      - run: tuist auth login
      - run: tuist test

Released under the MIT License.