Skip to main content

Split tests (parallelism)

With Harness CI, you can split tests for any language or test tool. This uses test splitting and the parallelism looping strategy to improve test times.

When you run tests in Harness CI, you use Run and Run Tests steps. You can enable test splitting on either of these steps.

warning

This topic explains how to set up parallelism and test splitting in Run steps. For information about test splitting with Test Intelligence (in Run Tests steps), go to Split tests (parallelism) with TI.

To split tests in a Run step, you need to:

  1. Define a parallelism strategy.
  2. Define a test splitting strategy.
  3. Produce test reports in JUnit XML format.
  4. Run the pipeline and inspect the test results.
Learn more about test splitting and parallelism

Most CI pipelines are set up to run tests for every new commit. By default, tests run sequentially. If there are a lot of tests to run, developers might have to wait longer than necessary for test results. Similarly, if there are frequent commits, your builds might start to queue due to concurrency limits.

Test splitting and parallelism can decrease test cycle time by dividing tests into multiple groups and running the groups in parallel. You can configure test splitting by number of tests, test timing, file size, and so on.

For example, suppose you have a pipeline that runs 100 tests, and each test takes about one second to run. By default, all 100 tests run in sequence, taking 100 seconds. If you use test splitting to create four workloads with 25 tests each, the four groups run at the same time, and then it takes only 25 seconds to run all 100 tests. While a savings of 75 seconds doesn't seem like much for a single run, assuming a rate of 50 commits per week, this amounts to a savings of just over one hour per week.

( 75 seconds x 50 commits ) / 60 seconds = 62.5 minutes saved

Note that this example only calculates the runtime for the tests. Additional time can be required for other commands in your Run or Run Tests step, such as initializing the step, installing dependencies, and so on.

Time saved can improve over subsequent runs. If you use a timing strategy to split tests, Harness must collect timing data during the first parallel run. Therefore, on the first parallel run, Harness divides tests by file size. Then, on subsequent runs, Harness uses the timing data from earlier runs to split tests by time. With each subsequent run, Harness further refines test splitting based on newer timing data.

This diagram demonstrates how parallelism can accelerate your CI pipelines. Without parallelism, the tests run one after the other. With parallelism enabled, Harness first splits tests into four groups based on file size, already significantly reducing the overall run time. Using timing data collected in the first parallel run, subsequent runs split tests by test time, further optimizing run time. With each subsequent run, test partitioning is refined based on the newest timing data.

Parallelism is one of the looping strategies available in Harness pipelines, and parallelism isn't limited to splitting tests. You can use parallelism to speed things up whenever it's possible to divide pipeline, step, or stage tasks into multiple sets and run them concurrently.

When using parallelism, it's important to take into account resource limitations that exist in your build infrastructure. For more information, go to Best Practices for Looping Strategies.

Here are examples of test splitting and parallelism applied to Run steps. For more YAML examples of test splitting, go to YAML Examples: Test splitting.

This example uses Harness Cloud build infrastructure.

    - stage:
name: tests
identifier: tests
description: ""
type: CI
spec:
cloneCodebase: true
platform:
os: Linux
arch: Amd64
runtime:
type: Cloud
spec: {}
execution:
steps:
- step:
type: Run ## Test splitting can be applied to any Run or Run Tests steps where you run tests.
name: tests
identifier: tests
strategy: ## This is the parallelism strategy for the step.
parallelism: 4 ## Tests are split into a maximum of four workloads.
maxConcurrency: 2 ## Optional. This setting limits the number of workloads that can run at once.
spec:
envVariables: ## These environment variables are used in the 'command' to get the index value for each parallel instance and the total number of parallel instances.
HARNESS_NODE_INDEX: <+strategy.iteration>
HARNESS_NODE_TOTAL: <+strategy.iterations>
shell: Sh
command: |- ## Split tests commands are included alongside the regular test commands.
# Install dependencies.
pip install -r requirements.txt

# Call split_tests, define splitting strategy, and generate the list of test files.
FILES=`./split_tests --glob "**/test_*.py" --split-by file_timing \
--split-index ${HARNESS_NODE_INDEX} \
--split-total ${HARNESS_NODE_TOTAL}`
echo $FILES

# Use the test files list as input for pytest and produce results in JUnit XML format.
pytest -v --junitxml="result_<+strategy.iteration>.xml" $FILES

reports:
type: JUnit
spec:
paths:
- "**/result_<+strategy.iteration>.xml" ## Using the expression '<+strategy.iteration>' in the file name ensures that the results of parallel runs don't overwrite each other.

Define a parallelism strategy

In the context of test splitting, the parallelism strategy defines the number of workloads into which tests can be divided. Each parallel instance (or workload) is a duplicate of the step or stage where you've defined a parallelism strategy, but each instance runs different tests according to the test splitting strategy.

Learn more about parallel workloads

To demonstrate how tests are split into concurrent (parallel) workloads, assume that you use the test_count splitting strategy to divide 20 tests into four workloads. To tell Harness to create four workloads, you would set your parallelism strategy to parallelism: 4. Then use the test_count splitting strategy in your Run step's commands. When the pipeline runs, test_count uses simple division to split the 20 tests into four workloads; therefore, each parallel instance runs five different tests.

20 tests / 4 workloads = 5 tests per workload

Parallel instances are zero-indexed, resulting in the following breakdown:

  • Instance 0 runs tests 1-5.
  • Instance 1 runs tests 6-10.
  • Instance 2 runs tests 11-15.
  • Instance 3 runs tests 16-20.

The four instances run concurrently, but they might finish at different times depending on individual test times. This example used test_count, which simply divides the tests based on the total number of tests; therefore, the instances might not finish at the same time if some tests run longer than others. To achieve more equal run times, you can use more nuanced splitting strategies, such as file size or timing.

  1. Use strategy.parallelism to define a parallelism strategy on either the step or stage where your tests run.

    • strategy: Declares a looping strategy
    • parallelism: Specify the number of workloads that you want to divide your tests into. For example, if you want to create four workloads, set parallelism: 4.
    • maxConcurrency: Optional strategy to optimize parallelism.

    This example shows parallelism applied to a Run step. For more YAML examples, go to YAML examples: Test splitting.

                  - step:
    type: Run
    name: tests
    identifier: tests
    strategy: ## Declares a looping strategy.
    parallelism: 8 ## Specify the number of workloads. This example creates 8 workloads.
    maxConcurrency: 2 ## Optional. Limit the number of workloads that can run at once.
    spec:
    ...
  2. Add the following environment variables to the same step or stage where you defined the parallelism strategy:

    • HARNESS_NODE_TOTAL: <+strategy.iterations> - This variable specifies the total number of parallel instances.
    • HARNESS_NODE_INDEX: <+strategy.iteration> - This variable specifies the index value of the currently-running parallel instance. Parallel instances are zero-indexed, so this value ranges from 0 to parallelism-1.

    This example shows these environment variables declared on a step. To declare them on a stage, add them to the stage variables.

                  - step:
    type: Run
    name: tests
    identifier: tests
    ...
    spec:
    envVariables:
    HARNESS_NODE_INDEX: <+strategy.iteration>
    HARNESS_NODE_TOTAL: <+strategy.iterations>
    ...

    You'll use these variables when you define a test splitting strategy to create commands that can be used for all parallel instances.

    FILES=`/addon/bin/split_tests --glob "**/test_*.py" --split-by file_timing \
    --split-index ${HARNESS_NODE_INDEX} \
    --split-total ${HARNESS_NODE_TOTAL}`
    echo $FILES

    You can also use them to create helpful step logs to help you differentiate parallel instances, such as echo "${HARNESS_NODE_INDEX} of ${HARNESS_NODE_TOTAL}".

Optimize parallelism

