uiprotect 7.6.0__tar.gz → 7.7.0__tar.gz

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.

Potentially problematic release.


This version of uiprotect might be problematic. Click here for more details.

Files changed (38) hide show
  1. {uiprotect-7.6.0 → uiprotect-7.7.0}/PKG-INFO +1 -1
  2. {uiprotect-7.6.0 → uiprotect-7.7.0}/pyproject.toml +5 -5
  3. {uiprotect-7.6.0 → uiprotect-7.7.0}/src/uiprotect/api.py +4 -0
  4. {uiprotect-7.6.0 → uiprotect-7.7.0}/src/uiprotect/data/devices.py +34 -0
  5. {uiprotect-7.6.0 → uiprotect-7.7.0}/src/uiprotect/data/nvr.py +5 -0
  6. {uiprotect-7.6.0 → uiprotect-7.7.0}/LICENSE +0 -0
  7. {uiprotect-7.6.0 → uiprotect-7.7.0}/README.md +0 -0
  8. {uiprotect-7.6.0 → uiprotect-7.7.0}/src/uiprotect/__init__.py +0 -0
  9. {uiprotect-7.6.0 → uiprotect-7.7.0}/src/uiprotect/__main__.py +0 -0
  10. {uiprotect-7.6.0 → uiprotect-7.7.0}/src/uiprotect/_compat.py +0 -0
  11. {uiprotect-7.6.0 → uiprotect-7.7.0}/src/uiprotect/cli/__init__.py +0 -0
  12. {uiprotect-7.6.0 → uiprotect-7.7.0}/src/uiprotect/cli/aiports.py +0 -0
  13. {uiprotect-7.6.0 → uiprotect-7.7.0}/src/uiprotect/cli/backup.py +0 -0
  14. {uiprotect-7.6.0 → uiprotect-7.7.0}/src/uiprotect/cli/base.py +0 -0
  15. {uiprotect-7.6.0 → uiprotect-7.7.0}/src/uiprotect/cli/cameras.py +0 -0
  16. {uiprotect-7.6.0 → uiprotect-7.7.0}/src/uiprotect/cli/chimes.py +0 -0
  17. {uiprotect-7.6.0 → uiprotect-7.7.0}/src/uiprotect/cli/doorlocks.py +0 -0
  18. {uiprotect-7.6.0 → uiprotect-7.7.0}/src/uiprotect/cli/events.py +0 -0
  19. {uiprotect-7.6.0 → uiprotect-7.7.0}/src/uiprotect/cli/lights.py +0 -0
  20. {uiprotect-7.6.0 → uiprotect-7.7.0}/src/uiprotect/cli/liveviews.py +0 -0
  21. {uiprotect-7.6.0 → uiprotect-7.7.0}/src/uiprotect/cli/nvr.py +0 -0
  22. {uiprotect-7.6.0 → uiprotect-7.7.0}/src/uiprotect/cli/sensors.py +0 -0
  23. {uiprotect-7.6.0 → uiprotect-7.7.0}/src/uiprotect/cli/viewers.py +0 -0
  24. {uiprotect-7.6.0 → uiprotect-7.7.0}/src/uiprotect/data/__init__.py +0 -0
  25. {uiprotect-7.6.0 → uiprotect-7.7.0}/src/uiprotect/data/base.py +0 -0
  26. {uiprotect-7.6.0 → uiprotect-7.7.0}/src/uiprotect/data/bootstrap.py +0 -0
  27. {uiprotect-7.6.0 → uiprotect-7.7.0}/src/uiprotect/data/convert.py +0 -0
  28. {uiprotect-7.6.0 → uiprotect-7.7.0}/src/uiprotect/data/types.py +0 -0
  29. {uiprotect-7.6.0 → uiprotect-7.7.0}/src/uiprotect/data/user.py +0 -0
  30. {uiprotect-7.6.0 → uiprotect-7.7.0}/src/uiprotect/data/websocket.py +0 -0
  31. {uiprotect-7.6.0 → uiprotect-7.7.0}/src/uiprotect/exceptions.py +0 -0
  32. {uiprotect-7.6.0 → uiprotect-7.7.0}/src/uiprotect/py.typed +0 -0
  33. {uiprotect-7.6.0 → uiprotect-7.7.0}/src/uiprotect/release_cache.json +0 -0
  34. {uiprotect-7.6.0 → uiprotect-7.7.0}/src/uiprotect/stream.py +0 -0
  35. {uiprotect-7.6.0 → uiprotect-7.7.0}/src/uiprotect/test_util/__init__.py +0 -0
  36. {uiprotect-7.6.0 → uiprotect-7.7.0}/src/uiprotect/test_util/anonymize.py +0 -0
  37. {uiprotect-7.6.0 → uiprotect-7.7.0}/src/uiprotect/utils.py +0 -0
  38. {uiprotect-7.6.0 → uiprotect-7.7.0}/src/uiprotect/websocket.py +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.3
2
2
  Name: uiprotect
3
- Version: 7.6.0
3
+ Version: 7.7.0
4
4
  Summary: Python API for Unifi Protect (Unofficial)
5
5
  License: MIT
6
6
  Author: UI Protect Maintainers
@@ -1,6 +1,6 @@
1
1
  [project]
2
2
  name = "uiprotect"
3
- version = "7.6.0"
3
+ version = "7.7.0"
4
4
  license = "MIT"
5
5
  description = "Python API for Unifi Protect (Unofficial)"
6
6
  authors = [{ name = "UI Protect Maintainers", email = "ui@koston.org" }]
@@ -58,10 +58,10 @@ asttokens = ">=2.4.1,<4.0.0"
58
58
  pytest-asyncio = ">=0.23.7,<0.27.0"
59
59
  pytest-benchmark = ">=4,<6"
60
60
  pytest-sugar = "^1.0.0"
61
- pytest-timeout = "^2.3.1"
61
+ pytest-timeout = "^2.4.0"
62
62
  pytest-xdist = "^3.6.1"
63
63
  types-aiofiles = ">=23.2.0.20240403,<25.0.0.0"
64
- types-dateparser = "^1.2.0.20240420"
64
+ types-dateparser = "^1.2.0.20250516"
65
65
  mypy = "^1.10.0"
66
66
 
67
67
  [tool.poetry.group.docs]
@@ -73,7 +73,7 @@ sphinx = { version = ">=4.0", python = ">=3.11"}
73
73
  furo = { version = ">=2023.5.20", python = ">=3.11"}
74
74
  sphinx-autobuild = { version = ">=2024.0.0", python = ">=3.11"}
75
75
  mike = "^2.1.1"
76
- mkdocs-material = "^9.5.26"
76
+ mkdocs-material = "^9.6.14"
77
77
  mkdocs-material-extensions = "^1.3.1"
78
78
  pymdown-extensions = "^10.8.1"
79
79
  mkdocs-git-revision-date-localized-plugin = "^1.2.6"
@@ -177,7 +177,7 @@ select = [
177
177
  "conftest.py" = ["D100"]
178
178
  "docs/conf.py" = ["D100"]
179
179
 
180
- [tool.ruff.isort]
180
+ [tool.ruff.lint.isort]
181
181
  known-first-party = ["uiprotect", "tests"]
182
182
 
183
183
  [tool.mypy]
@@ -1578,6 +1578,7 @@ class ProtectApiClient(BaseApiClient):
1578
1578
  raise_exception=False,
1579
1579
  )
1580
1580
 
1581
+ _LOGGER.debug("Requesting camera video: %s%s %s", self.api_path, path, params)
1581
1582
  r = await self.request(
1582
1583
  "get",
1583
1584
  f"{self.api_path}{path}",
@@ -1585,6 +1586,9 @@ class ProtectApiClient(BaseApiClient):
1585
1586
  timeout=0,
1586
1587
  params=params,
1587
1588
  )
1589
+ if r.status != 200:
1590
+ await self._raise_for_status(r, True)
1591
+
1588
1592
  if output_file is not None:
1589
1593
  async with aiofiles.open(output_file, "wb") as output:
1590
1594
 
@@ -1442,6 +1442,40 @@ class Camera(ProtectMotionDeviceModel):
1442
1442
  """Toggles vehicle smart detection. Requires camera to have smart detection"""
1443
1443
  return await self._set_object_detect(SmartDetectObjectType.VEHICLE, enabled)
1444
1444
 
1445
+ # endregion
1446
+ # region Face
1447
+
1448
+ @property
1449
+ def can_detect_face(self) -> bool:
1450
+ return SmartDetectObjectType.FACE in self.feature_flags.smart_detect_types
1451
+
1452
+ @property
1453
+ def is_face_detection_on(self) -> bool:
1454
+ """Is Face Detection available and enabled?"""
1455
+ return self._is_smart_enabled(SmartDetectObjectType.FACE)
1456
+
1457
+ @property
1458
+ def last_face_detect_event(self) -> Event | None:
1459
+ """Get the last face smart detection event."""
1460
+ return self.get_last_smart_detect_event(SmartDetectObjectType.FACE)
1461
+
1462
+ @property
1463
+ def last_face_detect(self) -> datetime | None:
1464
+ """Get the last face smart detection event."""
1465
+ return self.last_smart_detects.get(SmartDetectObjectType.FACE)
1466
+
1467
+ @property
1468
+ def is_face_currently_detected(self) -> bool:
1469
+ """Is face currently being detected"""
1470
+ return self._is_smart_detected(SmartDetectObjectType.FACE)
1471
+
1472
+ async def set_face_detection(self, enabled: bool) -> None:
1473
+ """Toggles face smart detection. Requires camera to have smart detection"""
1474
+ return await self._set_object_detect(
1475
+ SmartDetectObjectType.FACE,
1476
+ enabled,
1477
+ )
1478
+
1445
1479
  # endregion
1446
1480
  # region License Plate
1447
1481
 
@@ -1361,6 +1361,11 @@ class NVR(ProtectDeviceModel):
1361
1361
  """
1362
1362
  return self._is_smart_enabled(SmartDetectObjectType.VEHICLE)
1363
1363
 
1364
+ @property
1365
+ def is_global_face_detection_on(self) -> bool:
1366
+ """Is Face Detection available and enabled?"""
1367
+ return self._is_smart_enabled(SmartDetectObjectType.FACE)
1368
+
1364
1369
  @property
1365
1370
  def is_global_license_plate_detection_on(self) -> bool:
1366
1371
  """
File without changes
File without changes