packages/testing/playwright/docs/ORCHESTRATION.md
Capability-aware test distribution across CI shards.
| Step | What Happens |
|---|---|
| 1. Discovery | pnpm janitor discover (AST-based, detects test.fixme()/test.skip() automatically) |
| 2. Metrics | Get avgDuration per spec from Currents (last 30 days) |
| 3. Default | Missing specs get 60s default (accounts for container startup) |
| 4. Group | Group specs by @capability:xxx tag for worker reuse |
| 5. Effective Duration | Calculate actual time accounting for container reuse within groups |
| 6. Split | If a group exceeds 5 min, split into sub-groups |
| 7. Bin Pack | Greedy assign groups + standard specs to lightest shard |
Tests requiring containers (proxy, email, etc.) include ~20s startup overhead. When grouped on the same shard, only the first test pays this cost - the rest reuse the worker.
Example: 15 proxy tests across 8 shards = 8 container starts (160s). Grouped on 2 shards = 2 starts (40s). Saves 120s.
Metrics auto-correct over time. As grouped tests run, they report actual execution time (not startup overhead), so future distributions become more accurate.
// String capability - maps to predefined config
test.use({ capability: 'proxy' });
// Custom config - full control over container settings
test.use({
capability: {
proxyServerEnabled: true,
env: { MY_VAR: 'value' },
},
});
test('My feature @capability:proxy', async ({ page }) => {
// This test will be grouped with other proxy tests
});
// Or at describe level:
test.describe('Feature @capability:email', () => {
// All tests inherit the tag
});
| Capability | Tag | Containers |
|---|---|---|
'proxy' | @capability:proxy | Proxy server |
'email' | @capability:email | Mailpit |
'source-control' | @capability:source-control | Git server |
'task-runner' | @capability:task-runner | Task runner |
'oidc' | @capability:oidc | OIDC provider |
'observability' | @capability:observability | VictoriaLogs + VictoriaMetrics + Vector |
Capabilities (@capability:X) are add-on features you can combine with any infrastructure:
test.use({ capability: 'proxy' }) to configure the workerModes (@mode:X) define the infrastructure configuration itself:
@mode:postgres - n8n with PostgreSQL database (vs default sqlite)@mode:queue - n8n with EXECUTIONS_MODE=queue (workers via Bull, rarely used as tag)@mode:multi-main - n8n HA setup with leader election (implies queue mode)Most e2e tests run against ALL modes via projects (sqlite:e2e, postgres:e2e, etc).
Use @mode:X only for tests that ONLY work with a specific infrastructure.
// Capability - add-on feature
test.use({ capability: 'proxy' });
test('API mocking @capability:proxy', ...);
// Mode - infrastructure requirement (no test.use needed, project handles it)
test('Postgres-specific test @mode:postgres', ...);
// Combined - capability ON a specific mode
test.use({ capability: 'observability' });
test('Multi-main logs @capability:observability @mode:multi-main', ...);
Both @capability:X and @mode:X tests are skipped in local mode (they require containers).
Use test.fixme() to mark tests that need fixing. The janitor's discover command detects test.fixme() and test.skip() calls via AST analysis and automatically excludes them from CI distribution.
// Individual test
test.fixme('broken test', async ({ n8n }) => {
// Excluded from CI distribution automatically
});
// Entire describe block
test.describe('Feature', () => {
test.fixme(); // Marks all tests in this block
test('test 1', async ({ n8n }) => { ... });
test('test 2', async ({ n8n }) => { ... });
});
CURRENTS_API_KEY=<key> node packages/testing/playwright/scripts/fetch-currents-metrics.mjs --project=<id>
This fetches the last 30 days of test durations from Currents, aggregates by spec, and writes to .github/test-metrics/playwright.json.
When to refresh:
janitor orchestrate (generic) distribute-tests.mjs (n8n CI adapter)
┌──────────────────────────┐ ┌──────────────────────────┐
│ AST discovery │ │ Calls janitor orchestrate│
│ Metrics loading │ JSON │ Maps capabilities → │
│ Capability grouping │ ──────→ │ Docker images │
│ Group splitting │ │ Adds container overhead │
│ Greedy bin-packing │ │ Outputs GH Actions matrix│
└──────────────────────────┘ └──────────────────────────┘
The janitor handles generic orchestration (works for any Playwright project).
distribute-tests.mjs is n8n's CI adapter that maps capabilities to Docker images.
| Script | Purpose |
|---|---|
scripts/distribute-tests.mjs | CI adapter — calls janitor, maps images, outputs matrix |
scripts/fetch-currents-metrics.mjs | Fetches metrics from Currents API |
# Janitor orchestration (generic output)
pnpm janitor orchestrate --shards=14
# CI adapter (n8n-specific output with Docker images)
node scripts/distribute-tests.mjs --matrix 14 --orchestrate
# Get specs for shard 0
node scripts/distribute-tests.mjs 14 0
| Problem | Solution |
|---|---|
| Specs not running | Check path matches janitor test patterns in janitor.config.mjs |
| Unbalanced shards | Refresh metrics - durations may have drifted |
| Worker not reused | Use string capabilities like 'proxy', not inline objects |