Skip to content
Draft
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
123 commits
Select commit Hold shift + click to select a range
d20be82
Add plus_pair_request function
bouwew Feb 2, 2026
210d414
Document pairing process
bouwew Feb 7, 2026
29113e4
Add 0001-0002 req-resp-pair
bouwew Feb 7, 2026
13a5b13
Add init stick
bouwew Feb 7, 2026
8fb42a0
Improve docstring
bouwew Feb 7, 2026
9e79012
Improve pair_plus_device()
bouwew Feb 7, 2026
2dfad1c
Add todo for maybe needed functionality
bouwew Feb 7, 2026
e9f2e0b
Fix typos, return type
bouwew Feb 7, 2026
841332f
Correct imports, improve docstring
bouwew Feb 7, 2026
fccd283
Ruff fixes
bouwew Feb 7, 2026
98d8e9b
Make sure the Stick is ready to pair, as suggested
bouwew Feb 8, 2026
855d649
Fix spelling
bouwew Feb 8, 2026
94865bb
Remove quotes, move
bouwew Feb 8, 2026
1a96f09
Set output as bool and use
bouwew Feb 8, 2026
c4d62c0
Add missing await
bouwew Feb 8, 2026
b95eab4
Add 0003 response to docstring
bouwew Feb 8, 2026
7ec42a6
Start adding pairing test
bouwew Feb 8, 2026
f392df3
Add missing connected()
bouwew Feb 8, 2026
a4d36a4
Add pairing-test
bouwew Feb 8, 2026
f628158
Link to stick_pair_data
bouwew Feb 8, 2026
92b60f5
Fix stick_pair_data
bouwew Feb 8, 2026
9afa86a
Set network to offline
bouwew Feb 8, 2026
6768572
Try
bouwew Feb 8, 2026
226ccce
Try 2
bouwew Feb 8, 2026
dfa3450
Try 3
bouwew Feb 8, 2026
777121d
Add StickInitShortResponse
bouwew Feb 8, 2026
c1271e5
Remove commented-out in response
bouwew Feb 8, 2026
8cfff68
Update length
bouwew Feb 8, 2026
1b8c381
Adapt StickInitRequest send()
bouwew Feb 8, 2026
ae9208b
Update length StickInitResponse
bouwew Feb 8, 2026
97ab500
Clean up
bouwew Feb 8, 2026
315e26a
Full test-output - test_pairing
bouwew Feb 8, 2026
85a103e
Allow init to fail
bouwew Feb 8, 2026
68b3333
Add sleep
bouwew Feb 8, 2026
81ae16a
Call pair_plus_request()
bouwew Feb 8, 2026
d2467fa
Connected and initialized is not required
bouwew Feb 8, 2026
7683d78
fixup: pair-plus Python code fixed using Ruff
Feb 8, 2026
74f986e
There can only be one response
bouwew Feb 8, 2026
b91c324
Use inheritance for StickInitResponse
bouwew Feb 8, 2026
21299bd
Move pair_plus_device() to connection
bouwew Feb 8, 2026
e066b4b
Correct plus-mac
bouwew Feb 8, 2026
2a62536
Add stick-mac to 0002 response
bouwew Feb 8, 2026
5fbd81a
Move RESPONSE_MESSAGES
bouwew Feb 9, 2026
afedc85
Don't test network down first
bouwew Feb 9, 2026
f53b336
Add missing import
bouwew Feb 9, 2026
8f51745
Connect first
bouwew Feb 9, 2026
10e408e
Try
bouwew Feb 9, 2026
699f0b1
fixup: pair-plus Python code fixed using Ruff
Feb 9, 2026
f53f1f5
Try 3
bouwew Feb 10, 2026
01d787b
Try 4
bouwew Feb 10, 2026
e8ea755
Try not allowed
bouwew Feb 10, 2026
0f69e34
Extra bit
bouwew Feb 10, 2026
687de46
CirclePlusConnectReqyest: shorter args
bouwew Feb 10, 2026
e5335fb
fixup: pair-plus Python code fixed using Ruff
Feb 10, 2026
2e473e6
Add stick-mac to 0005-response, remove extra bit
bouwew Feb 10, 2026
988c764
Ruffed
bouwew Feb 10, 2026
9c56fbb
Shorten args, must be length=16
bouwew Feb 10, 2026
dac8315
Add missing CRC, can be corrected later
bouwew Feb 10, 2026
84906aa
Try
bouwew Feb 12, 2026
93637e3
Try 2
bouwew Feb 13, 2026
5049eba
Fixes
bouwew Feb 13, 2026
4d8c17a
Correct CRC
bouwew Feb 13, 2026
582f949
Try allowed
bouwew Feb 13, 2026
bccfb8b
Change to Circle+ mac in 0005-response
bouwew Feb 13, 2026
d31b921
Ruff-cleanup
bouwew Feb 13, 2026
6c8b4e7
Bump to v0.48.0a1 test-version
bouwew Feb 13, 2026
0ed5f70
Update CHANGELOG
bouwew Feb 13, 2026
18e5fed
Implement StickInitShortResponse-handling in class StickController
bouwew Feb 16, 2026
b190083
Back to full test-output
bouwew Feb 16, 2026
f245e9d
Improve
bouwew Feb 16, 2026
b07b1c3
Correct CHANGELOG after rebase
bouwew Feb 16, 2026
a0e2a95
Bump to v0.48.0a2 test-version
bouwew Feb 16, 2026
c9593d1
Run all test-files in case of failure
bouwew Feb 16, 2026
288e6e8
Revert back to python 3.13
bouwew Feb 17, 2026
fe97fd7
Bump to a3
bouwew Feb 17, 2026
1cb1aca
Try-except stick-initialize
bouwew Feb 17, 2026
9695e64
Ruff fix
bouwew Feb 17, 2026
4964264
Add log-warning
bouwew Feb 17, 2026
43912fe
Bump to a4
bouwew Feb 17, 2026
0430479
Remove is_connected requirement for mac_stick
bouwew Feb 18, 2026
9c9117d
Bump to a5
bouwew Feb 18, 2026
0fad600
Disable now invalid test
bouwew Feb 18, 2026
f3b5ecf
More debug-logging
bouwew Feb 19, 2026
b7421a9
Bump to a6
bouwew Feb 19, 2026
7afe29b
Move debug message
bouwew Feb 19, 2026
867e24a
Bump to a7
bouwew Feb 19, 2026
fd8c2f0
Revert adding try-except
bouwew Feb 19, 2026
851dcfe
Replace debuggers by distinct message
bouwew Feb 20, 2026
12be5ad
Bump to a8
bouwew Feb 20, 2026
6a9b5af
Remove unneeded StickError raises
bouwew Feb 20, 2026
2cf0fd6
Update relevant test-asserts
bouwew Feb 20, 2026
eb1f122
Ruff fixes
bouwew Feb 20, 2026
5cbcb5b
Correct -update test_stick_network_down()
bouwew Feb 20, 2026
04c8d3f
Fix pylint warnings
bouwew Feb 20, 2026
223e2fa
More adapting to StickInitShortResponse
bouwew Feb 20, 2026
39d3ca1
Bump to a9
bouwew Feb 20, 2026
d12880a
Update Stick properties mac_stick, mac_coordinator and name
bouwew Feb 20, 2026
4eb1b47
Update docstring
bouwew Feb 20, 2026
34256a7
Update network_online docstring
bouwew Feb 20, 2026
2a550f1
Bump to a10
bouwew Feb 20, 2026
f07895c
fixup: pair-plus Python code fixed using Ruff
Feb 20, 2026
416ebc9
Responses: line up Int() use
bouwew Feb 23, 2026
30d3342
Add missing decode_mac=False
bouwew Feb 23, 2026
9125ebb
Correct 0002-format in response
bouwew Feb 23, 2026
7c6c2cc
Bump to a11
bouwew Feb 23, 2026
045e178
Exit when network is not online
bouwew Feb 23, 2026
c92280d
Bump to a12
bouwew Feb 23, 2026
a632bd0
Revert "Exit when network is not online"
bouwew Feb 23, 2026
0a188ea
Remove logger-HOI-lines
bouwew Feb 23, 2026
5a355f2
Bump to a13
bouwew Feb 23, 2026
93bec4a
Don't collect NodeInfo during pairing
bouwew Feb 23, 2026
6bae1fd
Update 0004-request
bouwew Feb 23, 2026
a76519a
Update test-related
bouwew Feb 23, 2026
402fb2e
Ruffed
bouwew Feb 23, 2026
33167b5
Update docstring
bouwew Feb 23, 2026
e28f0af
Bump to a14
bouwew Feb 24, 2026
d727d0f
0005-response: add missing decode_mac=False
bouwew Feb 24, 2026
9268c3b
Improve docstring
bouwew Feb 24, 2026
3e9811d
Adapt 0005-test-response
bouwew Mar 2, 2026
f15f72c
Add init to StickNetworkInfoRequest
bouwew Mar 3, 2026
6455492
And adapt use
bouwew Mar 3, 2026
cd70505
Disable guarding that breaks 0004-0005 sequence detection
bouwew Mar 4, 2026
975a6c6
Fix CHANGELOG after rebase
bouwew Mar 4, 2026
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion .github/workflows/verify.yml
Original file line number Diff line number Diff line change
Expand Up @@ -132,7 +132,7 @@ jobs:
- commitcheck
strategy:
matrix:
python-version: ["3.14"]
python-version: ["3.13"]
steps:
- name: Check out committed code
uses: actions/checkout@v6
Expand Down
6 changes: 5 additions & 1 deletion CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,5 +1,9 @@
# Changelog

