uiprotect 1.17.0__tar.gz → 1.18.1__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 (36) hide show
  1. {uiprotect-1.17.0 → uiprotect-1.18.1}/PKG-INFO +1 -1
  2. {uiprotect-1.17.0 → uiprotect-1.18.1}/pyproject.toml +1 -1
  3. {uiprotect-1.17.0 → uiprotect-1.18.1}/src/uiprotect/data/bootstrap.py +35 -7
  4. {uiprotect-1.17.0 → uiprotect-1.18.1}/src/uiprotect/data/devices.py +15 -24
  5. {uiprotect-1.17.0 → uiprotect-1.18.1}/src/uiprotect/data/websocket.py +6 -0
  6. {uiprotect-1.17.0 → uiprotect-1.18.1}/LICENSE +0 -0
  7. {uiprotect-1.17.0 → uiprotect-1.18.1}/README.md +0 -0
  8. {uiprotect-1.17.0 → uiprotect-1.18.1}/src/uiprotect/__init__.py +0 -0
  9. {uiprotect-1.17.0 → uiprotect-1.18.1}/src/uiprotect/__main__.py +0 -0
  10. {uiprotect-1.17.0 → uiprotect-1.18.1}/src/uiprotect/api.py +0 -0
  11. {uiprotect-1.17.0 → uiprotect-1.18.1}/src/uiprotect/cli/__init__.py +0 -0
  12. {uiprotect-1.17.0 → uiprotect-1.18.1}/src/uiprotect/cli/backup.py +0 -0
  13. {uiprotect-1.17.0 → uiprotect-1.18.1}/src/uiprotect/cli/base.py +0 -0
  14. {uiprotect-1.17.0 → uiprotect-1.18.1}/src/uiprotect/cli/cameras.py +0 -0
  15. {uiprotect-1.17.0 → uiprotect-1.18.1}/src/uiprotect/cli/chimes.py +0 -0
  16. {uiprotect-1.17.0 → uiprotect-1.18.1}/src/uiprotect/cli/doorlocks.py +0 -0
  17. {uiprotect-1.17.0 → uiprotect-1.18.1}/src/uiprotect/cli/events.py +0 -0
  18. {uiprotect-1.17.0 → uiprotect-1.18.1}/src/uiprotect/cli/lights.py +0 -0
  19. {uiprotect-1.17.0 → uiprotect-1.18.1}/src/uiprotect/cli/liveviews.py +0 -0
  20. {uiprotect-1.17.0 → uiprotect-1.18.1}/src/uiprotect/cli/nvr.py +0 -0
  21. {uiprotect-1.17.0 → uiprotect-1.18.1}/src/uiprotect/cli/sensors.py +0 -0
  22. {uiprotect-1.17.0 → uiprotect-1.18.1}/src/uiprotect/cli/viewers.py +0 -0
  23. {uiprotect-1.17.0 → uiprotect-1.18.1}/src/uiprotect/data/__init__.py +0 -0
  24. {uiprotect-1.17.0 → uiprotect-1.18.1}/src/uiprotect/data/base.py +0 -0
  25. {uiprotect-1.17.0 → uiprotect-1.18.1}/src/uiprotect/data/convert.py +0 -0
  26. {uiprotect-1.17.0 → uiprotect-1.18.1}/src/uiprotect/data/nvr.py +0 -0
  27. {uiprotect-1.17.0 → uiprotect-1.18.1}/src/uiprotect/data/types.py +0 -0
  28. {uiprotect-1.17.0 → uiprotect-1.18.1}/src/uiprotect/data/user.py +0 -0
  29. {uiprotect-1.17.0 → uiprotect-1.18.1}/src/uiprotect/exceptions.py +0 -0
  30. {uiprotect-1.17.0 → uiprotect-1.18.1}/src/uiprotect/py.typed +0 -0
  31. {uiprotect-1.17.0 → uiprotect-1.18.1}/src/uiprotect/release_cache.json +0 -0
  32. {uiprotect-1.17.0 → uiprotect-1.18.1}/src/uiprotect/stream.py +0 -0
  33. {uiprotect-1.17.0 → uiprotect-1.18.1}/src/uiprotect/test_util/__init__.py +0 -0
  34. {uiprotect-1.17.0 → uiprotect-1.18.1}/src/uiprotect/test_util/anonymize.py +0 -0
  35. {uiprotect-1.17.0 → uiprotect-1.18.1}/src/uiprotect/utils.py +0 -0
  36. {uiprotect-1.17.0 → uiprotect-1.18.1}/src/uiprotect/websocket.py +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: uiprotect