In general, a higher parallelism value means a faster pipeline run time, because the tests can be divided into more parallel instances. However, this depends on your test suite and resource limitations in your build infrastructure. For example, if you try to run 10 groups of tests, but your build infrastructure can't handle 10 parallel instances, the pipeline can fail or take longer than expected.

To optimize your parallelism strategy:

  • Try different parallelism values to determine your infrastructure's limits.
    • It's important to understand that parallelism impacts resource allocation for each stage in the pipeline. A stage with five sequential steps can require fewer resources than a stage with running five parallel instances of one step, because the second stage has to run all five instances at once.
  • Use maxConcurrency to control the flow of parallel instances and avoid overtaxing infrastructure resources. Concurrency limits the number of parallel instances that can run at once and queues additional instances.
    • For example, if you set parallelism: 12, Harness attempts to run 12 instances of the step at once. If you set parallelism: 12 and maxConcurrency: 3, Harness generates 12 instances of the step, but only runs three instances at a time. The remaining nine instances are queued, and the queued instances start running as space clears in the concurrency limit (when prior instances finish).
    • Concurrency allows you to divide tests into more workloads without overloading your system resources.
    • Keep in mind that there are resource requirements for generating parallel instances (even if they are not all running at the same time) and handling queues. Try different combinations of parallelism and maxConcurrency values to determine your ideal configuration.
  • Review the Best practices for looping strategies, including how to calculate ideal concurrency.

Define a test splitting strategy

The test splitting strategy determines how you want to divide the tests, such as by number of tests or test timing.

To define and apply a test splitting strategy, use the split_tests binary on the Run step where your tests run, for example:

FILES=`[binary path] [--glob or --file-path] [--split-by] [--split-index] [--split-total]`

The split_tests tool outputs a list of test files based on your chosen splitting strategy. You then use this list as input for your test tool's commands so that each parallel instance only runs an assigned subset of tests. For example, these commands use split_tests with pytest:

# Install dependencies.
pip install -r requirements.txt

# Call split_tests, define splitting strategy, and generate the list of test files.
FILES=`/addon/bin/split_tests --glob "**/test_*.py" --split-by file_timing \
--split-index ${HARNESS_NODE_INDEX} \
--split-total ${HARNESS_NODE_TOTAL}`
echo $FILES

# Use the test files list as input for pytest and produce results in JUnit XML format.
pytest -v --junitxml="result_<+strategy.iteration>.xml" $FILES

Binary path

Call the split_tests binary. The path depends on your build infrastructure.

  • For Harness Cloud, use ./split_tests.
  • For other build infrastructures, use /addon/bin/split_tests.
  • For steps running in containers rather that on the host, use usr/bin/split_tests. You must also declare this path in shared paths.

Glob or file-path

Specify the set of all tests that you want to run across all parallel instances. Whether to use --glob or --file-path is determined by --split-by.

  • For --split-by file_timing, --split-by file_size, or --split-by test_count, you can use a glob expression to specify the set of files to split, such as --glob "**/test_*.py".

  • For --split-by class_timing, --split-by testcase_timing, and --split-by testsuite_timing, you must provide a text file of the elements to split. For example, if you want to split by Java class timing, you could specify the set of classes to split and test in a new-line-delineated string and then reference the text file in with --file-path FILE_NAME.txt:

    # Generate a list of classes and store in classname.txt
    echo 'io.harness.jhttp.server.PathResolverTest\nio.harness.jhttp.processor.DirectoryIndexTest\nio.harness.jhttp.functional.HttpClientTest\nio.harness.jhttp.processor.ResourceNotFoundTest'> classnames.txt

    # Run split_tests on elements in classname.txt
    CLASSES=`/addon/bin/split_tests --split-by class_timing --file-path classnames.txt`

Split-by

Specify a test splitting strategy, such as test timing or file size. If unspecified, the default is --split-by file_timing.

