Skip to content
Snippets Groups Projects
Commit ebdde7b4 authored by Anton Sarukhanov's avatar Anton Sarukhanov
Browse files

Add locking for thread safety.

bluepy is not inherently thread-safe. Concurrent attempts to read/write
will produce unexpected results. With this update, the TurnTouch library
can safely be used in multithreaded code.

See https://github.com/IanHarvey/bluepy/issues/126#issuecomment-213978728
parent 78213d89
Branches master
Tags 0.4.3
No related merge requests found
...@@ -9,7 +9,7 @@ def read(filename): ...@@ -9,7 +9,7 @@ def read(filename):
setup( setup(
name='TurnTouch', name='TurnTouch',
version='0.4.2', version='0.4.3',
url='https://github.com/antsar/python-turntouch', url='https://github.com/antsar/python-turntouch',
author='Anton Sarukhanov', author='Anton Sarukhanov',
author_email='code@ant.sr', author_email='code@ant.sr',
......
"""Classes related to the Turn Touch remote.""" """Classes related to the Turn Touch remote."""
from concurrent.futures import ThreadPoolExecutor from concurrent.futures import ThreadPoolExecutor
from threading import Lock
import time import time
import logging import logging
from functools import partial from functools import partial
...@@ -262,6 +263,7 @@ class TurnTouch(btle.Peripheral): ...@@ -262,6 +263,7 @@ class TurnTouch(btle.Peripheral):
self.debounce = debounce self.debounce = debounce
self._listening = False self._listening = False
self._combo_action = set() self._combo_action = set()
self._lock = Lock()
if listen: if listen:
self.listen_forever() self.listen_forever()
...@@ -305,11 +307,11 @@ class TurnTouch(btle.Peripheral): ...@@ -305,11 +307,11 @@ class TurnTouch(btle.Peripheral):
else: else:
return self._read_now(uuid) return self._read_now(uuid)
def _read_now(self, uuid) -> bytes: def _read_now(self, uuid) -> bytes:
"""Read some characteristic from the device.""" """Read some characteristic from the device."""
try: try:
read_bytes = self.getCharacteristics(uuid=uuid)[0].read() with self._lock:
read_bytes = self.getCharacteristics(uuid=uuid)[0].read()
except btle.BTLEException: except btle.BTLEException:
raise TurnTouchException("Failed to read device {address} " raise TurnTouchException("Failed to read device {address} "
"characteristic {uuid}" "characteristic {uuid}"
...@@ -320,13 +322,14 @@ class TurnTouch(btle.Peripheral): ...@@ -320,13 +322,14 @@ class TurnTouch(btle.Peripheral):
def _write(self, uuid, value_bytes): def _write(self, uuid, value_bytes):
"""Write some characteristic to the device.""" """Write some characteristic to the device."""
characteristic = self.getCharacteristics(uuid=uuid)[0] with self._lock:
try: characteristic = self.getCharacteristics(uuid=uuid)[0]
characteristic.write(value_bytes, withResponse=True) try:
except btle.BTLEException: characteristic.write(value_bytes, withResponse=True)
raise TurnTouchException("Failed to write device {address} " except btle.BTLEException:
"characteristic {uuid}" raise TurnTouchException("Failed to write device {address} "
.format(address=self.addr, uuid=uuid)) "characteristic {uuid}"
.format(address=self.addr, uuid=uuid))
logger.debug("Wrote device {address} characteristic {uuid}: '{value}'" logger.debug("Wrote device {address} characteristic {uuid}: '{value}'"
.format(address=self.addr, uuid=uuid, value=value_bytes)) .format(address=self.addr, uuid=uuid, value=value_bytes))
...@@ -345,10 +348,12 @@ class TurnTouch(btle.Peripheral): ...@@ -345,10 +348,12 @@ class TurnTouch(btle.Peripheral):
self.executor = ThreadPoolExecutor(5) self.executor = ThreadPoolExecutor(5)
try: try:
if only_one: if only_one:
self.waitForNotifications(0) with self._lock:
self.waitForNotifications(0)
else: else:
while True: while True:
self.waitForNotifications(self.LISTEN_PERIOD) with self._lock:
self.waitForNotifications(self.LISTEN_PERIOD)
if self._pending_read: if self._pending_read:
while self._read_value: while self._read_value:
# wait for previously read value to be consumed # wait for previously read value to be consumed
...@@ -375,16 +380,18 @@ class TurnTouch(btle.Peripheral): ...@@ -375,16 +380,18 @@ class TurnTouch(btle.Peripheral):
"""Tell the remote to start sending notifications for a particular """Tell the remote to start sending notifications for a particular
characteristic (uuid).""" characteristic (uuid)."""
try: try:
notification_handle = self.getCharacteristics( with self._lock:
uuid=uuid)[0].getHandle() notification_handle = self.getCharacteristics(
uuid=uuid)[0].getHandle()
notification_enable_handle = notification_handle + 1 notification_enable_handle = notification_handle + 1
logger.debug("{action} notifications for device {address}, " logger.debug("{action} notifications for device {address}, "
"characteristic {uuid}..." "characteristic {uuid}..."
.format(action="Enabling" if enabled else "Disabling", .format(action="Enabling" if enabled else "Disabling",
address=self.addr, uuid=uuid)) address=self.addr, uuid=uuid))
self.writeCharacteristic(notification_enable_handle, with self._lock:
bytes([0x01 if enabled else 0x00, 0x00]), self.writeCharacteristic(notification_enable_handle,
withResponse=True) bytes([1 if enabled else 0, 00]),
withResponse=True)
logger.debug("Notifications {action} for device {address}, " logger.debug("Notifications {action} for device {address}, "
"characteristic {uuid}" "characteristic {uuid}"
.format(action="enabled" if enabled else "disabled", .format(action="enabled" if enabled else "disabled",
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment