This is a reimplementation of the Tinkerforge Python bindings (original Python bindings) using Python 3 asyncio. The original bindings used threads to manage the blocking operations. A much cleaner implementation is possible using the await
syntax from asyncio.
Note: This API implementation is not an official Tinkerforge implementation. I am in no way affiliated with the Tinkerforge GmbH. Use at your own risk. If you find any bugs, please report them.
The library is fully type-hinted.
Brick | Supported | Tested | Comments |
---|---|---|---|
Master | ✔️ | ✔️ |
Bricklet | Supported | Tested |
---|---|---|
Ambient Light 2.0 | ✔️ | ✔️ |
Ambient Light 3.0 | ✔️ | ✔️ |
Analog In | ✔️ | ✔️ |
Barometer | ✔️ | ✔️ |
Barometer 2.0 | ✔️ | ✔️ |
Humidity | ✔️ | ✔️ |
Humidity 2.0 | ✔️ | ✔️ |
Industrial Dual Analog In 2.0 | ✔️ | ✔️ |
Industrial PTC | ✔️ | ✔️ |
IO-4 2.0 | ✔️ | ✔️ |
IO-16 | ✔️ | ✔️ |
Isolator | ✔️ | ✔️ |
Moisture | ✔️ | ✔️ |
Motion Detector 2.0 | ✔️ | ✔️ |
PTC | ✔️ | ✔️ |
PTC 2.0 | ✔️ | ✔️ |
RS232 2.0 | ✔️ | ✔️ |
Segment Display 4x7 | ✔️ | ✔️ |
Segment Display 4x7 2.0 | ✔️ | ✔️ |
Temperature | ✔️ | ✔️ |
Temperature 2.0 | ✔️ | ✔️ |
Thermocouple 2.0 | ✔️ | ✔️ |
The documentation is currently work in progress. The full documentation will be moved to https://patrickbaus.github.io/tinkerforge_async/. Below you can find the current state of the documentation. I use the Numpydoc style for documentation and Sphinx for compiling it.
To install the library in a virtual environment (always use venvs with every project):
python3 -m venv env # virtual environment, optional
source env/bin/activate # only if the virtual environment is used
pip install tinkerforge-async
Some design choices of the original Tinkerforge API are overly complex. I therefore replaced them with a simpler and more intuitive approach. A list of things that were changed can be found below:
-
Only Python 3 is supported (3.7+)
-
Replaced threads with an async event loop
-
Completely rewritten how responses from bricks/bricklets work. All setters now have a
response_expected
parameter, which is set toTrue
by default. If there is an error when calling the function, it will then raise an exception - either anAttributeError
if the function is unknown, or aValueError
if one or more parameters are invalid.Old style:
bricklet = BrickletHumidity(UID, ipcon) bricklet.set_response_expected( BrickletHumidity.FUNCTION_SET_HUMIDITY_CALLBACK_PERIOD, True ) bricklet.set_humidity_callback_period(1000)
New style:
bricklet = BrickletHumidity(UID, ipcon) await bricklet.set_humidity_callback_period( 1000, response_expected=True ) # Raises an exception if unsuccessful
-
Replaced all constants with Enums and enforced their use using assertions. This will allow beginners to spot their mistakes earlier and make the code more readable, including any debug output statements.
Old style:
class BrickletHumidity(Device): FUNCTION_GET_HUMIDITY = 1 FUNCTION_GET_ANALOG_VALUE = 2 FUNCTION_SET_HUMIDITY_CALLBACK_PERIOD = 3 FUNCTION_GET_HUMIDITY_CALLBACK_PERIOD = 4 FUNCTION_SET_ANALOG_VALUE_CALLBACK_PERIOD = 5 FUNCTION_GET_ANALOG_VALUE_CALLBACK_PERIOD = 6 FUNCTION_SET_HUMIDITY_CALLBACK_THRESHOLD = 7 FUNCTION_GET_HUMIDITY_CALLBACK_THRESHOLD = 8 FUNCTION_SET_ANALOG_VALUE_CALLBACK_THRESHOLD = 9 FUNCTION_GET_ANALOG_VALUE_CALLBACK_THRESHOLD = 10 FUNCTION_SET_DEBOUNCE_PERIOD = 11 FUNCTION_GET_DEBOUNCE_PERIOD = 12 FUNCTION_GET_IDENTITY = 255
New style:
class BrickletHumidity(Device): @unique class FunctionID(Enum): GET_HUMIDITY = 1 GET_ANALOG_VALUE = 2 SET_HUMIDITY_CALLBACK_PERIOD = 3 GET_HUMIDITY_CALLBACK_PERIOD = 4 SET_ANALOG_VALUE_CALLBACK_PERIOD = 5 GET_ANALOG_VALUE_CALLBACK_PERIOD = 6 SET_HUMIDITY_CALLBACK_THRESHOLD = 7 GET_HUMIDITY_CALLBACK_THRESHOLD = 8 SET_ANALOG_VALUE_CALLBACK_THRESHOLD = 9 GET_ANALOG_VALUE_CALLBACK_THRESHOLD = 10 SET_DEBOUNCE_PERIOD = 11 GET_DEBOUNCE_PERIOD = 12
-
Moved from base58 encoded uids to integers.
-
Moved from callbacks to queues in order to keep users out of the callback hell. It makes the code style more readable when using the
await
syntax anyway. -
Payloads will now be decoded by the
Device
object and no longer by theip_connection
. This makes the code a lot more readable. To do so, the payload and decoded header will be handed to the device. It will then decode it, if possible, and pass it on to the queue. -
If physical quantities are measured we will now return standard SI units, not some unexpected stuff like centi °C (Temperature Bricklet). To preserve the precision the Decimal package is used. The only exception to this rule is the use of °C for temperature. This is for convenience.
-
All callbacks now contain a timestamp (Unix timestamp) and the device object.
Example:
Event(timestamp=1658756708.6839857, sender=Temperature Bricklet 2.0 with uid 161085 connected at IPConnectionAsync(192.168.1.164:4223), sid=0, function_id=CallbackID.TEMPERATURE, payload=305.46)
-
Added the concept of secondary ids (
sid
). By default, the secondary id is0
. If there is more than one sensor on the bricklet, they will have asid
value of 1,2, etc. This is especially useful for sensors like the Industrial Dual Analog In Bricklet 2.0, which returns its two channels via the same callback. -
New functions:
BrickMaster.set_wpa_enterprise_username(username)
: Set the WPA enterprise username without callingBrickMaster.set_wifi_certificate()
. Takes astring
instead of an array ofint
.BrickMaster.set_wpa_enterprise_password(password)
: Set the WPA enterprise password without callingBrickMaster.set_wifi_certificate()
. Takes astring
instead of an array ofint
.BrickMaster.get_wpa_enterprise_username()
: Get the WPA enterprise password without callingBrickMaster.get_wifi_certificate()
. Also returns astring
instead of an array ofint
.BrickMaster.get_wpa_enterprise_password()
: Get the WPA enterprise password without callingBrickMaster.get_wifi_certificate()
. Also returns astring
instead of an array ofint
. -
IPConnection.authenticate(secret)
: removed. This can now be done through connect()IPConnection.set_timeout/IPConnection.get_timeout
: Replaced by a propertyIPConnection.register_callback(callback_id, function)
: Replaced byregister_event_queue()
IPConnection.connect(host, port=4223, authentication_secret='')
: Ifauthentication_secret
is not empty, try to authenticate.
-
BrickletIO4V2.set_pwm_configuration()
will now take the frequency in units of Hz and the duty cycle is normalized to 1, so it will take a float from [0...1].BrickletIO4V2.get_pwm_configuration()
will return the frequency in units of HZ and the duty cycle is normalized to 1.
-
BrickMaster.set_wifi_configuration()
/BrickMaster.get_wifi_configuration()
will take/return all ips in natural orderBrickMaster.set_ethernet_configuration()
/BrickMaster.get_ethernet_configuration()
will take/return all ips in natural orderBrickMaster.write_wifi2_serial_port()
will only accept abytestring
and no length argument. The length will be automatically determined from the string.BrickMaster.set_wifi2_status_led(enabled)
added. This allows setting the status led by value instead of callingenable_wifi2_status_led
/disable_wifi2_status_led
-
BrickletPtc()
takes an additional parameter to define the type of sensor. The options areBrickletPtc.SensorType.PT_100
andBrickletPtc.SensorType.PT_1000
. This only determines the resistance returned by the bricklet. The default isBrickletPtc.SensorType.PT_100
.BrickletPtc.sensor_type
getter and setter to change the type of sensor used.
-
BrickletPtcV2()
takes an additional parameter to define the type of sensor. The options areBrickletPtc.SensorType.PT_100
andBrickletPtc.SensorType.PT_1000
. This only determines the resistance returned by the bricklet. The default isBrickletPtc.SensorType.PT_100
.BrickletPtcV2.sensor_type
getter and setter to change the type of sensor used.
-
BrickletThermocoupleV2()
takes an additional parameter to define the type of sensor. The options are of typeBrickletThermocoupleV2.SensorType
. The default isBrickletPtc.SensorType.TYPE_K
.BrickletThermocoupleV2.sensor_type
getter and setter to change the type of sensor used.
-
BrickletSegmentDisplay4x7V2.set_segments()
takes alist
/tuple
of 4int
instead of digit0, digit1, digit2, digit3. This is the same API as the older Segment Display 4x7 Bricklet.
There are currently no packages available at the PyPi repository. To install the module, clone the repository and run:
python3 -m venv env # virtual environment, optional
source env/bin/activate # only if the virtual environment is used
python3 setup.py install
I use SemVer for versioning. For the versions available, see the tags on this repository.
- Patrick Baus - Initial work - PatrickBaus
This project is licensed under the GPL v3 license - see the LICENSE file for details