The split_tests binary supports these test splitting strategies:

  • --split-by class_timing: Split tests into groups based on the timing data for individual classes. This strategy requires timing data from the previous run. If timing data isn't available, split_tests falls back to --split-by file_size.
  • --split-by file_size: Split tests into groups based on the size of individual files.
  • --split-by file_timing: (Default) Split tests into groups based on the test times of individual files. split_tests uses the most recent timing data to ensure that all parallel test runs finish at approximately the same time. This strategy requires timing data from the previous run. If timing data isn't available, split_tests falls back to --split-by file_size.
  • --split-by test_count: Split tests into groups based on the total number of tests.
  • --split-by testcase_timing: Split tests into groups based on the timing data for individual test cases. This strategy requires timing data from the previous run. If timing data isn't available, split_tests falls back to --split-by file_size.
  • --split-by testsuite_timing: Split tests into groups based on the timing data for individual test suites. This strategy requires timing data from the previous run. If timing data isn't available, split_tests falls back to --split-by file_size.

Split-index and split-total

Include --split-index ${HARNESS_NODE_INDEX} and --split-total ${HARNESS_NODE_TOTAL}. These commands create groups of tests files based on the total number of parallel instances (--split-total) and assign specific test files for each individual instance (--split-index). These commands use the environment variables you declared when you defined a parallelism strategy.

Split tests output

split_tests stores list of test files assigned to the current instance in a variable. For example, the following command stores the list of tests in $FILES:

# Call split_tests, define splitting strategy, and generate the list of test files.
FILES=`/addon/bin/split_tests --glob "**/test_*.py" --split-by file_timing \
--split-index ${HARNESS_NODE_INDEX} \
--split-total ${HARNESS_NODE_TOTAL}`
echo $FILES

You then call the variable (such as $FILES) in your test tool's commands, for example:

pytest -v --junitxml="result_<+strategy.iteration>.xml" $FILES

You can include echo $FILES to print the list of assigned tests in each step's logs, for example:

+ FILES=test_file_1.py test_file_2.py test_file_6.py test_file_9.py test_file_10.py test_file_12.py test_file_13.py

Test splitting for Test Intelligence

If you define a parallelism strategy on a Run Tests step, Harness automatically splits tests by class timing. For information about enabling test splitting with Test Intelligence (in a Run Tests step), go to Split tests (parallelism) with TI. Test splitting is configured differently in Run Tests steps than in Run steps.

Produce test reports

  1. Edit the step where your tests run.

  2. Make sure your test tool's commands produce test results. The specific commands required to produce test results files depends on the specific language, test runner, and formatter you use.

  3. Use an expression or variable in the results file name, such as result_<+strategy.iteration>.xml or result_${HARNESS_NODE_INDEX}.xml, to ensure each parallel instance produces a uniquely-named results file.

    warning

    If you defined the parallelism strategy on a step (instead of a stage), you must use an expression or variable in the results file name, such as result_<+strategy.iteration>.xml or result_${HARNESS_NODE_INDEX}.xml, to ensure each parallel instance produces a uniquely-named results file. If you don't use an expression or variable in the results file name, the files overwrite each other or fail due to same-name conflicts.

  4. To publish your test results in the Harness UI, your test results must be in JUnit XML format.

    • Configure your test runner and formatter to publish your test reports in the JUnit XML format and include file names in the XML output.
    • For example, if you use pytest, you can set junit_family=xunit1 in your code repo's pytest.ini file, or you can include -o junit_family="xunit1" in the step's command.
  5. Add the reports section to the step.spec and include the paths for your test results files. For example:

                      type: Run
    name: tests
    identifier: tests
    spec:
    ...
    reports: ## Start of reports section.
    type: JUnit ## Specify format as JUnit.
    spec:
    paths: - "**/result_${HARNESS_NODE_INDEX}.xml" ## Specify the results file path. Use a variable or expression to generate uniquely-named results files for each parallel instance. Without a differentiating identifier, the results files can overwrite each other.

