9 Commits

Author SHA1 Message Date
RyanClark123
bd6a315b00 Bumping to pyowletapi 2023.5.25
#### Fix
* Bumping to pyowletapi 2023.5.25
2023-05-17 19:23:37 +01:00
RyanClark123
0cf3afef85 Bumping pyowlet, reauth config entry fix
#### Fix
* Bumping to pyowletapi 2023.5.24
* Reauthing now no longer re adds users' password to config entry
2023-05-17 18:59:38 +01:00
ryanbdclark
fa06157fe2 Update const.py
###Fix
* Error in sleep states constant corrected
2023-05-16 21:14:46 +01:00
RyanClark123
2b21188e73 Update CHANGELOG 2023-05-16 15:21:41 +01:00
RyanClark123
9b3392bdbc Added new sensors
###Feature
* Added new sensors, binary sensor awake, is on if baby awake, off otherwise. Sensor sleep state, shows baby sleep state, options are awake, light sleep, deep sleep
2023-05-16 15:06:01 +01:00
RyanClark123
dc710a1783 Integration now uses refresh token
-Makes use of pyowletapi 2023.5.23 to use the refresh token to reauthenticate, integration now no longer stores users password
-Default polling interval changed to 5 seconds to match owlet app
2023-05-16 14:05:26 +01:00
RyanClark123
8d9299a8f4 Updated changelog for new release
Updated changelog for new release
2023-05-15 15:56:37 +01:00
RyanClark123
228d54b641 bumping pyowletapi version
### Fix
* Bump pyowletapi to 2023.5.21, fix for unawaited authentication call believe to be causing an issue reauthenticating when token had expired
* Github username change reflected, won't change again
2023-05-15 11:21:50 +01:00
RyanClark123
3e147aa914 Added reauth flow, much better error handling on setup
### Feature
* Now supports reauthentication if credentials have expired/no longer work
* Better error handling when configuring integration, will notify user of incorrect credentials

### Fix
* Removed Owlet specific constants, now using homeassistant generic constants
* On initialisation the integration would crash when trying to update the auth token, the integration would then have to be deleted and setup again
2023-05-12 17:02:04 +01:00
11 changed files with 123 additions and 53 deletions

1
.gitignore vendored
View File

@@ -152,3 +152,4 @@ cython_debug/
#.idea/ #.idea/
/custom_components/owlet.zip /custom_components/owlet.zip
custom_components/owlet/owlet.zip

View File

