Skip to main content

Eliminate explicit waits in UI automation by detecting true UI stability

Project description

Waitless

CI

Zero-wait UI automation stabilization for Selenium

Eliminate explicit waits and sleeps by automatically detecting true UI stability.

Installation

pip install waitless

Quick Start

from selenium import webdriver
from selenium.webdriver.common.by import By
from waitless import stabilize

# Create driver as usual
driver = webdriver.Chrome()

# Enable automatic stabilization - ONE LINE
driver = stabilize(driver)

# All interactions now auto-wait for stability
driver.get("https://example.com")
driver.find_element(By.ID, "login-button").click()  # ← Auto-waits!
driver.find_element(By.ID, "username").send_keys("user")  # ← Auto-waits!

Why Waitless?

The Problem

Automation tests fail because interactions happen while the UI is still changing:

  • DOM mutations from React/Vue/Angular updates
  • In-flight AJAX requests
  • CSS animations and transitions
  • Layout shifts from lazy-loaded content

Traditional Solutions (and why they fail)

Approach Problem
time.sleep(2) Too slow, still fails sometimes
WebDriverWait Only checks one element, misses page-wide state
Retries Masks the real problem, adds flakiness

The Waitless Solution

Waitless monitors the entire page for stability signals:

  • ✅ DOM mutation activity (MutationObserver, including Shadow DOM)
  • ✅ Pending network requests (XHR/fetch interception)
  • ✅ CSS animations and transitions
  • ✅ Layout stability (element movement)
  • ✅ WebSocket/SSE activity (opt-in)
  • ✅ Framework hooks (React/Angular/Vue, opt-in)
  • ✅ iframe monitoring (opt-in)

When you interact, waitless ensures the page is truly ready.

Configuration

from waitless import stabilize, StabilizationConfig

config = StabilizationConfig(
    timeout=10,                    # Max wait time (seconds)
    mutation_rate_threshold=50,    # mutations/sec considered stable (allows animations)
    network_idle_threshold=2,      # Max pending requests (allows background traffic)
    animation_detection=True,      # Track CSS animations (non-blocking in normal mode)
    strictness='normal',           # 'strict' | 'normal' | 'relaxed'
    debug_mode=True                # Enable logging
)

driver = stabilize(driver, config=config)

Strictness Levels

Level What It Waits For
strict DOM + Network + Animations + Layout
normal DOM + Network (default)
relaxed DOM only

Factory Methods

# For strict testing
config = StabilizationConfig.strict()

# For apps with background traffic
config = StabilizationConfig.relaxed()

# For CI environments
config = StabilizationConfig.ci()

Manual Stabilization

If you don't want to wrap the driver:

from waitless import wait_for_stability

wait_for_stability(driver)
driver.find_element(By.ID, "button").click()

Disabling Stabilization

from waitless import unstabilize

driver = unstabilize(driver)  # Back to original behavior

Diagnostics

When tests fail, get detailed analysis:

from waitless import get_diagnostics, StabilizationTimeout
from waitless.diagnostics import print_report

try:
    driver.find_element(By.ID, "slow-button").click()
except StabilizationTimeout as e:
    diagnostics = get_diagnostics(driver)
    print_report(engine)  # Print detailed report

CLI Doctor Command

python -m waitless doctor --file diagnostics.json

Sample output:

+--------------------------------------------------------------------+
|                     WAITLESS STABILITY REPORT                      |
+--------------------------------------------------------------------+
| BLOCKING FACTORS:                                                  |
|   [!] NETWORK: 2 request(s) still pending                          |
|   -> GET /api/users                                                |
|   [!] ANIMATIONS: 1 active animation(s)                            |
+--------------------------------------------------------------------+
| SUGGESTIONS:                                                       |
|   1. Set network_idle_threshold=2 for background traffic           |
|   2. Use animation_detection=False for infinite spinners           |
+--------------------------------------------------------------------+

Important Notes

Network Threshold Warning

The default network_idle_threshold=2 allows some background traffic.

Many apps have background traffic that never stops:

  • Analytics calls
  • Long polling
  • Feature flags
  • WebSocket heartbeats

If tests timeout frequently, try:

config = StabilizationConfig(network_idle_threshold=2)

Wrapped Elements

The stabilized driver returns wrapped elements that auto-wait. They behave like WebElements but:

  • isinstance(element, WebElement) returns False
  • Use .unwrap() to get the original element if needed
element = driver.find_element(By.ID, "button")
original = element.unwrap()  # Gets the real WebElement

v1.0.0 New Features

  • WebSocket/SSE Awareness - Track WebSocket and Server-Sent Events activity
  • Framework Adapters - React, Angular, Vue hooks for framework-specific settling
  • iframe Support - Monitor same-origin iframes
  • Performance Benchmarks - Built-in benchmark suite
# Enable new v1.0 features
config = StabilizationConfig(
    track_websocket=True,         # WebSocket monitoring
    track_sse=True,               # SSE monitoring
    framework_hooks=['react'],    # React adapter
    track_iframes=True,           # iframe monitoring
)

Performance

Metric Typical Value
Instrumentation injection ~5-10ms
Per-poll overhead ~1-2ms
Poll interval (default) 50ms
Typical stabilization 50-200ms after activity

SPA Navigation Handling

Waitless automatically re-injects instrumentation after SPA route changes:

  1. Checks __waitless__.isAlive() before each wait
  2. Detects URL changes via driver.current_url
  3. Re-injects if instrumentation is missing

This works transparently with React Router, Vue Router, Angular Router, etc.

Current Limitations

  • Selenium only - Playwright support planned
  • Sync only - No async/await support yet
  • No Service Workers - SW network requests not intercepted

See CHANGELOG.md for version history.

API Reference

Functions

Function Description
stabilize(driver, config=None) Enable auto-stabilization
unstabilize(driver) Disable and return original driver
wait_for_stability(driver, timeout=None) Manual one-time wait
get_diagnostics(driver) Get diagnostic data

Classes

Class Description
StabilizationConfig Configuration options
StabilizedWebDriver Wrapped driver with auto-wait
StabilizedWebElement Wrapped element with auto-wait
StabilizationTimeout Exception when UI doesn't stabilize

License

MIT

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

waitless-1.0.0.tar.gz (30.8 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

waitless-1.0.0-py3-none-any.whl (33.5 kB view details)

Uploaded Python 3

File details

Details for the file waitless-1.0.0.tar.gz.

File metadata

  • Download URL: waitless-1.0.0.tar.gz
  • Upload date:
  • Size: 30.8 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.14.0

File hashes

Hashes for waitless-1.0.0.tar.gz
Algorithm Hash digest
SHA256 10afba1d76e07ada2bc3adda6fae3f292e69f330d99760eb650938deb09c9f96
MD5 e5718f7680b7db90ed2e2b4901aafc7c
BLAKE2b-256 5d5c831cdf3f164c37abd700e7930609e03c6dce279ae0cb97ca2a4d24cd36f2

See more details on using hashes here.

File details

Details for the file waitless-1.0.0-py3-none-any.whl.

File metadata

  • Download URL: waitless-1.0.0-py3-none-any.whl
  • Upload date:
  • Size: 33.5 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.14.0

File hashes

Hashes for waitless-1.0.0-py3-none-any.whl
Algorithm Hash digest
SHA256 4c13fc646e815cce5b1ec18a88cde439a21b4c425976d45df3bd003c83cdea30
MD5 a440728999ac4ba6099d246211a414fa
BLAKE2b-256 486362bf5678656fd46f85279a7f5b8f07ce49f6b59b0059237b564fa01e1e88

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page