Skip to main content

Library with protocol support for different 433MHz switches and weather stations for the RFControl Arduino library.

Project description

Python library rfcontrolpy

Python GitHub Release Licence Maintainer
GitHub Sponsors PayPal BuyMeCoffee Patreon

Introduction

rfcontrolpy is a Python library and port of the node.js rfcontroljs module for parsing and constructing 433mhz On-Off Keying (OOK) signals for various devices, switches and weather stations.

It works together with the RFControl Arduino library for receiving the signals.

The Python port now contains a working controller and a dozen of protocols. Not all protocols are ported yet due to low demand or lack of hardware.

You can find a list of all supported protocols here.

The Processing Pipeline

Receiving

The arduino is connected via serial bus to the processing computer (for example a raspberry pi) and waits for rf signal.

Mostly all 433mhzw OOK signals from devices are send multiple times directly in row and have a longer footer pulse in between. They differ by the pulse lengths used to encode the data and footer and the pulse count.

RFControl running on the Arduino detects the start of a signal by its longer footer pulse and verifies it one time by comparing it with the next signal. It is unaware of the specific protocol, it just uses the stated fact above. Also we are not interested in it if the pulse was a high or low pulse (presence or absence of a carrier wave), because the information is decoded in the pulse lengths.

We will call the received sequence of pulse lengths now timings sequence. For example a timing sequence in microseconds could look like this:

288  972  292  968  292  972  292  968  292  972  920  344  288  976  920  348  
284  976  288  976  284  976  288  976  288  976  916  348  284  980  916  348  
284  976  920  348  284  976  920  348  284  980  280  980  284  980  916  348  
284  9808

You can clearly see the two different pulse lengths (around 304 and 959 microseconds) for the data encoding and the longer footer pulse (9808 microseconds).

All observed protocols have less than 8 different pulse length and all pulse length do differ by at least a factor of 2. This makes a further compression and simplification possible: We map each pulse length to a number from 0 to 7 (a bucket) and calculate for a better accuracy the average of all timings mapped to each of the bucket. The result is something like that:

buckets: 304, 959, 9808
pulses: 01010101011001100101010101100110011001100101011002

To make the representation unique, we choose the buckets in ascending order (respectively we are sorting it after receiving from the Arduino).

We call the sorted buckets pulse lengths, the compressed timings pulse sequence and the length of the pulse sequence (inclusive footer) pulse count.

Protocol Matching

We detect possible protocols by two criteria. The pulse length must match with a small tolerance and the pulse count must match.

Protocol Parsing

If a protocol matches, its parse function is called with the pulse sequence. Most protocols are parsed almost the same way. First the pulse sequence must be converted to a binary representation.

In almost all cases there exist a mapping from pulse sequences to a binary 0 and 1. In this example the pulse sequence 0110 represents a binary 0 and 0101 maps to a binary 1:

pulses2binary_mapping = {
  ["0110": "0"], # Binary 0
  ["0101": "1"], # Binary 1 
  ["02": ""]     # Footer
}
binary = helpers.pulses2binary(pulses, pulses2binary_mapping)

The binary representation now looks like this:

110011000010

As last step the protocol dependent information must be extracted from the binary representation:

decoded = {
  "id": int(binary[:6], 2),
  "unit": int(binary[6:11], 2),
  "state": binary[12] == "1"
}

Details

RFControl is more sensitive than needed for most protocols. So we get sometimes, depending of the accuracy of the sender/remote, different bucket counts.

This is by design, to catch up further protocols that maybe need a higher sensitivity. The specific protocol has not to deal with this issue, because rfcontrolpy auto merges similar buckets before calling the decodePulses function of each protocol.

The algorithm is the following:

  1. Record the (maybe to many) buckets and compressed pulses with RFControl (Arduino / c++)
  2. Sort the buckets in rfcontrolpy prepare_compressed_pulses
  3. Try to find a matching protocol in rfcontrolpy decode_pulses
  4. If we have more than 3 buckets and two of the buckets are similar (b1*2 < b2) we merge them to just one bucket by averaging and adapting the pulses in rfcontrolpy fix_pulses
  5. Go to step 3

Contribution and appreciation

You can contribute to this library, or show your appreciation, in the following ways.

Add a protocol for you device

If the protocol for your device is not yet supported you are encouraged to add support for your protocol to this library.

Preparation

  1. Fork the rfcontrolpy repository and clone your fork into a local directory.
  2. unittest is used for automating tests.
  3. You should be able to run the tests with python3 -m unittest discover.
  4. Running python3 -m build let it compile all files.

Protocol development

  1. Create a new protocol file in rfcontrol/protocols/, you can use one of the other protocol files as a guideline.
  2. Add a test case in tests/protocols with the data from the Arduino.
  3. Adapt the protocol file, so that the test get passed.

Star this library

Help other users find this library by starring this GitHub page. Click ⭐ Star on the top right of the GitHub page.

Support my work

Do you enjoy using this Python library? Please consider supporting my work through one of the following platforms. Your contribution is greatly appreciated and keeps me motivated:

GitHub Sponsors PayPal BuyMeCoffee Patreon

Hire me

If you're in need of a freelance Python developer for your project please contact me, you can find my email address on my GitHub profile.

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

rfcontrolpy-0.0.10.tar.gz (37.8 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

rfcontrolpy-0.0.10-py3-none-any.whl (43.7 kB view details)

Uploaded Python 3

File details

Details for the file rfcontrolpy-0.0.10.tar.gz.

File metadata

  • Download URL: rfcontrolpy-0.0.10.tar.gz
  • Upload date:
  • Size: 37.8 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for rfcontrolpy-0.0.10.tar.gz
Algorithm Hash digest
SHA256 ce6b0afda0433df64588933a6178a578249c7ce0e71db9afedc99ef0716a70d1
MD5 a259a1199886ceb12ae8c0afd87e58c7
BLAKE2b-256 564f5b1e8e5cd03cbcf4c9c44ac8499ad7c41799e4fdc36ff0246eceacdf9c9c

See more details on using hashes here.

Provenance

The following attestation bundles were made for rfcontrolpy-0.0.10.tar.gz:

Publisher: pypi-publish.yml on rrooggiieerr/rfcontrolpy

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file rfcontrolpy-0.0.10-py3-none-any.whl.

File metadata

  • Download URL: rfcontrolpy-0.0.10-py3-none-any.whl
  • Upload date:
  • Size: 43.7 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for rfcontrolpy-0.0.10-py3-none-any.whl
Algorithm Hash digest
SHA256 91942dad45ca38e3e79cd78cc75dd39c8a5e61d50956479e86b61a072ddfe62b
MD5 4663e637bf45fe5be6ed640dc06fae18
BLAKE2b-256 dbf888e8d75bbec3a7981f3bf8bad05557ce5d00cc42bc622ef509ae8712e958

See more details on using hashes here.

Provenance

The following attestation bundles were made for rfcontrolpy-0.0.10-py3-none-any.whl:

Publisher: pypi-publish.yml on rrooggiieerr/rfcontrolpy

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page