Python library for controlling Jebao aquarium pumps
Project description
python-jebao
Python library for controlling Jebao aquarium pumps over local network.
Supported Models
- MDP-20000: Variable-speed circulation pump (30-100% speed control)
- MD-4.4: 4-channel dosing pump (planned)
Features
- 🔍 UDP Discovery - Automatic device discovery with multi-subnet support
- 🔌 Local Control - Direct TCP control, no cloud dependency
- ⚡ Full Protocol - On/Off, speed control, feed mode, status monitoring
- 🏠 Home Assistant Ready - Designed for HA integration
- 🔒 Type Safe - Full type hints for better IDE support
Installation
pip install python-jebao
For development:
git clone https://github.com/jrigling/python-jebao.git
cd python-jebao
pip install -e .
Quick Start
Discovery
import asyncio
from jebao import discover_devices
async def main():
# Discover all Jebao devices on network
devices = await discover_devices()
for device in devices:
print(f"Found: {device.model} at {device.ip_address}")
print(f" Device ID: {device.device_id}")
print(f" MAC: {device.mac_address}")
asyncio.run(main())
Multi-Subnet Discovery
If you have multiple network interfaces (e.g., IoT VLAN, main network):
# Discover on all interfaces
devices = await discover_devices()
# Or specify interfaces explicitly
devices = await discover_devices(interfaces=['eth0', 'eth1'])
Basic Control
from jebao import MDP20000Device
async def main():
# Connect to device
async with MDP20000Device(host="10.20.20.13") as pump:
# Ensure in manual mode (exits Program mode if needed)
await pump.ensure_manual_mode()
# Turn on
await pump.turn_on()
# Set speed to 75%
await pump.set_speed(75)
# Check status
await pump.update()
print(f"State: {pump.state.name}")
print(f"Speed: {pump.speed}%")
print(f"Is on: {pump.is_on}")
# Turn off
await pump.turn_off()
asyncio.run(main())
Feed Mode
async def feed_mode_example():
async with MDP20000Device(host="10.20.20.13") as pump:
await pump.ensure_manual_mode()
await pump.turn_on()
await pump.set_speed(75)
# Start 2-minute feed (pump stops temporarily)
await pump.start_feed(minutes=2)
print("Feed started, pump will auto-resume in 2 minutes")
# Check status during feed
await pump.update()
print(f"In feed mode: {pump.is_feed_mode}")
# Or cancel early
# await pump.cancel_feed(resume_speed=75)
Long-Running Connection
async def monitor_pump():
pump = MDP20000Device(host="10.20.20.13")
try:
await pump.connect()
await pump.ensure_manual_mode()
# Control pump
await pump.turn_on()
await pump.set_speed(50)
# Monitor status periodically
while True:
await pump.update()
print(f"Status: {pump.state.name} @ {pump.speed}%")
await asyncio.sleep(30)
finally:
await pump.disconnect()
Device States
from jebao import DeviceState
# State values
DeviceState.OFF # 0x10 - Manual mode, stopped
DeviceState.ON # 0x11 - Manual mode, running
DeviceState.FEED # 0x15 - Feed mode (temporary pause)
DeviceState.PROGRAM # 0x19 - Program mode (scheduled)
# Check state
if pump.state == DeviceState.FEED:
print("Pump is in feed mode")
# Helper properties
pump.is_on # True if ON or FEED
pump.is_off # True if OFF
pump.is_feed_mode # True if FEED
pump.is_program_mode # True if PROGRAM
pump.is_manual_mode # True if ON or OFF
API Reference
MDP20000Device
Connection
async connect(timeout=5.0)- Connect and authenticateasync disconnect()- Disconnect from deviceis_connected- Check connection statusasync update()- Refresh device status
Control
async turn_on()- Turn pump onasync turn_off()- Turn pump offasync set_speed(percentage)- Set speed (30-100%)
Feed Mode
async set_feed_duration(minutes)- Configure feed timer (1-10 minutes)async start_feed(minutes=None)- Start feed modeasync cancel_feed(resume_speed=None)- Cancel feed and resume
Program Mode
async ensure_manual_mode()- Exit Program mode if active
Properties
state: DeviceState- Current statespeed: int- Current speed (30-100)is_on: bool- Pump is runningis_off: bool- Pump is stoppedis_feed_mode: bool- In feed modeis_program_mode: bool- In program modeis_manual_mode: bool- In manual mode
Discovery
from jebao import JebaoDiscovery, discover_devices
# Convenience function
devices = await discover_devices(timeout=2.0, interfaces=['eth0'])
# Or use class directly
discovery = JebaoDiscovery()
devices = await discovery.discover(timeout=2.0, interfaces=['eth0'])
# Filter by model
mdp20000_devices = [d for d in devices if d.is_mdp20000]
md44_devices = [d for d in devices if d.is_md44]
DiscoveredDevice
device.device_id # Device ID (e.g., "POAKSFXJNJ")
device.ip_address # IP address
device.mac_address # MAC address
device.product_key # Product key
device.model # Model name (e.g., "MDP-20000")
device.is_mdp20000 # True if MDP-20000
device.is_md44 # True if MD-4.4
Home Assistant Integration
This library is designed for use with Home Assistant. See the separate homeassistant-jebao repository for the ready-to-use HA integration.
Reliability Features
The library includes automatic retry logic for enhanced reliability:
- Commands automatically retry up to 3 times on transient failures
- Handles garbage byte accumulation from pump firmware
- Exponential backoff for retry delays
- Typical success rate >97% even with flaky network conditions
Multi-Subnet Considerations
When Home Assistant runs on a system with multiple network interfaces:
- Automatic Discovery: The library will scan all interfaces by default
- Interface Selection: You can specify which interfaces to scan
- Broadcast Support: Each interface gets its own broadcast address
- IoT VLANs: Works great with isolated IoT networks
Example: HA connected to both 192.168.1.0/24 (main) and 10.20.20.0/24 (IoT):
# Discovers on both networks automatically
devices = await discover_devices()
# Or be explicit
devices = await discover_devices(interfaces=['eth0', 'eth1'])
Error Handling
from jebao import (
JebaoError,
JebaoConnectionError,
JebaoAuthenticationError,
JebaoCommandError,
JebaoTimeoutError,
JebaoInvalidStateError,
)
try:
async with MDP20000Device(host="10.20.20.13") as pump:
await pump.turn_on()
except JebaoConnectionError:
print("Failed to connect - check IP and network")
except JebaoAuthenticationError:
print("Authentication failed")
except JebaoCommandError:
print("Command execution failed")
except JebaoTimeoutError:
print("Operation timed out")
except JebaoError as e:
print(f"General error: {e}")
Logging
The library uses Python's standard logging:
import logging
# Enable debug logging
logging.basicConfig(level=logging.DEBUG)
# Or just for jebao
logging.getLogger('jebao').setLevel(logging.DEBUG)
Requirements
- Python 3.9+
netifaces- For multi-interface network discovery
License
MIT License - See LICENSE file
Contributing
Contributions welcome! Please:
- Fork the repository
- Create a feature branch
- Add tests for new functionality
- Ensure all tests pass
- Submit a pull request
Credits
This library builds upon the excellent work from:
- jebao-dosing-pump-md-4.4 by @tancou - Original Node.js implementation for MD 4.4 pumps that provided the foundation for understanding the GizWits protocol
Protocol for MDP-20000 was reverse-engineered through packet capture analysis of the official Jebao mobile app.
Disclaimer
This is an unofficial library not affiliated with Jebao. Use at your own risk. Device warranty may be affected by third-party control software.
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 python_jebao-0.1.6.tar.gz.
File metadata
- Download URL: python_jebao-0.1.6.tar.gz
- Upload date:
- Size: 23.7 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.11.15
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
44f70567ede740c078f62ad971eed07e987ae7025da9e990908e880e7f5dc3dd
|
|
| MD5 |
c3ca1a05522ecd6af97a2ef453b3dbf0
|
|
| BLAKE2b-256 |
e535e8face3e4a658d8a21e7a9904e48626fcd93bd50b1315f6870b6fe03c3c5
|
File details
Details for the file python_jebao-0.1.6-py3-none-any.whl.
File metadata
- Download URL: python_jebao-0.1.6-py3-none-any.whl
- Upload date:
- Size: 22.9 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.11.15
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
8e323996f3cebb9e48ba2d216cb60a0b6d224d02b628c8e267348aa9289db81d
|
|
| MD5 |
7eb4eb0b9160477ef797d2551715b2eb
|
|
| BLAKE2b-256 |
3694faf98d2f6bff51922ced1e6f95917061b6b4e4e3ec43d21082e24747d58e
|