Logs and results

When you run the pipeline, you can observe the parallel instances running on the Build details page.

info

If you use a timing split strategy, Harness must collect timing data during the first parallel run. Therefore, on the first parallel run, Harness divides tests by file size. Then, on subsequent runs, Harness can use the timing data from earlier runs to split tests by time.

Time saved can improve over subsequent runs. With each subsequent run, Harness further refines test splitting based on newer timing data.

Parallel steps in a build.

When the build finishes, go to the Tests tab to view the results. Use the Test Executions stage and step dropdown menu to view results for each parallel instance.

View results for individual runs.

YAML examples: Test splitting

With Harness CI, you can split tests for any language or tool. Here are some examples of test splitting for different languages and tools.

Go

              - step:
type: Run ## Test splitting can be applied to any Run or Run Tests steps where you run tests.
name: tests
identifier: tests
strategy:
parallelism: 4 ## Specify the number of workloads to split tests into.
maxConcurrency: 2 ## Optional. This setting limits the number of workloads that can run at once.
spec:
envVariables: ## These environment variables are used in 'command'.
HARNESS_NODE_INDEX: <+strategy.iteration>
HARNESS_NODE_TOTAL: <+strategy.iterations>
shell: Sh
command: |- ## Split tests commands are included alongside the regular test commands.
# Install dependencies.
go install github.com/jstemmer/go-junit-report/v2@latest

# Call split_tests, define splitting strategy, and generate the list of test files.
FILES=`/addon/bin/split_tests --glob "**/*_test.go" --split-by file_timing \
--split-index ${HARNESS_NODE_INDEX} \
--split-total ${HARNESS_NODE_TOTAL}`
echo $FILES

# Use the test files list as input and produce results in JUnit XML format.
go test -v $FILES | tee report_<+strategy.iteration>.out
cat report_<+strategy.iteration>.out | $HOME/go/bin/go-junit-report -set-exit-code > report_<+strategy.iteration>.xml

reports:
type: JUnit
spec:
paths:
- "report_<+strategy.iteration>.xml" ## Using the expression '<+strategy.iteration>' in the file name ensures that the results of parallel runs don't overwrite each other.

Java

This example use Maven.

              - step:
type: Run ## Test splitting can be applied to any Run or Run Tests steps where you run tests.
name: tests
identifier: tests
strategy:
parallelism: 4 ## Specify the number of workloads to split tests into.
maxConcurrency: 2 ## Optional. This setting limits the number of workloads that can run at once.
spec:
envVariables: ## These environment variables are used in 'command'.
HARNESS_NODE_INDEX: <+strategy.iteration>
HARNESS_NODE_TOTAL: <+strategy.iterations>
shell: Sh
command: |- ## Split tests commands are included alongside the regular test commands.
# Install dependencies.
mvn clean package dependency:copy-dependencies

# Call split_tests, define splitting strategy, and generate the list of test files.
FILES=`/addon/bin/split_tests --glob "PATH_TO_TEST_FILES" --split-by file_timing \
--split-index ${HARNESS_NODE_INDEX} \
--split-total ${HARNESS_NODE_TOTAL}`
echo $FILES

# Use the test files list as input. Make sure results are in JUnit XML format.
mvn -Dtest=$FILES test

