home / skills / bobmatnyc / claude-mpm-skills / condition-based-waiting

condition-based-waiting skill

/universal/testing/condition-based-waiting

This skill replaces arbitrary timeouts with condition polling to make asynchronous tests reliable across environments.

This is most likely a fork of the condition-based-waiting skill from yousufjoyian
npx playbooks add skill bobmatnyc/claude-mpm-skills --skill condition-based-waiting

Review the files below or copy the command above to add this skill to your agents.

Files (3)
SKILL.md
3.9 KB
---
name: Condition-Based Waiting
description: Replace arbitrary timeouts with condition polling for reliable async tests
when_to_use: when tests have race conditions, timing dependencies, or inconsistent pass/fail behavior
version: 1.1.0
languages: all
progressive_disclosure:
  entry_point:
    summary: "Replace arbitrary timeouts with condition polling for reliable async tests"
    when_to_use: "Tests with setTimeout/sleep, flaky tests, or timing-dependent async operations"
    quick_start: |
      1. Identify arbitrary delays in tests (setTimeout, sleep, time.sleep())
      2. Replace with condition-based waiting (waitFor pattern)
      3. Use domain-specific helpers for common scenarios
      See @example.ts for complete working implementation
    core_pattern: |
      // ❌ Guessing at timing
      await new Promise(r => setTimeout(r, 50));

      // ✅ Waiting for condition
      await waitFor(() => getResult() !== undefined);
  references:
    - path: references/patterns-and-implementation.md
      purpose: Detailed waiting patterns, implementation guide, and common mistakes
      when_to_read: When implementing waitFor or debugging timing issues
---

# Condition-Based Waiting

## Overview

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.

## When to Use

```dot
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:**
- Tests have arbitrary delays (`setTimeout`, `sleep`, `time.sleep()`)
- Tests are flaky (pass sometimes, fail under load)
- Tests timeout when run in parallel
- Waiting for async operations to complete

**Don't use when:**
- Testing actual timing behavior (debounce, throttle intervals)
- Always document WHY if using arbitrary timeout

## Core Pattern

```typescript
// ❌ 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();
```

## Quick Patterns

| 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)` |

## Implementation

Generic polling function:
```typescript
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`).

For detailed patterns, implementation guide, and common mistakes, see @references/patterns-and-implementation.md

## Real-World Impact

From debugging session (2025-10-03):
- Fixed 15 flaky tests across 3 files
- Pass rate: 60% → 100%
- Execution time: 40% faster
- No more race conditions

Overview

This skill replaces arbitrary time-based delays in tests with condition-based polling to eliminate race conditions and flakiness. It promotes waiting for the actual condition you care about (state, event, file, count) instead of guessing durations. The result is more reliable, faster, and deterministic asynchronous tests.

How this skill works

The skill provides a small polling utility that repeatedly evaluates a user-supplied condition until it returns a truthy value or a timeout elapses. It accepts a descriptive label and a configurable timeout, throws a clear error on timeout, and returns the condition result when met. Domain-specific helpers (events, counts, files, states) wrap the generic waitFor for convenient, readable assertions.

When to use it

  • Tests currently use arbitrary sleeps or setTimeout calls
  • Tests are flaky under CI or when run in parallel
  • Waiting for asynchronous operations, external IO, or state transitions
  • You want faster test suites by removing overlong safety delays

Best practices

  • Prefer waiting for a concrete observable condition instead of fixed delays
  • Use descriptive timeout messages to aid debugging when waits fail
  • Keep poll interval small (e.g., 10ms) but avoid busy-waiting in very tight loops
  • Set reasonable per-check timeouts and document any intentional timing tests
  • Create domain-specific wrappers (waitForEvent, waitForFile) for clarity

Example use cases

  • Replace time.sleep/setTimeout in unit and integration tests with waitFor(() => state === 'ready')
  • Wait for an event to appear: waitFor(() => events.find(e => e.type === 'DONE'))
  • Assert file creation: waitFor(() => fs.existsSync(path)) before reading the file
  • Ensure collections reach expected size: waitFor(() => items.length >= 5)
  • Test complex readiness: waitFor(() => obj.ready && obj.value > 10)

FAQ

What timeout should I use?

Start with a modest default (e.g., 5s) and increase only for slow external dependencies. Make timeouts explicit so failures are actionable.

Does polling slow tests down?

No—condition-based waiting is usually faster than fixed sleeps because it proceeds as soon as the condition is met; choose a reasonable poll interval to balance responsiveness and CPU usage.