From 32917e0756030168445076feea0f28508e156cf0 Mon Sep 17 00:00:00 2001 From: puddly <32534428+puddly@users.noreply.github.com> Date: Thu, 30 Mar 2023 13:59:29 -0400 Subject: [PATCH 1/7] 0.36.0.dev0 version bump --- bellows/__init__.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/bellows/__init__.py b/bellows/__init__.py index 52cdb99d..80d1a083 100644 --- a/bellows/__init__.py +++ b/bellows/__init__.py @@ -1,5 +1,5 @@ MAJOR_VERSION = 0 -MINOR_VERSION = 35 +MINOR_VERSION = 36 PATCH_VERSION = "0.dev0" __short_version__ = f"{MAJOR_VERSION}.{MINOR_VERSION}" __version__ = f"{__short_version__}.{PATCH_VERSION}" From a1bcb88a2072460208f4f5c335f8d108bc02e0d6 Mon Sep 17 00:00:00 2001 From: puddly <32534428+puddly@users.noreply.github.com> Date: Fri, 31 Mar 2023 13:46:23 -0400 Subject: [PATCH 2/7] Rename LQI to Energy for energy scanning (#542) --- bellows/zigbee/application.py | 4 ++-- bellows/zigbee/util.py | 20 +++++++++++++++++--- tests/test_util.py | 11 ++++++++--- 3 files changed, 27 insertions(+), 8 deletions(-) diff --git a/bellows/zigbee/application.py b/bellows/zigbee/application.py index 0ebe585b..2de3e1b7 100644 --- a/bellows/zigbee/application.py +++ b/bellows/zigbee/application.py @@ -688,9 +688,9 @@ async def energy_scan( for channel, rssi in results: all_results.setdefault(channel, []).append(rssi) - # Remap RSSI to LQI + # Remap RSSI to Energy return { - channel: util.remap_rssi_to_lqi(statistics.mean(rssis)) + channel: util.map_rssi_to_energy(statistics.mean(rssis)) for channel, rssis in all_results.items() } diff --git a/bellows/zigbee/util.py b/bellows/zigbee/util.py index 33aa7005..1d3779d0 100644 --- a/bellows/zigbee/util.py +++ b/bellows/zigbee/util.py @@ -110,12 +110,26 @@ def logistic(x: float, *, L: float = 1, x_0: float = 0, k: float = 1) -> float: return L / (1 + math.exp(-k * (x - x_0))) -def remap_rssi_to_lqi(rssi: int) -> float: - """Remaps RSSI (in dBm) to LQI (0-255).""" - +def map_rssi_to_energy(rssi: int) -> float: + """Remaps RSSI (in dBm) to Energy (0-255).""" return logistic( x=rssi, L=255, x_0=RSSI_MIN + 0.45 * (RSSI_MAX - RSSI_MIN), k=0.13, ) + + +def logit(y: float, *, L: float = 1, x_0: float = 0, k: float = 1) -> float: + """Logit function (inverse of logistic).""" + return x_0 - math.log(L / y - 1) / k + + +def map_energy_to_rssi(lqi: float) -> float: + """Remaps Energy (0-255) back to RSSI (in dBm).""" + return logit( + y=lqi, + L=255, + x_0=RSSI_MIN + 0.45 * (RSSI_MAX - RSSI_MIN), + k=0.13, + ) diff --git a/tests/test_util.py b/tests/test_util.py index 80e1e59c..da76e26b 100644 --- a/tests/test_util.py +++ b/tests/test_util.py @@ -145,6 +145,11 @@ def test_zigpy_key_to_ezsp_key(zigpy_key, ezsp_key, ezsp_mock): assert util.zigpy_key_to_ezsp_key(zigpy_key, ezsp_mock) == ezsp_key -def test_remap_rssi_to_lqi(): - assert 0 <= util.remap_rssi_to_lqi(-200) <= 0.01 - assert 254 <= util.remap_rssi_to_lqi(100) <= 255 +def test_map_rssi_to_energy(): + assert 0 <= util.map_rssi_to_energy(-200) <= 0.01 + assert 254 <= util.map_rssi_to_energy(100) <= 255 + + # Make sure the two functions are inverses + for rssi in range(-100, 100): + energy = util.map_rssi_to_energy(rssi) + assert abs(util.map_energy_to_rssi(energy) - rssi) < 0.1 From 9b1126ad905bef85b182a0d91c740906e82aa1f7 Mon Sep 17 00:00:00 2001 From: puddly <32534428+puddly@users.noreply.github.com> Date: Wed, 5 Apr 2023 14:52:34 -0400 Subject: [PATCH 3/7] Use `ConnectionResetError` instead of a bare `OSError` (#544) --- bellows/uart.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/bellows/uart.py b/bellows/uart.py index 2221daa5..03a7fc50 100644 --- a/bellows/uart.py +++ b/bellows/uart.py @@ -201,7 +201,7 @@ def _reset_cleanup(self, future): def eof_received(self): """Server gracefully closed its side of the connection.""" - self.connection_lost(OSError("Server closed connection")) + self.connection_lost(ConnectionResetError("Remote server closed connection")) def connection_lost(self, exc): """Port was closed unexpectedly.""" @@ -214,7 +214,7 @@ def connection_lost(self, exc): # `CancelledError` into the active event loop, breaking everything! if self._startup_reset_future: self._startup_reset_future.set_exception( - exc or OSError("Server closed connection") + exc or ConnectionResetError("Remote server closed connection") ) if self._connection_done_future: From 7b29e944f4b2fcbbcd2bfd9b055eb6e22ee9f094 Mon Sep 17 00:00:00 2001 From: puddly <32534428+puddly@users.noreply.github.com> Date: Wed, 5 Apr 2023 14:52:56 -0400 Subject: [PATCH 4/7] Only throw `NetworkNotFormed` when initialization fails with `EmberStatus.NOT_JOINED` (#545) --- bellows/zigbee/application.py | 10 +++++---- tests/test_application.py | 41 ++++++++++++++++++++++++++++++----- 2 files changed, 42 insertions(+), 9 deletions(-) diff --git a/bellows/zigbee/application.py b/bellows/zigbee/application.py index 2de3e1b7..aa75c055 100644 --- a/bellows/zigbee/application.py +++ b/bellows/zigbee/application.py @@ -144,10 +144,12 @@ async def _ensure_network_running(self) -> bool: return False (init_status,) = await self._ezsp.networkInit() - if init_status != t.EmberStatus.SUCCESS: - raise NetworkNotFormed(f"Failed to init network: {init_status!r}") - - return True + if init_status == t.EmberStatus.SUCCESS: + return True + elif init_status == t.EmberStatus.NOT_JOINED: + raise NetworkNotFormed("Node is not part of a network") + else: + raise ControllerError(f"Failed to initialize network: {init_status!r}") async def start_network(self): ezsp = self._ezsp diff --git a/tests/test_application.py b/tests/test_application.py index 58c12c96..026daddd 100644 --- a/tests/test_application.py +++ b/tests/test_application.py @@ -95,7 +95,14 @@ def ieee(init=0): @patch("zigpy.device.Device._initialize", new=AsyncMock()) @patch("bellows.zigbee.application.ControllerApplication._watchdog", new=AsyncMock()) async def _test_startup( - app, nwk_type, ieee, auto_form=False, init=0, ezsp_version=4, board_info=True + app, + nwk_type, + ieee, + auto_form=False, + init=0, + ezsp_version=4, + board_info=True, + network_state=t.EmberNetworkStatus.JOINED_NETWORK, ): nwk_params = bellows.types.struct.EmberNetworkParameters( extendedPanId=t.ExtendedPanId.convert("aa:bb:cc:dd:ee:ff:aa:bb"), @@ -144,9 +151,7 @@ async def mock_leave(*args, **kwargs): ezsp_mock.version = AsyncMock() ezsp_mock.getConfigurationValue = AsyncMock(return_value=(0, 1)) ezsp_mock.update_policies = AsyncMock() - ezsp_mock.networkState = AsyncMock( - return_value=[ezsp_mock.types.EmberNetworkStatus.JOINED_NETWORK] - ) + ezsp_mock.networkState = AsyncMock(return_value=[network_state]) ezsp_mock.getKey = AsyncMock( return_value=[ t.EmberStatus.SUCCESS, @@ -222,6 +227,32 @@ async def test_startup_no_status(app, ieee): ) +async def test_startup_status_not_joined(app, ieee): + """Test when NCP is a coordinator but isn't a part of a network.""" + with pytest.raises(zigpy.exceptions.NetworkNotFormed): + await _test_startup( + app, + t.EmberNodeType.COORDINATOR, + ieee, + auto_form=False, + init=t.EmberStatus.NOT_JOINED, + network_state=t.EmberNetworkStatus.NO_NETWORK, + ) + + +async def test_startup_status_unknown(app, ieee): + """Test when NCP is a coordinator but stack init fails.""" + with pytest.raises(zigpy.exceptions.ControllerException): + await _test_startup( + app, + t.EmberNodeType.COORDINATOR, + ieee, + auto_form=False, + init=t.EmberStatus.ERR_FATAL, + network_state=t.EmberNetworkStatus.NO_NETWORK, + ) + + async def test_startup_no_status_form(app, ieee): """Test when NCP is not a coordinator but allow auto forming.""" await _test_startup( @@ -1568,7 +1599,7 @@ async def test_ensure_network_running_not_joined_failure(app): ) ezsp.networkInit = AsyncMock(return_value=[ezsp.types.EmberStatus.INVALID_CALL]) - with pytest.raises(zigpy.exceptions.NetworkNotFormed): + with pytest.raises(zigpy.exceptions.ControllerException): await app._ensure_network_running() ezsp.networkState.assert_called_once() From 07b24f9beca5c6adf2c406b8693f9b60aad4e02f Mon Sep 17 00:00:00 2001 From: puddly <32534428+puddly@users.noreply.github.com> Date: Wed, 5 Apr 2023 14:53:22 -0400 Subject: [PATCH 5/7] Replace all `asyncio.wait_for`s with `asyncio.timeout` (#546) --- bellows/ezsp/__init__.py | 21 ++++++++++++++------- bellows/ezsp/protocol.py | 12 ++++++++++-- bellows/uart.py | 10 +++++++++- bellows/zigbee/application.py | 17 +++++++++++------ setup.py | 1 + tests/test_application.py | 2 +- tests/test_ezsp.py | 12 ++++++------ tests/test_ezsp_protocol.py | 5 ++++- tests/test_thread.py | 11 +++++++++-- 9 files changed, 65 insertions(+), 26 deletions(-) diff --git a/bellows/ezsp/__init__.py b/bellows/ezsp/__init__.py index a9bab33d..3ed1d874 100644 --- a/bellows/ezsp/__init__.py +++ b/bellows/ezsp/__init__.py @@ -5,9 +5,15 @@ import asyncio import functools import logging +import sys from typing import Any, Callable, Dict, List, Tuple, Union import urllib.parse +if sys.version_info[:2] < (3, 11): + from async_timeout import timeout as asyncio_timeout # pragma: no cover +else: + from asyncio import timeout as asyncio_timeout # pragma: no cover + import zigpy.config import bellows.config as conf @@ -58,7 +64,9 @@ async def probe(cls, device_config: Dict) -> bool | dict[str, int | str | bool]: ): ezsp = cls(conf.SCHEMA_DEVICE(config)) try: - await asyncio.wait_for(ezsp._probe(), timeout=PROBE_TIMEOUT) + async with asyncio_timeout(PROBE_TIMEOUT): + await ezsp._probe() + return config except Exception as exc: LOGGER.debug( @@ -83,10 +91,8 @@ async def _startup_reset(self): parsed_path = urllib.parse.urlparse(self._config[conf.CONF_DEVICE_PATH]) if parsed_path.scheme == "socket": try: - await asyncio.wait_for( - self._gw.wait_for_startup_reset(), - NETWORK_COORDINATOR_STARTUP_RESET_WAIT, - ) + async with asyncio_timeout(NETWORK_COORDINATOR_STARTUP_RESET_WAIT): + await self._gw.wait_for_startup_reset() except asyncio.TimeoutError: pass else: @@ -233,8 +239,9 @@ def cb(frame_name: str, response: List) -> None: (status,) = await self._command("leaveNetwork") if status != t.EmberStatus.SUCCESS: raise EzspError(f"failed to leave network: {status.name}") - result = await asyncio.wait_for(stack_status, timeout=timeout) - return result + + async with asyncio_timeout(timeout): + return await stack_status finally: self.remove_callback(cb_id) diff --git a/bellows/ezsp/protocol.py b/bellows/ezsp/protocol.py index 2f67b9b5..dcab7b61 100644 --- a/bellows/ezsp/protocol.py +++ b/bellows/ezsp/protocol.py @@ -3,8 +3,14 @@ import binascii import functools import logging +import sys from typing import Any, Callable, Dict, Optional, Tuple +if sys.version_info[:2] < (3, 11): + from async_timeout import timeout as asyncio_timeout # pragma: no cover +else: + from asyncio import timeout as asyncio_timeout # pragma: no cover + from bellows.config import CONF_EZSP_CONFIG, CONF_EZSP_POLICIES from bellows.exception import EzspError from bellows.typing import GatewayType @@ -111,7 +117,7 @@ async def get_free_buffers(self) -> Optional[int]: return int.from_bytes(value, byteorder="little") - def command(self, name, *args) -> asyncio.Future: + async def command(self, name, *args) -> Any: """Serialize command and send it.""" LOGGER.debug("Send command %s: %s", name, args) data = self._ezsp_frame(name, *args) @@ -120,7 +126,9 @@ def command(self, name, *args) -> asyncio.Future: future = asyncio.Future() self._awaiting[self._seq] = (c[0], c[2], future) self._seq = (self._seq + 1) % 256 - return asyncio.wait_for(future, timeout=EZSP_CMD_TIMEOUT) + + async with asyncio_timeout(EZSP_CMD_TIMEOUT): + return await future async def set_source_routing(self) -> None: """Enable source routing on NCP.""" diff --git a/bellows/uart.py b/bellows/uart.py index 03a7fc50..cd67d59c 100644 --- a/bellows/uart.py +++ b/bellows/uart.py @@ -1,6 +1,12 @@ import asyncio import binascii import logging +import sys + +if sys.version_info[:2] < (3, 11): + from async_timeout import timeout as asyncio_timeout # pragma: no cover +else: + from asyncio import timeout as asyncio_timeout # pragma: no cover import zigpy.serial @@ -257,7 +263,9 @@ async def reset(self): self._reset_future = asyncio.get_event_loop().create_future() self._reset_future.add_done_callback(self._reset_cleanup) self.write(self._rst_frame()) - return await asyncio.wait_for(self._reset_future, timeout=RESET_TIMEOUT) + + async with asyncio_timeout(RESET_TIMEOUT): + return await self._reset_future async def _send_loop(self): """Send queue handler""" diff --git a/bellows/zigbee/application.py b/bellows/zigbee/application.py index aa75c055..7afba46d 100644 --- a/bellows/zigbee/application.py +++ b/bellows/zigbee/application.py @@ -4,8 +4,14 @@ import logging import os import statistics +import sys from typing import Dict, Optional +if sys.version_info[:2] < (3, 11): + from async_timeout import timeout as asyncio_timeout # pragma: no cover +else: + from asyncio import timeout as asyncio_timeout # pragma: no cover + import zigpy.application import zigpy.config import zigpy.device @@ -816,9 +822,8 @@ async def send_packet(self, packet: zigpy.types.ZigbeePacket) -> None: return # Wait for `messageSentHandler` message - send_status, _ = await asyncio.wait_for( - req.result, timeout=APS_ACK_TIMEOUT - ) + async with asyncio_timeout(APS_ACK_TIMEOUT): + send_status, _ = await req.result if send_status != t.EmberStatus.SUCCESS: raise zigpy.exceptions.DeliveryError( @@ -880,9 +885,9 @@ async def _watchdog(self): await asyncio.sleep(WATCHDOG_WAKE_PERIOD) while True: try: - await asyncio.wait_for( - self.controller_event.wait(), timeout=WATCHDOG_WAKE_PERIOD * 2 - ) + async with asyncio_timeout(WATCHDOG_WAKE_PERIOD * 2): + await self.controller_event.wait() + if self._ezsp.ezsp_version == 4: await self._ezsp.nop() else: diff --git a/setup.py b/setup.py index 05b610d0..90989abb 100644 --- a/setup.py +++ b/setup.py @@ -20,6 +20,7 @@ "pure_pcapy3==1.0.1", "voluptuous", "zigpy>=0.54.0", + 'async-timeout; python_version<"3.11"', ], dependency_links=[ "https://codeload.github.com/rcloran/pure-pcapy-3/zip/master", diff --git a/tests/test_application.py b/tests/test_application.py index 026daddd..2a23d7d4 100644 --- a/tests/test_application.py +++ b/tests/test_application.py @@ -60,7 +60,7 @@ def inner(config): app = bellows.zigbee.application.ControllerApplication(app_cfg) app._ezsp = ezsp_mock - monkeypatch.setattr(bellows.zigbee.application, "APS_ACK_TIMEOUT", 0.01) + monkeypatch.setattr(bellows.zigbee.application, "APS_ACK_TIMEOUT", 0.05) app._ctrl_event.set() app._in_flight_msg = asyncio.Semaphore() app.handle_message = MagicMock() diff --git a/tests/test_ezsp.py b/tests/test_ezsp.py index 115eecb1..5616332b 100644 --- a/tests/test_ezsp.py +++ b/tests/test_ezsp.py @@ -70,21 +70,21 @@ def test_attr(ezsp_f): assert callable(m) -def test_non_existent_attr(ezsp_f): +async def test_non_existent_attr(ezsp_f): with pytest.raises(AttributeError): - ezsp_f.nonexistentMethod() + await ezsp_f.nonexistentMethod() -def test_command(ezsp_f): +async def test_command(ezsp_f): ezsp_f.start_ezsp() with patch.object(ezsp_f._protocol, "command") as cmd_mock: - ezsp_f.nop() + await ezsp_f.nop() assert cmd_mock.call_count == 1 -def test_command_ezsp_stopped(ezsp_f): +async def test_command_ezsp_stopped(ezsp_f): with pytest.raises(EzspError): - ezsp_f._command("version") + await ezsp_f._command("version") async def _test_list_command(ezsp_f, mockcommand): diff --git a/tests/test_ezsp_protocol.py b/tests/test_ezsp_protocol.py index 640788cb..591388ad 100644 --- a/tests/test_ezsp_protocol.py +++ b/tests/test_ezsp_protocol.py @@ -29,7 +29,10 @@ def prot_hndl(): async def test_command(prot_hndl): coro = prot_hndl.command("nop") - prot_hndl._awaiting[prot_hndl._seq - 1][2].set_result(True) + asyncio.get_running_loop().call_soon( + lambda: prot_hndl._awaiting[prot_hndl._seq - 1][2].set_result(True) + ) + await coro assert prot_hndl._gw.data.call_count == 1 diff --git a/tests/test_thread.py b/tests/test_thread.py index 9789083c..a7c37723 100644 --- a/tests/test_thread.py +++ b/tests/test_thread.py @@ -3,6 +3,11 @@ import threading from unittest import mock +if sys.version_info[:2] < (3, 11): + from async_timeout import timeout as asyncio_timeout # pragma: no cover +else: + from asyncio import timeout as asyncio_timeout # pragma: no cover + import pytest from bellows.thread import EventLoopThread, ThreadsafeProxy @@ -48,7 +53,8 @@ async def thread(): yield thread thread.force_stop() if thread.thread_complete is not None: - await asyncio.wait_for(thread.thread_complete, 1) + async with asyncio_timeout(1): + await thread.thread_complete [t.join(1) for t in threading.enumerate() if "bellows" in t.name] threads = [t for t in threading.enumerate() if "bellows" in t.name] assert len(threads) == 0 @@ -178,4 +184,5 @@ async def wait_forever(): # The cancellation should propagate to the outer event loop with pytest.raises(asyncio.CancelledError): # This will stall forever without the patch - await asyncio.wait_for(proxy.wait_forever(), 1) + async with asyncio_timeout(1): + await proxy.wait_forever() From a73588dc6e458c5a82c5d719bd1caf0b48727705 Mon Sep 17 00:00:00 2001 From: "David F. Mulcahey" Date: Tue, 11 Apr 2023 20:43:23 -0400 Subject: [PATCH 6/7] Update to shared CI and add ruff (#547) * update CI * shared CI * contextlib supress * dangling tasks * remove mypy until we can clean types up * update Zigpy requirement * ignore mypy a different way to account for CI --- .github/workflows/ci.yml | 393 +----------------------- .github/workflows/matchers/ruff.json | 30 ++ .pre-commit-config.yaml | 31 +- .travis.yml | 19 -- bellows/cli/application.py | 24 +- bellows/cli/backup.py | 2 +- bellows/cli/dump.py | 4 +- bellows/cli/ncp.py | 16 +- bellows/cli/network.py | 14 +- bellows/cli/util.py | 15 +- bellows/ezsp/__init__.py | 20 +- bellows/ezsp/v10/types/named.py | 5 +- bellows/ezsp/v6/types/named.py | 5 +- bellows/ezsp/v7/types/named.py | 5 +- bellows/ezsp/v8/types/named.py | 5 +- bellows/ezsp/v9/types/named.py | 5 +- bellows/types/basic.py | 4 +- bellows/zigbee/application.py | 12 +- bellows/zigbee/device.py | 4 +- requirements_test.txt | 16 +- ruff.toml | 71 +++++ script/setup | 15 + setup.cfg | 3 + setup.py | 2 +- tests/test_application_device.py | 2 - tests/test_application_network_state.py | 2 - 26 files changed, 237 insertions(+), 487 deletions(-) create mode 100644 .github/workflows/matchers/ruff.json delete mode 100644 .travis.yml create mode 100644 ruff.toml create mode 100755 script/setup diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 7eab0507..894f25c3 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -1,392 +1,15 @@ name: CI -# yamllint disable-line rule:truthy on: push: pull_request: ~ -env: - CACHE_VERSION: 1 - DEFAULT_PYTHON: 3.8 - PRE_COMMIT_HOME: ~/.cache/pre-commit - jobs: - # Separate job to pre-populate the base dependency cache - # This prevent upcoming jobs to do the same individually - prepare-base: - name: Prepare base dependencies - runs-on: ubuntu-latest - strategy: - matrix: - python-version: [3.8, 3.9, "3.10", "3.11"] - steps: - - name: Check out code from GitHub - uses: actions/checkout@v2 - - name: Set up Python ${{ matrix.python-version }} - id: python - uses: actions/setup-python@v2.1.4 - with: - python-version: ${{ matrix.python-version }} - - name: Restore base Python virtual environment - id: cache-venv - uses: actions/cache@v2 - with: - path: venv - key: >- - ${{ env.CACHE_VERSION}}-${{ runner.os }}-base-venv-${{ - steps.python.outputs.python-version }}-${{ - hashFiles('setup.py') }}-${{ - hashFiles('requirements_test.txt') }}-${{ - hashFiles('.pre-commit-config.yaml') }} - restore-keys: | - ${{ env.CACHE_VERSION}}-${{ runner.os }}-base-venv-${{ steps.python.outputs.python-version }}- - - name: Create Python virtual environment - if: steps.cache-venv.outputs.cache-hit != 'true' - run: | - python -m venv venv - . venv/bin/activate - pip install -U pip setuptools pre-commit - pip install -r requirements_test.txt - pip install -e . - - pre-commit: - name: Prepare pre-commit environment - runs-on: ubuntu-latest - needs: prepare-base - steps: - - name: Check out code from GitHub - uses: actions/checkout@v2 - - name: Set up Python ${{ env.DEFAULT_PYTHON }} - uses: actions/setup-python@v2.1.4 - id: python - with: - python-version: ${{ env.DEFAULT_PYTHON }} - - name: Restore base Python virtual environment - id: cache-venv - uses: actions/cache@v2 - with: - path: venv - key: >- - ${{ env.CACHE_VERSION}}-${{ runner.os }}-base-venv-${{ - steps.python.outputs.python-version }}-${{ - hashFiles('setup.py') }}-${{ - hashFiles('requirements_test.txt') }}-${{ - hashFiles('.pre-commit-config.yaml') }} - - name: Fail job if Python cache restore failed - if: steps.cache-venv.outputs.cache-hit != 'true' - run: | - echo "Failed to restore Python virtual environment from cache" - exit 1 - - name: Restore pre-commit environment from cache - id: cache-precommit - uses: actions/cache@v2 - with: - path: ${{ env.PRE_COMMIT_HOME }} - key: | - ${{ env.CACHE_VERSION}}-${{ runner.os }}-pre-commit-${{ hashFiles('.pre-commit-config.yaml') }} - restore-keys: | - ${{ env.CACHE_VERSION}}-${{ runner.os }}-pre-commit- - - name: Install pre-commit dependencies - if: steps.cache-precommit.outputs.cache-hit != 'true' - run: | - . venv/bin/activate - pre-commit install-hooks - - lint-black: - name: Check black - runs-on: ubuntu-latest - needs: pre-commit - steps: - - name: Check out code from GitHub - uses: actions/checkout@v2 - - name: Set up Python ${{ env.DEFAULT_PYTHON }} - uses: actions/setup-python@v2.1.4 - id: python - with: - python-version: ${{ env.DEFAULT_PYTHON }} - - name: Restore base Python virtual environment - id: cache-venv - uses: actions/cache@v2 - with: - path: venv - key: >- - ${{ env.CACHE_VERSION}}-${{ runner.os }}-base-venv-${{ - steps.python.outputs.python-version }}-${{ - hashFiles('setup.py') }}-${{ - hashFiles('requirements_test.txt') }}-${{ - hashFiles('.pre-commit-config.yaml') }} - - name: Fail job if Python cache restore failed - if: steps.cache-venv.outputs.cache-hit != 'true' - run: | - echo "Failed to restore Python virtual environment from cache" - exit 1 - - name: Restore pre-commit environment from cache - id: cache-precommit - uses: actions/cache@v2 - with: - path: ${{ env.PRE_COMMIT_HOME }} - key: | - ${{ env.CACHE_VERSION}}-${{ runner.os }}-pre-commit-${{ hashFiles('.pre-commit-config.yaml') }} - - name: Fail job if cache restore failed - if: steps.cache-venv.outputs.cache-hit != 'true' - run: | - echo "Failed to restore Python virtual environment from cache" - exit 1 - - name: Run black - run: | - . venv/bin/activate - pre-commit run --hook-stage manual black --all-files --show-diff-on-failure - - lint-flake8: - name: Check flake8 - runs-on: ubuntu-latest - needs: pre-commit - steps: - - name: Check out code from GitHub - uses: actions/checkout@v2 - - name: Set up Python ${{ env.DEFAULT_PYTHON }} - uses: actions/setup-python@v2.1.4 - id: python - with: - python-version: ${{ env.DEFAULT_PYTHON }} - - name: Restore base Python virtual environment - id: cache-venv - uses: actions/cache@v2 - with: - path: venv - key: >- - ${{ env.CACHE_VERSION}}-${{ runner.os }}-base-venv-${{ - steps.python.outputs.python-version }}-${{ - hashFiles('setup.py') }}-${{ - hashFiles('requirements_test.txt') }}-${{ - hashFiles('.pre-commit-config.yaml') }} - - name: Fail job if Python cache restore failed - if: steps.cache-venv.outputs.cache-hit != 'true' - run: | - echo "Failed to restore Python virtual environment from cache" - exit 1 - - name: Restore pre-commit environment from cache - id: cache-precommit - uses: actions/cache@v2 - with: - path: ${{ env.PRE_COMMIT_HOME }} - key: | - ${{ env.CACHE_VERSION}}-${{ runner.os }}-pre-commit-${{ hashFiles('.pre-commit-config.yaml') }} - - name: Fail job if cache restore failed - if: steps.cache-venv.outputs.cache-hit != 'true' - run: | - echo "Failed to restore Python virtual environment from cache" - exit 1 - - name: Register flake8 problem matcher - run: | - echo "::add-matcher::.github/workflows/matchers/flake8.json" - - name: Run flake8 - run: | - . venv/bin/activate - pre-commit run --hook-stage manual flake8 --all-files - - lint-isort: - name: Check isort - runs-on: ubuntu-latest - needs: pre-commit - steps: - - name: Check out code from GitHub - uses: actions/checkout@v2 - - name: Set up Python ${{ env.DEFAULT_PYTHON }} - uses: actions/setup-python@v2.1.4 - id: python - with: - python-version: ${{ env.DEFAULT_PYTHON }} - - name: Restore base Python virtual environment - id: cache-venv - uses: actions/cache@v2 - with: - path: venv - key: >- - ${{ env.CACHE_VERSION}}-${{ runner.os }}-base-venv-${{ - steps.python.outputs.python-version }}-${{ - hashFiles('setup.py') }}-${{ - hashFiles('requirements_test.txt') }}-${{ - hashFiles('.pre-commit-config.yaml') }} - - name: Fail job if Python cache restore failed - if: steps.cache-venv.outputs.cache-hit != 'true' - run: | - echo "Failed to restore Python virtual environment from cache" - exit 1 - - name: Restore pre-commit environment from cache - id: cache-precommit - uses: actions/cache@v2 - with: - path: ${{ env.PRE_COMMIT_HOME }} - key: | - ${{ env.CACHE_VERSION}}-${{ runner.os }}-pre-commit-${{ hashFiles('.pre-commit-config.yaml') }} - - name: Fail job if cache restore failed - if: steps.cache-venv.outputs.cache-hit != 'true' - run: | - echo "Failed to restore Python virtual environment from cache" - exit 1 - - name: Run isort - run: | - . venv/bin/activate - pre-commit run --hook-stage manual isort --all-files --show-diff-on-failure - - lint-codespell: - name: Check codespell - runs-on: ubuntu-latest - needs: pre-commit - steps: - - name: Check out code from GitHub - uses: actions/checkout@v2 - - name: Set up Python ${{ env.DEFAULT_PYTHON }} - uses: actions/setup-python@v2.1.4 - id: python - with: - python-version: ${{ env.DEFAULT_PYTHON }} - - name: Restore base Python virtual environment - id: cache-venv - uses: actions/cache@v2 - with: - path: venv - key: >- - ${{ env.CACHE_VERSION}}-${{ runner.os }}-base-venv-${{ - steps.python.outputs.python-version }}-${{ - hashFiles('setup.py') }}-${{ - hashFiles('requirements_test.txt') }}-${{ - hashFiles('.pre-commit-config.yaml') }} - - name: Fail job if Python cache restore failed - if: steps.cache-venv.outputs.cache-hit != 'true' - run: | - echo "Failed to restore Python virtual environment from cache" - exit 1 - - name: Restore pre-commit environment from cache - id: cache-precommit - uses: actions/cache@v2 - with: - path: ${{ env.PRE_COMMIT_HOME }} - key: | - ${{ env.CACHE_VERSION}}-${{ runner.os }}-pre-commit-${{ hashFiles('.pre-commit-config.yaml') }} - - name: Fail job if cache restore failed - if: steps.cache-venv.outputs.cache-hit != 'true' - run: | - echo "Failed to restore Python virtual environment from cache" - exit 1 - - name: Register codespell problem matcher - run: | - echo "::add-matcher::.github/workflows/matchers/codespell.json" - - name: Run codespell - run: | - . venv/bin/activate - pre-commit run --hook-stage manual codespell --all-files --show-diff-on-failure - - pytest: - runs-on: ubuntu-latest - needs: prepare-base - strategy: - matrix: - python-version: [3.8, 3.9, "3.10", "3.11"] - name: >- - Run tests Python ${{ matrix.python-version }} - steps: - - name: Check out code from GitHub - uses: actions/checkout@v2 - - name: Set up Python ${{ matrix.python-version }} - uses: actions/setup-python@v2.1.4 - id: python - with: - python-version: ${{ matrix.python-version }} - - name: Restore base Python virtual environment - id: cache-venv - uses: actions/cache@v2 - with: - path: venv - key: >- - ${{ env.CACHE_VERSION}}-${{ runner.os }}-base-venv-${{ - steps.python.outputs.python-version }}-${{ - hashFiles('setup.py') }}-${{ - hashFiles('requirements_test.txt') }}-${{ - hashFiles('.pre-commit-config.yaml') }} - - name: Fail job if Python cache restore failed - if: steps.cache-venv.outputs.cache-hit != 'true' - run: | - echo "Failed to restore Python virtual environment from cache" - exit 1 - - name: Register Python problem matcher - run: | - echo "::add-matcher::.github/workflows/matchers/python.json" - - name: Install Pytest Annotation plugin - run: | - . venv/bin/activate - # Ideally this should be part of our dependencies - # However this plugin is fairly new and doesn't run correctly - # on a non-GitHub environment. - pip install pytest-github-actions-annotate-failures - - name: Run pytest - run: | - . venv/bin/activate - pytest \ - -qq \ - --timeout=9 \ - --durations=10 \ - --cov bellows \ - -o console_output_style=count \ - -p no:sugar \ - tests - - name: Upload coverage artifact - uses: actions/upload-artifact@v2.2.0 - with: - name: coverage-${{ matrix.python-version }} - path: .coverage - - name: Coveralls - env: - GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} - COVERALLS_FLAG_NAME: ${{ matrix.python-version }} - COVERALLS_PARALLEL: true - run: | - . venv/bin/activate - coveralls - - - coverage: - name: Process test coverage - runs-on: ubuntu-latest - needs: pytest - steps: - - name: Check out code from GitHub - uses: actions/checkout@v2 - - name: Set up Python ${{ matrix.python-version }} - uses: actions/setup-python@v2.1.4 - id: python - with: - python-version: ${{ env.DEFAULT_PYTHON }} - - name: Restore base Python virtual environment - id: cache-venv - uses: actions/cache@v2 - with: - path: venv - key: >- - ${{ env.CACHE_VERSION}}-${{ runner.os }}-base-venv-${{ - steps.python.outputs.python-version }}-${{ - hashFiles('setup.py') }}-${{ - hashFiles('requirements_test.txt') }}-${{ - hashFiles('.pre-commit-config.yaml') }} - - name: Fail job if Python cache restore failed - if: steps.cache-venv.outputs.cache-hit != 'true' - run: | - echo "Failed to restore Python virtual environment from cache" - exit 1 - - name: Download all coverage artifacts - uses: actions/download-artifact@v2 - - name: Combine coverage results - run: | - . venv/bin/activate - coverage combine coverage*/.coverage* - coverage report --fail-under=99 - coverage xml - - name: Upload coverage to Codecov - uses: codecov/codecov-action@v1 - - name: Upload coverage to Coveralls - env: - GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} - run: | - . venv/bin/activate - coveralls --finish + shared-ci: + uses: zigpy/workflows/.github/workflows/ci.yml@main + with: + CODE_FOLDER: bellows + CACHE_VERSION: 2 + PYTHON_VERSION_DEFAULT: 3.9.15 + PRE_COMMIT_CACHE_PATH: ~/.cache/pre-commit + MINIMUM_COVERAGE_PERCENTAGE: 99 \ No newline at end of file diff --git a/.github/workflows/matchers/ruff.json b/.github/workflows/matchers/ruff.json new file mode 100644 index 00000000..4411fc5e --- /dev/null +++ b/.github/workflows/matchers/ruff.json @@ -0,0 +1,30 @@ +{ + "problemMatcher": [ + { + "owner": "ruff-error", + "severity": "error", + "pattern": [ + { + "regexp": "^(.*):(\\d+):(\\d+):\\s([EF]\\d{3}\\s.*)$", + "file": 1, + "line": 2, + "column": 3, + "message": 4 + } + ] + }, + { + "owner": "ruff-warning", + "severity": "warning", + "pattern": [ + { + "regexp": "^(.*):(\\d+):(\\d+):\\s([CDNW]\\d{3}\\s.*)$", + "file": 1, + "line": 2, + "column": 3, + "message": 4 + } + ] + } + ] + } \ No newline at end of file diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml index 201b5d80..38a1a2d4 100644 --- a/.pre-commit-config.yaml +++ b/.pre-commit-config.yaml @@ -1,19 +1,32 @@ repos: + - repo: https://github.com/asottile/pyupgrade + rev: v3.3.1 + hooks: + - id: pyupgrade + args: [--py38-plus] + + - repo: https://github.com/PyCQA/autoflake + rev: v2.0.2 + hooks: + - id: autoflake + - repo: https://github.com/psf/black - rev: 23.1.0 + rev: 23.3.0 hooks: - id: black args: - - --safe - --quiet + - repo: https://github.com/pycqa/flake8 rev: 6.0.0 hooks: - id: flake8 + - repo: https://github.com/PyCQA/isort rev: 5.12.0 hooks: - id: isort + - repo: https://github.com/codespell-project/codespell rev: v2.2.4 hooks: @@ -22,3 +35,17 @@ repos: - --ignore-words-list=zigpy,hass - --skip="./.*" - --quiet-level=2 + + - repo: https://github.com/pre-commit/mirrors-mypy + rev: v1.2.0 + hooks: + - id: mypy + additional_dependencies: + - zigpy + + - repo: https://github.com/charliermarsh/ruff-pre-commit + rev: v0.0.261 + hooks: + - id: ruff + args: + - --fix diff --git a/.travis.yml b/.travis.yml deleted file mode 100644 index e3e1757f..00000000 --- a/.travis.yml +++ /dev/null @@ -1,19 +0,0 @@ -language: python -dist: bionic - -matrix: - fast_finish: true - include: - - python: "3.7" - env: TOXENV=lint - - python: "3.7" - env: TOXENV=black - - python: "3.7" - env: TOXENV=py37 - - python: "3.8" - env: TOXENV=py38 - - python: "3.8.6" - env: TOXENV=py38 -install: pip install -U setuptools tox coveralls -script: tox -after_success: coveralls diff --git a/bellows/cli/application.py b/bellows/cli/application.py index 36d85f74..9d175df6 100644 --- a/bellows/cli/application.py +++ b/bellows/cli/application.py @@ -57,7 +57,7 @@ async def inner(ctx): app = ctx.obj["app"] await app.permit(duration_s) - click.echo("Joins are permitted for the next %ss..." % (duration_s,)) + click.echo(f"Joins are permitted for the next {duration_s}s...") await asyncio.sleep(duration_s + 1) click.echo("Done") @@ -80,7 +80,7 @@ async def inner(ctx): try: await app.permit_with_key(node, code, duration_s) - click.echo("Joins are permitted for the next %ss..." % (duration_s,)) + click.echo(f"Joins are permitted for the next {duration_s}s...") await asyncio.sleep(duration_s + 1) click.echo("Done") except Exception as e: @@ -96,11 +96,11 @@ def devices(ctx, database): """Show device database""" def print_clusters(title, clusters): - clusters = sorted(list(clusters.items())) + clusters = sorted(clusters.items()) if clusters: - click.echo(" %s:" % (title,)) + click.echo(f" {title}:") for cluster_id, cluster in clusters: - click.echo(" %s (%s)" % (cluster.name, cluster_id)) + click.echo(f" {cluster.name} ({cluster_id})") loop = asyncio.get_event_loop() config = { @@ -113,8 +113,8 @@ def print_clusters(title, clusters): ) for ieee, dev in app.devices.items(): click.echo("Device:") - click.echo(" NWK: 0x%04x" % (dev.nwk,)) - click.echo(" IEEE: %s" % (ieee,)) + click.echo(f" NWK: 0x{dev.nwk:04x}") + click.echo(f" IEEE: {ieee}") click.echo(" Endpoints:") for epid, ep in dev.endpoints.items(): if epid == 0: @@ -155,7 +155,7 @@ async def endpoints(ctx): try: v = await dev.zdo.request(0x0005, dev.nwk) if v[0] != t.EmberStatus.SUCCESS: - click.echo("Non-success response: %s" % (v,)) + click.echo(f"Non-success response: {v}") else: click.echo(v[2]) except zigpy.exceptions.ZigbeeException as e: @@ -178,7 +178,7 @@ async def get_endpoint(ctx, endpoint): try: v = await dev.zdo.request(0x0004, dev.nwk, endpoint) if v[0] != t.EmberStatus.SUCCESS: - click.echo("Non-success response: %s" % (v,)) + click.echo(f"Non-success response: {v}") else: click.echo(v[2]) except zigpy.exceptions.ZigbeeException as e: @@ -278,10 +278,12 @@ async def read_attribute(ctx, attribute, manufacturer): click.echo("Received empty response") elif attribute not in v[0]: click.echo( - "Attribute %s not successful. Status=%s" % (attribute, v[1][attribute]) + "Attribute {} not successful. Status={}".format( + attribute, v[1][attribute] + ) ) else: - click.echo("%s=%s" % (attribute, v[0][attribute])) + click.echo(f"{attribute}={v[0][attribute]}") except zigpy.exceptions.ZigbeeException as e: click.echo(e) diff --git a/bellows/cli/backup.py b/bellows/cli/backup.py index daad2330..8f3c5af2 100644 --- a/bellows/cli/backup.py +++ b/bellows/cli/backup.py @@ -167,7 +167,7 @@ async def restore( """Backup NCP config to stdio.""" click.echo("Restoring NCP") try: - with open(backup_file, "r") as file: + with open(backup_file) as file: backup_data = json.load(file) LOGGER.debug("loaded: %s", backup_data) backup_data = SCHEMA_BAK(backup_data) diff --git a/bellows/cli/dump.py b/bellows/cli/dump.py index c315c4d5..65c6d67f 100644 --- a/bellows/cli/dump.py +++ b/bellows/cli/dump.py @@ -33,9 +33,7 @@ def dump(ctx, channel, outfile): start_time = ctx.obj.get("start_time", None) if start_time: duration = time.time() - start_time - click.echo( - "\nCaptured %s frames in %0.2fs" % (captured, duration), err=True - ) + click.echo(f"\nCaptured {captured} frames in {duration:0.2f}s", err=True) finally: if "ezsp" in ctx.obj: loop.run_until_complete(ctx.obj["ezsp"].mfglibEnd()) diff --git a/bellows/cli/ncp.py b/bellows/cli/ncp.py index 9cc1509f..c4c458ee 100644 --- a/bellows/cli/ncp.py +++ b/bellows/cli/ncp.py @@ -29,7 +29,7 @@ async def config(ctx, config, all_): v = await s.getConfigurationValue(config) if v[0] == t.EzspStatus.ERROR_INVALID_ID: continue - click.echo("%s=%s" % (config.name, v[1])) + click.echo(f"{config.name}={v[1]}") s.close() return @@ -39,18 +39,18 @@ async def config(ctx, config, all_): try: config = s.types.EzspConfigId(int(config)) except ValueError: - raise click.BadArgumentUsage("Invalid config ID: %s" % (config,)) + raise click.BadArgumentUsage(f"Invalid config ID: {config}") else: try: config = s.types.EzspConfigId[config] except KeyError: - raise click.BadArgumentUsage("Invalid config name: %s" % (config,)) + raise click.BadArgumentUsage(f"Invalid config name: {config}") try: value = t.uint16_t(value) if not (0 <= value <= 65535): - raise ValueError("%s out of allowed range 0..65535" % (value,)) + raise ValueError(f"{value} out of allowed range 0..65535") except ValueError as e: - raise click.BadArgumentUsage("Invalid value: %s" % (e,)) + raise click.BadArgumentUsage(f"Invalid value: {e}") v = await s.setConfigurationValue(config, value) click.echo(v) @@ -109,10 +109,8 @@ async def bootloader(ctx): return click.echo( - ( - f"bootloader version: 0x{version:04x}, nodePlat: 0x{plat:02x}, " - f"nodeMicro: 0x{micro:02x}, nodePhy: 0x{phy:02x}" - ) + f"bootloader version: 0x{version:04x}, nodePlat: 0x{plat:02x}, " + f"nodeMicro: 0x{micro:02x}, nodePhy: 0x{phy:02x}" ) res = await ezsp.launchStandaloneBootloader(0x00) diff --git a/bellows/cli/network.py b/bellows/cli/network.py index bd371507..f44762a4 100644 --- a/bellows/cli/network.py +++ b/bellows/cli/network.py @@ -63,9 +63,7 @@ def cb(fut, frame_name, response): extended_pan_id = network.extendedPanId channel = network.channel - click.echo( - "Found network %s %s on channel %s" % (pan_id, extended_pan_id, channel) - ) + click.echo(f"Found network {pan_id} {extended_pan_id} on channel {channel}") if pan_id is None: pan_id = t.uint16_t(0) @@ -81,12 +79,12 @@ def cb(fut, frame_name, response): if v[0] == t.EmberStatus.SUCCESS: LOGGER.debug("Network was up, leaving...") v = await s.leaveNetwork() - util.check(v[0], "Failure leaving network: %s" % (v[0],)) + util.check(v[0], f"Failure leaving network: {v[0]}") await asyncio.sleep(1) # TODO initial_security_state = zutil.zha_security(SCHEMA_NETWORK({})) v = await s.setInitialSecurityState(initial_security_state) - util.check(v[0], "Setting security state failed: %s" % (v[0],)) + util.check(v[0], f"Setting security state failed: {v[0]}") parameters = t.EmberNetworkParameters() parameters.extendedPanId = extended_pan_id @@ -102,7 +100,7 @@ def cb(fut, frame_name, response): fut = asyncio.Future() cbid = s.add_callback(functools.partial(cb, fut)) v = await s.joinNetwork(t.EmberNodeType.END_DEVICE, parameters) - util.check(v[0], "Joining network failed: %s" % (v[0],)) + util.check(v[0], f"Joining network failed: {v[0]}") v = await fut click.echo(v) @@ -124,7 +122,7 @@ async def leave(ctx): v = await s.leaveNetwork() util.check( v[0], - "Failure leaving network: %s" % (v[0],), + f"Failure leaving network: {v[0]}", expected=t.EmberStatus.NETWORK_DOWN, ) @@ -142,7 +140,7 @@ async def scan(ctx, channels, duration_ms, energy_scan): s = await util.setup(ctx.obj["device"], ctx.obj["baudrate"]) channel_mask = util.channel_mask(channels) - click.echo("Scanning channels %s" % (" ".join(map(str, channels)),)) + click.echo("Scanning channels {}".format(" ".join(map(str, channels)))) # TFM says: # Sets the exponent of the number of scan periods, where a scan period is diff --git a/bellows/cli/util.py b/bellows/cli/util.py index 2ff6942c..eac1d7a8 100644 --- a/bellows/cli/util.py +++ b/bellows/cli/util.py @@ -1,4 +1,5 @@ import asyncio +import contextlib import functools import logging @@ -65,10 +66,8 @@ async def async_inner(ctx, *args, **kwargs): await application.shutdown() def shutdown(): - try: + with contextlib.suppress(Exception): application._ezsp.close() - except: # noqa: E722 - pass @functools.wraps(f) def inner(*args, **kwargs): @@ -88,7 +87,7 @@ def inner(*args, **kwargs): def print_cb(frame_name, response): - click.echo("Callback: %s %s" % (frame_name, response)) + click.echo(f"Callback: {frame_name} {response}") def channel_mask(channels): @@ -121,7 +120,7 @@ async def setup(dev, baudrate, cbh=None, configure=True): async def cfg(config_id, value): v = await s.setConfigurationValue(config_id, value) - check(v[0], "Setting config %s to %s: %s" % (config_id, value, v[0])) + check(v[0], f"Setting config {config_id} to {value}: {v[0]}") c = s.types.EzspConfigId @@ -155,7 +154,7 @@ async def network_init(s): v = await s.networkInit() check( v[0], - "Failure initializing network: %s" % (v[0],), + f"Failure initializing network: {v[0]}", [0, t.EmberStatus.NOT_JOINED], ) return v @@ -170,7 +169,7 @@ def parse_epan(epan): async def basic_tc_permits(s): async def set_policy(policy, decision): v = await s.setPolicy(policy, decision) - check(v[0], "Failed to set policy %s to %s: %s" % (policy, decision, v[0])) + check(v[0], f"Failed to set policy {policy} to {decision}: {v[0]}") await set_policy( s.types.EzspPolicyId.TC_KEY_REQUEST_POLICY, @@ -188,7 +187,7 @@ async def set_policy(policy, decision): def get_device(app, node): if node not in app.devices: - click.echo("Device %s is not in the device database" % (node,)) + click.echo(f"Device {node} is not in the device database") return None return app.devices[node] diff --git a/bellows/ezsp/__init__.py b/bellows/ezsp/__init__.py index 3ed1d874..0e5a36a9 100644 --- a/bellows/ezsp/__init__.py +++ b/bellows/ezsp/__init__.py @@ -6,7 +6,7 @@ import functools import logging import sys -from typing import Any, Callable, Dict, List, Tuple, Union +from typing import Any, Callable import urllib.parse if sys.version_info[:2] < (3, 11): @@ -47,7 +47,7 @@ class EZSP: v11.EZSP_VERSION: v11.EZSPv11, } - def __init__(self, device_config: Dict): + def __init__(self, device_config: dict): self._config = device_config self._callbacks = {} self._ezsp_event = asyncio.Event() @@ -56,7 +56,7 @@ def __init__(self, device_config: Dict): self._protocol = None @classmethod - async def probe(cls, device_config: Dict) -> bool | dict[str, int | str | bool]: + async def probe(cls, device_config: dict) -> bool | dict[str, int | str | bool]: """Probe port for the device presence.""" for config in ( {**device_config, conf.CONF_DEVICE_BAUDRATE: 115200}, @@ -103,7 +103,7 @@ async def _startup_reset(self): await self.reset() @classmethod - async def initialize(cls, zigpy_config: Dict) -> "EZSP": + async def initialize(cls, zigpy_config: dict) -> EZSP: """Return initialized EZSP instance.""" ezsp = cls(zigpy_config[conf.CONF_DEVICE]) await ezsp.connect(use_thread=zigpy_config[conf.CONF_USE_THREAD]) @@ -163,7 +163,7 @@ def close(self): self._gw.close() self._gw = None - def _command(self, name: str, *args: Tuple[Any, ...]) -> asyncio.Future: + def _command(self, name: str, *args: tuple[Any, ...]) -> asyncio.Future: if not self.is_ezsp_running: LOGGER.debug( "Couldn't send command %s(%s). EZSP is not running", name, args @@ -221,13 +221,11 @@ def cb(frame_name, response): 0, ) - async def leaveNetwork( - self, timeout: Union[float, int] = NETWORK_OPS_TIMEOUT - ) -> List: + async def leaveNetwork(self, timeout: float | int = NETWORK_OPS_TIMEOUT) -> list: """Send leaveNetwork command and wait for stackStatusHandler frame.""" stack_status = asyncio.Future() - def cb(frame_name: str, response: List) -> None: + def cb(frame_name: str, response: list) -> None: if ( frame_name == "stackStatusHandler" and response[0] == t.EmberStatus.NETWORK_DOWN @@ -308,7 +306,7 @@ def frame_received(self, data: bytes) -> None: self._protocol(data) - async def get_board_info(self) -> Tuple[str, str, str]: + async def get_board_info(self) -> tuple[str, str, str]: """Return board info.""" tokens = [] @@ -359,7 +357,7 @@ def remove_callback(self, id_): return self._callbacks.pop(id_) def handle_callback(self, *args): - for callback_id, handler in self._callbacks.items(): + for _callback_id, handler in self._callbacks.items(): try: handler(*args) except Exception as e: diff --git a/bellows/ezsp/v10/types/named.py b/bellows/ezsp/v10/types/named.py index b1e2cb7f..828d89ef 100644 --- a/bellows/ezsp/v10/types/named.py +++ b/bellows/ezsp/v10/types/named.py @@ -675,10 +675,11 @@ class EmberEntropySource(basic.enum8): class EmberDutyCycleHectoPct(basic.uint16_t): - """ "The percent of duty cycle for a limit. + """The percent of duty cycle for a limit. Duty Cycle, Limits, and Thresholds are reported in units of Percent * 100 - (i.e. 10000 = 100.00%, 1 = 0.01%)""" + (i.e. 10000 = 100.00%, 1 = 0.01%) + """ class EmberGpProxyTableEntryStatus(basic.uint8_t): diff --git a/bellows/ezsp/v6/types/named.py b/bellows/ezsp/v6/types/named.py index fac5e0f5..c6ffa969 100644 --- a/bellows/ezsp/v6/types/named.py +++ b/bellows/ezsp/v6/types/named.py @@ -587,10 +587,11 @@ class EmberRadioPowerMode(basic.enum8): class EmberDutyCycleHectoPct(basic.uint16_t): - """ "The percent of duty cycle for a limit. + """The percent of duty cycle for a limit. Duty Cycle, Limits, and Thresholds are reported in units of Percent * 100 - (i.e. 10000 = 100.00%, 1 = 0.01%)""" + (i.e. 10000 = 100.00%, 1 = 0.01%) + """ class EmberGpProxyTableEntryStatus(basic.uint8_t): diff --git a/bellows/ezsp/v7/types/named.py b/bellows/ezsp/v7/types/named.py index 65042a06..00e24e97 100644 --- a/bellows/ezsp/v7/types/named.py +++ b/bellows/ezsp/v7/types/named.py @@ -661,10 +661,11 @@ class EmberEntropySource(basic.enum8): class EmberDutyCycleHectoPct(basic.uint16_t): - """ "The percent of duty cycle for a limit. + """The percent of duty cycle for a limit. Duty Cycle, Limits, and Thresholds are reported in units of Percent * 100 - (i.e. 10000 = 100.00%, 1 = 0.01%)""" + (i.e. 10000 = 100.00%, 1 = 0.01%) + """ class EmberGpProxyTableEntryStatus(basic.uint8_t): diff --git a/bellows/ezsp/v8/types/named.py b/bellows/ezsp/v8/types/named.py index f1104521..9a3a946a 100644 --- a/bellows/ezsp/v8/types/named.py +++ b/bellows/ezsp/v8/types/named.py @@ -666,10 +666,11 @@ class EmberEntropySource(basic.enum8): class EmberDutyCycleHectoPct(basic.uint16_t): - """ "The percent of duty cycle for a limit. + """The percent of duty cycle for a limit. Duty Cycle, Limits, and Thresholds are reported in units of Percent * 100 - (i.e. 10000 = 100.00%, 1 = 0.01%)""" + (i.e. 10000 = 100.00%, 1 = 0.01%) + """ class EmberGpProxyTableEntryStatus(basic.uint8_t): diff --git a/bellows/ezsp/v9/types/named.py b/bellows/ezsp/v9/types/named.py index 2a95b49e..e727ca7d 100644 --- a/bellows/ezsp/v9/types/named.py +++ b/bellows/ezsp/v9/types/named.py @@ -674,10 +674,11 @@ class EmberEntropySource(basic.enum8): class EmberDutyCycleHectoPct(basic.uint16_t): - """ "The percent of duty cycle for a limit. + """The percent of duty cycle for a limit. Duty Cycle, Limits, and Thresholds are reported in units of Percent * 100 - (i.e. 10000 = 100.00%, 1 = 0.01%)""" + (i.e. 10000 = 100.00%, 1 = 0.01%) + """ class EmberGpProxyTableEntryStatus(basic.uint8_t): diff --git a/bellows/types/basic.py b/bellows/types/basic.py index e3e890c8..f3d6e722 100644 --- a/bellows/types/basic.py +++ b/bellows/types/basic.py @@ -123,7 +123,7 @@ def serialize(self): def deserialize(cls, data): r = cls() length, data = data[0], data[1:] - for i in range(length): + for _i in range(length): item, data = r._itemtype.deserialize(data) r.append(item) return r, data @@ -147,7 +147,7 @@ class _FixedList(_List): @classmethod def deserialize(cls, data): r = cls() - for i in range(r._length): + for _i in range(r._length): item, data = r._itemtype.deserialize(data) r.append(item) return r, data diff --git a/bellows/zigbee/application.py b/bellows/zigbee/application.py index 7afba46d..580e2cad 100644 --- a/bellows/zigbee/application.py +++ b/bellows/zigbee/application.py @@ -5,7 +5,6 @@ import os import statistics import sys -from typing import Dict, Optional if sys.version_info[:2] < (3, 11): from async_timeout import timeout as asyncio_timeout # pragma: no cover @@ -69,7 +68,7 @@ class ControllerApplication(zigpy.application.ControllerApplication): SCHEMA = CONFIG_SCHEMA SCHEMA_DEVICE = SCHEMA_DEVICE - def __init__(self, config: Dict): + def __init__(self, config: dict): super().__init__(config) self._ctrl_event = asyncio.Event() self._ezsp = None @@ -140,8 +139,7 @@ async def connect(self): LOGGER.info("EmberZNet version: %s", version) async def _ensure_network_running(self) -> bool: - """ - Ensures the network is currently running and returns whether or not the network + """Ensures the network is currently running and returns whether or not the network was started. """ (state,) = await self._ezsp.networkState() @@ -606,7 +604,7 @@ def _handle_tc_join_handler( return if device_update_status == EmberDeviceUpdate.STANDARD_SECURITY_UNSECURED_JOIN: - asyncio.create_task(self.cleanup_tc_link_key(ieee)) + self.create_task(self.cleanup_tc_link_key(ieee), "cleanup_tc_link_key") if decision == t.EmberJoinDecision.DENY_JOIN: # no point in handling the join if it was denied @@ -832,7 +830,7 @@ async def send_packet(self, packet: zigpy.types.ZigbeePacket) -> None: async def permit(self, time_s: int = 60, node: t.EmberNodeId = None) -> None: """Permit joining.""" - asyncio.create_task(self._ezsp.pre_permit(time_s)) + self.create_task(self._ezsp.pre_permit(time_s), "pre_permit") await super().permit(time_s, node) def permit_ncp(self, time_s=60): @@ -933,7 +931,7 @@ async def _watchdog(self): self.state.counters[COUNTERS_CTRL][COUNTER_WATCHDOG].increment() self._handle_reset_request(f"Watchdog timeout. Heartbeat timeouts: {failures}") - async def _get_free_buffers(self) -> Optional[int]: + async def _get_free_buffers(self) -> int | None: status, value = await self._ezsp.getValue( self._ezsp.types.EzspValueId.VALUE_FREE_BUFFERS ) diff --git a/bellows/zigbee/device.py b/bellows/zigbee/device.py index 69952488..1adbc307 100644 --- a/bellows/zigbee/device.py +++ b/bellows/zigbee/device.py @@ -62,9 +62,7 @@ def app(self) -> zigpy.application.ControllerApplication: return self.device.application def make_zdo_reply(self, cmd: zdo_t.ZDOCmd, **kwargs): - """ - Provides a way to create ZDO commands with schemas. Currently does nothing. - """ + """Provides a way to create ZDO commands with schemas. Currently does nothing.""" return list(kwargs.values()) diff --git a/requirements_test.txt b/requirements_test.txt index 6b9647b0..5fbb4e49 100644 --- a/requirements_test.txt +++ b/requirements_test.txt @@ -1,9 +1,19 @@ # Test dependencies. asynctest -coveralls==2.1.2 +isort +black +flake8 +codecov +colorlog +codespell +mypy==1.2.0 pre-commit -pytest -pytest-asyncio>=0.17 +pylint pytest-cov +pytest-sugar pytest-timeout +pytest-asyncio>=0.17 +pytest>=7.1.3 +zigpy>=0.54.1 +ruff==0.0.261 \ No newline at end of file diff --git a/ruff.toml b/ruff.toml new file mode 100644 index 00000000..9c27612c --- /dev/null +++ b/ruff.toml @@ -0,0 +1,71 @@ +target-version = "py38" + +select = [ + "B007", # Loop control variable {name} not used within loop body + "B014", # Exception handler with duplicate exception + "C", # complexity + "D", # docstrings + "E", # pycodestyle + "F", # pyflakes/autoflake + "ICN001", # import concentions; {name} should be imported as {asname} + "PGH004", # Use specific rule codes when using noqa + "PLC0414", # Useless import alias. Import alias does not rename original package. + "SIM105", # Use contextlib.suppress({exception}) instead of try-except-pass + "SIM117", # Merge with-statements that use the same scope + "SIM118", # Use {key} in {dict} instead of {key} in {dict}.keys() + "SIM201", # Use {left} != {right} instead of not {left} == {right} + "SIM212", # Use {a} if {a} else {b} instead of {b} if not {a} else {a} + "SIM300", # Yoda conditions. Use 'age == 42' instead of '42 == age'. + "SIM401", # Use get from dict with default instead of an if block + "T20", # flake8-print + "TRY004", # Prefer TypeError exception for invalid type + "RUF006", # Store a reference to the return value of asyncio.create_task + "UP", # pyupgrade + "W", # pycodestyle +] + +ignore = [ + "D100", # Missing docstring in public module + "D101", # Missing docstring in public class + "D102", # Missing docstring in public method + "D103", # Missing docstring in public function + "D104", # Missing docstring in public package + "D105", # Missing docstring in magic method + "D106", # Missing docstring in public nested class + "D107", # Missing docstring in `__init__` + "D202", # No blank lines allowed after function docstring + "D203", # 1 blank line required before class docstring + "D205", # 1 blank line required between summary line and description + "D213", # Multi-line docstring summary should start at the second line + "D400", # First line should end with a period + "D401", # First line of docstring should be in imperative mood: + "D406", # Section name should end with a newline + "D407", # Section name underlining + "D415", # First line should end with a period, question mark, or exclamation point + "E501", # line too long + # the rules below this line should be corrected + "PGH004", # Use specific rule codes when using `noqa` +] + +extend-exclude = [ + "tests" +] + +[flake8-pytest-style] +fixture-parentheses = false + +[pyupgrade] +keep-runtime-typing = true + +[isort] +# will group `import x` and `from x import` of the same module. +force-sort-within-sections = true +known-first-party = [ + "bellows", + "tests", +] +forced-separate = ["tests"] +combine-as-imports = true + +[mccabe] +max-complexity = 25 \ No newline at end of file diff --git a/script/setup b/script/setup new file mode 100755 index 00000000..3531e5bb --- /dev/null +++ b/script/setup @@ -0,0 +1,15 @@ +#!/usr/bin/env bash +# Setup the repository. + +# Stop on errors +set -e + +cd "$(dirname "$0")/.." + +python3 -m venv venv +source venv/bin/activate + +pip install -r requirements_test.txt +pre-commit install + +python3 -m pip install -e . diff --git a/setup.cfg b/setup.cfg index 51ea8f48..a19f6c80 100644 --- a/setup.cfg +++ b/setup.cfg @@ -25,3 +25,6 @@ combine_as_imports = true [tool:pytest] asyncio_mode = auto + +[mypy] +ignore_errors = True \ No newline at end of file diff --git a/setup.py b/setup.py index 90989abb..030acf6d 100644 --- a/setup.py +++ b/setup.py @@ -19,7 +19,7 @@ "click-log>=0.2.1", "pure_pcapy3==1.0.1", "voluptuous", - "zigpy>=0.54.0", + "zigpy>=0.54.1", 'async-timeout; python_version<"3.11"', ], dependency_links=[ diff --git a/tests/test_application_device.py b/tests/test_application_device.py index 7814cdae..37f6e3d6 100644 --- a/tests/test_application_device.py +++ b/tests/test_application_device.py @@ -1,5 +1,3 @@ -import asyncio - import pytest import zigpy.endpoint import zigpy.types as t diff --git a/tests/test_application_network_state.py b/tests/test_application_network_state.py index 8cd050cb..108df3b7 100644 --- a/tests/test_application_network_state.py +++ b/tests/test_application_network_state.py @@ -1,5 +1,3 @@ -import logging - import pytest import zigpy.state import zigpy.types as zigpy_t From 9b1111ac06afe2978dfb90b8377f117cfc560911 Mon Sep 17 00:00:00 2001 From: puddly <32534428+puddly@users.noreply.github.com> Date: Tue, 11 Apr 2023 20:44:19 -0400 Subject: [PATCH 7/7] 0.35.1 version bump --- bellows/__init__.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/bellows/__init__.py b/bellows/__init__.py index 80d1a083..0315dc82 100644 --- a/bellows/__init__.py +++ b/bellows/__init__.py @@ -1,5 +1,5 @@ MAJOR_VERSION = 0 -MINOR_VERSION = 36 -PATCH_VERSION = "0.dev0" +MINOR_VERSION = 35 +PATCH_VERSION = "1" __short_version__ = f"{MAJOR_VERSION}.{MINOR_VERSION}" __version__ = f"{__short_version__}.{PATCH_VERSION}"