MTR receiver readings to MQTT server
Project description
mtr2mqtt
A CLI tool for reading Nokeval MTR wireless receivers and forwarding readings as JSON objects to MQTT topics. This allows integration with home automation systems, data logging platforms, and visualization tools.
Overview
mtr2mqtt connects to Nokeval MTR series wireless receivers (such as RTR970, FTR980, etc.) via serial connection, reads the measurements from wireless transmitters, and publishes the data to MQTT topics in a structured JSON format.
It can also publish Home Assistant MQTT Discovery messages so that each detected transmitter appears automatically as a Home Assistant device with reading, battery, and rsl sensor entities.
Installation
Using pip
The simplest way to install mtr2mqtt:
pip install mtr2mqtt
Basic Command Line Usage
Pass in serial port settings and MQTT server address. If not provided, serial port autodetection is used and localhost is used for MQTT.
Example with metadata file:
mtr2mqtt -f metadata.yml
Or define the serial port and mqtt host as parameters:
mtr2mqtt --serial-port /dev/ttyUSB12345 --mqtt-host 192.168.1.2
By default, runtime logs are emitted as one JSON object per line. When running in an interactive terminal, JSON keys and values are syntax colored, with standard fields such as timestamp, level, logger, and message highlighted consistently. When output is redirected or piped, ANSI color is suppressed so the log stream remains valid JSON.
For a human-focused live view, use --output table. In this mode, the console shows the latest reading for each sensor in a continuously refreshed table instead of printing each measurement as a log line. The table starts with a stable set of core columns and automatically adds extra columns for additional measurement or metadata fields when they appear. The nested ha metadata block is excluded from the table. Table output requires an interactive terminal.
The table view also includes sensor availability status. It shows the latest textual status and numeric status code, and timeout-driven offline transitions are reflected even when no fresh readings arrive.
Enable Home Assistant discovery:
mtr2mqtt -f metadata.yml --ha-discovery
Use a custom discovery prefix or node id:
mtr2mqtt --ha-discovery --ha-discovery-prefix ha --ha-discovery-node-id mtr-bridge-1
Configure the offline timeout used by retained status topics and table status:
mtr2mqtt --offline-timeout 1800
Use the live table view:
mtr2mqtt -f metadata.yml --output table
Using Docker
You can use the pre-built Docker images from Docker Hub. Specify the latest or a specific version.
- Pull the latest Docker image:
docker pull tvallas/mtr2mqtt:latest
Or pull a specific version:
docker pull tvallas/mtr2mqtt:0.5.4
Run the Docker container:
docker run --rm -it -v /dev/ttyUSB12345:/dev/ttyUSB12345 tvallas/mtr2mqtt:latest --serial-port /dev/ttyUSB12345 --mqtt-host 192.168.1.2
Note: On macOS, Docker cannot access the serial port because it runs in a virtual machine. You will need to run the tool natively on macOS or use a Linux-based system for Docker.
Using Docker Compose
You can also use Docker Compose to run the application along with an MQTT broker.
- Create a
docker-compose.ymlfile:
version: '3.8'
services:
mosquitto:
image: eclipse-mosquitto:latest
container_name: mosquitto
restart: always
ports:
- "1883:1883"
command: mosquitto -c /mosquitto-no-auth.conf
mtr2mqtt:
image: tvallas/mtr2mqtt:latest
container_name: mtr2mqtt
restart: always
depends_on:
- mosquitto
volumes:
- type: bind
source: ./metadata.yml
target: /tmp/metadata.yml
environment:
MTR2MQTT_MQTT_HOST: mosquitto
MTR2MQTT_METADATA_FILE: /tmp/metadata.yml
MTR2MQTT_QUIET: "true"
devices:
- "/dev/ttyUSB0:/dev/ttyUSB0"
- Run Docker Compose:
docker-compose up -d
Metadata
The metadata.yml file is used to provide additional configuration for the mtr2mqtt tool. This file allows you to define the structure and details of the data being read from the Nokeval MTR wireless receivers.
Structure of metadata.yml
The metadata.yml file should be structured as follows:
- id: 12345
location: "Living room"
description: "Ambient air temperature"
unit: "°C"
quantity: "Temperature"
ha:
name: "Ambient"
device_class: "temperature"
state_class: "measurement"
suggested_display_precision: 1
icon: "mdi:thermometer"
- id: 54321
location: "Living room"
unit: "%"
description: "Ambient air humidity"
quantity: "Humidity"
Each sensor entry should include:
id: A unique identifier for the sensor.description: A descriptive label for the sensor.unit: The unit of measurement for the sensor's readings.
Note: Other metadata fields can be freely added and those are added to the JSON object.
The optional ha: block is used only for Home Assistant discovery customization. It can override the main reading entity metadata with fields such as name, device_class, state_class, suggested_display_precision, and icon. Existing metadata files continue to work unchanged.
Using the metadata file
To use the metadata file with mtr2mqtt, pass the file path as an argument:
mtr2mqtt -f metadata.yml
MQTT Topics and message format
The messages are published to the MQTT broker in a structured JSON format. The structure of the topics and message format is as follows:
Topic Structure
The MQTT topics are structured based on the receiver serial number and sensor ID. For example:
measurements/<receiver_serial_number>/<sensor_id>
Where <receiver_serial_number> is the serial number of the receiver and <sensor_id> is the unique identifier for the sensor.
This measurement topic structure remains unchanged even when Home Assistant discovery is enabled.
Status Topics
mtr2mqtt also publishes retained status payloads for observed receivers and sensors:
status/<receiver_serial_number>
status/<receiver_serial_number>/<sensor_id>
Receiver status example:
{
"entity_type": "receiver",
"receiver": "receiver-a",
"status": "online",
"status_code": 1,
"last_received_at": "2026-04-26T10:15:32Z",
"last_publish_at": "2026-04-26T10:15:33Z",
"error_count": 0
}
Sensor status example:
{
"entity_type": "sensor",
"receiver": "receiver-a",
"sensor": "sensor-123",
"status": "offline",
"status_code": 0,
"last_received_at": "2026-04-26T08:42:10Z",
"last_publish_at": "2026-04-26T08:42:10Z",
"error_count": 2
}
Status values use this exact numeric mapping:
offline=0online=1
An entity is online after it has been observed at least once and the last valid traffic is within the configured offline timeout. It becomes offline after the timeout passes. The default offline timeout is 30 minutes (1800 seconds) and can be changed with --offline-timeout or MTR2MQTT_OFFLINE_TIMEOUT.
Status topics are retained so downstream tooling can evaluate current receiver and sensor availability immediately after subscribing. Numeric status_code values are intended for simple alert conditions in tools such as tvallas/mqtt-alerts.
Never-seen sensors are not fabricated at startup and do not publish offline status. Measurement topics and payloads are intentionally left untouched when a receiver or sensor goes offline: mtr2mqtt does not publish synthetic null, 0, "offline", or any other fake reading to measurements/....
Home Assistant MQTT Discovery
When --ha-discovery is enabled, mtr2mqtt publishes retained Home Assistant device discovery messages the first time it sees a real measurement for a transmitter. One Home Assistant device is created per physical transmitter, using the transmitter id as the Home Assistant device identifier and entity unique id base. The MQTT state topic itself remains unchanged and still includes the receiver serial number.
Discovery creates these entities for each transmitter:
reading: the primary sensor entitybattery: a diagnostic battery voltage sensorrsl: a diagnostic signal sensor
Discovery topics use Home Assistant device discovery:
<discovery_prefix>/device/<object_id>/config
<discovery_prefix>/device/<node_id>/<object_id>/config
For example:
homeassistant/device/15006/config
measurements/RTR970123/15006
The discovery payload points each entity to the existing measurement topic, so measurement payloads continue to be published as non-retained JSON messages.
The primary reading entity also exposes the measurement JSON as Home Assistant attributes using json_attributes_topic, which means metadata fields such as location, description, quantity, and zone are visible in Home Assistant automatically.
Home Assistant discovery configuration
CLI flags:
--ha-discovery: enable Home Assistant discovery--ha-discovery-prefix: set the discovery prefix, defaulthomeassistant--ha-discovery-retainand--no-ha-discovery-retain: control retained discovery messages, default retained--ha-discovery-node-id: add an optional node id segment to the discovery topic
Environment variables:
MTR2MQTT_HA_DISCOVERYMTR2MQTT_HA_DISCOVERY_PREFIXMTR2MQTT_HA_DISCOVERY_RETAINMTR2MQTT_HA_DISCOVERY_NODE_ID
The reading entity infers conservative Home Assistant metadata from the existing metadata file when possible. For example, quantity: Temperature or unit: °C maps to device_class: temperature, humidity-like metadata maps to device_class: humidity, and pressure-like metadata maps to device_class: pressure.
To align better with typical Home Assistant sensor setups, the primary reading entity also publishes:
expire_after: 900suggested_areafrom metadatalocation- a default icon inferred from the measurement type, unless overridden in metadata
ha:
Message Format
{
"battery": 2.8,
"type": "FT10",
"rsl": -69,
"id": "15006",
"reading": 21.2,
"timestamp": "2025-03-14 20:57:49.152063+00:00"
}
Fields Explanation
battery: The battery voltage of the sensor.type: The type of the sensor.rsl: The received signal level (RSL) of the sensor.id: The unique identifier for the sensor.reading: The current reading from the sensor.timestamp: The timestamp of the reading in ISO 8601 format.
Note: Other metadata fields can be freely added and those are added to the JSON object.
Preparing the Development Environment
- Install
uv. - Clone the repository:
git clone git@github.com:tvallas/mtr2mqtt
- Change into the repository directory:
cd mtr2mqtt
- Create or update the virtual environment and install runtime and development dependencies:
uv sync --group dev
- Run the CLI locally:
uv run python -m mtr2mqtt.cli
Running tests
uv run pytest -v
Linting
uv run pylint $(find mtr2mqtt -name "*.py" -type f)
Building distributions
uv build
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 mtr2mqtt-0.9.0.tar.gz.
File metadata
- Download URL: mtr2mqtt-0.9.0.tar.gz
- Upload date:
- Size: 39.8 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
198751336c9dccc52b4d7fc492637ab620bf28a7461767e17541f09ce67e1a5e
|
|
| MD5 |
65fd2065d9f193223525a7147527507b
|
|
| BLAKE2b-256 |
cdb7cfa4a0354e945faedff3ed487e82122ff040614eb87734fe986dcc89ae8b
|
Provenance
The following attestation bundles were made for mtr2mqtt-0.9.0.tar.gz:
Publisher:
semantic_release.yml on tvallas/mtr2mqtt
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
mtr2mqtt-0.9.0.tar.gz -
Subject digest:
198751336c9dccc52b4d7fc492637ab620bf28a7461767e17541f09ce67e1a5e - Sigstore transparency entry: 1389528649
- Sigstore integration time:
-
Permalink:
tvallas/mtr2mqtt@d41e910ab1898d5394411bae5ade34a0358c0bc4 -
Branch / Tag:
refs/heads/master - Owner: https://github.com/tvallas
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
semantic_release.yml@d41e910ab1898d5394411bae5ade34a0358c0bc4 -
Trigger Event:
push
-
Statement type:
File details
Details for the file mtr2mqtt-0.9.0-py3-none-any.whl.
File metadata
- Download URL: mtr2mqtt-0.9.0-py3-none-any.whl
- Upload date:
- Size: 26.3 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
296df1a8b920acea8d5c05965ed6fce65305e8054b4bb48e551587d3e5bc070a
|
|
| MD5 |
84fc7298205b5828f11a49273dffdc8d
|
|
| BLAKE2b-256 |
88f19fa69bed8e7fa55df68a5cf1775d7d88aa03d079cf982885bf9fafd58ece
|
Provenance
The following attestation bundles were made for mtr2mqtt-0.9.0-py3-none-any.whl:
Publisher:
semantic_release.yml on tvallas/mtr2mqtt
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
mtr2mqtt-0.9.0-py3-none-any.whl -
Subject digest:
296df1a8b920acea8d5c05965ed6fce65305e8054b4bb48e551587d3e5bc070a - Sigstore transparency entry: 1389528658
- Sigstore integration time:
-
Permalink:
tvallas/mtr2mqtt@d41e910ab1898d5394411bae5ade34a0358c0bc4 -
Branch / Tag:
refs/heads/master - Owner: https://github.com/tvallas
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
semantic_release.yml@d41e910ab1898d5394411bae5ade34a0358c0bc4 -
Trigger Event:
push
-
Statement type: