Only log one warning per unknown clean error code and status (#1462)
Some checks are pending
CI / Perform linting checks (3.13) (push) Waiting to run
CI / Python ${{ matrix.python-version}} on ${{ matrix.os }}${{ fromJSON('[" (extras)", ""]')[matrix.extras == ''] }} (false, macos-latest, 3.11) (push) Blocked by required conditions
CI / Python ${{ matrix.python-version}} on ${{ matrix.os }}${{ fromJSON('[" (extras)", ""]')[matrix.extras == ''] }} (false, macos-latest, 3.12) (push) Blocked by required conditions
CI / Python ${{ matrix.python-version}} on ${{ matrix.os }}${{ fromJSON('[" (extras)", ""]')[matrix.extras == ''] }} (false, macos-latest, 3.13) (push) Blocked by required conditions
CI / Python ${{ matrix.python-version}} on ${{ matrix.os }}${{ fromJSON('[" (extras)", ""]')[matrix.extras == ''] }} (false, ubuntu-latest, 3.11) (push) Blocked by required conditions
CI / Python ${{ matrix.python-version}} on ${{ matrix.os }}${{ fromJSON('[" (extras)", ""]')[matrix.extras == ''] }} (false, ubuntu-latest, 3.12) (push) Blocked by required conditions
CI / Python ${{ matrix.python-version}} on ${{ matrix.os }}${{ fromJSON('[" (extras)", ""]')[matrix.extras == ''] }} (false, ubuntu-latest, 3.13) (push) Blocked by required conditions
CI / Python ${{ matrix.python-version}} on ${{ matrix.os }}${{ fromJSON('[" (extras)", ""]')[matrix.extras == ''] }} (false, windows-latest, 3.11) (push) Blocked by required conditions
CI / Python ${{ matrix.python-version}} on ${{ matrix.os }}${{ fromJSON('[" (extras)", ""]')[matrix.extras == ''] }} (false, windows-latest, 3.12) (push) Blocked by required conditions
CI / Python ${{ matrix.python-version}} on ${{ matrix.os }}${{ fromJSON('[" (extras)", ""]')[matrix.extras == ''] }} (false, windows-latest, 3.13) (push) Blocked by required conditions
CI / Python ${{ matrix.python-version}} on ${{ matrix.os }}${{ fromJSON('[" (extras)", ""]')[matrix.extras == ''] }} (true, ubuntu-latest, 3.11) (push) Blocked by required conditions
CI / Python ${{ matrix.python-version}} on ${{ matrix.os }}${{ fromJSON('[" (extras)", ""]')[matrix.extras == ''] }} (true, ubuntu-latest, 3.12) (push) Blocked by required conditions
CI / Python ${{ matrix.python-version}} on ${{ matrix.os }}${{ fromJSON('[" (extras)", ""]')[matrix.extras == ''] }} (true, ubuntu-latest, 3.13) (push) Blocked by required conditions
CodeQL checks / Analyze (python) (push) Waiting to run

Co-authored-by: Steven B. <51370195+sdb9696@users.noreply.github.com>
This commit is contained in:
Teemu R. 2025-01-22 17:58:04 +01:00 committed by GitHub
parent 7b1b14d1e6
commit 307173487a
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
2 changed files with 70 additions and 13 deletions

View File

@ -37,6 +37,7 @@ class ErrorCode(IntEnum):
SideBrushStuck = 2 SideBrushStuck = 2
MainBrushStuck = 3 MainBrushStuck = 3
WheelBlocked = 4 WheelBlocked = 4
Trapped = 6
DustBinRemoved = 14 DustBinRemoved = 14
UnableToMove = 15 UnableToMove = 15
LidarBlocked = 16 LidarBlocked = 16
@ -79,6 +80,8 @@ class Clean(SmartModule):
REQUIRED_COMPONENT = "clean" REQUIRED_COMPONENT = "clean"
_error_code = ErrorCode.Ok _error_code = ErrorCode.Ok
_logged_error_code_warnings: set | None = None
_logged_status_code_warnings: set
def _initialize_features(self) -> None: def _initialize_features(self) -> None:
"""Initialize features.""" """Initialize features."""
@ -229,12 +232,17 @@ class Clean(SmartModule):
async def _post_update_hook(self) -> None: async def _post_update_hook(self) -> None:
"""Set error code after update.""" """Set error code after update."""
if self._logged_error_code_warnings is None:
self._logged_error_code_warnings = set()
self._logged_status_code_warnings = set()
errors = self._vac_status.get("err_status") errors = self._vac_status.get("err_status")
if errors is None or not errors: if errors is None or not errors:
self._error_code = ErrorCode.Ok self._error_code = ErrorCode.Ok
return return
if len(errors) > 1: if len(errors) > 1 and "multiple" not in self._logged_error_code_warnings:
self._logged_error_code_warnings.add("multiple")
_LOGGER.warning( _LOGGER.warning(
"Multiple error codes, using the first one only: %s", errors "Multiple error codes, using the first one only: %s", errors
) )
@ -243,10 +251,13 @@ class Clean(SmartModule):
try: try:
self._error_code = ErrorCode(error) self._error_code = ErrorCode(error)
except ValueError: except ValueError:
_LOGGER.warning( if error not in self._logged_error_code_warnings:
"Unknown error code, please create an issue describing the error: %s", self._logged_error_code_warnings.add(error)
error, _LOGGER.warning(
) "Unknown error code, please create an issue "
"describing the error: %s",
error,
)
self._error_code = ErrorCode.UnknownInternal self._error_code = ErrorCode.UnknownInternal
def query(self) -> dict: def query(self) -> dict:
@ -360,7 +371,11 @@ class Clean(SmartModule):
try: try:
return Status(status_code) return Status(status_code)
except ValueError: except ValueError:
_LOGGER.warning("Got unknown status code: %s (%s)", status_code, self.data) if status_code not in self._logged_status_code_warnings:
self._logged_status_code_warnings.add(status_code)
_LOGGER.warning(
"Got unknown status code: %s (%s)", status_code, self.data
)
return Status.UnknownInternal return Status.UnknownInternal
@property @property

View File

@ -104,21 +104,39 @@ async def test_actions(
@pytest.mark.parametrize( @pytest.mark.parametrize(
("err_status", "error"), ("err_status", "error", "warning_msg"),
[ [
pytest.param([], ErrorCode.Ok, id="empty error"), pytest.param([], ErrorCode.Ok, None, id="empty error"),
pytest.param([0], ErrorCode.Ok, id="no error"), pytest.param([0], ErrorCode.Ok, None, id="no error"),
pytest.param([3], ErrorCode.MainBrushStuck, id="known error"), pytest.param([3], ErrorCode.MainBrushStuck, None, id="known error"),
pytest.param([123], ErrorCode.UnknownInternal, id="unknown error"), pytest.param(
pytest.param([3, 4], ErrorCode.MainBrushStuck, id="multi-error"), [123],
ErrorCode.UnknownInternal,
"Unknown error code, please create an issue describing the error: 123",
id="unknown error",
),
pytest.param(
[3, 4],
ErrorCode.MainBrushStuck,
"Multiple error codes, using the first one only: [3, 4]",
id="multi-error",
),
], ],
) )
@clean @clean
async def test_post_update_hook(dev: SmartDevice, err_status: list, error: ErrorCode): async def test_post_update_hook(
dev: SmartDevice,
err_status: list,
error: ErrorCode,
warning_msg: str | None,
caplog: pytest.LogCaptureFixture,
):
"""Test that post update hook sets error states correctly.""" """Test that post update hook sets error states correctly."""
clean = next(get_parent_and_child_modules(dev, Module.Clean)) clean = next(get_parent_and_child_modules(dev, Module.Clean))
assert clean assert clean
caplog.set_level(logging.DEBUG)
# _post_update_hook will pop an item off the status list so create a copy. # _post_update_hook will pop an item off the status list so create a copy.
err_status = [e for e in err_status] err_status = [e for e in err_status]
clean.data["getVacStatus"]["err_status"] = err_status clean.data["getVacStatus"]["err_status"] = err_status
@ -130,6 +148,16 @@ async def test_post_update_hook(dev: SmartDevice, err_status: list, error: Error
if error is not ErrorCode.Ok: if error is not ErrorCode.Ok:
assert clean.status is Status.Error assert clean.status is Status.Error
if warning_msg:
assert warning_msg in caplog.text
# Check doesn't log twice
caplog.clear()
await clean._post_update_hook()
if warning_msg:
assert warning_msg not in caplog.text
@clean @clean
async def test_resume(dev: SmartDevice, mocker: MockerFixture): async def test_resume(dev: SmartDevice, mocker: MockerFixture):
@ -164,6 +192,20 @@ async def test_unknown_status(
assert clean.status is Status.UnknownInternal assert clean.status is Status.UnknownInternal
assert "Got unknown status code: 123" in caplog.text assert "Got unknown status code: 123" in caplog.text
# Check only logs once
caplog.clear()
assert clean.status is Status.UnknownInternal
assert "Got unknown status code: 123" not in caplog.text
# Check logs again for other errors
caplog.clear()
clean.data["getVacStatus"]["status"] = 123456
assert clean.status is Status.UnknownInternal
assert "Got unknown status code: 123456" in caplog.text
@clean @clean
@pytest.mark.parametrize( @pytest.mark.parametrize(