mirror of
https://github.com/home-assistant/core.git
synced 2025-07-23 13:17:32 +00:00
Owlet baby monitor component (#21108)
This commit is contained in:
parent
4509caefde
commit
4c23ccad98
@ -376,6 +376,7 @@ omit =
|
|||||||
homeassistant/components/openuv/__init__.py
|
homeassistant/components/openuv/__init__.py
|
||||||
homeassistant/components/openuv/binary_sensor.py
|
homeassistant/components/openuv/binary_sensor.py
|
||||||
homeassistant/components/openuv/sensor.py
|
homeassistant/components/openuv/sensor.py
|
||||||
|
homeassistant/components/owlet/*
|
||||||
homeassistant/components/pilight/*
|
homeassistant/components/pilight/*
|
||||||
homeassistant/components/plum_lightpad/*
|
homeassistant/components/plum_lightpad/*
|
||||||
homeassistant/components/point/*
|
homeassistant/components/point/*
|
||||||
|
70
homeassistant/components/owlet/__init__.py
Normal file
70
homeassistant/components/owlet/__init__.py
Normal file
@ -0,0 +1,70 @@
|
|||||||
|
"""Support for Owlet baby monitors."""
|
||||||
|
import logging
|
||||||
|
|
||||||
|
import voluptuous as vol
|
||||||
|
|
||||||
|
from homeassistant.const import (CONF_USERNAME, CONF_PASSWORD, CONF_NAME)
|
||||||
|
import homeassistant.helpers.config_validation as cv
|
||||||
|
from homeassistant.helpers.discovery import load_platform
|
||||||
|
|
||||||
|
from .const import SENSOR_MOVEMENT, SENSOR_BASE_STATION, SENSOR_HEART_RATE, \
|
||||||
|
SENSOR_OXYGEN_LEVEL
|
||||||
|
|
||||||
|
REQUIREMENTS = ['pyowlet==1.0.2']
|
||||||
|
|
||||||
|
_LOGGER = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
DOMAIN = 'owlet'
|
||||||
|
|
||||||
|
SENSOR_TYPES = [
|
||||||
|
SENSOR_OXYGEN_LEVEL,
|
||||||
|
SENSOR_HEART_RATE,
|
||||||
|
SENSOR_BASE_STATION,
|
||||||
|
SENSOR_MOVEMENT
|
||||||
|
]
|
||||||
|
|
||||||
|
CONFIG_SCHEMA = vol.Schema({
|
||||||
|
DOMAIN: vol.Schema({
|
||||||
|
vol.Required(CONF_USERNAME): cv.string,
|
||||||
|
vol.Required(CONF_PASSWORD): cv.string,
|
||||||
|
vol.Optional(CONF_NAME): cv.string,
|
||||||
|
}),
|
||||||
|
}, extra=vol.ALLOW_EXTRA)
|
||||||
|
|
||||||
|
|
||||||
|
def setup(hass, config):
|
||||||
|
"""Set up owlet component."""
|
||||||
|
from pyowlet.PyOwlet import PyOwlet
|
||||||
|
|
||||||
|
username = config[DOMAIN][CONF_USERNAME]
|
||||||
|
password = config[DOMAIN][CONF_PASSWORD]
|
||||||
|
name = config[DOMAIN].get(CONF_NAME)
|
||||||
|
|
||||||
|
try:
|
||||||
|
device = PyOwlet(username, password)
|
||||||
|
except KeyError:
|
||||||
|
_LOGGER.error('Owlet authentication failed. Please verify your '
|
||||||
|
'credentials are correct.')
|
||||||
|
return False
|
||||||
|
|
||||||
|
device.update_properties()
|
||||||
|
|
||||||
|
if not name:
|
||||||
|
name = '{}\'s Owlet'.format(device.baby_name)
|
||||||
|
|
||||||
|
hass.data[DOMAIN] = OwletDevice(device, name, SENSOR_TYPES)
|
||||||
|
|
||||||
|
load_platform(hass, 'sensor', DOMAIN, {}, config)
|
||||||
|
load_platform(hass, 'binary_sensor', DOMAIN, {}, config)
|
||||||
|
|
||||||
|
return True
|
||||||
|
|
||||||
|
|
||||||
|
class OwletDevice():
|
||||||
|
"""Represents a configured Owlet device."""
|
||||||
|
|
||||||
|
def __init__(self, device, name, monitor):
|
||||||
|
"""Initialize device."""
|
||||||
|
self.name = name
|
||||||
|
self.monitor = monitor
|
||||||
|
self.device = device
|
82
homeassistant/components/owlet/binary_sensor.py
Normal file
82
homeassistant/components/owlet/binary_sensor.py
Normal file
@ -0,0 +1,82 @@
|
|||||||
|
"""Support for Owlet binary sensors."""
|
||||||
|
from datetime import timedelta
|
||||||
|
|
||||||
|
from homeassistant.components.binary_sensor import BinarySensorDevice
|
||||||
|
from homeassistant.components.owlet import DOMAIN as OWLET_DOMAIN
|
||||||
|
from homeassistant.util import dt as dt_util
|
||||||
|
|
||||||
|
from .const import SENSOR_BASE_STATION, SENSOR_MOVEMENT
|
||||||
|
|
||||||
|
SCAN_INTERVAL = timedelta(seconds=120)
|
||||||
|
|
||||||
|
BINARY_CONDITIONS = {
|
||||||
|
SENSOR_BASE_STATION: {
|
||||||
|
'name': 'Base Station',
|
||||||
|
'device_class': 'power'
|
||||||
|
},
|
||||||
|
SENSOR_MOVEMENT: {
|
||||||
|
'name': 'Movement',
|
||||||
|
'device_class': 'motion'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
def setup_platform(hass, config, add_entities, discovery_info=None):
|
||||||
|
"""Set up owlet binary sensor."""
|
||||||
|
if discovery_info is None:
|
||||||
|
return
|
||||||
|
|
||||||
|
device = hass.data[OWLET_DOMAIN]
|
||||||
|
|
||||||
|
entities = []
|
||||||
|
for condition in BINARY_CONDITIONS:
|
||||||
|
if condition in device.monitor:
|
||||||
|
entities.append(OwletBinarySensor(device, condition))
|
||||||
|
|
||||||
|
add_entities(entities, True)
|
||||||
|
|
||||||
|
|
||||||
|
class OwletBinarySensor(BinarySensorDevice):
|
||||||
|
"""Representation of owlet binary sensor."""
|
||||||
|
|
||||||
|
def __init__(self, device, condition):
|
||||||
|
"""Init owlet binary sensor."""
|
||||||
|
self._device = device
|
||||||
|
self._condition = condition
|
||||||
|
self._state = None
|
||||||
|
self._base_on = False
|
||||||
|
self._prop_expiration = None
|
||||||
|
self._is_charging = None
|
||||||
|
|
||||||
|
@property
|
||||||
|
def name(self):
|
||||||
|
"""Return sensor name."""
|
||||||
|
return '{} {}'.format(self._device.name,
|
||||||
|
BINARY_CONDITIONS[self._condition]['name'])
|
||||||
|
|
||||||
|
@property
|
||||||
|
def is_on(self):
|
||||||
|
"""Return current state of sensor."""
|
||||||
|
return self._state
|
||||||
|
|
||||||
|
@property
|
||||||
|
def device_class(self):
|
||||||
|
"""Return the device class."""
|
||||||
|
return BINARY_CONDITIONS[self._condition]['device_class']
|
||||||
|
|
||||||
|
def update(self):
|
||||||
|
"""Update state of sensor."""
|
||||||
|
self._base_on = self._device.device.base_station_on
|
||||||
|
self._prop_expiration = self._device.device.prop_expire_time
|
||||||
|
self._is_charging = self._device.device.charge_status > 0
|
||||||
|
|
||||||
|
# handle expired values
|
||||||
|
if self._prop_expiration < dt_util.now().timestamp():
|
||||||
|
self._state = False
|
||||||
|
return
|
||||||
|
|
||||||
|
if self._condition == 'movement':
|
||||||
|
if not self._base_on or self._is_charging:
|
||||||
|
return False
|
||||||
|
|
||||||
|
self._state = getattr(self._device.device, self._condition)
|
6
homeassistant/components/owlet/const.py
Normal file
6
homeassistant/components/owlet/const.py
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
"""Constants for Owlet component."""
|
||||||
|
SENSOR_OXYGEN_LEVEL = 'oxygen_level'
|
||||||
|
SENSOR_HEART_RATE = 'heart_rate'
|
||||||
|
|
||||||
|
SENSOR_BASE_STATION = 'base_station_on'
|
||||||
|
SENSOR_MOVEMENT = 'movement'
|
103
homeassistant/components/owlet/sensor.py
Normal file
103
homeassistant/components/owlet/sensor.py
Normal file
@ -0,0 +1,103 @@
|
|||||||
|
"""Support for Owlet sensors."""
|
||||||
|
from datetime import timedelta
|
||||||
|
|
||||||
|
from homeassistant.components.owlet import DOMAIN as OWLET_DOMAIN
|
||||||
|
from homeassistant.helpers.entity import Entity
|
||||||
|
from homeassistant.util import dt as dt_util
|
||||||
|
|
||||||
|
from .const import SENSOR_HEART_RATE, SENSOR_OXYGEN_LEVEL
|
||||||
|
|
||||||
|
SCAN_INTERVAL = timedelta(seconds=120)
|
||||||
|
|
||||||
|
SENSOR_CONDITIONS = {
|
||||||
|
SENSOR_OXYGEN_LEVEL: {
|
||||||
|
'name': 'Oxygen Level',
|
||||||
|
'device_class': None
|
||||||
|
},
|
||||||
|
SENSOR_HEART_RATE: {
|
||||||
|
'name': 'Heart Rate',
|
||||||
|
'device_class': None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
def setup_platform(hass, config, add_entities, discovery_info=None):
|
||||||
|
"""Set up owlet binary sensor."""
|
||||||
|
if discovery_info is None:
|
||||||
|
return
|
||||||
|
|
||||||
|
device = hass.data[OWLET_DOMAIN]
|
||||||
|
|
||||||
|
entities = []
|
||||||
|
for condition in SENSOR_CONDITIONS:
|
||||||
|
if condition in device.monitor:
|
||||||
|
entities.append(OwletSensor(device, condition))
|
||||||
|
|
||||||
|
add_entities(entities, True)
|
||||||
|
|
||||||
|
|
||||||
|
class OwletSensor(Entity):
|
||||||
|
"""Representation of Owlet sensor."""
|
||||||
|
|
||||||
|
def __init__(self, device, condition):
|
||||||
|
"""Init owlet binary sensor."""
|
||||||
|
self._device = device
|
||||||
|
self._condition = condition
|
||||||
|
self._state = None
|
||||||
|
self._prop_expiration = None
|
||||||
|
self.is_charging = None
|
||||||
|
self.battery_level = None
|
||||||
|
self.sock_off = None
|
||||||
|
self.sock_connection = None
|
||||||
|
self._movement = None
|
||||||
|
|
||||||
|
@property
|
||||||
|
def name(self):
|
||||||
|
"""Return sensor name."""
|
||||||
|
return '{} {}'.format(self._device.name,
|
||||||
|
SENSOR_CONDITIONS[self._condition]['name'])
|
||||||
|
|
||||||
|
@property
|
||||||
|
def state(self):
|
||||||
|
"""Return current state of sensor."""
|
||||||
|
return self._state
|
||||||
|
|
||||||
|
@property
|
||||||
|
def device_class(self):
|
||||||
|
"""Return the device class."""
|
||||||
|
return SENSOR_CONDITIONS[self._condition]['device_class']
|
||||||
|
|
||||||
|
@property
|
||||||
|
def device_state_attributes(self):
|
||||||
|
"""Return state attributes."""
|
||||||
|
attributes = {
|
||||||
|
'battery_charging': self.is_charging,
|
||||||
|
'battery_level': self.battery_level,
|
||||||
|
'sock_off': self.sock_off,
|
||||||
|
'sock_connection': self.sock_connection
|
||||||
|
}
|
||||||
|
|
||||||
|
return attributes
|
||||||
|
|
||||||
|
def update(self):
|
||||||
|
"""Update state of sensor."""
|
||||||
|
self.is_charging = self._device.device.charge_status
|
||||||
|
self.battery_level = self._device.device.batt_level
|
||||||
|
self.sock_off = self._device.device.sock_off
|
||||||
|
self.sock_connection = self._device.device.sock_connection
|
||||||
|
self._movement = self._device.device.movement
|
||||||
|
self._prop_expiration = self._device.device.prop_expire_time
|
||||||
|
|
||||||
|
value = getattr(self._device.device, self._condition)
|
||||||
|
|
||||||
|
if self._condition == 'batt_level':
|
||||||
|
self._state = min(100, value)
|
||||||
|
return
|
||||||
|
|
||||||
|
if not self._device.device.base_station_on \
|
||||||
|
or self._device.device.charge_status > 0 \
|
||||||
|
or self._prop_expiration < dt_util.now().timestamp() \
|
||||||
|
or self._movement:
|
||||||
|
value = None
|
||||||
|
|
||||||
|
self._state = value
|
@ -1188,6 +1188,9 @@ pyotgw==0.4b1
|
|||||||
# homeassistant.components.sensor.otp
|
# homeassistant.components.sensor.otp
|
||||||
pyotp==2.2.6
|
pyotp==2.2.6
|
||||||
|
|
||||||
|
# homeassistant.components.owlet
|
||||||
|
pyowlet==1.0.2
|
||||||
|
|
||||||
# homeassistant.components.sensor.openweathermap
|
# homeassistant.components.sensor.openweathermap
|
||||||
# homeassistant.components.weather.openweathermap
|
# homeassistant.components.weather.openweathermap
|
||||||
pyowm==2.10.0
|
pyowm==2.10.0
|
||||||
|
Loading…
x
Reference in New Issue
Block a user