3
- Version: 1.17.0
3
+ Version: 1.18.1
4
4
  Summary: Python API for Unifi Protect (Unofficial)
5
5
  Home-page: https://github.com/uilibs/uiprotect
6
6
  License: MIT
@@ -1,6 +1,6 @@
1
1
  [tool.poetry]
2
2
  name = "uiprotect"
3
- version = "1.17.0"
3
+ version = "1.18.1"
4
4
  description = "Python API for Unifi Protect (Unofficial)"
5
5
  authors = ["UI Protect Maintainers <ui@koston.org>"]
6
6
  license = "MIT"
@@ -66,9 +66,34 @@ STATS_KEYS = {
66
66
  "lastSeen",
67
67
  "recordingSchedules",
68
68
  }
69
+
69
70
  IGNORE_DEVICE_KEYS = {"nvrMac", "guid"}
70
71
  STATS_AND_IGNORE_DEVICE_KEYS = STATS_KEYS | IGNORE_DEVICE_KEYS
71
72
 
73
+ _IGNORE_KEYS_BY_MODEL_TYPE = {
74
+ #
75
+ # `lastMotion` from cameras update every 100 milliseconds when a motion event is active
76
+ # this overrides the behavior to only update `lastMotion` when a new event starts
77
+ #
78
+ ModelType.CAMERA: {"lastMotion"},
79
+ #
80
+ # `cameraIds` is updated every 10s, but we don't need to process it since bootstrap
81
+ # is resynced every so often anyways.
82
+ #
83
+ ModelType.CHIME: {"cameraIds"},
84
+ }
85
+
86
+
87
+ IGNORE_DEVICE_KEYS_BY_MODEL_TYPE = {
88
+ model_type: IGNORE_DEVICE_KEYS | keys
89
+ for model_type, keys in _IGNORE_KEYS_BY_MODEL_TYPE.items()
90
+ }
91
+ STATS_AND_IGNORE_DEVICE_KEYS_BY_MODEL_TYPE = {
92
+ model_type: STATS_AND_IGNORE_DEVICE_KEYS | keys
93
+ for model_type, keys in _IGNORE_KEYS_BY_MODEL_TYPE.items()
94
+ }
95
+
96
+
72
97
  CAMERA_EVENT_ATTR_MAP: dict[EventType, tuple[str, str]] = {
73
98
  EventType.MOTION: ("last_motion", "last_motion_event_id"),
74
99
  EventType.SMART_DETECT: ("last_smart_detect", "last_smart_detect_event_id"),
@@ -453,15 +478,18 @@ class Bootstrap(ProtectBaseObject):
453
478
  that was generated internally as a result of an event
454
479
  that will expire and result in a state change.
455
480
  """
456
- remove_keys = (
457
- STATS_AND_IGNORE_DEVICE_KEYS if ignore_stats else IGNORE_DEVICE_KEYS
458
- )
481
+ if ignore_stats:
482
+ remove_keys = STATS_AND_IGNORE_DEVICE_KEYS_BY_MODEL_TYPE.get(
483
+ model_type, STATS_AND_IGNORE_DEVICE_KEYS
484
+ )
485
+ else:
486
+ remove_keys = IGNORE_DEVICE_KEYS_BY_MODEL_TYPE.get(
487
+ model_type, IGNORE_DEVICE_KEYS
488
+ )
489
+
459
490
  for key in remove_keys.intersection(data):
460
491
  del data[key]
461
- # `last_motion` from cameras update every 100 milliseconds when a motion event is active
462
- # this overrides the behavior to only update `last_motion` when a new event starts
463
- if model_type is ModelType.CAMERA and "lastMotion" in data:
464
- del data["lastMotion"]
492
+
465
493
  # nothing left to process
466
494
  if not data and not is_ping_back:
467
495
  self._create_stat(packet, None, True)
@@ -474,14 +474,9 @@ class SmartDetectSettings(ProtectBaseObject):
474
474
 
475
475
  @classmethod
476
476
  def unifi_dict_to_dict(cls, data: dict[str, Any]) -> dict[str, Any]:
477
- if "objectTypes" in data:
478
- data["objectTypes"] = convert_smart_types(data.pop("objectTypes"))
479
- if "audioTypes" in data:
480
- data["audioTypes"] = convert_smart_audio_types(data.pop("audioTypes"))
481
- if "autoTrackingObjectTypes" in data:
482
- data["autoTrackingObjectTypes"] = convert_smart_types(
483
- data.pop("autoTrackingObjectTypes"),
484
- )
477
+ for key in ("objectTypes", "audioTypes", "autoTrackingObjectTypes"):
478
+ if key in data:
479
+ data[key] = convert_smart_types(data[key])
485
480
 
486
481
  return super().unifi_dict_to_dict(data)
487
482
 
@@ -578,22 +573,18 @@ class VideoStats(ProtectBaseObject):
578
573
 
579
574
  @classmethod
580
575
  def unifi_dict_to_dict(cls, data: dict[str, Any]) -> dict[str, Any]:
581
- if "recordingStart" in data:
582
- data["recordingStart"] = convert_to_datetime(data["recordingStart"])
583
- if "recordingEnd" in data:
584
- data["recordingEnd"] = convert_to_datetime(data["recordingEnd"])
585
- if "recordingStartLQ" in data:
586
- data["recordingStartLQ"] = convert_to_datetime(data["recordingStartLQ"])
587
- if "recordingEndLQ" in data:
588
- data["recordingEndLQ"] = convert_to_datetime(data["recordingEndLQ"])
589
- if "timelapseStart" in data:
590
- data["timelapseStart"] = convert_to_datetime(data["timelapseStart"])
591
- if "timelapseEnd" in data:
592
- data["timelapseEnd"] = convert_to_datetime(data["timelapseEnd"])
593
- if "timelapseStartLQ" in data:
594
- data["timelapseStartLQ"] = convert_to_datetime(data["timelapseStartLQ"])
595
- if "timelapseEndLQ" in data:
596
- data["timelapseEndLQ"] = convert_to_datetime(data["timelapseEndLQ"])
576
+ for key in (
577
+ "recordingStart",
578
+ "recordingEnd",
579
+ "recordingStartLQ",
580
+ "recordingEndLQ",
581
+ "timelapseStart",
582
+ "timelapseEnd",
583
+ "timelapseStartLQ",
584
+ "timelapseEndLQ",
585
+ ):
586
+ if key in data:
587
+ data[key] = convert_to_datetime(data[key])
597
588
 
598
589
  return super().unifi_dict_to_dict(data)
599
590
 
@@ -56,6 +56,9 @@ class BaseWSPacketFrame:
56
56
  is_deflated: bool = False
57
57
  length: int = 0
58
58
 
59
+ def __repr__(self) -> str:
60
+ return f"<{self.__class__.__name__} header={self.header} data={self.data}>"
61
+
59
62
  def set_data_from_binary(self, data: bytes) -> None:
60
63
  self.data = data
61
64
  if self.header is not None and self.header.deflated:
@@ -188,6 +191,9 @@ class WSPacket:
188
191
  def __init__(self, data: bytes) -> None:
189
192
  self._raw = data
190
193
 
194
+ def __repr__(self) -> str:
195
+ return f"<{self.__class__.__name__} action_frame={self.action_frame} data_frame={self.data_frame}>"
196
+
191
197
  def decode(self) -> None:
192
198
  data = self._raw
193
199
  self._action_frame = WSRawPacketFrame.from_binary(data)
File without changes
File without changes