reports:
type: JUnit
spec:
paths:
- target/surefire-reports/*_<+strategy.iteration>.xml" ## Using the expression '<+strategy.iteration>' in the file name ensures that the results of parallel runs don't overwrite each other. Depending on your test tool, you might need to specify the report name in your test suite config file.

JavaScript

This example uses Mocha.

              - step:
type: Run ## Test splitting can be applied to any Run or Run Tests steps where you run tests.
name: tests
identifier: tests
strategy:
parallelism: 4 ## Specify the number of workloads to split tests into.
maxConcurrency: 2 ## Optional. This setting limits the number of workloads that can run at once.
spec:
envVariables: ## These environment variables are used in 'command'.
HARNESS_NODE_INDEX: <+strategy.iteration>
HARNESS_NODE_TOTAL: <+strategy.iterations>
shell: Sh
command: |- ## Split tests commands are included alongside the regular test commands.
# Install dependencies.
npm install
mkdir /harness/junit
npm run build --if-present

# Call split_tests, define splitting strategy, and generate the list of test files.
FILES=`/addon/bin/split_tests --glob "PATH_TO_TEST_FILES" --split-by file_timing \
--split-index ${HARNESS_NODE_INDEX} \
--split-total ${HARNESS_NODE_TOTAL}`
echo $FILES

# Use the test files list as input and produce results in JUnit XML format.
mocha test $FILES --reporter mocha-junit-reporter --reporter-options mochaFile=/harness/junit/test-results-<+strategy.iteration>.xml

reports:
type: JUnit
spec:
paths:
- "/harness/junit/test-results-<+strategy.iteration>.xml" ## Using the expression '<+strategy.iteration>' in the file name ensures that the results of parallel runs don't overwrite each other.

PHP

              - step:
type: Run ## Test splitting can be applied to any Run or Run Tests steps where you run tests.
name: tests
identifier: tests
strategy:
parallelism: 4 ## Specify the number of workloads to split tests into.
maxConcurrency: 2 ## Optional. This setting limits the number of workloads that can run at once.
spec:
envVariables: ## These environment variables are used in 'command'.
HARNESS_NODE_INDEX: <+strategy.iteration>
HARNESS_NODE_TOTAL: <+strategy.iterations>
shell: Sh
command: |- ## Split tests commands are included alongside the regular test commands.
# Install dependencies.
mkdir -p /harness/phpunit

# Call split_tests, define splitting strategy, and generate the list of test files.
FILES=`/addon/bin/split_tests --glob "**/test_*.php" --split-by file_timing \
--split-index ${HARNESS_NODE_INDEX} \
--split-total ${HARNESS_NODE_TOTAL}`
echo $FILES

# Use the test files list as input and produce results in JUnit XML format.
phpunit --filter $FILES --log-junit /harness/phpunit/junit_<+strategy.iteration>.xml

reports:
type: JUnit
spec:
paths:
- "/harness/phpunit/junit_<+strategy.iteration>.xml" ## Using the expression '<+strategy.iteration>' in the file name ensures that the results of parallel runs don't overwrite each other.

Python

This example uses pytest.

              - step:
type: Run ## Test splitting can be applied to any Run or Run Tests steps where you run tests.
name: tests
identifier: tests
strategy:
parallelism: 4 ## Specify the number of workloads to split tests into.
maxConcurrency: 2 ## Optional. This setting limits the number of workloads that can run at once.
spec:
envVariables: ## These environment variables are used in 'command'.
HARNESS_NODE_INDEX: <+strategy.iteration>
HARNESS_NODE_TOTAL: <+strategy.iterations>
shell: Sh
command: |- ## Split tests commands are included alongside the regular test commands.
# Install dependencies.
pip install -r requirements.txt

# Call split_tests, define splitting strategy, and generate the list of test files.
FILES=`/addon/bin/split_tests --glob "**/test_*.py" --split-by file_timing \
--split-index ${HARNESS_NODE_INDEX} \
--split-total ${HARNESS_NODE_TOTAL}`
echo $FILES

# Use the test files list as input for pytest and produce results in JUnit XML format.
pytest -v --junitxml="result_<+strategy.iteration>.xml" $FILES

reports:
type: JUnit
spec:
paths:
- "**/result_<+strategy.iteration>.xml" ## Using the expression '<+strategy.iteration>' in the file name ensures that the results of parallel runs don't overwrite each other.

Ruby

This example uses Minitest.

              - step:
