Kermi x-center to MQTT bridge for heat pumps with Home Assistant auto-discovery
Project description
kermi2mqtt
Modbus-to-MQTT bridge for Kermi heat pumps with Home Assistant auto-discovery.
Features
- Full monitoring - All heat pump sensors published via MQTT
- Bidirectional control - Change settings (mode, preset, DHW temp) via MQTT
- Home Assistant integration - Zero-config auto-discovery with climate & water_heater entities
- Unified device - All entities grouped under single "Kermi X-Center" device in HA
- Safety-first - Only exposes user-safe controls with validation
- Kubernetes ready - Helm chart included for easy deployment
- Async/efficient - Low resource usage (<50MB RAM)
Quick Start
Prerequisites
- Kermi heat pump with Modbus TCP/RTU interface
- MQTT broker (Mosquitto, Home Assistant, etc.)
- Python 3.12+ or Docker
Installation
Option 1: Docker (Recommended)
# Create config file
cp config.example.yaml config.yaml
# Edit config.yaml with your Modbus and MQTT settings
# Build and run
docker build -t kermi2mqtt .
docker run -d --name kermi2mqtt \
-v $(pwd)/config.yaml:/config/config.yaml:ro \
kermi2mqtt
Or with Docker Compose:
docker-compose up -d
Option 2: Kubernetes (Helm)
# Add your values
cat > my-values.yaml << EOF
config:
modbus:
host: "xcenter.local"
port: 502
mqtt:
host: "mqtt.local"
port: 8883
tlsEnabled: true
mqttAuth:
username: "kermi"
password: "your-password"
EOF
# Install
helm install kermi2mqtt ./charts/kermi2mqtt -f my-values.yaml
See charts/kermi2mqtt/values.yaml for all options.
Option 3: Python Package
# Install from PyPI
pip install kermi2mqtt
# Or install from source
git clone https://github.com/jr42/kermi2mqtt
cd kermi2mqtt
pip install -e .
Configuration
- Copy
config.example.yamltoconfig.yaml - Configure your Modbus connection:
modbus: host: 192.168.1.100 # Your heat pump IP port: 502 mode: tcp
- Configure your MQTT broker:
mqtt: host: localhost port: 1883
- Set device ID (or leave blank for auto-detection):
integration: device_id: my_heat_pump poll_interval: 30.0
Running
Docker
docker-compose up -d
docker-compose logs -f kermi2mqtt
Python
python -m kermi2mqtt --config config.yaml
Systemd Service
# Copy service file
sudo cp kermi2mqtt.service /etc/systemd/system/
sudo systemctl daemon-reload
# Enable and start
sudo systemctl enable kermi2mqtt
sudo systemctl start kermi2mqtt
# Check status
sudo systemctl status kermi2mqtt
MQTT Topics
State Topics (Published by kermi2mqtt)
kermi/{device_id}/sensors/outdoor_temp → Outdoor temperature
kermi/{device_id}/sensors/supply_temp → Supply temperature
kermi/{device_id}/sensors/cop → Coefficient of Performance
kermi/{device_id}/sensors/power_total → Thermal power output
kermi/{device_id}/sensors/power_electrical → Electrical power consumption
kermi/{device_id}/heating/actual → Current heating temperature
kermi/{device_id}/heating/setpoint → Heating setpoint
kermi/{device_id}/heating/circuit_status → Heating circuit status
kermi/{device_id}/water_heater/actual → DHW actual temperature
kermi/{device_id}/water_heater/setpoint → DHW setpoint
kermi/{device_id}/water_heater/single_charge → One-time heating active
kermi/{device_id}/availability → online/offline
Command Topics (Subscribe with MQTT client)
# Monitor all topics
mosquitto_sub -h localhost -t 'kermi/#' -v
# Monitor specific sensor
mosquitto_sub -h localhost -t 'kermi/my_heat_pump/sensors/outdoor_temp'
Home Assistant Integration
Entities automatically appear in Home Assistant with appropriate types:
- Climate entities for heating/cooling control
- Water Heater entities for domestic hot water
- Sensor entities for temperature, power, COP readings
- Switch entities for on/off controls
- Binary Sensor entities for status indicators
All entities are grouped under a single Device in Home Assistant.
Architecture
┌─────────────────┐ ┌──────────────┐ ┌─────────────────┐
│ Kermi Heat │ Modbus │ kermi2mqtt │ MQTT │ Home Assistant │
│ Pump (x-center)│◄───────►│ Bridge │◄───────►│ / MQTT Clients │
└─────────────────┘ └──────────────┘ └─────────────────┘
│
┌───────▼────────┐
│ py-kermi-xcenter│
│ (Modbus lib) │
└─────────────────┘
Development
Setup
# Clone repository
git clone https://github.com/jr42/kermi2mqtt
cd kermi2mqtt
# Create virtual environment
python -m venv .venv
source .venv/bin/activate # or `.venv\Scripts\activate` on Windows
# Install with dev dependencies
pip install -e ".[dev]"
Testing
# Run tests
pytest
# Run tests with coverage
pytest --cov=src/kermi2mqtt --cov-report=html
# Run linters
ruff check src/ tests/
mypy src/kermi2mqtt/
black --check src/ tests/
Project Structure
kermi2mqtt/
├── src/kermi2mqtt/ # Main package
│ ├── __main__.py # Entry point
│ ├── config.py # Configuration loading
│ ├── modbus_client.py # Modbus wrapper
│ ├── mqtt_client.py # MQTT wrapper
│ ├── bridge.py # Main bridge logic
│ ├── safety.py # Safety validation
│ ├── ha_discovery.py # HA discovery payloads
│ ├── mappings.py # Attribute definitions
│ └── models/ # Data models
├── charts/kermi2mqtt/ # Helm chart for Kubernetes
├── tests/ # Test suite
├── Dockerfile # Container image
└── config.example.yaml # Example configuration
Safety
This integration only exposes user-safe controls equivalent to the heat pump's physical interface:
✅ Safe to modify:
- Temperature setpoints (40-60°C for DHW)
- Operating modes (heating/cooling)
- One-time water heating
- Heating schedules
❌ Not exposed (hardware safety):
- Compressor controls
- Refrigerant valve positions
- System pressure
- Low-level firmware parameters
See specs/001-modbus-mqtt/safety.md for detailed safety documentation.
Troubleshooting
Connection Issues
# Test Modbus connection
python -c "from kermi_xcenter import KermiModbusClient, HeatPump; import asyncio; asyncio.run(test())"
# Check MQTT broker
mosquitto_sub -h localhost -t '#' -v
Logs
# Docker logs
docker-compose logs -f kermi2mqtt
# Systemd logs
journalctl -u kermi2mqtt -f
# Increase log verbosity in config.yaml
logging:
level: DEBUG
Common Issues
-
"No response from heat pump"
- Check network connectivity:
ping <heat_pump_ip> - Verify Modbus port 502 is accessible
- Check firewall rules
- Check network connectivity:
-
"MQTT connection failed"
- Verify broker is running:
systemctl status mosquitto - Test broker:
mosquitto_sub -h localhost -t test - Check credentials in config.yaml
- Verify broker is running:
-
"Entities not appearing in Home Assistant"
- Check MQTT discovery prefix matches HA config (default:
homeassistant) - Verify kermi2mqtt is publishing:
mosquitto_sub -t 'homeassistant/#' - Restart Home Assistant after first discovery
- Check MQTT discovery prefix matches HA config (default:
Contributing
Contributions welcome! Please:
- Fork the repository
- Create a feature branch
- Make your changes with tests
- Run linters and tests
- Submit a pull request
License
Apache License 2.0 - see LICENSE for details.
Credits
- Built with py-kermi-xcenter by @jr42
- Uses aiomqtt for async MQTT
- Designed for Home Assistant
Disclaimer
This software is not affiliated with or endorsed by Kermi. Use at your own risk. Always ensure changes are safe for your equipment.
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 kermi2mqtt-0.1.2.tar.gz.
File metadata
- Download URL: kermi2mqtt-0.1.2.tar.gz
- Upload date:
- Size: 52.4 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
2fe05c804e9e96ff5fd5fefa007c0eed1212fbdbf9c86c9d662556f7d319d89d
|
|
| MD5 |
0654b0021fc6a469d6bfcef323d1c71a
|
|
| BLAKE2b-256 |
a18085658653b1cb43e3a91ef868963f2955f90e29f37d80276e915c09432232
|
Provenance
The following attestation bundles were made for kermi2mqtt-0.1.2.tar.gz:
Publisher:
release.yml on jr42/kermi2mqtt
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
kermi2mqtt-0.1.2.tar.gz -
Subject digest:
2fe05c804e9e96ff5fd5fefa007c0eed1212fbdbf9c86c9d662556f7d319d89d - Sigstore transparency entry: 805157800
- Sigstore integration time:
-
Permalink:
jr42/kermi2mqtt@29d5c42ef4e6c96e028a93136a71b72ba953574c -
Branch / Tag:
refs/tags/v0.1.2 - Owner: https://github.com/jr42
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@29d5c42ef4e6c96e028a93136a71b72ba953574c -
Trigger Event:
push
-
Statement type:
File details
Details for the file kermi2mqtt-0.1.2-py3-none-any.whl.
File metadata
- Download URL: kermi2mqtt-0.1.2-py3-none-any.whl
- Upload date:
- Size: 50.8 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
c90348a7b0934ffe8dd0701d7b3184837d65e05bfcdbce13f06cacdfc1763989
|
|
| MD5 |
202c44e4aef4ce3498443dfc45b87340
|
|
| BLAKE2b-256 |
0ef116d27100e82b5ccf3062e28e17fb25c658832cdd7e7f406838cd80ee3023
|
Provenance
The following attestation bundles were made for kermi2mqtt-0.1.2-py3-none-any.whl:
Publisher:
release.yml on jr42/kermi2mqtt
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
kermi2mqtt-0.1.2-py3-none-any.whl -
Subject digest:
c90348a7b0934ffe8dd0701d7b3184837d65e05bfcdbce13f06cacdfc1763989 - Sigstore transparency entry: 805157803
- Sigstore integration time:
-
Permalink:
jr42/kermi2mqtt@29d5c42ef4e6c96e028a93136a71b72ba953574c -
Branch / Tag:
refs/tags/v0.1.2 - Owner: https://github.com/jr42
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@29d5c42ef4e6c96e028a93136a71b72ba953574c -
Trigger Event:
push
-
Statement type: