Compare commits
39 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
45e65f384b | ||
|
|
df6b45621e | ||
|
|
1244bffcb4 | ||
|
|
d323cbfd11 | ||
|
|
2accec2b49 | ||
|
|
6b343a76ca | ||
|
|
fa2e06dcf4 | ||
|
|
975e98c337 | ||
|
|
268365ccd4 | ||
|
|
ac9c8c6111 | ||
|
|
4fa40f8621 | ||
|
|
c04d6b7bf8 | ||
|
|
dd17aca283 | ||
|
|
91578464de | ||
|
|
e28b9ddf3e | ||
|
|
f63e0a6dfe | ||
|
|
82823be1c8 | ||
|
|
14787e03c4 | ||
|
|
0991eb31d9 | ||
|
|
ad91a851fc | ||
|
|
339dc43d6d | ||
|
|
f3c853e2d7 | ||
|
|
dfc2ffc0e1 | ||
|
|
dc28ebb02f | ||
|
|
52710ba7de | ||
|
|
d8449c14a4 | ||
|
|
3610262855 | ||
|
|
904023e93a | ||
|
|
ceade24851 | ||
|
|
ab486d5519 | ||
|
|
5e17ecdeb2 | ||
|
|
4b90ce0d61 | ||
|
|
50c55dcfd3 | ||
|
|
faefd0b18b | ||
|
|
1192b833ca | ||
|
|
d440fed621 | ||
|
|
50fe1a8765 | ||
|
|
1cfff537d7 | ||
|
|
3acf847352 |
63
CHANGELOG.md
63
CHANGELOG.md
@@ -1,6 +1,69 @@
|
|||||||
# Changelog
|
# Changelog
|
||||||
|
|
||||||
<!--next-version-placeholder-->
|
<!--next-version-placeholder-->
|
||||||
|
## 2025.4.3 (2025-04-15)
|
||||||
|
### Fix
|
||||||
|
* Bumping pyowletapi to 2025.4.8.
|
||||||
|
* Changes to how the sensors are stored to solve the issue where only one device is added, thanks [`@MarjovanLier`](https://github.com/MarjovanLier). ([`1244bff`](https://github.com/ryanbdclark/owlet/commit/1244bffcb48d7337a9d7a0da518959fe4b31a230))
|
||||||
|
|
||||||
|
## 2025.4.2 (2025-04-14)
|
||||||
|
### Fix
|
||||||
|
* Bumping pyowletapi to 2025.4.1, should hopefully stop issue where only one device was added to HA. ([`d323cbf`](https://github.com/ryanbdclark/owlet/commit/d323cbfd11411ff34866ead492de10c109c72689))
|
||||||
|
|
||||||
|
## 2025.4.1 (2025-04-11)
|
||||||
|
### Fix
|
||||||
|
* Changes to stop errors after refactoring pyowletapi ([`6b343a7`](https://github.com/ryanbdclark/owlet/commit/6b343a76caad3375e10c80f4d26942a1bbbb831d))
|
||||||
|
|
||||||
|
## 2025.4.0 (2025-04-11)
|
||||||
|
### Fix
|
||||||
|
* Bumping pyowletapi to 2025.4.0 ([`268365c`](https://github.com/ryanbdclark/owlet/commit/268365ccd428418dd5707f0569ce738b54a12fdd))
|
||||||
|
|
||||||
|
## 2024.10.1 (2024-10-09)
|
||||||
|
### Feature
|
||||||
|
* Base station has now been removed from binary sensors and added as a switch ([`f63e0a6`](https://github.com/ryanbdclark/owlet/commit/f63e0a6dfeab1a05ba09ef3e0087cb404ba0dac4))
|
||||||
|
### Fix
|
||||||
|
* Bump pyowletapi to 2024.10.1 ([`82823be`](https://github.com/ryanbdclark/owlet/commit/82823be1c8265d2b9431771136853febef648650))
|
||||||
|
* Fix strings for password and connection error in all languages ([`14787e0`](https://github.com/ryanbdclark/owlet/commit/14787e03c4d275f46f446921a3ee133fc7cfd1b1))
|
||||||
|
* Add state class to battery minutes and O2 saturation 10 minute average ([`0991eb3`](https://github.com/ryanbdclark/owlet/commit/0991eb31d919f3ee9f65ece793166d7ee3e33c38))
|
||||||
|
|
||||||
|
## 2024.9.1 (2024-09-26)
|
||||||
|
### Feature
|
||||||
|
* Now includes French translation, thanks [`@Julien80`](https://github.com/Julien80) ([`f3c853e`](https://github.com/ryanbdclark/owlet/commit/f3c853e2d7243d766889f2d18c718819da30e4be))
|
||||||
|
|
||||||
|
## 2024.6.1 (2024-06-17)
|
||||||
|
### Fix
|
||||||
|
* Bumping pyowletapi to 2024.6.1 to resolve setup errors ([`52710ba`](https://github.com/ryanbdclark/owlet/commit/52710ba7de53fd07195537c2a5fd2f95bc7dfd1a))
|
||||||
|
|
||||||
|
## 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
|
||||||
|
* 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))
|
||||||
|
### Fix
|
||||||
|
* Bumping pyowletapi to 2023.11.4 to allow V2 support ([`50fe1a8`](https://github.com/ryanbdclark/owlet/commit/50fe1a87656b7d6413d06f06f3650fd0bfb48e02))
|
||||||
|
* Refactoring ([`50fe1a8`](https://github.com/ryanbdclark/owlet/commit/50fe1a87656b7d6413d06f06f3650fd0bfb48e02))
|
||||||
|
* Corrected spelling of sock disconnected sensor ([`50fe1a8`](https://github.com/ryanbdclark/owlet/commit/50fe1a87656b7d6413d06f06f3650fd0bfb48e02))
|
||||||
|
|
||||||
|
## 2023.11.1 (2023-11-16)
|
||||||
|
### Fix
|
||||||
|
* Bumping pyowletapi to 2023.11.1 ([`3acf847`](https://github.com/ryanbdclark/owlet/commit/3acf8473526665382b44ef6325d708a6c62fff45))
|
||||||
|
* Sensors and binary sensors are now only created where the sock contains that property, this stops errors where different sock versions have different properties ([`3acf847`](https://github.com/ryanbdclark/owlet/commit/3acf8473526665382b44ef6325d708a6c62fff45))
|
||||||
|
|
||||||
## 2023.9.1 (2023-09-20)
|
## 2023.9.1 (2023-09-20)
|
||||||
### Fix
|
### Fix
|
||||||
* Bumping pyowletapi to 2023.9.1 to allow for revisions ([`0a7f703`](https://github.com/ryanbdclark/owlet/commit/0a7f70310080a129c988e9607331baa2f6c691e0))
|
* Bumping pyowletapi to 2023.9.1 to allow for revisions ([`0a7f703`](https://github.com/ryanbdclark/owlet/commit/0a7f70310080a129c988e9607331baa2f6c691e0))
|
||||||
|
|||||||
@@ -8,13 +8,11 @@
|
|||||||
[![hacs][hacsbadge]][hacs]
|
[![hacs][hacsbadge]][hacs]
|
||||||
[![Project Maintenance][maintenance-shield]][user_profile]
|
[![Project Maintenance][maintenance-shield]][user_profile]
|
||||||
|
|
||||||
A custom component for the Owlet smart sock, currently this only supports the owlet smart sock 3.
|
A custom component for the Owlet smart sock
|
||||||
|
|
||||||
If you have a smart sock 2 and would like to contribute then please do so.
|
|
||||||
|
|
||||||
## Installation
|
## Installation
|
||||||
|
|
||||||
1. Use [HACS](https://hacs.xyz/docs/setup/download), in `HACS > Integrations > Explore & Add Repositories` search for "Owlet". After adding this `https://github.com/ryanbdclark/owlet` as a custom repository.
|
1. Use [HACS](https://hacs.xyz/docs/use/download/download/), in `HACS > Integrations > Explore & Add Repositories` search for "Owlet".
|
||||||
2. Restart Home Assistant.
|
2. Restart Home Assistant.
|
||||||
3. [![Add Integration][add-integration-badge]][add-integration] or in the HA UI go to "Settings" -> "Devices & Services" then click "+" and search for "Owlet Smart Sock".
|
3. [![Add Integration][add-integration-badge]][add-integration] or in the HA UI go to "Settings" -> "Devices & Services" then click "+" and search for "Owlet Smart Sock".
|
||||||
|
|
||||||
@@ -47,4 +45,4 @@ This integration provides the following entities:
|
|||||||
[releases]: https://github.com/ryanbdclark/owlet/releases
|
[releases]: https://github.com/ryanbdclark/owlet/releases
|
||||||
[user_profile]: https://github.com/ryanbdclark
|
[user_profile]: https://github.com/ryanbdclark
|
||||||
[add-integration]: https://my.home-assistant.io/redirect/config_flow_start?domain=owlet
|
[add-integration]: https://my.home-assistant.io/redirect/config_flow_start?domain=owlet
|
||||||
[add-integration-badge]: https://my.home-assistant.io/badges/config_flow_start.svg
|
[add-integration-badge]: https://my.home-assistant.io/badges/config_flow_start.svg
|
||||||
|
|||||||
@@ -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
|
||||||
@@ -29,7 +30,7 @@ from homeassistant.helpers.aiohttp_client import async_get_clientsession
|
|||||||
from .const import CONF_OWLET_EXPIRY, CONF_OWLET_REFRESH, DOMAIN, SUPPORTED_VERSIONS
|
from .const import CONF_OWLET_EXPIRY, CONF_OWLET_REFRESH, DOMAIN, SUPPORTED_VERSIONS
|
||||||
from .coordinator import OwletCoordinator
|
from .coordinator import OwletCoordinator
|
||||||
|
|
||||||
PLATFORMS: list[Platform] = [Platform.BINARY_SENSOR, Platform.SENSOR]
|
PLATFORMS: list[Platform] = [Platform.BINARY_SENSOR, Platform.SENSOR, Platform.SWITCH]
|
||||||
|
|
||||||
_LOGGER = logging.getLogger(__name__)
|
_LOGGER = logging.getLogger(__name__)
|
||||||
|
|
||||||
@@ -67,20 +68,17 @@ async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
|||||||
_LOGGER.error("No owlet devices found to set up")
|
_LOGGER.error("No owlet devices found to set up")
|
||||||
return False
|
return False
|
||||||
|
|
||||||
if devices["tokens"]:
|
if "tokens" in devices:
|
||||||
hass.config_entries.async_update_entry(
|
hass.config_entries.async_update_entry(
|
||||||
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,61 +18,79 @@ 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(
|
||||||
|
key="critical_oxygen_alert",
|
||||||
|
translation_key="crit_ox_alrt",
|
||||||
|
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(
|
||||||
|
key="critical_battery_alert",
|
||||||
|
translation_key="crit_batt_alrt",
|
||||||
|
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(
|
|
||||||
key="sleep_state",
|
|
||||||
translation_key="awake",
|
|
||||||
icon="mdi:sleep",
|
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -85,11 +104,18 @@ async def async_setup_entry(
|
|||||||
|
|
||||||
coordinators: OwletCoordinator = hass.data[DOMAIN][config_entry.entry_id].values()
|
coordinators: OwletCoordinator = hass.data[DOMAIN][config_entry.entry_id].values()
|
||||||
|
|
||||||
async_add_entities(
|
sensors = []
|
||||||
OwletBinarySensor(coordinator, sensor)
|
for coordinator in coordinators:
|
||||||
for coordinator in coordinators
|
sensors.extend([
|
||||||
for sensor in SENSORS
|
OwletBinarySensor(coordinator, sensor)
|
||||||
)
|
for sensor in SENSORS
|
||||||
|
if sensor.key in coordinator.sock.properties
|
||||||
|
])
|
||||||
|
|
||||||
|
if OwletAwakeSensor.entity_description.key in coordinator.sock.properties:
|
||||||
|
sensors.append(OwletAwakeSensor(coordinator))
|
||||||
|
|
||||||
|
async_add_entities(sensors)
|
||||||
|
|
||||||
|
|
||||||
class OwletBinarySensor(OwletBaseEntity, BinarySensorEntity):
|
class OwletBinarySensor(OwletBaseEntity, BinarySensorEntity):
|
||||||
@@ -105,17 +131,39 @@ 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":
|
return self.sock.properties[self.entity_description.key]
|
||||||
if self.sock.properties["charging"]:
|
|
||||||
return None
|
|
||||||
if state in [8, 15]:
|
|
||||||
state = False
|
|
||||||
else:
|
|
||||||
state = True
|
|
||||||
|
|
||||||
return state
|
|
||||||
|
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 self.sock.properties[self.entity_description.key] not in [8, 15]
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
"""Config flow for Owlet Smart Sock integration."""
|
"""Config flow for Owlet Smart Sock integration."""
|
||||||
|
|
||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
|
|
||||||
from collections.abc import Mapping
|
from collections.abc import Mapping
|
||||||
@@ -12,23 +13,20 @@ from pyowletapi.exceptions import (
|
|||||||
OwletEmailError,
|
OwletEmailError,
|
||||||
OwletPasswordError,
|
OwletPasswordError,
|
||||||
)
|
)
|
||||||
from pyowletapi.sock import Sock
|
|
||||||
import voluptuous as vol
|
import voluptuous as vol
|
||||||
|
|
||||||
from homeassistant import config_entries, exceptions
|
from homeassistant import config_entries, exceptions
|
||||||
from homeassistant.config_entries import ConfigEntry
|
from homeassistant.config_entries import ConfigEntry, ConfigFlowResult
|
||||||
from homeassistant.const import (
|
from homeassistant.const import (
|
||||||
CONF_API_TOKEN,
|
|
||||||
CONF_PASSWORD,
|
CONF_PASSWORD,
|
||||||
CONF_REGION,
|
CONF_REGION,
|
||||||
CONF_SCAN_INTERVAL,
|
CONF_SCAN_INTERVAL,
|
||||||
CONF_USERNAME,
|
CONF_USERNAME,
|
||||||
)
|
)
|
||||||
from homeassistant.core import callback
|
from homeassistant.core import callback
|
||||||
from homeassistant.data_entry_flow import FlowResult
|
|
||||||
from homeassistant.helpers.aiohttp_client import async_get_clientsession
|
from homeassistant.helpers.aiohttp_client import async_get_clientsession
|
||||||
|
|
||||||
from .const import CONF_OWLET_EXPIRY, CONF_OWLET_REFRESH, DOMAIN, POLLING_INTERVAL
|
from .const import DOMAIN, POLLING_INTERVAL
|
||||||
|
|
||||||
_LOGGER = logging.getLogger(__name__)
|
_LOGGER = logging.getLogger(__name__)
|
||||||
|
|
||||||
@@ -52,7 +50,7 @@ class OwletConfigFlow(config_entries.ConfigFlow, domain=DOMAIN):
|
|||||||
|
|
||||||
async def async_step_user(
|
async def async_step_user(
|
||||||
self, user_input: dict[str, Any] | None = None
|
self, user_input: dict[str, Any] | None = None
|
||||||
) -> FlowResult:
|
) -> ConfigFlowResult:
|
||||||
"""Handle the initial step."""
|
"""Handle the initial step."""
|
||||||
errors: dict[str, str] = {}
|
errors: dict[str, str] = {}
|
||||||
if user_input is not None:
|
if user_input is not None:
|
||||||
@@ -86,10 +84,8 @@ class OwletConfigFlow(config_entries.ConfigFlow, domain=DOMAIN):
|
|||||||
title=user_input[CONF_USERNAME],
|
title=user_input[CONF_USERNAME],
|
||||||
data={
|
data={
|
||||||
CONF_REGION: user_input[CONF_REGION],
|
CONF_REGION: user_input[CONF_REGION],
|
||||||
CONF_USERNAME: user_input[CONF_PASSWORD],
|
CONF_USERNAME: user_input[CONF_USERNAME],
|
||||||
CONF_API_TOKEN: token[CONF_API_TOKEN],
|
**token,
|
||||||
CONF_OWLET_EXPIRY: token[CONF_OWLET_EXPIRY],
|
|
||||||
CONF_OWLET_REFRESH: token[CONF_OWLET_REFRESH],
|
|
||||||
},
|
},
|
||||||
options={CONF_SCAN_INTERVAL: POLLING_INTERVAL},
|
options={CONF_SCAN_INTERVAL: POLLING_INTERVAL},
|
||||||
)
|
)
|
||||||
@@ -104,7 +100,9 @@ class OwletConfigFlow(config_entries.ConfigFlow, domain=DOMAIN):
|
|||||||
"""Get the options flow for this handler."""
|
"""Get the options flow for this handler."""
|
||||||
return OptionsFlowHandler(config_entry)
|
return OptionsFlowHandler(config_entry)
|
||||||
|
|
||||||
async def async_step_reauth(self, user_input: Mapping[str, Any]) -> FlowResult:
|
async def async_step_reauth(
|
||||||
|
self, user_input: Mapping[str, Any]
|
||||||
|
) -> ConfigFlowResult:
|
||||||
"""Handle reauth."""
|
"""Handle reauth."""
|
||||||
self.reauth_entry = self.hass.config_entries.async_get_entry(
|
self.reauth_entry = self.hass.config_entries.async_get_entry(
|
||||||
self.context["entry_id"]
|
self.context["entry_id"]
|
||||||
@@ -113,7 +111,7 @@ class OwletConfigFlow(config_entries.ConfigFlow, domain=DOMAIN):
|
|||||||
|
|
||||||
async def async_step_reauth_confirm(
|
async def async_step_reauth_confirm(
|
||||||
self, user_input: dict[str, Any] | None = None
|
self, user_input: dict[str, Any] | None = None
|
||||||
) -> FlowResult:
|
) -> ConfigFlowResult:
|
||||||
"""Dialog that informs the user that reauth is required."""
|
"""Dialog that informs the user that reauth is required."""
|
||||||
assert self.reauth_entry is not None
|
assert self.reauth_entry is not None
|
||||||
errors: dict[str, str] = {}
|
errors: dict[str, str] = {}
|
||||||
@@ -127,15 +125,16 @@ class OwletConfigFlow(config_entries.ConfigFlow, domain=DOMAIN):
|
|||||||
session=async_get_clientsession(self.hass),
|
session=async_get_clientsession(self.hass),
|
||||||
)
|
)
|
||||||
try:
|
try:
|
||||||
token = await owlet_api.authenticate()
|
if token := await owlet_api.authenticate():
|
||||||
if token:
|
|
||||||
self.hass.config_entries.async_update_entry(
|
self.hass.config_entries.async_update_entry(
|
||||||
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"
|
||||||
@@ -158,7 +157,7 @@ class OptionsFlowHandler(config_entries.OptionsFlow):
|
|||||||
|
|
||||||
async def async_step_init(
|
async def async_step_init(
|
||||||
self, user_input: dict[str, Any] | None = None
|
self, user_input: dict[str, Any] | None = None
|
||||||
) -> FlowResult:
|
) -> ConfigFlowResult:
|
||||||
"""Handle options flow."""
|
"""Handle options flow."""
|
||||||
if user_input is not None:
|
if user_input is not None:
|
||||||
return self.async_create_entry(title="", data=user_input)
|
return self.async_create_entry(title="", data=user_input)
|
||||||
@@ -168,7 +167,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)),
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|||||||
@@ -5,7 +5,7 @@ DOMAIN = "owlet"
|
|||||||
CONF_OWLET_EXPIRY = "expiry"
|
CONF_OWLET_EXPIRY = "expiry"
|
||||||
CONF_OWLET_REFRESH = "refresh"
|
CONF_OWLET_REFRESH = "refresh"
|
||||||
|
|
||||||
SUPPORTED_VERSIONS = [3]
|
SUPPORTED_VERSIONS = [2, 3]
|
||||||
POLLING_INTERVAL = 5
|
POLLING_INTERVAL = 5
|
||||||
MANUFACTURER = "Owlet Baby Care"
|
MANUFACTURER = "Owlet Baby Care"
|
||||||
SLEEP_STATES = {0: "unknown", 1: "awake", 8: "light_sleep", 15: "deep_sleep"}
|
SLEEP_STATES = {0: "unknown", 1: "awake", 8: "light_sleep", 15: "deep_sleep"}
|
||||||
|
|||||||
@@ -36,13 +36,13 @@ class OwletCoordinator(DataUpdateCoordinator):
|
|||||||
update_interval=timedelta(seconds=interval),
|
update_interval=timedelta(seconds=interval),
|
||||||
)
|
)
|
||||||
self.sock = sock
|
self.sock = sock
|
||||||
self.config_entry = entry
|
self.config_entry: ConfigEntry = entry
|
||||||
|
|
||||||
async def _async_update_data(self) -> None:
|
async def _async_update_data(self) -> None:
|
||||||
"""Fetch the data from the device."""
|
"""Fetch the data from the device."""
|
||||||
try:
|
try:
|
||||||
properties = await self.sock.update_properties()
|
properties = await self.sock.update_properties()
|
||||||
if properties["tokens"]:
|
if "tokens" in properties:
|
||||||
self.hass.config_entries.async_update_entry(
|
self.hass.config_entries.async_update_entry(
|
||||||
self.config_entry,
|
self.config_entry,
|
||||||
data={**self.config_entry.data, **properties["tokens"]},
|
data={**self.config_entry.data, **properties["tokens"]},
|
||||||
|
|||||||
@@ -1,11 +1,11 @@
|
|||||||
"""Base class for Owlet entities."""
|
"""Base class for Owlet entities."""
|
||||||
|
|
||||||
|
from homeassistant.helpers.device_registry import DeviceInfo
|
||||||
from homeassistant.helpers.entity import Entity
|
from homeassistant.helpers.entity import Entity
|
||||||
from homeassistant.helpers.update_coordinator import CoordinatorEntity
|
from homeassistant.helpers.update_coordinator import CoordinatorEntity
|
||||||
from homeassistant.helpers.entity import DeviceInfo
|
|
||||||
|
|
||||||
from .coordinator import OwletCoordinator
|
|
||||||
from .const import DOMAIN, MANUFACTURER
|
from .const import DOMAIN, MANUFACTURER
|
||||||
|
from .coordinator import OwletCoordinator
|
||||||
|
|
||||||
|
|
||||||
class OwletBaseEntity(CoordinatorEntity[OwletCoordinator], Entity):
|
class OwletBaseEntity(CoordinatorEntity[OwletCoordinator], Entity):
|
||||||
@@ -19,16 +19,20 @@ class OwletBaseEntity(CoordinatorEntity[OwletCoordinator], Entity):
|
|||||||
) -> None:
|
) -> None:
|
||||||
"""Initialize the base entity."""
|
"""Initialize the base entity."""
|
||||||
super().__init__(coordinator)
|
super().__init__(coordinator)
|
||||||
|
self.coordinator = coordinator
|
||||||
self.sock = coordinator.sock
|
self.sock = coordinator.sock
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def device_info(self) -> DeviceInfo:
|
def device_info(self) -> DeviceInfo:
|
||||||
"""Return the device info of the device"""
|
"""Return the device info of the device."""
|
||||||
return DeviceInfo(
|
return DeviceInfo(
|
||||||
identifiers={(DOMAIN, self.sock.serial)},
|
identifiers={(DOMAIN, self.sock.serial)},
|
||||||
name="Owlet Baby Care Sock",
|
name=f"Owlet Sock {self.sock.serial}",
|
||||||
manufacturer=MANUFACTURER,
|
connections={("mac", getattr(self.sock, "mac", "unknown"))},
|
||||||
model=self.sock.model,
|
suggested_area="Nursery",
|
||||||
sw_version=self.sock.sw_version,
|
configuration_url="https://my.owletcare.com/",
|
||||||
hw_version=f"{self.sock.version}r{self.sock.revision}",
|
manufacturer="Owlet Baby Care",
|
||||||
|
model=getattr(self.sock, "model", None),
|
||||||
|
sw_version=getattr(self.sock, "sw_version", None),
|
||||||
|
hw_version=getattr(self.sock, "hw_version", "3r8"),
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -5,11 +5,11 @@
|
|||||||
"@ryanbdclark"
|
"@ryanbdclark"
|
||||||
],
|
],
|
||||||
"config_flow": true,
|
"config_flow": true,
|
||||||
"documentation": "https://www.home-assistant.io/integrations/owlet",
|
"documentation":"https://github.com/ryanbdclark/owlet",
|
||||||
"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.9.1"
|
"pyowletapi==2025.4.8"
|
||||||
],
|
],
|
||||||
"version": "2023.9.1"
|
"version": "2025.4.3"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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,18 +26,21 @@ 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_ALL: tuple[OwletSensorEntityDescription, ...] = (
|
|
||||||
|
SENSORS: tuple[OwletSensorEntityDescription, ...] = (
|
||||||
OwletSensorEntityDescription(
|
OwletSensorEntityDescription(
|
||||||
key="battery_percentage",
|
key="battery_percentage",
|
||||||
translation_key="batterypercent",
|
translation_key="batterypercent",
|
||||||
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_ALL: 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,6 +56,7 @@ SENSORS_ALL: 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",
|
||||||
@@ -58,6 +64,7 @@ SENSORS_ALL: tuple[OwletSensorEntityDescription, ...] = (
|
|||||||
native_unit_of_measurement=UnitOfTime.MINUTES,
|
native_unit_of_measurement=UnitOfTime.MINUTES,
|
||||||
device_class=SensorDeviceClass.DURATION,
|
device_class=SensorDeviceClass.DURATION,
|
||||||
state_class=SensorStateClass.MEASUREMENT,
|
state_class=SensorStateClass.MEASUREMENT,
|
||||||
|
available_during_charging=False,
|
||||||
),
|
),
|
||||||
OwletSensorEntityDescription(
|
OwletSensorEntityDescription(
|
||||||
key="signal_strength",
|
key="signal_strength",
|
||||||
@@ -65,6 +72,7 @@ SENSORS_ALL: 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 +80,7 @@ SENSORS_ALL: 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,16 +88,7 @@ SENSORS_ALL: 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,
|
||||||
)
|
|
||||||
|
|
||||||
SENSORS_OLD: tuple[OwletSensorEntityDescription, ...] = (
|
|
||||||
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",
|
||||||
@@ -101,6 +96,7 @@ SENSORS_OLD: 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,
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -118,13 +114,20 @@ async def async_setup_entry(
|
|||||||
|
|
||||||
sensors = []
|
sensors = []
|
||||||
|
|
||||||
sensor_list = SENSORS_ALL
|
|
||||||
for coordinator in coordinators:
|
for coordinator in coordinators:
|
||||||
if coordinator.sock.revision < 5:
|
sensors.extend([
|
||||||
sensor_list += SENSORS_OLD
|
OwletSensor(coordinator, sensor)
|
||||||
|
for sensor in SENSORS
|
||||||
|
if sensor.key in coordinator.sock.properties
|
||||||
|
])
|
||||||
|
|
||||||
for sensor in sensor_list:
|
if OwletSleepSensor.entity_description.key in coordinator.sock.properties:
|
||||||
sensors.append(OwletSensor(coordinator, sensor))
|
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)
|
||||||
|
|
||||||
@@ -142,32 +145,75 @@ 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]:
|
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,
|
||||||
|
state_class=SensorStateClass.MEASUREMENT,
|
||||||
|
)
|
||||||
|
|
||||||
|
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%]"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@@ -54,14 +54,20 @@
|
|||||||
"low_ox_alrt": {
|
"low_ox_alrt": {
|
||||||
"name": "Low oxygen alert"
|
"name": "Low oxygen alert"
|
||||||
},
|
},
|
||||||
|
"crit_ox_alrt": {
|
||||||
|
"name": "Critical oxygen alert"
|
||||||
|
},
|
||||||
"low_batt_alrt": {
|
"low_batt_alrt": {
|
||||||
"name": "Low battery alert"
|
"name": "Low battery alert"
|
||||||
},
|
},
|
||||||
|
"crit_batt_alrt": {
|
||||||
|
"name": "Critical battery alert"
|
||||||
|
},
|
||||||
"lost_pwr_alrt": {
|
"lost_pwr_alrt": {
|
||||||
"name": "Lost power alert"
|
"name": "Lost power alert"
|
||||||
},
|
},
|
||||||
"sock_discon_alrt": {
|
"sock_discon_alrt": {
|
||||||
"name": "Sock diconnected alert"
|
"name": "Sock disconnected alert"
|
||||||
},
|
},
|
||||||
"sock_off": {
|
"sock_off": {
|
||||||
"name": "Sock off"
|
"name": "Sock off"
|
||||||
@@ -107,6 +113,11 @@
|
|||||||
"movementbucket": {
|
"movementbucket": {
|
||||||
"name": "Movement bucket"
|
"name": "Movement bucket"
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
"switch": {
|
||||||
|
"base_on": {
|
||||||
|
"name": "Base station on"
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
94
custom_components/owlet/switch.py
Normal file
94
custom_components/owlet/switch.py
Normal file
@@ -0,0 +1,94 @@
|
|||||||
|
"""Support for Owlet switches."""
|
||||||
|
|
||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
from collections.abc import Callable, Coroutine
|
||||||
|
from dataclasses import dataclass
|
||||||
|
from datetime import timedelta
|
||||||
|
from typing import Any
|
||||||
|
|
||||||
|
from pyowletapi.sock import Sock
|
||||||
|
|
||||||
|
from homeassistant.components.switch import SwitchEntity, SwitchEntityDescription
|
||||||
|
from homeassistant.config_entries import ConfigEntry
|
||||||
|
from homeassistant.core import HomeAssistant
|
||||||
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
||||||
|
|
||||||
|
from .const import DOMAIN
|
||||||
|
from .coordinator import OwletCoordinator
|
||||||
|
from .entity import OwletBaseEntity
|
||||||
|
|
||||||
|
SCAN_INTERVAL = timedelta(seconds=5)
|
||||||
|
PARALLEL_UPDATES = 0
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass(frozen=True, kw_only=True)
|
||||||
|
class OwletSwitchEntityDescription(SwitchEntityDescription):
|
||||||
|
"""Describes Owlet switch entity."""
|
||||||
|
|
||||||
|
turn_on_fn: Callable[[Sock], Callable[[bool], Coroutine[Any, Any, None]]]
|
||||||
|
turn_off_fn: Callable[[Sock], Callable[[bool], Coroutine[Any, Any, None]]]
|
||||||
|
available_during_charging: bool
|
||||||
|
|
||||||
|
|
||||||
|
SWITCHES: tuple[OwletSwitchEntityDescription, ...] = (
|
||||||
|
OwletSwitchEntityDescription(
|
||||||
|
key="base_station_on",
|
||||||
|
translation_key="base_on",
|
||||||
|
turn_on_fn=lambda sock: (lambda state: sock.control_base_station(state)),
|
||||||
|
turn_off_fn=lambda sock: (lambda state: sock.control_base_station(state)),
|
||||||
|
available_during_charging=False,
|
||||||
|
),
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
async def async_setup_entry(
|
||||||
|
hass: HomeAssistant,
|
||||||
|
config_entry: ConfigEntry,
|
||||||
|
async_add_entities: AddEntitiesCallback,
|
||||||
|
) -> None:
|
||||||
|
"""Set up Owlet switch based on a config entry."""
|
||||||
|
coordinators: OwletCoordinator = hass.data[DOMAIN][config_entry.entry_id].values()
|
||||||
|
|
||||||
|
switches = []
|
||||||
|
for coordinator in coordinators:
|
||||||
|
switches.extend([OwletBaseSwitch(coordinator, switch) for switch in SWITCHES])
|
||||||
|
async_add_entities(switches)
|
||||||
|
|
||||||
|
|
||||||
|
class OwletBaseSwitch(OwletBaseEntity, SwitchEntity):
|
||||||
|
"""Defines a Owlet switch."""
|
||||||
|
|
||||||
|
entity_description: OwletSwitchEntityDescription
|
||||||
|
|
||||||
|
def __init__(
|
||||||
|
self,
|
||||||
|
coordinator: OwletCoordinator,
|
||||||
|
description: OwletSwitchEntityDescription,
|
||||||
|
) -> None:
|
||||||
|
"""Initialize owlet switch platform."""
|
||||||
|
super().__init__(coordinator)
|
||||||
|
self.entity_description = description
|
||||||
|
self._attr_unique_id = f"{self.sock.serial}-{description.key}"
|
||||||
|
self._attr_is_on = False
|
||||||
|
|
||||||
|
@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
|
||||||
|
def is_on(self) -> bool:
|
||||||
|
"""Return if switch is on or off."""
|
||||||
|
return self.sock.properties[self.entity_description.key]
|
||||||
|
|
||||||
|
async def async_turn_on(self, **kwargs: Any) -> None:
|
||||||
|
"""Turn on the switch."""
|
||||||
|
await self.entity_description.turn_on_fn(self.sock)(True)
|
||||||
|
|
||||||
|
async def async_turn_off(self, **kwargs: Any) -> None:
|
||||||
|
"""Turn off the switch."""
|
||||||
|
await self.entity_description.turn_off_fn(self.sock)(False)
|
||||||
@@ -1,112 +1,123 @@
|
|||||||
{
|
{
|
||||||
"config": {
|
"config": {
|
||||||
"step": {
|
"abort": {
|
||||||
"user": {
|
"already_configured": "Device already configured",
|
||||||
"data": {
|
"reauth_successful": "Reauthentication successful"
|
||||||
"region": "Region",
|
},
|
||||||
"username": "Email",
|
"error": {
|
||||||
"password": "Password"
|
"cannot_connect": "Failed to connect",
|
||||||
|
"invalid_credentials": "Entered credentials are incorrect",
|
||||||
|
"invalid_email": "Entered email address is incorrect",
|
||||||
|
"invalid_password": "Entered password is incorrect",
|
||||||
|
"unknown": "Unknown error occured"
|
||||||
|
},
|
||||||
|
"step": {
|
||||||
|
"reauth_confirm": {
|
||||||
|
"data": {
|
||||||
|
"password": "Password"
|
||||||
|
},
|
||||||
|
"title": "Reauthentiaction required for Owlet"
|
||||||
|
},
|
||||||
|
"user": {
|
||||||
|
"data": {
|
||||||
|
"password": "Password",
|
||||||
|
"region": "Region",
|
||||||
|
"username": "Email"
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
},
|
|
||||||
"reauth_confirm": {
|
|
||||||
"title": "Reauthentiaction required for Owlet",
|
|
||||||
"data": {
|
|
||||||
"password": "Password"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
"error": {
|
"entity": {
|
||||||
"cannot_connect": "[%key:common::config_flow::error::cannot_connect%]",
|
"binary_sensor": {
|
||||||
"invalid_email": "Entered email address is incorrect",
|
"awake": {
|
||||||
"invalid_password": "Entered password is incorrect",
|
"name": "Awake"
|
||||||
"invalid_credentials": "Entered credentials are incorrect",
|
},
|
||||||
"unknown": "Unknown error occured"
|
"charging": {
|
||||||
},
|
"name": "Charging"
|
||||||
"abort": {
|
},
|
||||||
"already_configured": "Device already configured",
|
"crit_batt_alrt": {
|
||||||
"reauth_successful": "Reauthentication successful"
|
"name": "Critical battery alert"
|
||||||
}
|
},
|
||||||
},
|
"crit_ox_alrt": {
|
||||||
"options": {
|
"name": "Critical oxygen alert"
|
||||||
"step": {
|
},
|
||||||
"init": {
|
"high_hr_alrt": {
|
||||||
"title": "Configure options for Owlet",
|
"name": "High heart rate alert"
|
||||||
"data": {
|
},
|
||||||
"pollinterval": "Polling interval in seconds, min 10"
|
"high_ox_alrt": {
|
||||||
|
"name": "High oxygen alert"
|
||||||
|
},
|
||||||
|
"lost_pwr_alrt": {
|
||||||
|
"name": "Lost power alert"
|
||||||
|
},
|
||||||
|
"low_batt_alrt": {
|
||||||
|
"name": "Low battery alert"
|
||||||
|
},
|
||||||
|
"low_hr_alrt": {
|
||||||
|
"name": "Low heart rate alert"
|
||||||
|
},
|
||||||
|
"low_ox_alrt": {
|
||||||
|
"name": "Low oxygen alert"
|
||||||
|
},
|
||||||
|
"sock_discon_alrt": {
|
||||||
|
"name": "Sock disconnected alert"
|
||||||
|
},
|
||||||
|
"sock_off": {
|
||||||
|
"name": "Sock off"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"sensor": {
|
||||||
|
"batterymin": {
|
||||||
|
"name": "Battery remaining"
|
||||||
|
},
|
||||||
|
"batterypercent": {
|
||||||
|
"name": "Battery percentage"
|
||||||
|
},
|
||||||
|
"heartrate": {
|
||||||
|
"name": "Heart rate"
|
||||||
|
},
|
||||||
|
"movement": {
|
||||||
|
"name": "Movement"
|
||||||
|
},
|
||||||
|
"movementbucket": {
|
||||||
|
"name": "Movement bucket"
|
||||||
|
},
|
||||||
|
"o2saturation": {
|
||||||
|
"name": "O2 saturation"
|
||||||
|
},
|
||||||
|
"o2saturation10a": {
|
||||||
|
"name": "O2 saturation 10 minute average"
|
||||||
|
},
|
||||||
|
"signalstrength": {
|
||||||
|
"name": "Signal strength"
|
||||||
|
},
|
||||||
|
"skintemp": {
|
||||||
|
"name": "Skin temperature"
|
||||||
|
},
|
||||||
|
"sleepstate": {
|
||||||
|
"name": "Sleep state",
|
||||||
|
"state": {
|
||||||
|
"awake": "Awake",
|
||||||
|
"deep_sleep": "Deep sleep",
|
||||||
|
"light_sleep": "Light sleep",
|
||||||
|
"unknown": "Unknown"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"switch": {
|
||||||
|
"base_on": {
|
||||||
|
"name": "Base station on"
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"entity": {
|
|
||||||
"binary_sensor": {
|
|
||||||
"charging": {
|
|
||||||
"name": "Charging"
|
|
||||||
},
|
|
||||||
"high_hr_alrt": {
|
|
||||||
"name": "High heart rate alert"
|
|
||||||
},
|
|
||||||
"low_hr_alrt": {
|
|
||||||
"name": "Low heart rate alert"
|
|
||||||
},
|
|
||||||
"high_ox_alrt": {
|
|
||||||
"name": "High oxygen alert"
|
|
||||||
},
|
|
||||||
"low_ox_alrt": {
|
|
||||||
"name": "Low oxygen alert"
|
|
||||||
},
|
|
||||||
"low_batt_alrt": {
|
|
||||||
"name": "Low battery alert"
|
|
||||||
},
|
|
||||||
"lost_pwr_alrt": {
|
|
||||||
"name": "Lost power alert"
|
|
||||||
},
|
|
||||||
"sock_discon_alrt": {
|
|
||||||
"name": "Sock diconnected alert"
|
|
||||||
},
|
|
||||||
"sock_off": {
|
|
||||||
"name": "Sock off"
|
|
||||||
},
|
|
||||||
"awake": {
|
|
||||||
"name": "Awake"
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
"sensor": {
|
"options": {
|
||||||
"batterypercent": {
|
"step": {
|
||||||
"name": "Battery percentage"
|
"init": {
|
||||||
},
|
"data": {
|
||||||
"signalstrength": {
|
"pollinterval": "Polling interval in seconds, min 10"
|
||||||
"name": "Signal strength"
|
},
|
||||||
},
|
"title": "Configure options for Owlet"
|
||||||
"o2saturation": {
|
}
|
||||||
"name": "O2 saturation"
|
|
||||||
},
|
|
||||||
"o2saturation10a": {
|
|
||||||
"name": "O2 saturation 10 minute average"
|
|
||||||
},
|
|
||||||
"heartrate": {
|
|
||||||
"name": "Heart rate"
|
|
||||||
},
|
|
||||||
"batterymin": {
|
|
||||||
"name": "Battery remaining"
|
|
||||||
},
|
|
||||||
"skintemp": {
|
|
||||||
"name": "Skin temperature"
|
|
||||||
},
|
|
||||||
"sleepstate": {
|
|
||||||
"name": "Sleep state",
|
|
||||||
"state": {
|
|
||||||
"unknown": "Unknown",
|
|
||||||
"awake": "Awake",
|
|
||||||
"light_sleep": "Light sleep",
|
|
||||||
"deep_sleep": "Deep sleep"
|
|
||||||
}
|
}
|
||||||
},
|
|
||||||
"movement": {
|
|
||||||
"name": "Movement"
|
|
||||||
},
|
|
||||||
"movementbucket": {
|
|
||||||
"name": "Movement bucket"
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
}
|
||||||
123
custom_components/owlet/translations/fr.json
Normal file
123
custom_components/owlet/translations/fr.json
Normal file
@@ -0,0 +1,123 @@
|
|||||||
|
{
|
||||||
|
"config": {
|
||||||
|
"step": {
|
||||||
|
"user": {
|
||||||
|
"data": {
|
||||||
|
"region": "Région",
|
||||||
|
"username": "Email",
|
||||||
|
"password": "Mot de passe"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"reauth_confirm": {
|
||||||
|
"title": "Réauthentification requise pour Owlet",
|
||||||
|
"data": {
|
||||||
|
"password": "Mot de passe"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"error": {
|
||||||
|
"cannot_connect": "N'a pas réussi à se connecter",
|
||||||
|
"invalid_email": "L'adresse e-mail saisie est incorrecte",
|
||||||
|
"invalid_password": "Le mot de passe saisi est incorrect",
|
||||||
|
"invalid_credentials": "Les informations d'identification saisies sont incorrectes",
|
||||||
|
"unknown": "Une erreur inconnue est survenue"
|
||||||
|
},
|
||||||
|
"abort": {
|
||||||
|
"already_configured": "Appareil déjà configuré",
|
||||||
|
"reauth_successful": "Réauthentification réussie"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"options": {
|
||||||
|
"step": {
|
||||||
|
"init": {
|
||||||
|
"title": "Configurer les options pour Owlet",
|
||||||
|
"data": {
|
||||||
|
"pollinterval": "Intervalle de sondage en secondes, minimum 10"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"entity": {
|
||||||
|
"binary_sensor": {
|
||||||
|
"charging": {
|
||||||
|
"name": "En charge"
|
||||||
|
},
|
||||||
|
"high_hr_alrt": {
|
||||||
|
"name": "Alerte fréquence cardiaque élevée"
|
||||||
|
},
|
||||||
|
"low_hr_alrt": {
|
||||||
|
"name": "Alerte fréquence cardiaque basse"
|
||||||
|
},
|
||||||
|
"high_ox_alrt": {
|
||||||
|
"name": "Alerte oxygène élevé"
|
||||||
|
},
|
||||||
|
"low_ox_alrt": {
|
||||||
|
"name": "Alerte oxygène faible"
|
||||||
|
},
|
||||||
|
"crit_ox_alrt": {
|
||||||
|
"name": "Alerte oxygène critique"
|
||||||
|
},
|
||||||
|
"low_batt_alrt": {
|
||||||
|
"name": "Alerte batterie faible"
|
||||||
|
},
|
||||||
|
"crit_batt_alrt": {
|
||||||
|
"name": "Alerte batterie critique"
|
||||||
|
},
|
||||||
|
"lost_pwr_alrt": {
|
||||||
|
"name": "Alerte perte d'alimentation"
|
||||||
|
},
|
||||||
|
"sock_discon_alrt": {
|
||||||
|
"name": "Alerte chaussette déconnectée"
|
||||||
|
},
|
||||||
|
"sock_off": {
|
||||||
|
"name": "Chaussette retirée"
|
||||||
|
},
|
||||||
|
"awake": {
|
||||||
|
"name": "Réveillé"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"sensor": {
|
||||||
|
"batterypercent": {
|
||||||
|
"name": "Pourcentage de batterie"
|
||||||
|
},
|
||||||
|
"signalstrength": {
|
||||||
|
"name": "Force du signal"
|
||||||
|
},
|
||||||
|
"o2saturation": {
|
||||||
|
"name": "Saturation O2"
|
||||||
|
},
|
||||||
|
"o2saturation10a": {
|
||||||
|
"name": "Moyenne de saturation O2 sur 10 minutes"
|
||||||
|
},
|
||||||
|
"heartrate": {
|
||||||
|
"name": "Fréquence cardiaque"
|
||||||
|
},
|
||||||
|
"batterymin": {
|
||||||
|
"name": "Autonomie de la batterie restante"
|
||||||
|
},
|
||||||
|
"skintemp": {
|
||||||
|
"name": "Température de la peau"
|
||||||
|
},
|
||||||
|
"sleepstate": {
|
||||||
|
"name": "État de sommeil",
|
||||||
|
"state": {
|
||||||
|
"unknown": "Inconnu",
|
||||||
|
"awake": "Réveillé",
|
||||||
|
"light_sleep": "Sommeil léger",
|
||||||
|
"deep_sleep": "Sommeil profond"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"movement": {
|
||||||
|
"name": "Mouvement"
|
||||||
|
},
|
||||||
|
"movementbucket": {
|
||||||
|
"name": "Seuil de mouvement"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"switch": {
|
||||||
|
"base_on": {
|
||||||
|
"name": "Station de base allumée"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -16,7 +16,7 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"error": {
|
"error": {
|
||||||
"cannot_connect": "[%key:common::config_flow::error::cannot_connect%]",
|
"cannot_connect": "Failed to connect",
|
||||||
"invalid_email": "Entered email address is incorrect",
|
"invalid_email": "Entered email address is incorrect",
|
||||||
"invalid_password": "Entered password is incorrect",
|
"invalid_password": "Entered password is incorrect",
|
||||||
"invalid_credentials": "Entered credentials are incorrect",
|
"invalid_credentials": "Entered credentials are incorrect",
|
||||||
@@ -54,14 +54,20 @@
|
|||||||
"low_ox_alrt": {
|
"low_ox_alrt": {
|
||||||
"name": "Low oxygen alert"
|
"name": "Low oxygen alert"
|
||||||
},
|
},
|
||||||
|
"crit_ox_alrt": {
|
||||||
|
"name": "Critical oxygen alert"
|
||||||
|
},
|
||||||
"low_batt_alrt": {
|
"low_batt_alrt": {
|
||||||
"name": "Low battery alert"
|
"name": "Low battery alert"
|
||||||
},
|
},
|
||||||
|
"crit_batt_alrt": {
|
||||||
|
"name": "Critical battery alert"
|
||||||
|
},
|
||||||
"lost_pwr_alrt": {
|
"lost_pwr_alrt": {
|
||||||
"name": "Lost power alert"
|
"name": "Lost power alert"
|
||||||
},
|
},
|
||||||
"sock_discon_alrt": {
|
"sock_discon_alrt": {
|
||||||
"name": "Sock diconnected alert"
|
"name": "Sock disconnected alert"
|
||||||
},
|
},
|
||||||
"sock_off": {
|
"sock_off": {
|
||||||
"name": "Sock off"
|
"name": "Sock off"
|
||||||
@@ -107,6 +113,11 @@
|
|||||||
"movementbucket": {
|
"movementbucket": {
|
||||||
"name": "Movement bucket"
|
"name": "Movement bucket"
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
"switch": {
|
||||||
|
"base_on": {
|
||||||
|
"name": "Base station on"
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"name": "Owlet",
|
"name": "Owlet",
|
||||||
"hacs": "1.32.1",
|
"hacs": "1.32.1",
|
||||||
"homeassistant": "2023.04.1",
|
"homeassistant": "2024.1",
|
||||||
"zip_release": true,
|
"zip_release": true,
|
||||||
"filename": "owlet.zip"
|
"filename": "owlet.zip"
|
||||||
}
|
}
|
||||||
|
|||||||
6
info.md
6
info.md
@@ -8,9 +8,7 @@
|
|||||||
[![hacs][hacsbadge]][hacs]
|
[![hacs][hacsbadge]][hacs]
|
||||||
[![Project Maintenance][maintenance-shield]][user_profile]
|
[![Project Maintenance][maintenance-shield]][user_profile]
|
||||||
|
|
||||||
A custom component for the Owlet smart sock, currently this only supports the owlet smart sock 3.
|
A custom component for the Owlet smart sock
|
||||||
|
|
||||||
If you have a smart sock 2 and would like to contribute then please do so.
|
|
||||||
|
|
||||||
## Installation
|
## Installation
|
||||||
|
|
||||||
@@ -34,4 +32,4 @@ If you have a smart sock 2 and would like to contribute then please do so.
|
|||||||
[releases]: https://github.com/ryanbdclark/owlet/releases
|
[releases]: https://github.com/ryanbdclark/owlet/releases
|
||||||
[user_profile]: https://github.com/ryanbdclark
|
[user_profile]: https://github.com/ryanbdclark
|
||||||
[add-integration]: https://my.home-assistant.io/redirect/config_flow_start?domain=owlet
|
[add-integration]: https://my.home-assistant.io/redirect/config_flow_start?domain=owlet
|
||||||
[add-integration-badge]: https://my.home-assistant.io/badges/config_flow_start.svg
|
[add-integration-badge]: https://my.home-assistant.io/badges/config_flow_start.svg
|
||||||
|
|||||||
@@ -1052,4 +1052,4 @@
|
|||||||
"expiry": 200,
|
"expiry": 200,
|
||||||
"refresh": "new_refresh_token"
|
"refresh": "new_refresh_token"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
1211
tests/fixtures/update_properties_v2.json
vendored
Normal file
1211
tests/fixtures/update_properties_v2.json
vendored
Normal file
File diff suppressed because it is too large
Load Diff
188
tests/test_binary_sensor.py
Normal file
188
tests/test_binary_sensor.py
Normal file
@@ -0,0 +1,188 @@
|
|||||||
|
"""Test Owlet Sensor."""
|
||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
from homeassistant.core import HomeAssistant
|
||||||
|
|
||||||
|
from . import async_init_integration
|
||||||
|
|
||||||
|
|
||||||
|
async def test_sensors_asleep(hass: HomeAssistant) -> None:
|
||||||
|
"""Test sensor values."""
|
||||||
|
await async_init_integration(
|
||||||
|
hass, properties_fixture="update_properties_asleep.json"
|
||||||
|
)
|
||||||
|
|
||||||
|
assert len(hass.states.async_all("binary_sensor")) == 10
|
||||||
|
|
||||||
|
assert hass.states.get("binary_sensor.owlet_baby_care_sock_charging").state == "off"
|
||||||
|
assert (
|
||||||
|
hass.states.get(
|
||||||
|
"binary_sensor.owlet_baby_care_sock_high_heart_rate_alert"
|
||||||
|
).state
|
||||||
|
== "off"
|
||||||
|
)
|
||||||
|
assert (
|
||||||
|
hass.states.get("binary_sensor.owlet_baby_care_sock_low_heart_rate_alert").state
|
||||||
|
== "off"
|
||||||
|
)
|
||||||
|
assert (
|
||||||
|
hass.states.get("binary_sensor.owlet_baby_care_sock_high_oxygen_alert").state
|
||||||
|
== "off"
|
||||||
|
)
|
||||||
|
assert (
|
||||||
|
hass.states.get("binary_sensor.owlet_baby_care_sock_low_oxygen_alert").state
|
||||||
|
== "off"
|
||||||
|
)
|
||||||
|
assert (
|
||||||
|
hass.states.get("binary_sensor.owlet_baby_care_sock_low_battery_alert").state
|
||||||
|
== "off"
|
||||||
|
)
|
||||||
|
assert (
|
||||||
|
hass.states.get("binary_sensor.owlet_baby_care_sock_lost_power_alert").state
|
||||||
|
== "off"
|
||||||
|
)
|
||||||
|
assert (
|
||||||
|
hass.states.get(
|
||||||
|
"binary_sensor.owlet_baby_care_sock_sock_disconnected_alert"
|
||||||
|
).state
|
||||||
|
== "off"
|
||||||
|
)
|
||||||
|
assert hass.states.get("binary_sensor.owlet_baby_care_sock_sock_off").state == "off"
|
||||||
|
assert hass.states.get("binary_sensor.owlet_baby_care_sock_awake").state == "off"
|
||||||
|
|
||||||
|
|
||||||
|
async def test_sensors_awake(hass: HomeAssistant) -> None:
|
||||||
|
"""Test sensor values."""
|
||||||
|
await async_init_integration(
|
||||||
|
hass, properties_fixture="update_properties_awake.json"
|
||||||
|
)
|
||||||
|
|
||||||
|
assert len(hass.states.async_all("binary_sensor")) == 10
|
||||||
|
|
||||||
|
assert hass.states.get("binary_sensor.owlet_baby_care_sock_charging").state == "off"
|
||||||
|
assert (
|
||||||
|
hass.states.get(
|
||||||
|
"binary_sensor.owlet_baby_care_sock_high_heart_rate_alert"
|
||||||
|
).state
|
||||||
|
== "on"
|
||||||
|
)
|
||||||
|
assert (
|
||||||
|
hass.states.get("binary_sensor.owlet_baby_care_sock_low_heart_rate_alert").state
|
||||||
|
== "on"
|
||||||
|
)
|
||||||
|
assert (
|
||||||
|
hass.states.get("binary_sensor.owlet_baby_care_sock_high_oxygen_alert").state
|
||||||
|
== "on"
|
||||||
|
)
|
||||||
|
assert (
|
||||||
|
hass.states.get("binary_sensor.owlet_baby_care_sock_low_oxygen_alert").state
|
||||||
|
== "on"
|
||||||
|
)
|
||||||
|
assert (
|
||||||
|
hass.states.get("binary_sensor.owlet_baby_care_sock_low_battery_alert").state
|
||||||
|
== "on"
|
||||||
|
)
|
||||||
|
assert (
|
||||||
|
hass.states.get("binary_sensor.owlet_baby_care_sock_lost_power_alert").state
|
||||||
|
== "on"
|
||||||
|
)
|
||||||
|
assert (
|
||||||
|
hass.states.get(
|
||||||
|
"binary_sensor.owlet_baby_care_sock_sock_disconnected_alert"
|
||||||
|
).state
|
||||||
|
== "off"
|
||||||
|
)
|
||||||
|
assert hass.states.get("binary_sensor.owlet_baby_care_sock_sock_off").state == "off"
|
||||||
|
assert hass.states.get("binary_sensor.owlet_baby_care_sock_awake").state == "on"
|
||||||
|
|
||||||
|
|
||||||
|
async def test_sensors_charging(hass: HomeAssistant) -> None:
|
||||||
|
"""Test sensor values."""
|
||||||
|
await async_init_integration(
|
||||||
|
hass, properties_fixture="update_properties_charging.json"
|
||||||
|
)
|
||||||
|
|
||||||
|
assert len(hass.states.async_all("binary_sensor")) == 10
|
||||||
|
|
||||||
|
assert hass.states.get("binary_sensor.owlet_baby_care_sock_charging").state == "on"
|
||||||
|
assert (
|
||||||
|
hass.states.get(
|
||||||
|
"binary_sensor.owlet_baby_care_sock_high_heart_rate_alert"
|
||||||
|
).state
|
||||||
|
== "off"
|
||||||
|
)
|
||||||
|
assert (
|
||||||
|
hass.states.get("binary_sensor.owlet_baby_care_sock_low_heart_rate_alert").state
|
||||||
|
== "off"
|
||||||
|
)
|
||||||
|
assert (
|
||||||
|
hass.states.get("binary_sensor.owlet_baby_care_sock_high_oxygen_alert").state
|
||||||
|
== "off"
|
||||||
|
)
|
||||||
|
assert (
|
||||||
|
hass.states.get("binary_sensor.owlet_baby_care_sock_low_oxygen_alert").state
|
||||||
|
== "off"
|
||||||
|
)
|
||||||
|
assert (
|
||||||
|
hass.states.get("binary_sensor.owlet_baby_care_sock_low_battery_alert").state
|
||||||
|
== "off"
|
||||||
|
)
|
||||||
|
assert (
|
||||||
|
hass.states.get("binary_sensor.owlet_baby_care_sock_lost_power_alert").state
|
||||||
|
== "off"
|
||||||
|
)
|
||||||
|
assert (
|
||||||
|
hass.states.get(
|
||||||
|
"binary_sensor.owlet_baby_care_sock_sock_disconnected_alert"
|
||||||
|
).state
|
||||||
|
== "off"
|
||||||
|
)
|
||||||
|
assert hass.states.get("binary_sensor.owlet_baby_care_sock_sock_off").state == "off"
|
||||||
|
assert (
|
||||||
|
hass.states.get("binary_sensor.owlet_baby_care_sock_awake").state == "unknown"
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
async def test_sensors_v2(hass: HomeAssistant) -> None:
|
||||||
|
"""Test sensor values."""
|
||||||
|
await async_init_integration(hass, properties_fixture="update_properties_v2.json")
|
||||||
|
assert len(hass.states.async_all("binary_sensor")) == 9
|
||||||
|
|
||||||
|
assert hass.states.get("binary_sensor.owlet_baby_care_sock_charging").state == "off"
|
||||||
|
assert (
|
||||||
|
hass.states.get(
|
||||||
|
"binary_sensor.owlet_baby_care_sock_high_heart_rate_alert"
|
||||||
|
).state
|
||||||
|
== "off"
|
||||||
|
)
|
||||||
|
assert (
|
||||||
|
hass.states.get("binary_sensor.owlet_baby_care_sock_low_heart_rate_alert").state
|
||||||
|
== "off"
|
||||||
|
)
|
||||||
|
assert (
|
||||||
|
hass.states.get("binary_sensor.owlet_baby_care_sock_low_oxygen_alert").state
|
||||||
|
== "off"
|
||||||
|
)
|
||||||
|
assert (
|
||||||
|
hass.states.get("binary_sensor.owlet_baby_care_sock_low_battery_alert").state
|
||||||
|
== "off"
|
||||||
|
)
|
||||||
|
assert (
|
||||||
|
hass.states.get(
|
||||||
|
"binary_sensor.owlet_baby_care_sock_critical_battery_alert"
|
||||||
|
).state
|
||||||
|
== "off"
|
||||||
|
)
|
||||||
|
assert (
|
||||||
|
hass.states.get(
|
||||||
|
"binary_sensor.owlet_baby_care_sock_critical_oxygen_alert"
|
||||||
|
).state
|
||||||
|
== "off"
|
||||||
|
)
|
||||||
|
assert (
|
||||||
|
hass.states.get(
|
||||||
|
"binary_sensor.owlet_baby_care_sock_sock_disconnected_alert"
|
||||||
|
).state
|
||||||
|
== "off"
|
||||||
|
)
|
||||||
|
assert hass.states.get("binary_sensor.owlet_baby_care_sock_sock_off").state == "off"
|
||||||
@@ -70,7 +70,7 @@ async def test_flow_wrong_password(hass: HomeAssistant) -> None:
|
|||||||
user_input=CONF_INPUT,
|
user_input=CONF_INPUT,
|
||||||
)
|
)
|
||||||
assert result["type"] == FlowResultType.FORM
|
assert result["type"] == FlowResultType.FORM
|
||||||
assert result["errors"] == {"base": "invalid_password"}
|
assert result["errors"] == {"password": "invalid_password"}
|
||||||
|
|
||||||
|
|
||||||
async def test_flow_wrong_email(hass: HomeAssistant) -> None:
|
async def test_flow_wrong_email(hass: HomeAssistant) -> None:
|
||||||
@@ -88,7 +88,7 @@ async def test_flow_wrong_email(hass: HomeAssistant) -> None:
|
|||||||
user_input=CONF_INPUT,
|
user_input=CONF_INPUT,
|
||||||
)
|
)
|
||||||
assert result["type"] == FlowResultType.FORM
|
assert result["type"] == FlowResultType.FORM
|
||||||
assert result["errors"] == {"base": "invalid_email"}
|
assert result["errors"] == {"username": "invalid_email"}
|
||||||
|
|
||||||
|
|
||||||
async def test_flow_credentials_error(hass: HomeAssistant) -> None:
|
async def test_flow_credentials_error(hass: HomeAssistant) -> None:
|
||||||
@@ -193,7 +193,7 @@ async def test_reauth_invalid_password(hass: HomeAssistant) -> None:
|
|||||||
|
|
||||||
assert result["type"] == FlowResultType.FORM
|
assert result["type"] == FlowResultType.FORM
|
||||||
assert result["step_id"] == "reauth_confirm"
|
assert result["step_id"] == "reauth_confirm"
|
||||||
assert result["errors"] == {"base": "invalid_password"}
|
assert result["errors"] == {"password": "invalid_password"}
|
||||||
|
|
||||||
|
|
||||||
async def test_reauth_unknown_error(hass: HomeAssistant) -> None:
|
async def test_reauth_unknown_error(hass: HomeAssistant) -> None:
|
||||||
|
|||||||
@@ -43,7 +43,7 @@ async def test_async_setup_entry(hass: HomeAssistant) -> None:
|
|||||||
|
|
||||||
entities = er.async_entries_for_device(entity_registry, device_entry.id)
|
entities = er.async_entries_for_device(entity_registry, device_entry.id)
|
||||||
|
|
||||||
assert len(entities) == 8
|
assert len(entities) == 18
|
||||||
|
|
||||||
await entry.async_unload(hass)
|
await entry.async_unload(hass)
|
||||||
|
|
||||||
|
|||||||
@@ -107,3 +107,20 @@ async def test_sensors_charging(hass: HomeAssistant) -> None:
|
|||||||
== "unknown"
|
== "unknown"
|
||||||
)
|
)
|
||||||
assert hass.states.get("sensor.owlet_baby_care_sock_sleep_state").state == "unknown"
|
assert hass.states.get("sensor.owlet_baby_care_sock_sleep_state").state == "unknown"
|
||||||
|
|
||||||
|
|
||||||
|
async def test_sensors_v2(hass: HomeAssistant) -> None:
|
||||||
|
"""Test sensor values."""
|
||||||
|
await async_init_integration(hass, properties_fixture="update_properties_v2.json")
|
||||||
|
assert len(hass.states.async_all("sensor")) == 4
|
||||||
|
|
||||||
|
assert (
|
||||||
|
hass.states.get("sensor.owlet_baby_care_sock_battery_percentage").state == "29"
|
||||||
|
)
|
||||||
|
assert hass.states.get("sensor.owlet_baby_care_sock_heart_rate").state == "145"
|
||||||
|
assert hass.states.get("sensor.owlet_baby_care_sock_o2_saturation").state == "99"
|
||||||
|
|
||||||
|
assert (
|
||||||
|
hass.states.get("sensor.owlet_baby_care_sock_signal_strength").state == "98.0"
|
||||||
|
)
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user