A python client for uHoo APIs
Project description
uhooapi - Python Client for uHoo API
A modern, asynchronous Python client for the uHoo air quality API. This library provides an intuitive, type-safe interface to access your uHoo device data, manage devices, and retrieve real-time air quality metrics with automatic token management and comprehensive error handling.
โจ Features
- ๐ Async/Await Native: Built on
aiohttpfor high-performance, non-blocking API calls - ๐ Automatic Token Management: Handles authentication, token refresh, and retry logic automatically
- ๐ Full Type Annotations: Complete type hints for better IDE support and reliability
- ๐ฏ Production Ready: 100% test coverage with comprehensive unit and integration tests
- ๐ Smart Error Handling: Custom exceptions with automatic retry for 401/403 errors
- ๐ Complete Sensor Coverage: Access to all uHoo metrics (temperature, humidity, COโ, PM2.5, virus index, etc.)
- โก Efficient Data Processing: Automatic averaging and rounding of sensor readings
๐ฆ Installation
From PyPI (Recommended)
pip install uhooapi
Development Installation
# Clone the repository
git clone https://github.com/yourusername/uhooapi.git
cd uhooapi
# 2. Create virtual environment
python -m venv venv
source venv/bin/activate # On Windows: venv\Scripts\activate
# 3. Install with dev dependencies
pip install -e ".[dev]"
# 4. Install pre-commit hooks
pre-commit install
# 5. Run tests to verify
pytest
๐ Quick Start
import asyncio
import aiohttp
from uhooapi import Client
async def main():
# Create a session and client
async with aiohttp.ClientSession() as session:
client = Client(
api_key="your_uhoo_api_key_here", # Get from uHoo dashboard
websession=session,
debug=True # Enable debug logging
)
# Authenticate and get token
await client.login()
# Discover and set up your devices
await client.setup_devices()
# Get all devices
devices = client.get_devices()
print(f"๐ฑ Found {len(devices)} uHoo device(s)")
# Get latest data for the first device
if devices:
first_device_serial = list(devices.keys())[0]
await client.get_latest_data(first_device_serial)
# Access the device data
device = devices[first_device_serial]
print(f"\n๐ Device: {device.device_name}")
print(f"๐ Location: {device.room_name}")
print(f"๐ก๏ธ Temperature: {device.temperature}ยฐC")
print(f"๐ง Humidity: {device.humidity}%")
print(f"โ๏ธ COโ: {device.co2} ppm")
print(f"๐จ PM2.5: {device.pm25} ยตg/mยณ")
print(f"๐ฆ Virus Risk Index: {device.virus_index}")
# Run the async function
asyncio.run(main())
๐ Usage Examples
๐ Continuous Monitoring
import asyncio
from datetime import datetime
from uhooapi import Client
async def monitor_air_quality(api_key: str, update_interval: int = 300):
"""Continuously monitor air quality and log changes."""
async with aiohttp.ClientSession() as session:
client = Client(api_key=api_key, websession=session)
await client.login()
await client.setup_devices()
print("Starting air quality monitoring...")
while True:
for serial_number, device in client.get_devices().items():
await client.get_latest_data(serial_number)
print(f"\n[{datetime.now().strftime('%Y-%m-%d %H:%M:%S')}]")
print(f"Device: {device.device_name} ({device.room_name})")
print("-" * 40)
print(f"Temperature: {device.temperature:5.1f}ยฐC")
print(f"Humidity: {device.humidity:5.1f}%")
print(f"COโ: {device.co2:5.0f} ppm")
print(f"PM2.5: {device.pm25:5.1f} ยตg/mยณ")
print(f"Virus Index: {device.virus_index:5.1f}")
# Add alerts for poor air quality
if device.co2 > 1000:
print("โ ๏ธ Warning: High COโ levels detected!")
if device.pm25 > 35:
print("โ ๏ธ Warning: Elevated PM2.5 levels!")
await asyncio.sleep(update_interval)
๐ก๏ธ Robust Error Handling
from uhooapi.errors import UnauthorizedError, ForbiddenError, RequestError
async def fetch_with_retry(client: Client, serial_number: str, max_retries: int = 3):
"""Fetch data with exponential backoff retry logic."""
for attempt in range(max_retries):
try:
await client.get_latest_data(serial_number)
return True
except UnauthorizedError as e:
print(f"โ Authentication failed: {e}")
# Re-authenticate and retry
await client.login()
continue
except ForbiddenError as e:
print(f"๐ Permission denied: {e}")
return False
except RequestError as e:
if attempt < max_retries - 1:
wait_time = 2 ** attempt # Exponential backoff
print(f"๐ Request failed (attempt {attempt + 1}/{max_retries}), "
f"retrying in {wait_time}s...")
await asyncio.sleep(wait_time)
else:
print(f"๐ฅ Max retries exceeded: {e}")
return False
return False
๐ Multi-Device Data Aggregation
async def get_environmental_summary(api_key: str):
"""Get summary statistics across all devices."""
async with aiohttp.ClientSession() as session:
client = Client(api_key=api_key, websession=session)
await client.login()
await client.setup_devices()
devices = client.get_devices()
# Fetch data for all devices concurrently
tasks = [
client.get_latest_data(serial)
for serial in devices.keys()
]
await asyncio.gather(*tasks)
# Calculate averages
temps = [d.temperature for d in devices.values()]
humidities = [d.humidity for d in devices.values()]
co2_levels = [d.co2 for d in devices.values()]
print("\n๐ Environmental Summary")
print("=" * 40)
print(f"Total Devices: {len(devices)}")
print(f"Avg Temperature: {sum(temps)/len(temps):.1f}ยฐC")
print(f"Avg Humidity: {sum(humidities)/len(humidities):.1f}%")
print(f"Avg COโ: {sum(co2_levels)/len(co2_levels):.0f} ppm")
# Identify problem areas
worst_co2 = max(devices.values(), key=lambda d: d.co2)
if worst_co2.co2 > 800:
print(f"\nโ ๏ธ Highest COโ in: {worst_co2.room_name} ({worst_co2.co2} ppm)")
๐๏ธ Architecture
Client Class (uhooapi.client.Client)
Client(
api_key: str, # Your uHoo API key
websession: aiohttp.ClientSession, # aiohttp session
**kwargs # Optional: debug=True for debug logging
)
Device Class (uhooapi.device.Device)
device.device_name # "Living Room"
device.serial_number # "UHOO12345"
device.mac_address # "AA:BB:CC:DD:EE:FF"
device.room_name # "Living Room"
device.floor_number # 1
device.temperature # 22.5ยฐC
device.humidity # 45.0%
device.co2 # 800 ppm
device.pm25 # 12.3 ยตg/mยณ
device.virus_index # 2.5
device.mold_index # 1.8
device.tvoc # 150.0 ppb
# ... and 15+ more sensors
๐จ Error Handling
The library defines custom exceptions for different error scenarios:
from uhooapi.errors import (
UhooError, # Base exception
RequestError, # General API failures
UnauthorizedError, # 401 - Invalid/expired token
ForbiddenError # 403 - Insufficient permissions
)
try:
await client.get_latest_data("UHOO12345")
except UnauthorizedError:
# Automatic retry with fresh login is built-in
print("Token expired, re-authenticating...")
except ForbiddenError as e:
print(f"Access denied: {e.message}")
except RequestError as e:
print(f"API request failed (status: {e.status}): {e}")
except KeyError:
print("Device not found. Did you call setup_devices()?")
except Exception as e:
print(f"Unexpected error: {e}")
๐งช Testing
The project includes a comprehensive test suite:
# Run all tests
pytest
# Run with coverage report
pytest --cov=src/uhooapi --cov-report=html
# Run specific test categories
pytest tests/unit/ -v # Unit tests
pytest tests/integration/ -v # Integration tests
# Run tests in parallel
pytest -n auto
๐ง Building and Publishing
# Update version in pyproject.toml first!
# Build distribution packages
python -m build
# Check build quality
twine check dist/*
# Upload to TestPyPI (for testing)
python -m twine upload --repository-url https://test.pypi.org/legacy/ dist/*
# Upload to PyPI
python -m twine upload dist/*
๐ Project Structure
uhooapi/
โโโ src/uhooapi/ # Source code
โ โโโ __init__.py # Package exports
โ โโโ client.py # Main Client class
โ โโโ api.py # Low-level API wrapper
โ โโโ device.py # Device data model (22+ sensors)
โ โโโ errors.py # Custom exceptions
โ โโโ const.py # Constants and defaults
โ โโโ endpoints.py # API endpoint configurations
โ โโโ util.py # Utility functions
โโโ tests/ # Test suite
โ โโโ unit/ # Unit tests (mocked)
โ โ โโโ test_client.py # Client tests
โ โ โโโ test_api.py # API tests
โ โ โโโ test_device.py # Device model tests
โ โโโ integration/ # Integration tests
โ โโโ conftest.py # Test fixtures
โโโ pyproject.toml # Package configuration
โโโ README.md # This file
โโโ pre-commit-config.yaml # Code quality hooks
โโโ .github/workflows/ # CI/CD pipelines (optional)
๐ค Contributing
We welcome contributions! Here's how to help:
-
Fork the repository
-
Clone your fork: git clone https://github.com/yourusername/uhooapi.git
-
Create a branch: git checkout -b feature/amazing-feature
-
Make your changes and add tests
-
Run tests: pytest && pre-commit run --all-files
-
Commit: git commit -m 'Add amazing feature'
-
Push: git push origin feature/amazing-feature
-
Open a Pull Request
Project details
Release history Release notifications | RSS feed
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 uhooapi-1.2.3.tar.gz.
File metadata
- Download URL: uhooapi-1.2.3.tar.gz
- Upload date:
- Size: 14.1 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.14.0
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
f4c936bcda7ed663c2b6b0bfcbf30b7d4cb3aa505cb29c70e7214f8195f66102
|
|
| MD5 |
b9c364d0bfa975961e4d970679973477
|
|
| BLAKE2b-256 |
1fc15cf5c4ccfbf32177b4992f0c75c1c4abead9a48dce2bed84d3708c7de830
|
File details
Details for the file uhooapi-1.2.3-py3-none-any.whl.
File metadata
- Download URL: uhooapi-1.2.3-py3-none-any.whl
- Upload date:
- Size: 11.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 |
984a15e88c6cc3ca7d74a86203dc62bb771a36462773039505ada7dce6a5dfac
|
|
| MD5 |
a649d525cce7f425a325254644436477
|
|
| BLAKE2b-256 |
8702b9727f88b207b3d5951f6a8892ffaf8b405e81480ed2ccf0306e454f3d7c
|