Coverage for src/bluetooth_sig/gatt/descriptors/manufacturer_limits.py: 78%
27 statements
« prev ^ index » next coverage.py v7.11.0, created at 2025-10-30 00:10 +0000
« prev ^ index » next coverage.py v7.11.0, created at 2025-10-30 00:10 +0000
1"""Manufacturer Limits Descriptor implementation."""
3from __future__ import annotations
5import msgspec
7from ..characteristics.utils import DataParser
8from .base import BaseDescriptor
11class ManufacturerLimitsData(msgspec.Struct, frozen=True, kw_only=True):
12 """Manufacturer Limits descriptor data."""
14 min_limit: int | float
15 max_limit: int | float
18class ManufacturerLimitsDescriptor(BaseDescriptor):
19 """Manufacturer Limits Descriptor (0x2913).
21 Defines manufacturer-specified limits for characteristic values.
22 Contains minimum and maximum limits set by the manufacturer.
23 """
25 def _has_structured_data(self) -> bool:
26 return True
28 def _get_data_format(self) -> str:
29 return "struct"
31 def _parse_descriptor_value(self, data: bytes) -> ManufacturerLimitsData:
32 """Parse Manufacturer Limits descriptor value.
34 The format depends on the characteristic's value type.
35 For simplicity, this implementation assumes uint16 min/max values.
37 Args:
38 data: Raw bytes containing min and max limit values
40 Returns:
41 ManufacturerLimitsData with manufacturer limits
43 Raises:
44 ValueError: If data length is incorrect
45 """
46 # Manufacturer Limits format: min_limit + max_limit
47 # For now, assume 4 bytes total (2 bytes each for uint16)
48 if len(data) != 4:
49 raise ValueError(f"Manufacturer Limits data expected 4 bytes, got {len(data)}")
51 min_limit = DataParser.parse_int16(data, offset=0, endian="little")
52 max_limit = DataParser.parse_int16(data, offset=2, endian="little")
54 return ManufacturerLimitsData(
55 min_limit=min_limit,
56 max_limit=max_limit,
57 )
59 def get_min_limit(self, data: bytes) -> int | float:
60 """Get the minimum manufacturer limit."""
61 parsed = self._parse_descriptor_value(data)
62 return parsed.min_limit
64 def get_max_limit(self, data: bytes) -> int | float:
65 """Get the maximum manufacturer limit."""
66 parsed = self._parse_descriptor_value(data)
67 return parsed.max_limit
69 def is_value_within_limits(self, data: bytes, value: int | float) -> bool:
70 """Check if a value is within manufacturer limits."""
71 parsed = self._parse_descriptor_value(data)
72 return parsed.min_limit <= value <= parsed.max_limit