Compare commits
8 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
d8449c14a4 | ||
|
|
3610262855 | ||
|
|
904023e93a | ||
|
|
ceade24851 | ||
|
|
ab486d5519 | ||
|
|
5e17ecdeb2 | ||
|
|
4b90ce0d61 | ||
|
|
50c55dcfd3 |
18
CHANGELOG.md
18
CHANGELOG.md
@@ -1,7 +1,23 @@
|
|||||||
# Changelog
|
# Changelog
|
||||||
|
|
||||||
<!--next-version-placeholder-->
|
<!--next-version-placeholder-->
|
||||||
##2023.11.2 (2023-11-23)
|
## 2024.5.1 (2024-05-13)
|
||||||
|
### Feature
|
||||||
|
* As per HA core patterns, certain sensors will now show as unavailable when sock is charging ([`ceade24`](https://github.com/ryanbdclark/owlet/commit/ceade24851479b8c9bc60b7b8bed74a7bdb927e9))
|
||||||
|
* Oxygen 10 minute average now only shows a figure if it is between 0 and 100 this avoids skewing by 255 values before the 10 minutes is reached, thanks @coreywillwhat ([`5e17ecd`](https://github.com/ryanbdclark/owlet/commit/5e17ecdeb2aca5bbb35f19ca5795a2c5e0f776ab))
|
||||||
|
### Fix
|
||||||
|
* Refactoring as per core maintainers suggestions ([`ceade24`](https://github.com/ryanbdclark/owlet/commit/ceade24851479b8c9bc60b7b8bed74a7bdb927e9))
|
||||||
|
|
||||||
|
|
||||||
|
## 2024.3.1
|
||||||
|
### Feature
|
||||||
|
* Base station on added as binary sensor ([`50c55dc`](https://github.com/ryanbdclark/owlet/commit/50c55dcfd30d15027155a8f1d05340238501522d))
|
||||||
|
|
||||||
|
### Fix
|
||||||
|
* Bumping pyowletapi to 2024.3.2 ([`50c55dc`](https://github.com/ryanbdclark/owlet/commit/50c55dcfd30d15027155a8f1d05340238501522d))
|
||||||
|
* UI config now allows you to set interval to 5 seconds, previously the minimum was 10 ([`50c55dc`](https://github.com/ryanbdclark/owlet/commit/50c55dcfd30d15027155a8f1d05340238501522d))
|
||||||
|
|
||||||
|
## 2023.11.2 (2023-11-23)
|
||||||
### Feature
|
### Feature
|
||||||
* Support added for V2 sock ([`50fe1a8`](https://github.com/ryanbdclark/owlet/commit/50fe1a87656b7d6413d06f06f3650fd0bfb48e02))
|
* Support added for V2 sock ([`50fe1a8`](https://github.com/ryanbdclark/owlet/commit/50fe1a87656b7d6413d06f06f3650fd0bfb48e02))
|
||||||
* Added tests for binary sensors ([`50fe1a8`](https://github.com/ryanbdclark/owlet/commit/50fe1a87656b7d6413d06f06f3650fd0bfb48e02))
|
* Added tests for binary sensors ([`50fe1a8`](https://github.com/ryanbdclark/owlet/commit/50fe1a87656b7d6413d06f06f3650fd0bfb48e02))
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
"""The Owlet Smart Sock integration."""
|
"""The Owlet Smart Sock integration."""
|
||||||
|
|
||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
|
|
||||||
import asyncio
|
import asyncio
|
||||||
@@ -72,15 +73,12 @@ async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
|||||||
entry, data={**entry.data, **devices["tokens"]}
|
entry, data={**entry.data, **devices["tokens"]}
|
||||||
)
|
)
|
||||||
|
|
||||||
socks = {
|
|
||||||
device["device"]["dsn"]: Sock(owlet_api, device["device"])
|
|
||||||
for device in devices["response"]
|
|
||||||
}
|
|
||||||
|
|
||||||
scan_interval = entry.options.get(CONF_SCAN_INTERVAL)
|
scan_interval = entry.options.get(CONF_SCAN_INTERVAL)
|
||||||
coordinators = {
|
coordinators = {
|
||||||
serial: OwletCoordinator(hass, sock, scan_interval, entry)
|
device["device"]["dsn"]: OwletCoordinator(
|
||||||
for (serial, sock) in socks.items()
|
hass, Sock(owlet_api, device["device"]), scan_interval, entry
|
||||||
|
)
|
||||||
|
for device in devices["response"]
|
||||||
}
|
}
|
||||||
|
|
||||||
await asyncio.gather(
|
await asyncio.gather(
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
"""Support for Owlet binary sensors."""
|
"""Support for Owlet binary sensors."""
|
||||||
|
|
||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
|
|
||||||
from dataclasses import dataclass
|
from dataclasses import dataclass
|
||||||
@@ -17,71 +18,85 @@ from .coordinator import OwletCoordinator
|
|||||||
from .entity import OwletBaseEntity
|
from .entity import OwletBaseEntity
|
||||||
|
|
||||||
|
|
||||||
@dataclass
|
@dataclass(kw_only=True)
|
||||||
class OwletBinarySensorEntityDescription(BinarySensorEntityDescription):
|
class OwletBinarySensorEntityDescription(BinarySensorEntityDescription):
|
||||||
"""Represent the owlet binary sensor entity description."""
|
"""Represent the owlet binary sensor entity description."""
|
||||||
|
|
||||||
|
available_during_charging: bool
|
||||||
|
|
||||||
|
|
||||||
SENSORS: tuple[OwletBinarySensorEntityDescription, ...] = (
|
SENSORS: tuple[OwletBinarySensorEntityDescription, ...] = (
|
||||||
OwletBinarySensorEntityDescription(
|
OwletBinarySensorEntityDescription(
|
||||||
key="charging",
|
key="charging",
|
||||||
translation_key="charging",
|
translation_key="charging",
|
||||||
device_class=BinarySensorDeviceClass.BATTERY_CHARGING,
|
device_class=BinarySensorDeviceClass.BATTERY_CHARGING,
|
||||||
|
available_during_charging=True,
|
||||||
),
|
),
|
||||||
OwletBinarySensorEntityDescription(
|
OwletBinarySensorEntityDescription(
|
||||||
key="high_heart_rate_alert",
|
key="high_heart_rate_alert",
|
||||||
translation_key="high_hr_alrt",
|
translation_key="high_hr_alrt",
|
||||||
device_class=BinarySensorDeviceClass.SOUND,
|
device_class=BinarySensorDeviceClass.SOUND,
|
||||||
|
available_during_charging=True,
|
||||||
),
|
),
|
||||||
OwletBinarySensorEntityDescription(
|
OwletBinarySensorEntityDescription(
|
||||||
key="low_heart_rate_alert",
|
key="low_heart_rate_alert",
|
||||||
translation_key="low_hr_alrt",
|
translation_key="low_hr_alrt",
|
||||||
device_class=BinarySensorDeviceClass.SOUND,
|
device_class=BinarySensorDeviceClass.SOUND,
|
||||||
|
available_during_charging=True,
|
||||||
),
|
),
|
||||||
OwletBinarySensorEntityDescription(
|
OwletBinarySensorEntityDescription(
|
||||||
key="high_oxygen_alert",
|
key="high_oxygen_alert",
|
||||||
translation_key="high_ox_alrt",
|
translation_key="high_ox_alrt",
|
||||||
device_class=BinarySensorDeviceClass.SOUND,
|
device_class=BinarySensorDeviceClass.SOUND,
|
||||||
|
available_during_charging=True,
|
||||||
),
|
),
|
||||||
OwletBinarySensorEntityDescription(
|
OwletBinarySensorEntityDescription(
|
||||||
key="low_oxygen_alert",
|
key="low_oxygen_alert",
|
||||||
translation_key="low_ox_alrt",
|
translation_key="low_ox_alrt",
|
||||||
device_class=BinarySensorDeviceClass.SOUND,
|
device_class=BinarySensorDeviceClass.SOUND,
|
||||||
|
available_during_charging=True,
|
||||||
),
|
),
|
||||||
OwletBinarySensorEntityDescription(
|
OwletBinarySensorEntityDescription(
|
||||||
key="critical_oxygen_alert",
|
key="critical_oxygen_alert",
|
||||||
translation_key="crit_ox_alrt",
|
translation_key="crit_ox_alrt",
|
||||||
device_class=BinarySensorDeviceClass.SOUND,
|
device_class=BinarySensorDeviceClass.SOUND,
|
||||||
|
available_during_charging=True,
|
||||||
),
|
),
|
||||||
OwletBinarySensorEntityDescription(
|
OwletBinarySensorEntityDescription(
|
||||||
key="low_battery_alert",
|
key="low_battery_alert",
|
||||||
translation_key="low_batt_alrt",
|
translation_key="low_batt_alrt",
|
||||||
device_class=BinarySensorDeviceClass.SOUND,
|
device_class=BinarySensorDeviceClass.SOUND,
|
||||||
|
available_during_charging=True,
|
||||||
),
|
),
|
||||||
OwletBinarySensorEntityDescription(
|
OwletBinarySensorEntityDescription(
|
||||||
key="critical_battery_alert",
|
key="critical_battery_alert",
|
||||||
translation_key="crit_batt_alrt",
|
translation_key="crit_batt_alrt",
|
||||||
device_class=BinarySensorDeviceClass.SOUND,
|
device_class=BinarySensorDeviceClass.SOUND,
|
||||||
|
available_during_charging=True,
|
||||||
),
|
),
|
||||||
OwletBinarySensorEntityDescription(
|
OwletBinarySensorEntityDescription(
|
||||||
key="lost_power_alert",
|
key="lost_power_alert",
|
||||||
translation_key="lost_pwr_alrt",
|
translation_key="lost_pwr_alrt",
|
||||||
device_class=BinarySensorDeviceClass.SOUND,
|
device_class=BinarySensorDeviceClass.SOUND,
|
||||||
|
available_during_charging=True,
|
||||||
),
|
),
|
||||||
OwletBinarySensorEntityDescription(
|
OwletBinarySensorEntityDescription(
|
||||||
key="sock_disconnected",
|
key="sock_disconnected",
|
||||||
translation_key="sock_discon_alrt",
|
translation_key="sock_discon_alrt",
|
||||||
device_class=BinarySensorDeviceClass.SOUND,
|
device_class=BinarySensorDeviceClass.SOUND,
|
||||||
|
available_during_charging=True,
|
||||||
),
|
),
|
||||||
OwletBinarySensorEntityDescription(
|
OwletBinarySensorEntityDescription(
|
||||||
key="sock_off",
|
key="sock_off",
|
||||||
translation_key="sock_off",
|
translation_key="sock_off",
|
||||||
device_class=BinarySensorDeviceClass.POWER,
|
device_class=BinarySensorDeviceClass.POWER,
|
||||||
|
available_during_charging=True,
|
||||||
),
|
),
|
||||||
OwletBinarySensorEntityDescription(
|
OwletBinarySensorEntityDescription(
|
||||||
key="sleep_state",
|
key="base_station_on",
|
||||||
translation_key="awake",
|
translation_key="base_on",
|
||||||
icon="mdi:sleep",
|
device_class=BinarySensorDeviceClass.POWER,
|
||||||
|
available_during_charging=True,
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -97,11 +112,13 @@ async def async_setup_entry(
|
|||||||
|
|
||||||
sensors = []
|
sensors = []
|
||||||
for coordinator in coordinators:
|
for coordinator in coordinators:
|
||||||
print(coordinator.sock.properties)
|
|
||||||
for sensor in SENSORS:
|
for sensor in SENSORS:
|
||||||
if sensor.key in coordinator.sock.properties:
|
if sensor.key in coordinator.sock.properties:
|
||||||
sensors.append(OwletBinarySensor(coordinator, sensor))
|
sensors.append(OwletBinarySensor(coordinator, sensor))
|
||||||
|
|
||||||
|
if OwletAwakeSensor.entity_description.key in coordinator.sock.properties:
|
||||||
|
sensors.append(OwletAwakeSensor(coordinator))
|
||||||
|
|
||||||
async_add_entities(sensors)
|
async_add_entities(sensors)
|
||||||
|
|
||||||
|
|
||||||
@@ -118,11 +135,17 @@ class OwletBinarySensor(OwletBaseEntity, BinarySensorEntity):
|
|||||||
self.entity_description = description
|
self.entity_description = description
|
||||||
self._attr_unique_id = f"{self.sock.serial}-{description.key}"
|
self._attr_unique_id = f"{self.sock.serial}-{description.key}"
|
||||||
|
|
||||||
|
@property
|
||||||
|
def available(self) -> bool:
|
||||||
|
"""Return if entity is available."""
|
||||||
|
return super().available and (
|
||||||
|
not self.sock.properties["charging"]
|
||||||
|
or self.entity_description.available_during_charging
|
||||||
|
)
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def is_on(self) -> bool:
|
def is_on(self) -> bool:
|
||||||
"""Return true if the binary sensor is on."""
|
"""Return true if the binary sensor is on."""
|
||||||
state = self.sock.properties[self.entity_description.key]
|
|
||||||
|
|
||||||
if self.entity_description.key == "sleep_state":
|
if self.entity_description.key == "sleep_state":
|
||||||
if self.sock.properties["charging"]:
|
if self.sock.properties["charging"]:
|
||||||
return None
|
return None
|
||||||
@@ -131,4 +154,27 @@ class OwletBinarySensor(OwletBaseEntity, BinarySensorEntity):
|
|||||||
else:
|
else:
|
||||||
state = True
|
state = True
|
||||||
|
|
||||||
return state
|
return self.sock.properties[self.entity_description.key]
|
||||||
|
|
||||||
|
|
||||||
|
class OwletAwakeSensor(OwletBinarySensor):
|
||||||
|
"""Representation of an Owlet sleep sensor."""
|
||||||
|
|
||||||
|
entity_description = OwletBinarySensorEntityDescription(
|
||||||
|
key="sleep_state",
|
||||||
|
translation_key="awake",
|
||||||
|
icon="mdi:sleep",
|
||||||
|
available_during_charging=False,
|
||||||
|
)
|
||||||
|
|
||||||
|
def __init__(
|
||||||
|
self,
|
||||||
|
coordinator: OwletCoordinator,
|
||||||
|
) -> None:
|
||||||
|
"""Initialize the sensor."""
|
||||||
|
super().__init__(coordinator, self.entity_description)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def is_on(self) -> bool:
|
||||||
|
"""Return true if the binary sensor is on."""
|
||||||
|
return False if self.sock.properties[self.entity_description.key] in [8, 15] else True
|
||||||
|
|||||||
@@ -130,9 +130,9 @@ class OwletConfigFlow(config_entries.ConfigFlow, domain=DOMAIN):
|
|||||||
self.reauth_entry, data={**entry_data, **token}
|
self.reauth_entry, data={**entry_data, **token}
|
||||||
)
|
)
|
||||||
|
|
||||||
await self.hass.config_entries.async_reload(self.reauth_entry.entry_id)
|
await self.hass.config_entries.async_reload(self.reauth_entry.entry_id)
|
||||||
|
|
||||||
return self.async_abort(reason="reauth_successful")
|
return self.async_abort(reason="reauth_successful")
|
||||||
|
|
||||||
except OwletPasswordError:
|
except OwletPasswordError:
|
||||||
errors[CONF_PASSWORD] = "invalid_password"
|
errors[CONF_PASSWORD] = "invalid_password"
|
||||||
@@ -165,7 +165,7 @@ class OptionsFlowHandler(config_entries.OptionsFlow):
|
|||||||
vol.Required(
|
vol.Required(
|
||||||
CONF_SCAN_INTERVAL,
|
CONF_SCAN_INTERVAL,
|
||||||
default=self.config_entry.options.get(CONF_SCAN_INTERVAL),
|
default=self.config_entry.options.get(CONF_SCAN_INTERVAL),
|
||||||
): vol.All(vol.Coerce(int), vol.Range(min=10)),
|
): vol.All(vol.Coerce(int), vol.Range(min=5)),
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|||||||
@@ -9,7 +9,7 @@
|
|||||||
"iot_class": "cloud_polling",
|
"iot_class": "cloud_polling",
|
||||||
"issue_tracker": "https://github.com/ryanbdclark/owlet/issues",
|
"issue_tracker": "https://github.com/ryanbdclark/owlet/issues",
|
||||||
"requirements": [
|
"requirements": [
|
||||||
"pyowletapi==2023.11.4"
|
"pyowletapi==2024.3.2"
|
||||||
],
|
],
|
||||||
"version": "2023.11.2"
|
"version": "2024.5.1"
|
||||||
}
|
}
|
||||||
@@ -1,4 +1,5 @@
|
|||||||
"""Support for Owlet sensors."""
|
"""Support for Owlet sensors."""
|
||||||
|
|
||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
|
|
||||||
from dataclasses import dataclass
|
from dataclasses import dataclass
|
||||||
@@ -25,10 +26,12 @@ from .coordinator import OwletCoordinator
|
|||||||
from .entity import OwletBaseEntity
|
from .entity import OwletBaseEntity
|
||||||
|
|
||||||
|
|
||||||
@dataclass
|
@dataclass(kw_only=True)
|
||||||
class OwletSensorEntityDescription(SensorEntityDescription):
|
class OwletSensorEntityDescription(SensorEntityDescription):
|
||||||
"""Represent the owlet sensor entity description."""
|
"""Represent the owlet sensor entity description."""
|
||||||
|
|
||||||
|
available_during_charging: bool
|
||||||
|
|
||||||
|
|
||||||
SENSORS: tuple[OwletSensorEntityDescription, ...] = (
|
SENSORS: tuple[OwletSensorEntityDescription, ...] = (
|
||||||
OwletSensorEntityDescription(
|
OwletSensorEntityDescription(
|
||||||
@@ -37,6 +40,7 @@ SENSORS: tuple[OwletSensorEntityDescription, ...] = (
|
|||||||
native_unit_of_measurement=PERCENTAGE,
|
native_unit_of_measurement=PERCENTAGE,
|
||||||
device_class=SensorDeviceClass.BATTERY,
|
device_class=SensorDeviceClass.BATTERY,
|
||||||
state_class=SensorStateClass.MEASUREMENT,
|
state_class=SensorStateClass.MEASUREMENT,
|
||||||
|
available_during_charging=True,
|
||||||
),
|
),
|
||||||
OwletSensorEntityDescription(
|
OwletSensorEntityDescription(
|
||||||
key="oxygen_saturation",
|
key="oxygen_saturation",
|
||||||
@@ -44,6 +48,7 @@ SENSORS: tuple[OwletSensorEntityDescription, ...] = (
|
|||||||
native_unit_of_measurement=PERCENTAGE,
|
native_unit_of_measurement=PERCENTAGE,
|
||||||
state_class=SensorStateClass.MEASUREMENT,
|
state_class=SensorStateClass.MEASUREMENT,
|
||||||
icon="mdi:leaf",
|
icon="mdi:leaf",
|
||||||
|
available_during_charging=False,
|
||||||
),
|
),
|
||||||
OwletSensorEntityDescription(
|
OwletSensorEntityDescription(
|
||||||
key="heart_rate",
|
key="heart_rate",
|
||||||
@@ -51,13 +56,14 @@ SENSORS: tuple[OwletSensorEntityDescription, ...] = (
|
|||||||
native_unit_of_measurement="bpm",
|
native_unit_of_measurement="bpm",
|
||||||
state_class=SensorStateClass.MEASUREMENT,
|
state_class=SensorStateClass.MEASUREMENT,
|
||||||
icon="mdi:heart-pulse",
|
icon="mdi:heart-pulse",
|
||||||
|
available_during_charging=False,
|
||||||
),
|
),
|
||||||
OwletSensorEntityDescription(
|
OwletSensorEntityDescription(
|
||||||
key="battery_minutes",
|
key="battery_minutes",
|
||||||
translation_key="batterymin",
|
translation_key="batterymin",
|
||||||
native_unit_of_measurement=UnitOfTime.MINUTES,
|
native_unit_of_measurement=UnitOfTime.MINUTES,
|
||||||
device_class=SensorDeviceClass.DURATION,
|
device_class=SensorDeviceClass.DURATION,
|
||||||
state_class=SensorStateClass.MEASUREMENT,
|
available_during_charging=False,
|
||||||
),
|
),
|
||||||
OwletSensorEntityDescription(
|
OwletSensorEntityDescription(
|
||||||
key="signal_strength",
|
key="signal_strength",
|
||||||
@@ -65,6 +71,7 @@ SENSORS: tuple[OwletSensorEntityDescription, ...] = (
|
|||||||
native_unit_of_measurement=SIGNAL_STRENGTH_DECIBELS_MILLIWATT,
|
native_unit_of_measurement=SIGNAL_STRENGTH_DECIBELS_MILLIWATT,
|
||||||
device_class=SensorDeviceClass.SIGNAL_STRENGTH,
|
device_class=SensorDeviceClass.SIGNAL_STRENGTH,
|
||||||
state_class=SensorStateClass.MEASUREMENT,
|
state_class=SensorStateClass.MEASUREMENT,
|
||||||
|
available_during_charging=True,
|
||||||
),
|
),
|
||||||
OwletSensorEntityDescription(
|
OwletSensorEntityDescription(
|
||||||
key="skin_temperature",
|
key="skin_temperature",
|
||||||
@@ -72,11 +79,7 @@ SENSORS: tuple[OwletSensorEntityDescription, ...] = (
|
|||||||
native_unit_of_measurement=UnitOfTemperature.CELSIUS,
|
native_unit_of_measurement=UnitOfTemperature.CELSIUS,
|
||||||
device_class=SensorDeviceClass.TEMPERATURE,
|
device_class=SensorDeviceClass.TEMPERATURE,
|
||||||
state_class=SensorStateClass.MEASUREMENT,
|
state_class=SensorStateClass.MEASUREMENT,
|
||||||
),
|
available_during_charging=False,
|
||||||
OwletSensorEntityDescription(
|
|
||||||
key="sleep_state",
|
|
||||||
translation_key="sleepstate",
|
|
||||||
device_class=SensorDeviceClass.ENUM,
|
|
||||||
),
|
),
|
||||||
OwletSensorEntityDescription(
|
OwletSensorEntityDescription(
|
||||||
key="movement",
|
key="movement",
|
||||||
@@ -84,13 +87,7 @@ SENSORS: tuple[OwletSensorEntityDescription, ...] = (
|
|||||||
state_class=SensorStateClass.MEASUREMENT,
|
state_class=SensorStateClass.MEASUREMENT,
|
||||||
icon="mdi:cursor-move",
|
icon="mdi:cursor-move",
|
||||||
entity_registry_enabled_default=False,
|
entity_registry_enabled_default=False,
|
||||||
),
|
available_during_charging=False,
|
||||||
OwletSensorEntityDescription(
|
|
||||||
key="oxygen_10_av",
|
|
||||||
translation_key="o2saturation10a",
|
|
||||||
native_unit_of_measurement=PERCENTAGE,
|
|
||||||
state_class=SensorStateClass.MEASUREMENT,
|
|
||||||
icon="mdi:leaf",
|
|
||||||
),
|
),
|
||||||
OwletSensorEntityDescription(
|
OwletSensorEntityDescription(
|
||||||
key="movement_bucket",
|
key="movement_bucket",
|
||||||
@@ -98,6 +95,7 @@ SENSORS: tuple[OwletSensorEntityDescription, ...] = (
|
|||||||
state_class=SensorStateClass.MEASUREMENT,
|
state_class=SensorStateClass.MEASUREMENT,
|
||||||
icon="mdi:bucket-outline",
|
icon="mdi:bucket-outline",
|
||||||
entity_registry_enabled_default=False,
|
entity_registry_enabled_default=False,
|
||||||
|
available_during_charging=False,
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -120,6 +118,11 @@ async def async_setup_entry(
|
|||||||
if sensor.key in coordinator.sock.properties:
|
if sensor.key in coordinator.sock.properties:
|
||||||
sensors.append(OwletSensor(coordinator, sensor))
|
sensors.append(OwletSensor(coordinator, sensor))
|
||||||
|
|
||||||
|
if OwletSleepSensor.entity_description.key in coordinator.sock.properties:
|
||||||
|
sensors.append(OwletSleepSensor(coordinator))
|
||||||
|
if OwletOxygenAverageSensor.entity_description.key in coordinator.sock.properties:
|
||||||
|
sensors.append(OwletOxygenAverageSensor(coordinator))
|
||||||
|
|
||||||
async_add_entities(sensors)
|
async_add_entities(sensors)
|
||||||
|
|
||||||
|
|
||||||
@@ -136,32 +139,68 @@ class OwletSensor(OwletBaseEntity, SensorEntity):
|
|||||||
self.entity_description: OwletSensorEntityDescription = description
|
self.entity_description: OwletSensorEntityDescription = description
|
||||||
self._attr_unique_id = f"{self.sock.serial}-{description.key}"
|
self._attr_unique_id = f"{self.sock.serial}-{description.key}"
|
||||||
|
|
||||||
|
@property
|
||||||
|
def available(self) -> bool:
|
||||||
|
"""Return if entity is available."""
|
||||||
|
return super().available and (
|
||||||
|
not self.sock.properties["charging"]
|
||||||
|
or self.entity_description.available_during_charging
|
||||||
|
)
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def native_value(self) -> StateType:
|
def native_value(self) -> StateType:
|
||||||
"""Return sensor value."""
|
"""Return sensor value."""
|
||||||
|
|
||||||
if (
|
|
||||||
self.entity_description.key
|
|
||||||
in [
|
|
||||||
"heart_rate",
|
|
||||||
"battery_minutes",
|
|
||||||
"oxygen_saturation",
|
|
||||||
"skin_temperature",
|
|
||||||
"oxygen_10_av",
|
|
||||||
"sleep_state",
|
|
||||||
]
|
|
||||||
and self.sock.properties["charging"]
|
|
||||||
):
|
|
||||||
return None
|
|
||||||
|
|
||||||
if self.entity_description.key == "sleep_state":
|
|
||||||
return SLEEP_STATES[self.sock.properties["sleep_state"]]
|
|
||||||
|
|
||||||
return self.sock.properties[self.entity_description.key]
|
return self.sock.properties[self.entity_description.key]
|
||||||
|
|
||||||
|
|
||||||
|
class OwletSleepSensor(OwletSensor):
|
||||||
|
"""Representation of an Owlet sleep sensor."""
|
||||||
|
|
||||||
|
_attr_options = list(SLEEP_STATES.values())
|
||||||
|
entity_description = OwletSensorEntityDescription(
|
||||||
|
key="sleep_state",
|
||||||
|
translation_key="sleepstate",
|
||||||
|
device_class=SensorDeviceClass.ENUM,
|
||||||
|
available_during_charging=False,
|
||||||
|
)
|
||||||
|
|
||||||
|
def __init__(
|
||||||
|
self,
|
||||||
|
coordinator: OwletCoordinator,
|
||||||
|
) -> None:
|
||||||
|
"""Initialize the sensor."""
|
||||||
|
super().__init__(coordinator, self.entity_description)
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def options(self) -> list[str] | None:
|
def native_value(self) -> StateType:
|
||||||
"""Set options for sleep state."""
|
"""Return sensor value."""
|
||||||
if self.entity_description.key != "sleep_state":
|
return SLEEP_STATES[self.sock.properties["sleep_state"]]
|
||||||
return None
|
|
||||||
return list(SLEEP_STATES.values())
|
|
||||||
|
class OwletOxygenAverageSensor(OwletSensor):
|
||||||
|
"""Representation of an Owlet sleep sensor."""
|
||||||
|
|
||||||
|
entity_description = OwletSensorEntityDescription(
|
||||||
|
key="oxygen_10_av",
|
||||||
|
translation_key="o2saturation10a",
|
||||||
|
native_unit_of_measurement=PERCENTAGE,
|
||||||
|
icon="mdi:leaf",
|
||||||
|
available_during_charging=False,
|
||||||
|
)
|
||||||
|
|
||||||
|
def __init__(
|
||||||
|
self,
|
||||||
|
coordinator: OwletCoordinator,
|
||||||
|
) -> None:
|
||||||
|
"""Initialize the sensor."""
|
||||||
|
super().__init__(coordinator, self.entity_description)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def available(self) -> bool:
|
||||||
|
"""Return if entity is available."""
|
||||||
|
return super().available and (
|
||||||
|
not self.sock.properties["charging"]
|
||||||
|
or self.entity_description.available_during_charging
|
||||||
|
) and (self.sock.properties["oxygen_10_av"] >= 0 and self.sock.properties["oxygen_10_av"] <= 100)
|
||||||
|
|
||||||
|
|||||||
@@ -5,13 +5,13 @@
|
|||||||
"data": {
|
"data": {
|
||||||
"region": "Region",
|
"region": "Region",
|
||||||
"username": "Email",
|
"username": "Email",
|
||||||
"password": "Password"
|
"password": "[%key:common::config_flow::data::password%]"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"reauth_confirm": {
|
"reauth_confirm": {
|
||||||
"title": "Reauthentiaction required for Owlet",
|
"title": "Reauthentiaction required for Owlet",
|
||||||
"data": {
|
"data": {
|
||||||
"password": "Password"
|
"password": "[%key:common::config_flow::data::password%]"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@@ -74,6 +74,9 @@
|
|||||||
},
|
},
|
||||||
"awake": {
|
"awake": {
|
||||||
"name": "Awake"
|
"name": "Awake"
|
||||||
|
},
|
||||||
|
"base_on": {
|
||||||
|
"name": "Base station on"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"sensor": {
|
"sensor": {
|
||||||
|
|||||||
@@ -5,13 +5,13 @@
|
|||||||
"data": {
|
"data": {
|
||||||
"region": "Region",
|
"region": "Region",
|
||||||
"username": "Email",
|
"username": "Email",
|
||||||
"password": "Password"
|
"password": "[%key:common::config_flow::data::password%]"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"reauth_confirm": {
|
"reauth_confirm": {
|
||||||
"title": "Reauthentiaction required for Owlet",
|
"title": "Reauthentiaction required for Owlet",
|
||||||
"data": {
|
"data": {
|
||||||
"password": "Password"
|
"password": "[%key:common::config_flow::data::password%]"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@@ -74,6 +74,9 @@
|
|||||||
},
|
},
|
||||||
"awake": {
|
"awake": {
|
||||||
"name": "Awake"
|
"name": "Awake"
|
||||||
|
},
|
||||||
|
"base_on": {
|
||||||
|
"name": "Base station on"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"sensor": {
|
"sensor": {
|
||||||
|
|||||||
@@ -5,13 +5,13 @@
|
|||||||
"data": {
|
"data": {
|
||||||
"region": "Region",
|
"region": "Region",
|
||||||
"username": "Email",
|
"username": "Email",
|
||||||
"password": "Password"
|
"password": "[%key:common::config_flow::data::password%]"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"reauth_confirm": {
|
"reauth_confirm": {
|
||||||
"title": "Reauthentiaction required for Owlet",
|
"title": "Reauthentiaction required for Owlet",
|
||||||
"data": {
|
"data": {
|
||||||
"password": "Password"
|
"password": "[%key:common::config_flow::data::password%]"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@@ -74,6 +74,9 @@
|
|||||||
},
|
},
|
||||||
"awake": {
|
"awake": {
|
||||||
"name": "Awake"
|
"name": "Awake"
|
||||||
|
},
|
||||||
|
"base_on": {
|
||||||
|
"name": "Base station on"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"sensor": {
|
"sensor": {
|
||||||
|
|||||||
Reference in New Issue
Block a user