type: Run ## Test splitting can be applied to any Run or Run Tests steps where you run tests.
name: tests
identifier: tests
strategy:
parallelism: 4 ## Specify the number of workloads to split tests into.
maxConcurrency: 2 ## Optional. This setting limits the number of workloads that can run at once.
spec:
envVariables: ## These environment variables are used in 'command'.
HARNESS_NODE_INDEX: <+strategy.iteration>
HARNESS_NODE_TOTAL: <+strategy.iterations>
shell: Sh
command: |- ## Split tests commands are included alongside the regular test commands.
# Install dependencies.
bundle check || bundle install

# Call split_tests, define splitting strategy, and generate the list of test files.
FILES=`/addon/bin/split_tests --glob "**/test_*.rb" --split-by file_timing \
--split-index ${HARNESS_NODE_INDEX} \
--split-total ${HARNESS_NODE_TOTAL}`
echo $FILES

# Use the test files list as input for rake test and produce results in JUnit XML format.
bundle exec rake test -n $FILES --junit --junit-filename="result_<+strategy.iteration>.xml"

reports:
type: JUnit
spec:
paths:
- "**/result_<+strategy.iteration>.xml" ## Using the expression '<+strategy.iteration>' in the file name ensures that the results of parallel runs don't overwrite each other.

C/C++

This example uses CTest. Note that CTest has parallelize functions built-in as well.

              - step:
type: Run ## Test splitting can be applied to any Run or Run Tests steps where you run tests.
name: tests
identifier: tests
strategy:
parallelism: 4 ## Specify the number of workloads to split tests into.
maxConcurrency: 2 ## Optional. This setting limits the number of workloads that can run at once.
spec:
envVariables: ## These environment variables are used in 'command'.
HARNESS_NODE_INDEX: <+strategy.iteration>
HARNESS_NODE_TOTAL: <+strategy.iterations>
shell: Sh
command: |- ## Split tests commands are included alongside the regular test commands.
# Install dependencies.
mkdir build
cmake -S . -B build

# Call split_tests, define splitting strategy, and generate the list of test files.
FILES=`/addon/bin/split_tests --glob "/build" --split-by file_timing \
--split-index ${HARNESS_NODE_INDEX} \
--split-total ${HARNESS_NODE_TOTAL}`
echo $FILES

# Use the test files list as input for ctest and produce results in JUnit XML format.
ctest --test-dir $FILES --output-junit out_<+strategy.iteration>.xml

reports:
type: JUnit
spec:
paths:
- "/harness/build/out_<+strategy.iteration>.xml" ## Using the expression '<+strategy.iteration>' in the file name ensures that the results of parallel runs don't overwrite each other.

C#

              - step:
type: Run ## Test splitting can be applied to any Run or Run Tests steps where you run tests.
name: tests
identifier: tests
strategy:
parallelism: 4 ## Specify the number of workloads to split tests into.
maxConcurrency: 2 ## Optional. This setting limits the number of workloads that can run at once.
spec:
envVariables: ## These environment variables are used in 'command'.
HARNESS_NODE_INDEX: <+strategy.iteration>
HARNESS_NODE_TOTAL: <+strategy.iterations>
shell: Sh
command: |- ## Split tests commands are included alongside the regular test commands.
# Install dependencies
dotnet tool install -g trx2junit
export PATH="$PATH:/root/.dotnet/tools"

dotnet restore
dotnet build

# Call split_tests, define splitting strategy, and generate the list of test files.
FILES=`/addon/bin/split_tests --glob "PATH_TO_TEST_PROJECT" --split-by file_timing \
--split-index ${HARNESS_NODE_INDEX} \
--split-total ${HARNESS_NODE_TOTAL}`
echo $FILES

# Use the test files list as input and produce results in JUnit XML format.
dotnet test --filter $FILES --no-build --verbosity normal
trx2junit results_<+strategy.iteration>.trx

reports:
type: JUnit
spec:
paths:
- "./results_<+strategy.iteration>.trx" ## Using the expression '<+strategy.iteration>' in the file name ensures that the results of parallel runs don't overwrite each other.