@@ -1,6 +1,26 @@
# Changelog # Changelog
<!--next-version-placeholder--> <!--next-version-placeholder-->
## 2023-05-4 (2023-05-17)
#### Fix
* Bumping to pyowletapi 2023.5.25
## 2023-05-3 (2023-05-17)
#### Fix
* Bumping to pyowletapi 2023.5.24
* Reauthing now no longer re adds users' password to config entry
## 2023-05-2 (2023-05-16)
#### Feature
* Integration now makes use of refresh token from pyowletapi to reauthenticate, user password in no longer stored by integration ([`dc710a1`](https://github.com/ryanbdclark/owlet/commit/dc710a1783a4cad9d6cf355240fe12ac779a87ef))
* New sensors create for baby sleep state ([`9b3392b`](https://github.com/ryanbdclark/owlet/commit/9b3392bdbcd82015ed31d3a50a517e4e22905684))
## 2023-05-1 (2023-05-15)
#### Feature
* Changed versioning to date based
### Fix
* Bumping to pyowletapi 2023.5.21 to fix issue with unawaited authentication call, should resolve issue with refreshing authentication ([`228d54b`](https://github.com/ryanbdclark/owlet/commit/228d54b6414e0b9171064254246d1f36c3af8f5b))
## v1.5.0 (2023-05-12) ## v1.5.0 (2023-05-12)
### Feature ### Feature

View File

@@ -1,8 +1,6 @@
# Owlet Custom Integration # Owlet Custom Integration
[![GitHub Release][releases-shield]][releases] [![GitHub Release][releases-shield]][releases]
![GitHub all releases][download-all]
![GitHub release (latest by RyanClark123)][download-latest]
[![GitHub Activity][commits-shield]][commits] [![GitHub Activity][commits-shield]][commits]
[![License][license-shield]][license] [![License][license-shield]][license]
@@ -16,7 +14,7 @@ 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/RyanClark123/owlet` as a custom repository. 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.
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".
@@ -34,21 +32,19 @@ This integration provides the following entities:
## Options ## Options
- Seconds between polling - Number of seconds between each call for data from the owlet cloud service, default is 10 seconds. - Seconds between polling - Number of seconds between each call for data from the owlet cloud service, default is 5 seconds.
--- ---
[commits-shield]: https://img.shields.io/github/commit-activity/w/RyanClark123/owlet?style=for-the-badge [commits-shield]: https://img.shields.io/github/commit-activity/w/ryanbdclark/owlet?style=for-the-badge
[commits]: https://github.com/RyanClark123/owlet/commits/main [commits]: https://github.com/ryanbdclark/owlet/commits/main
[hacs]: https://github.com/hacs/integration [hacs]: https://github.com/hacs/integration
[hacsbadge]: https://img.shields.io/badge/HACS-Custom-orange.svg?style=for-the-badge [hacsbadge]: https://img.shields.io/badge/HACS-Custom-orange.svg?style=for-the-badge
[license]: LICENSE [license]: LICENSE
[license-shield]: https://img.shields.io/github/license/RyanClark123/owlet.svg?style=for-the-badge [license-shield]: https://img.shields.io/github/license/ryanbdclark/owlet.svg?style=for-the-badge
[maintenance-shield]: https://img.shields.io/badge/maintainer-Ryan%20Clark%20%40RyanClark123-blue.svg?style=for-the-badge [maintenance-shield]: https://img.shields.io/badge/maintainer-Ryan%20Clark%20%40ryanbdclark-blue.svg?style=for-the-badge
[releases-shield]: https://img.shields.io/github/release/RyanClark123/owlet.svg?style=for-the-badge [releases-shield]: https://img.shields.io/github/release/ryanbdclark/owlet.svg?style=for-the-badge
[releases]: https://github.com/RyanClark123/owlet/releases [releases]: https://github.com/ryanbdclark/owlet/releases
[user_profile]: https://github.com/RyanClark123 [user_profile]: https://github.com/ryanbdclark
[download-all]: https://img.shields.io/github/downloads/RyanClark123/Owlet/total?style=for-the-badge
[download-latest]: https://img.shields.io/github/downloads/RyanClark123/Owlet/latest/total?style=for-the-badge
[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

View File

@@ -5,7 +5,7 @@ import logging
from pyowletapi.api import OwletAPI from pyowletapi.api import OwletAPI
from pyowletapi.sock import Sock from pyowletapi.sock import Sock
from pyowletapi.exceptions import OwletConnectionError from pyowletapi.exceptions import OwletAuthenticationError
from homeassistant.config_entries import ConfigEntry, ConfigEntryAuthFailed from homeassistant.config_entries import ConfigEntry, ConfigEntryAuthFailed
from homeassistant.const import ( from homeassistant.const import (
@@ -21,6 +21,7 @@ from homeassistant.helpers.aiohttp_client import async_get_clientsession
from .const import ( from .const import (
DOMAIN, DOMAIN,
CONF_OWLET_EXPIRY, CONF_OWLET_EXPIRY,
CONF_OWLET_REFRESH,
SUPPORTED_VERSIONS, SUPPORTED_VERSIONS,
) )
from .coordinator import OwletCoordinator from .coordinator import OwletCoordinator
@@ -35,12 +36,11 @@ async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
hass.data.setdefault(DOMAIN, {}) hass.data.setdefault(DOMAIN, {})
owlet_api = OwletAPI( owlet_api = OwletAPI(
entry.data[CONF_REGION], region=entry.data[CONF_REGION],
entry.data[CONF_USERNAME], token=entry.data[CONF_API_TOKEN],
entry.data[CONF_PASSWORD], expiry=entry.data[CONF_OWLET_EXPIRY],
entry.data[CONF_API_TOKEN], refresh=entry.data[CONF_OWLET_REFRESH],
entry.data[CONF_OWLET_EXPIRY], session=async_get_clientsession(hass),
async_get_clientsession(hass),
) )
try: try:
@@ -54,7 +54,7 @@ async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
for device in await owlet_api.get_devices(SUPPORTED_VERSIONS) for device in await owlet_api.get_devices(SUPPORTED_VERSIONS)
} }
except OwletConnectionError as err: except OwletAuthenticationError as err:
_LOGGER.error("Credentials no longer valid, please setup owlet again") _LOGGER.error("Credentials no longer valid, please setup owlet again")
raise ConfigEntryAuthFailed( raise ConfigEntryAuthFailed(
f"Credentials expired for {entry.data[CONF_USERNAME]}" f"Credentials expired for {entry.data[CONF_USERNAME]}"

View File

@@ -86,6 +86,12 @@ SENSORS: tuple[OwletBinarySensorEntityDescription, ...] = (
device_class=BinarySensorDeviceClass.POWER, device_class=BinarySensorDeviceClass.POWER,
element="sock_off", element="sock_off",
), ),
OwletBinarySensorEntityDescription(
key="awake",
name="Awake",
element="sleep_state",
icon="mdi:sleep",
),
) )
@@ -119,4 +125,14 @@ class OwletBinarySensor(OwletBaseEntity, BinarySensorEntity):
@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."""
return self.sock.properties[self.entity_description.element] state = self.sock.properties[self.entity_description.element]
if self.entity_description.element == "sleep_state":
if self.sock.properties["charging"]:
return None
if state in [8, 15]:
state = False
else:
state = True
return state

View File

@@ -32,6 +32,7 @@ from .const import (
CONF_OWLET_EXPIRY, CONF_OWLET_EXPIRY,
POLLING_INTERVAL, POLLING_INTERVAL,
SUPPORTED_VERSIONS, SUPPORTED_VERSIONS,
CONF_OWLET_REFRESH,
) )
_LOGGER = logging.getLogger(__name__) _LOGGER = logging.getLogger(__name__)
@@ -87,9 +88,9 @@ class OwletConfigFlow(config_entries.ConfigFlow, domain=DOMAIN):
data={ data={
CONF_REGION: self._region, CONF_REGION: self._region,
CONF_USERNAME: self._username, CONF_USERNAME: self._username,
CONF_PASSWORD: self._password,
CONF_API_TOKEN: token[CONF_API_TOKEN], CONF_API_TOKEN: token[CONF_API_TOKEN],
CONF_OWLET_EXPIRY: token[CONF_OWLET_EXPIRY], 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},
) )
@@ -137,10 +138,8 @@ class OwletConfigFlow(config_entries.ConfigFlow, domain=DOMAIN):
try: try:
token = await owlet_api.authenticate() token = await owlet_api.authenticate()
if token: if token:
user_input[CONF_API_TOKEN] = token[CONF_API_TOKEN]
user_input[CONF_OWLET_EXPIRY] = token[CONF_OWLET_EXPIRY]
self.hass.config_entries.async_update_entry( self.hass.config_entries.async_update_entry(
self.reauth_entry, data={**entry_data, **user_input} 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)

View File

@@ -3,7 +3,9 @@
DOMAIN = "owlet" DOMAIN = "owlet"
CONF_OWLET_EXPIRY = "expiry" CONF_OWLET_EXPIRY = "expiry"
CONF_OWLET_REFRESH = "refresh"
SUPPORTED_VERSIONS = [3] SUPPORTED_VERSIONS = [3]
POLLING_INTERVAL = 10 POLLING_INTERVAL = 5
MANUFACTURER = "Owlet Baby Care" MANUFACTURER = "Owlet Baby Care"
SLEEP_STATES = {0: "Unknown", 1: "Awake", 8: "Light Sleep", 15: "Deep Sleep"}

View File

@@ -5,16 +5,18 @@ from datetime import timedelta
import logging import logging
from pyowletapi.sock import Sock from pyowletapi.sock import Sock
from pyowletapi.exceptions import OwletError from pyowletapi.exceptions import (
OwletError,
OwletConnectionError,
OwletAuthenticationError,
)
from homeassistant.core import HomeAssistant from homeassistant.core import HomeAssistant
from homeassistant.helpers.entity import DeviceInfo from homeassistant.helpers.entity import DeviceInfo
from homeassistant.helpers.update_coordinator import DataUpdateCoordinator, UpdateFailed from homeassistant.helpers.update_coordinator import DataUpdateCoordinator, UpdateFailed
from homeassistant.const import CONF_API_TOKEN
from .const import ( from .const import DOMAIN, MANUFACTURER, CONF_OWLET_EXPIRY, CONF_OWLET_REFRESH
DOMAIN,
MANUFACTURER,
)
_LOGGER = logging.getLogger(__name__) _LOGGER = logging.getLogger(__name__)
@@ -49,5 +51,16 @@ class OwletCoordinator(DataUpdateCoordinator):
"""Fetch the data from the device.""" """Fetch the data from the device."""
try: try:
await self.sock.update_properties() await self.sock.update_properties()
except OwletError as err: tokens = await self.sock.api.tokens_changed(
{
CONF_API_TOKEN: self.config_entry.data[CONF_API_TOKEN],
CONF_OWLET_EXPIRY: self.config_entry.data[CONF_OWLET_EXPIRY],
CONF_OWLET_REFRESH: self.config_entry.data[CONF_OWLET_REFRESH],
}
)
if tokens:
self.hass.config_entries.async_update_entry(
self.config_entry, data={**self.config_entry.data, **tokens}
)
except (OwletError, OwletConnectionError, OwletAuthenticationError) as err:
raise UpdateFailed(err) from err raise UpdateFailed(err) from err

View File

@@ -2,7 +2,7 @@
"domain": "owlet", "domain": "owlet",
"name": "Owlet Smart Sock", "name": "Owlet Smart Sock",
"codeowners": [ "codeowners": [
"@RyanClark123" "@ryanbdclark"
], ],
"config_flow": true, "config_flow": true,
"dependencies": [], "dependencies": [],
@@ -10,7 +10,7 @@
"homekit": {}, "homekit": {},
"iot_class": "cloud_polling", "iot_class": "cloud_polling",
"requirements": [ "requirements": [
"pyowletapi==2023.5.20" "pyowletapi==2023.5.25"
], ],
"version":"1.5.0" "version":"2023.5.4"
} }

View File

@@ -18,7 +18,7 @@ from homeassistant.const import (
UnitOfTemperature, UnitOfTemperature,
) )
from .const import DOMAIN from .const import DOMAIN, SLEEP_STATES
from .coordinator import OwletCoordinator from .coordinator import OwletCoordinator
from .entity import OwletBaseEntity from .entity import OwletBaseEntity
@@ -107,6 +107,7 @@ async def async_setup_entry(
coordinator: OwletCoordinator = hass.data[DOMAIN][config_entry.entry_id] coordinator: OwletCoordinator = hass.data[DOMAIN][config_entry.entry_id]
entities = [OwletSensor(coordinator, sensor) for sensor in SENSORS] entities = [OwletSensor(coordinator, sensor) for sensor in SENSORS]
entities.append(OwletSleepStateSensor(coordinator))
async_add_entities(entities) async_add_entities(entities)
@@ -142,3 +143,31 @@ class OwletSensor(OwletBaseEntity, SensorEntity):
return None return None
return self.sock.properties[self.entity_description.element] return self.sock.properties[self.entity_description.element]
class OwletSleepStateSensor(OwletBaseEntity, SensorEntity):
"""Representation of an Owlet sleep state sensor."""
def __init__(
self,
coordinator: OwletCoordinator,
) -> None:
"""Initialize the sensor."""
super().__init__(coordinator)
self._attr_unique_id = f"{self.sock.serial}-Sleep State"
self._attr_icon = "mdi:sleep"
self._attr_device_class = SensorDeviceClass.ENUM
self._attr_translation_key = "sleepstate"
self._attr_name = "Sleep State"
@property
def native_value(self):
"""Return sensor value"""
if self.sock.properties["charging"]:
return None
return SLEEP_STATES[self.sock.properties["sleep_state"]]
@property
def options(self) -> list[str]:
return SLEEP_STATES.values()

20
info.md
View File

@@ -1,8 +1,6 @@
# Owlet Custom Integration # Owlet Custom Integration
[![GitHub Release][releases-shield]][releases] [![GitHub Release][releases-shield]][releases]
![GitHub all releases][download-all]
![GitHub release (latest by RyanClark123)][download-latest]
[![GitHub Activity][commits-shield]][commits] [![GitHub Activity][commits-shield]][commits]
[![License][license-shield]][license] [![License][license-shield]][license]
@@ -21,23 +19,19 @@ If you have a smart sock 2 and would like to contribute then please do so.
3. Hard refresh browser cache. 3. Hard refresh browser cache.
4. [![Add Integration][add-integration-badge]][add-integration] or in the HA UI go to "Configuration" -> "Integrations" click "+" and search for "Owlet Smart Sock". 4. [![Add Integration][add-integration-badge]][add-integration] or in the HA UI go to "Configuration" -> "Integrations" click "+" and search for "Owlet Smart Sock".
{% endif %}
<!----> <!---->
--- ---
[commits-shield]: https://img.shields.io/github/commit-activity/w/RyanClark123/owlet?style=for-the-badge [commits-shield]: https://img.shields.io/github/commit-activity/w/ryanbdclark/owlet?style=for-the-badge
[commits]: https://github.com/RyanClark123/owlet/commits/main [commits]: https://github.com/ryanbdclark/owlet/commits/main
[hacs]: https://github.com/hacs/integration [hacs]: https://github.com/hacs/integration
[hacsbadge]: https://img.shields.io/badge/HACS-Custom-orange.svg?style=for-the-badge [hacsbadge]: https://img.shields.io/badge/HACS-Custom-orange.svg?style=for-the-badge
[license]: LICENSE [license]: LICENSE
[license-shield]: https://img.shields.io/github/license/RyanClark123/owlet.svg?style=for-the-badge [license-shield]: https://img.shields.io/github/license/ryanbdclark/owlet.svg?style=for-the-badge
[maintenance-shield]: https://img.shields.io/badge/maintainer-Ryan%20Clark%20%40RyanClark123-blue.svg?style=for-the-badge [maintenance-shield]: https://img.shields.io/badge/maintainer-Ryan%20Clark%20%40ryanbdclark-blue.svg?style=for-the-badge
[releases-shield]: https://img.shields.io/github/release/RyanClark123/owlet.svg?style=for-the-badge [releases-shield]: https://img.shields.io/github/release/ryanbdclark/owlet.svg?style=for-the-badge
[releases]: https://github.com/RyanClark123/owlet/releases [releases]: https://github.com/ryanbdclark/owlet/releases
[user_profile]: https://github.com/RyanClark123 [user_profile]: https://github.com/ryanbdclark
[download-all]: https://img.shields.io/github/downloads/RyanClark123/Owlet/total?style=for-the-badge
[download-latest]: https://img.shields.io/github/downloads/RyanClark123/Owlet/latest/total?style=for-the-badge
[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