testrail-integration
Syncs test runs / results / cases between an automated test suite and TestRail (Gurock / Idera) - opens a Test Run for the build (`add_run`), batches per-case results back via `add_results_for_cases` (preferred over per-test `add_result_for_case` - N+1 API calls vs 1), maps the test framework's pass/fail/skip to TestRail status IDs, and attaches build URL + version + elapsed time. Use when the team uses TestRail for test management and wants automated suites to update TestRail without a human-driven copy-paste step.
testrail-integration
Overview
Teams that use TestRail as the source of truth for test cases and runs need automation result sync - without it, automated runs don't update TestRail and the test-management view drifts from reality. This skill wires that sync.
The official documentation is at support.testrail.com/hc/en-us/articles/... (the canonical support knowledge base) and testrail.com/docs/api/ (the API reference). At the time of this skill's authoring (2026-05-05), both were behind a 403 to automated WebFetch; the URLs above are the canonical references that authors should consult in a real browser.
The patterns documented below are the stable, well-known TestRail API shapes that have been documented for many years across the Gurock blog, multiple versions of the support KB, and the per-language client libraries (Python testrail, Java testrail-java-client, JavaScript testrail-api).
When to use
Step 1 - Authentication
TestRail uses HTTP Basic auth with email + API key (preferred over password - the API key is per-user, revocable):
# Generated in TestRail: My Settings → API Keys
TESTRAIL_API_KEY=<generated>
TESTRAIL_USER=test-runner@example.com
TESTRAIL_HOST=https://yourcompany.testrail.ioAll requests use:
Authorization: Basic <base64(email:api_key)>
Content-Type: application/jsonThe base API URL is ${TESTRAIL_HOST}/index.php?/api/v2. Every endpoint is appended after ?/api/v2.
Step 2 - Map test names to TestRail case IDs
Two common patterns:
Pattern A - Embed case ID in test name
def test_C1234_can_add_to_cart():
...A regex extracts C1234 (the TestRail case ID) at sync time.
Pattern B - Annotation / metadata
@Test
@TestRailCase(id = 1234)
void canAddToCart() { ... }Or in JS:
test('can add to cart [C1234]', async () => {
// ...
});Pattern A is the lowest-friction; Pattern B is cleaner when the test framework supports custom annotations. Either way, the sync script needs a way to find the case ID from the test result.
Step 3 - Open a Test Run for the build
# scripts/testrail_sync.py
import base64, json, requests
from os import environ as env
API = f"{env['TESTRAIL_HOST']}/index.php?/api/v2"
AUTH = base64.b64encode(f"{env['TESTRAIL_USER']}:{env['TESTRAIL_API_KEY']}".encode()).decode()
HEADERS = {'Authorization': f'Basic {AUTH}', 'Content-Type': 'application/json'}
def open_run(project_id, suite_id, name, case_ids):
r = requests.post(
f'{API}/add_run/{project_id}',
headers=HEADERS,
json={
'suite_id': suite_id,
'name': name,
'include_all': False,
'case_ids': case_ids,
},
)
r.raise_for_status()
return r.json()['id'] # Run IDinclude_all: False + case_ids: [...] opens a run scoped to the exact cases the automated suite covers. Without this, a 5,000-case project produces a 5,000-row Test Run with thousands of empty cells.
Step 4 - Batch results back
The well-known status ID convention for stock TestRail installations:
| Status | ID |
|---|---|
| Passed | 1 |
| Blocked | 2 |
| Untested | 3 |
| Retest | 4 |
| Failed | 5 |
Custom status IDs (added by the project admin) follow 6+. Read the get_statuses endpoint at sync-script init to confirm - don't hard-code.
def add_results(run_id, results):
"""results = [{'case_id': 1234, 'status_id': 1, 'comment': '...', 'elapsed': '12s'}]"""
r = requests.post(
f'{API}/add_results_for_cases/{run_id}',
headers=HEADERS,
json={'results': results},
)
r.raise_for_status()
return r.json()Use add_results_for_cases (batch), not add_result_for_case (per-case). A 200-test run is one POST instead of 200 POSTs; TestRail's rate limit (180 req/min on shared cloud) makes per-case posting flaky.
Common per-result fields:
| Field | Use |
|---|---|
case_id | Required. The TestRail case ID. |
status_id | Required. Per the convention above. |
comment | The test framework's failure message + stack trace. |
elapsed | Format: '1h 30m 45s' or '45s'. Optional. |
version | Build version / commit SHA. Searchable in the UI. |
defects | Comma-separated Jira / GitHub issue keys. |
assignedto_id | Auto-assign failures to a specific user. |
Step 5 - Close the run
After all results are in:
def close_run(run_id):
requests.post(f'{API}/close_run/{run_id}', headers=HEADERS)Closed runs are read-only - no further results can be added. Useful for release-stamp runs; skip for runs that get re-run.
Step 6 - Wire into a CI pipeline
- name: Run tests
run: npm test -- --reporters=jest-junit
env:
JEST_JUNIT_OUTPUT_FILE: junit.xml
- name: Sync to TestRail
if: always()
env:
TESTRAIL_HOST: ${{ secrets.TESTRAIL_HOST }}
TESTRAIL_USER: ${{ secrets.TESTRAIL_USER }}
TESTRAIL_API_KEY: ${{ secrets.TESTRAIL_API_KEY }}
TESTRAIL_PROJECT_ID: '42'
TESTRAIL_SUITE_ID: '7'
BUILD_VERSION: ${{ github.sha }}
run: python scripts/testrail_sync.py junit.xmlThe sync script:
Step 7 - Handling untested case IDs
Tests that have no TestRail case ID (case removed; new test; intentional sync-skip) need explicit handling:
unmapped = [t for t in tests if extract_case_id(t['name']) is None]
if unmapped:
print(f"Warning: {len(unmapped)} tests have no TestRail case ID:")
for t in unmapped:
print(f" - {t['name']}")Don't silently drop unmapped tests - they're candidates for either new TestRail cases or naming-pattern fixes.
Anti-patterns
| Anti-pattern | Why it fails | Fix |
|---|---|---|
Per-test add_result_for_case calls | N API calls; rate limit (180 req/min on Cloud) trips on suites >180 cases. | add_results_for_cases batch (Step 4). |
Hard-coded status IDs without get_statuses | Custom statuses break the mapping; "Failed" goes to "Custom Status" silently. | Fetch get_statuses at script init; build the map dynamically. |
include_all: True on add_run | The run includes every case in the suite, most as Untested; runs become noise. | include_all: False + explicit case_ids: [...]. |
| Posting credentials as URL params | Secrets leak in proxy logs. | Always Basic auth header (Step 1). |
| No retry on 5xx | TestRail Cloud has occasional 502s; one transient failure loses the whole run. | Retry with exponential backoff on 5xx; cap at 3 attempts. |
| Closing every run, including PR runs | Closed runs can't accept reruns; a PR retest after fixing flake fails to update. | Close only main runs (Step 6); PR runs stay open. |
| Storing case IDs in test code AND in TestRail | Two sources of truth; renames drift. | TestRail is canonical; test code references via ID only (Step 2 Pattern A). |