Eliminate explicit waits in UI automation by detecting true UI stability
Project description
Waitless
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)returnsFalse- 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:
- Checks
__waitless__.isAlive()before each wait - Detects URL changes via
driver.current_url - 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
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
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
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
10afba1d76e07ada2bc3adda6fae3f292e69f330d99760eb650938deb09c9f96
|
|
| MD5 |
e5718f7680b7db90ed2e2b4901aafc7c
|
|
| BLAKE2b-256 |
5d5c831cdf3f164c37abd700e7930609e03c6dce279ae0cb97ca2a4d24cd36f2
|
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
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
4c13fc646e815cce5b1ec18a88cde439a21b4c425976d45df3bd003c83cdea30
|
|
| MD5 |
a440728999ac4ba6099d246211a414fa
|
|
| BLAKE2b-256 |
486362bf5678656fd46f85279a7f5b8f07ce49f6b59b0059237b564fa01e1e88
|