## Ongoing

- PR [405](https://github.com/plugwise/python-plugwise-usb/pull/405): Try adding plus-device pairing (untested!!)

## v0.47.7 - 2026-05-18

PR [443](https://github.com/plugwise/python-plugwise-usb/pull/443): Migrate to serialx
Expand All @@ -19,7 +23,7 @@ PR [422](https://github.com/plugwise/python-plugwise-usb/pull/422): Add missing
## v0.47.3 - 2026-03-04

- PR [418](https://github.com/plugwise/python-plugwise-usb/pull/418): Improve raise-message for better debugging
- PR [405](https://github.com/plugwise/python-plugwise-usb/pull/405): Fix recent Ruff errors
- PR [409](https://github.com/plugwise/python-plugwise-usb/pull/409): Fix recent Ruff errors

## v0.47.2 - 2026-01-29

Expand Down
37 changes: 28 additions & 9 deletions plugwise_usb/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -130,18 +130,27 @@ def hardware(self) -> str:
return self._controller.hardware_stick

@property
def mac_stick(self) -> str:
"""MAC address of USB-Stick. Raises StickError is connection is missing."""
def mac_stick(self) -> str | None:
"""MAC address of USB-Stick.

Returns None when the connection to the Stick fails.
"""
return self._controller.mac_stick

@property
def mac_coordinator(self) -> str:
"""MAC address of the network coordinator (Circle+). Raises StickError is connection is missing."""
def mac_coordinator(self) -> str | None:
"""MAC address of the network coordinator (Circle+).

Returns none when there is no connection, not paired, not present in the network.
"""
return self._controller.mac_coordinator

@property
def name(self) -> str:
"""Return name of Stick."""
def name(self) -> str | None:
"""Return name of Stick.

Returns None when the connection to the Stick fails.
"""
return self._controller.stick_name

@property
Expand Down Expand Up @@ -237,8 +246,8 @@ async def setup(self, discover: bool = True, load: bool = True) -> None:
if not self.is_connected:
await self.connect()
if not self.is_initialized:
await self.initialize()
if discover:
initialized = await self.initialize()
if initialized and discover:
await self.start_network()
await self.discover_coordinator()
await self.discover_nodes()
Expand Down Expand Up @@ -266,10 +275,18 @@ async def connect(self, port: str | None = None) -> None:
self._port,
)

async def plus_pair_request(self, mac: str) -> bool:
"""Send a pair request to a Plus device."""
return await self._controller.pair_plus_device(mac)

@raise_not_connected
async def initialize(self, create_root_cache_folder: bool = False) -> None:
async def initialize(self, create_root_cache_folder: bool = False) -> bool:
"""Initialize connection to USB-Stick."""
await self._controller.initialize_stick()
# Check if network is offline = StickInitShortResponse
if self._controller.mac_coordinator is None:
return False

if self._network is None:
self._network = StickNetwork(self._controller)
self._network.cache_folder = self._cache_folder
Expand All @@ -278,6 +295,8 @@ async def initialize(self, create_root_cache_folder: bool = False) -> None:
if self._cache_enabled:
await self._network.initialize_cache()

return True

@raise_not_connected
@raise_not_initialized
async def start_network(self) -> None:
Expand Down
118 changes: 83 additions & 35 deletions plugwise_usb/connection/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -8,19 +8,22 @@

from ..api import StickEvent
from ..constants import UTF8
from ..exceptions import NodeError, StickError
from ..helpers.util import version_to_model
from ..exceptions import MessageError, NodeError, StickError
from ..helpers.util import validate_mac, version_to_model
from ..messages.requests import (
CirclePlusConnectRequest,
NodeInfoRequest,
NodePingRequest,
PlugwiseRequest,
StickInitRequest,
StickNetworkInfoRequest,
)
from ..messages.responses import (
NodeInfoResponse,
NodePingResponse,
PlugwiseResponse,
StickInitResponse,
StickInitShortResponse,
)
from .manager import StickConnectionManager
from .queue import StickQueue
Expand Down Expand Up @@ -69,38 +72,27 @@ def hardware_stick(self) -> str | None:
return self._hw_stick

@property
def mac_stick(self) -> str:
"""MAC address of USB-Stick. Raises StickError when not connected."""
if not self._manager.is_connected or self._mac_stick is None:
raise StickError(
"No mac address available. Connect and initialize USB-Stick first."
)
def mac_stick(self) -> str | None:
"""MAC address of USB-Stick."""
return self._mac_stick

@property
def mac_coordinator(self) -> str:
"""Return MAC address of the Zigbee network coordinator (Circle+).

Raises StickError when not connected.
"""
if not self._manager.is_connected or self._mac_nc is None:
raise StickError(
"No mac address available. Connect and initialize USB-Stick first."
)
def mac_coordinator(self) -> str | None:
"""Return MAC address of the Zigbee network coordinator (Circle+)."""
return self._mac_nc

@property
def network_id(self) -> int:
"""Returns the Zigbee network ID. Raises StickError when not connected."""
if not self._manager.is_connected or self._network_id is None:
raise StickError(
"No network ID available. Connect and initialize USB-Stick first."
)
def network_id(self) -> int | None:
"""Returns the Zigbee network ID."""
return self._network_id

@property
def network_online(self) -> bool:
"""Return the network state."""
"""Return the network state.

The ZigBee network is online when the Stick is connected and a
StickInitResponse indicates that the ZigBee network is online.
"""
if not self._manager.is_connected:
raise StickError(
"Network status not available. Connect and initialize USB-Stick first."
Expand Down Expand Up @@ -159,7 +151,7 @@ async def _handle_stick_event(self, event: StickEvent) -> None:
elif event == StickEvent.DISCONNECTED and self._queue.is_running:
await self._queue.stop()

async def initialize_stick(self) -> None:
async def initialize_stick(self, node_info=True) -> None:
"""Initialize connection to the USB-stick."""
if not self._manager.is_connected:
raise StickError(
Expand All @@ -170,7 +162,9 @@ async def initialize_stick(self) -> None:

try:
request = StickInitRequest(self.send)
init_response: StickInitResponse | None = await request.send()
init_response: (
StickInitResponse | StickInitShortResponse | None
) = await request.send()
except StickError as err:
raise StickError(
"No response from USB-Stick to initialization request."
Expand All @@ -186,26 +180,80 @@ async def initialize_stick(self) -> None:
self._mac_stick = init_response.mac_decoded
self.stick_name = f"Stick {self._mac_stick[-5:]}"
self._network_online = init_response.network_online
if self._network_online:
# Replace first 2 characters by 00 for mac of circle+ node
self._mac_nc = init_response.mac_network_controller
self._network_id = init_response.network_id

# Replace first 2 characters by 00 for mac of circle+ node
self._mac_nc = init_response.mac_network_controller
self._network_id = init_response.network_id
self._is_initialized = True

# Add Stick NodeInfoRequest
if not node_info:
return

# Collect Stick NodeInfo
node_info, _ = await self.get_node_details(self._mac_stick, ping_first=False)
if node_info is not None:
self._fw_stick = node_info.firmware
self._fw_stick = node_info.firmware # type: ignore
hardware, _ = version_to_model(node_info.hardware)
self._hw_stick = hardware

if not self._network_online:
raise StickError("Zigbee network connection to Circle+ is down.")
async def pair_plus_device(self, mac: str) -> bool:
"""Pair Plus-device to Plugwise Stick.

According to https://roheve.wordpress.com/author/roheve/page/2/
The pairing process should look like:
0001 - 0002 - 0003: StickNetworkInfoRequest - StickNetworkInfoResponse - NodeSpecificResponse,
000A - 0011: StickInitRequest - StickInitShortResponse/StickInitResponse,
0004 - 0005: CirclePlusConnectRequest - CirclePlusConnectResponse,
the Plus-device will then send a NodeRejoinResponse (0061).

In the first occurrence of this process a 0004 0001 .... message is sent.
A StickInitShortResponse is received indicating the network is offline.
In the second occurrence of this process a 0004 0101 .... message is sent.
Again a StickInitShortResponse is received.
In the third occurrence only 000A is sent and a StickInitResponse indicating the network is online, is received.
"""
_LOGGER.debug("Pair Plus-device with mac: %s", mac)
if not validate_mac(mac):
raise NodeError(f"Pairing failed: MAC {mac} invalid")

# Collect network info
try:
request = StickNetworkInfoRequest(self.send)
info_response = await request.send()
except MessageError as exc:
raise NodeError(f"Pairing failed: {exc}") from exc
if info_response is None:
raise NodeError(
"Pairing failed, StickNetworkInfoResponse is None"
) from None

# Init Stick
try:
await self.initialize_stick(node_info=False)
except StickError as exc:
raise NodeError(
f"Pairing failed, failed to initialize Stick: {exc}"
) from exc

try:
request = CirclePlusConnectRequest(self.send, bytes(mac, UTF8))
response = await request.send()
except MessageError as exc:
raise NodeError(f"Pairing failed: {exc}") from exc
if response is None:
raise NodeError(
"Pairing failed, CirclePlusConnectResponse is None"
) from None
if response.allowed.value != 1:
raise NodeError("Pairing failed, not allowed")

return True

async def get_node_details(
self, mac: str, ping_first: bool
) -> tuple[NodeInfoResponse | None, NodePingResponse | None]:
"""Return node discovery type."""
"""Collect NodeInfo data from the Stick."""
ping_response: NodePingResponse | None = None
if ping_first:
# Define ping request with one retry
Expand Down Expand Up @@ -234,7 +282,7 @@ async def send(
return await self._queue.submit(request)
try:
return await self._queue.submit(request)
except (NodeError, StickError):
except NodeError, StickError:
return None

def _reset_states(self) -> None:
Expand Down
10 changes: 5 additions & 5 deletions plugwise_usb/connection/receiver.py
Original file line number Diff line number Diff line change
Expand Up @@ -441,11 +441,11 @@ async def _notify_node_response_subscribers(

notify_tasks: list[Coroutine[Any, Any, bool]] = []
for node_subscription in self._node_response_subscribers.values():
if (
node_subscription.mac is not None
and node_subscription.mac != node_response.mac
):
continue
# if (
# node_subscription.mac is not None
# and node_subscription.mac != node_response.mac
# ):
# continue
if (
node_subscription.response_ids is not None
and node_response.identifier not in node_subscription.response_ids
Expand Down
38 changes: 26 additions & 12 deletions plugwise_usb/messages/requests.py
Original file line number Diff line number Diff line change
Expand Up @@ -38,6 +38,7 @@
NodeSpecificResponse,
PlugwiseResponse,
StickInitResponse,
StickInitShortResponse,
StickNetworkInfoResponse,
StickResponse,
StickResponseType,
Expand Down Expand Up @@ -363,8 +364,18 @@ class StickNetworkInfoRequest(PlugwiseRequest):
_identifier = b"0001"
_reply_identifier = b"0002"

def __init__(
self,
send_fn: Callable[[PlugwiseRequest, bool], Awaitable[PlugwiseResponse | None]],
) -> None:
"""Initialize StickInitRequest message object."""
super().__init__(send_fn, None)
self._max_retries = 1

async def send(self) -> StickNetworkInfoResponse | None:
"""Send request."""
if self._send_fn is None:
raise MessageError("Send function missing")
result = await self._send_request()
if isinstance(result, StickNetworkInfoResponse):
return result
Expand Down Expand Up @@ -399,14 +410,17 @@ async def send(self) -> CirclePlusConnectResponse | None:
# This message has an exceptional format and therefore
# need to override the serialize method
def serialize(self) -> bytes:
"""Convert message to serialized list of bytes."""
# This command has
# args: byte
# key, byte
# network info.index, ulong
# network key = 0
args = b"00000000000000000000"
msg: bytes = self._identifier + args
"""Convert message to serialized list of bytes.

Parameters
----------
- special_id: byte - observed sequence with retry: b"0001", B"0101",
- and args: byte.

"""
special_id = b"0001"
args = b"0000000000000000"
msg: bytes = self._identifier + special_id + args
if self._mac is not None:
msg += self._mac
checksum = self.calculate_checksum(msg)
Expand Down Expand Up @@ -523,7 +537,7 @@ class StickInitRequest(PlugwiseRequest):
"""Initialize USB-Stick.

Supported protocols : 1.0, 2.0
Response message : StickInitResponse
Response message : StickInitResponse or StickInitShortResponse
"""

_identifier = b"000A"
Expand All @@ -537,17 +551,17 @@ def __init__(
super().__init__(send_fn, None)
self._max_retries = 1

async def send(self) -> StickInitResponse | None:
async def send(self) -> StickInitResponse | StickInitShortResponse | None:
"""Send request."""
if self._send_fn is None:
raise MessageError("Send function missing")
result = await self._send_request()
if isinstance(result, StickInitResponse):
if isinstance(result, StickInitResponse | StickInitShortResponse):
return result
if result is None:
return None
raise MessageError(
f"Invalid response message. Received {result.__class__.__name__}, expected StickInitResponse"
f"Invalid response message. Received {result.__class__.__name__}, expected StickInitResponse/StickInitShortResponse"
)


Expand Down
Loading
Loading