.wrangler/memory/knowledge-base/reference-prompts/skills/condition-based-waiting/SKILL.md
Use when tests have race conditions, timing dependencies, or inconsistent pass/fail behavior - replaces arbitrary timeouts with condition polling; prefer event-based waiting when available (WebSockets, EventEmitters, Observables)
npx skillsauth add bacchus-labs/wrangler condition-based-waitingInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
MANDATORY: When using this skill, announce it at the start with:
🔧 Using Skill: condition-based-waiting | [brief purpose based on context]
Example:
🔧 Using Skill: condition-based-waiting | [Provide context-specific example of what you're doing]
This creates an audit trail showing which skills were applied during the session.
Flaky tests often guess at timing with arbitrary delays. This creates race conditions where tests pass on fast machines but fail under load or in CI.
Core principle: Wait for the actual condition you care about, not a guess about how long it takes.
digraph when_to_use {
"Test uses setTimeout/sleep?" [shape=diamond];
"Testing timing behavior?" [shape=diamond];
"Document WHY timeout needed" [shape=box];
"Use condition-based waiting" [shape=box];
"Test uses setTimeout/sleep?" -> "Testing timing behavior?" [label="yes"];
"Testing timing behavior?" -> "Document WHY timeout needed" [label="yes"];
"Testing timing behavior?" -> "Use condition-based waiting" [label="no"];
}
Use when:
setTimeout, sleep, time.sleep())Don't use when:
Polling is NOT appropriate when:
Use event listeners instead of polling for:
Example:
// ❌ BAD: Polling for WebSocket message
let lastMessage;
ws.on('message', msg => lastMessage = msg);
await waitForCondition(() => lastMessage?.type === 'ready');
// ✅ GOOD: Event-based waiting
await new Promise((resolve, reject) => {
const timeout = setTimeout(() => reject(new Error('Timeout')), 5000);
ws.once('message', (msg) => {
if (msg.type === 'ready') {
clearTimeout(timeout);
resolve(msg);
}
});
});
Use await instead of polling for:
Example:
// ❌ BAD: Polling to check if promise resolved
let result;
apiCall().then(r => result = r);
await waitForCondition(() => result !== undefined);
// ✅ GOOD: Just await the promise
const result = await apiCall();
Use reactive patterns instead of polling for:
Example:
// ❌ BAD: Polling Redux store
await waitForCondition(() => store.getState().user.loggedIn);
// ✅ GOOD: Subscribe to store changes
await new Promise(resolve => {
const unsubscribe = store.subscribe(() => {
if (store.getState().user.loggedIn) {
unsubscribe();
resolve();
}
});
});
Use polling when:
// ❌ BEFORE: Guessing at timing
await new Promise(r => setTimeout(r, 50));
const result = getResult();
expect(result).toBeDefined();
// ✅ AFTER: Waiting for condition
await waitFor(() => getResult() !== undefined);
const result = getResult();
expect(result).toBeDefined();
| Scenario | Pattern |
|----------|---------|
| Wait for event | waitFor(() => events.find(e => e.type === 'DONE')) |
| Wait for state | waitFor(() => machine.state === 'ready') |
| Wait for count | waitFor(() => items.length >= 5) |
| Wait for file | waitFor(() => fs.existsSync(path)) |
| Complex condition | waitFor(() => obj.ready && obj.value > 10) |
Generic polling function:
async function waitFor<T>(
condition: () => T | undefined | null | false,
description: string,
timeoutMs = 5000
): Promise<T> {
const startTime = Date.now();
while (true) {
const result = condition();
if (result) return result;
if (Date.now() - startTime > timeoutMs) {
throw new Error(`Timeout waiting for ${description} after ${timeoutMs}ms`);
}
await new Promise(r => setTimeout(r, 10)); // Poll every 10ms
}
}
See @example.ts for complete implementation with domain-specific helpers (waitForEvent, waitForEventCount, waitForEventMatch) from actual debugging session.
Default: 10-50ms (recommended for most test cases)
Shorter intervals (1-10ms):
Longer intervals (100-1000ms):
Exponential backoff:
Example: Exponential backoff
async function waitWithBackoff<T>(
condition: () => T | undefined | null | false,
description: string,
maxTimeoutMs = 30000
): Promise<T> {
const startTime = Date.now();
let interval = 100; // Start at 100ms
while (true) {
const result = condition();
if (result) return result;
if (Date.now() - startTime > maxTimeoutMs) {
throw new Error(`Timeout waiting for ${description} after ${maxTimeoutMs}ms`);
}
await new Promise(r => setTimeout(r, interval));
interval = Math.min(interval * 2, 5000); // Double, max 5s
}
}
❌ Polling too fast: setTimeout(check, 1) - wastes CPU
✅ Fix: Poll every 10ms
❌ No timeout: Loop forever if condition never met ✅ Fix: Always include timeout with clear error
❌ Stale data: Cache state before loop ✅ Fix: Call getter inside loop for fresh data
// Tool ticks every 100ms - need 2 ticks to verify partial output
await waitForEvent(manager, 'TOOL_STARTED'); // First: wait for condition
await new Promise(r => setTimeout(r, 200)); // Then: wait for timed behavior
// 200ms = 2 ticks at 100ms intervals - documented and justified
Requirements:
From debugging session (2025-10-03):
tools
Use when creating technical specifications for features, systems, or architectural designs. Creates comprehensive specification documents using the Wrangler MCP issue management system with proper structure and completeness checks.
testing
Creates and refines agent skills using TDD methodology with pressure testing and rationalization detection. Use when creating new skills, editing existing skills, testing skills with pressure scenarios, or verifying skills work before deployment.
tools
Use when design is complete and you need detailed implementation tasks - creates tracked MCP issues with exact file paths, complete code examples, and verification steps. Optional reference plan file for architecture overview.
development
Validates governance file completeness, format compliance, and metric accuracy. Use when auditing governance health, after bulk changes, or ensuring documentation integrity.