* First cut of Rollease Acmeda Pulse Hub integration. * Acmeda integration improvements: - Moved common code into a base entity - Battery level sensor added - Localisation now working * Added requirement for aiopulse now that it has been uploaded to PyPI. * Exclude acmeda integration from coverage check as it relies on a hub being present. * Fix Travis CI build issues. * Remove unused constants. * Remove unused group logic from cover.py * Removed commented code from base.py * Remove sensors (battery entities) on removal of hub. * Remove unused groups from sensor.py * Acmeda device and entity update made fully asynchronous using subscriptions to remove need for config polling. * Updated aiopulse version dependency. Removed non-functional battery charging indication. * Rationalised common code to update entities into helpers.py * Fix linting issue. * Correct additional CI pylint errors. * Index config_entries by entry_id. Move entity loading and unloading to __init__.py Add entry_id to dispatcher signal Removed now unused polling code hub Added config_flow unit tests * Tweak to integration config_entry title. * Bumped aiopulse module to 0.3.2. Reduced verbosity of aiopulse module. * Changed to using direct write of device state. Removed old style async_step_init config_flow step. * Remove superfluous battery_level and device_state_attributes from battery entity. * Removal of unused strings. Removal of unused create_config_flow helper. Removal of stale comment. * Remove use of shared container to track existing enities. Moved removal and deregistration of entities to base class through use of dispatch helper. * Fixed strings.json * Fix incorrect use of remove instead of pop on dict. * Add support for tilting covers, bump aiopulse version number. * Bump aiopulse version to v0.3.4. Fixed bug in cover supported_features. * Bumped aiopulse version to 0.4.0 Update acmeda .coveragerc exclusions * Removed already configured hub check from __init__.py async_setup_entry Removed passing in hass reference to base entity class Renamed entity async_reset to async_will_remove_from_hass Changed device_info and properties Migrated to CoveEntity from CoverDevice Added dispatched_connect cleanup on hub removal Removed unused entries from manifest Removed override of battery icon Renamed translations folder * Reversed unintended change to .coveragerc * Fixed config flow for multi-hub discovery. * Acmeda enhancements as requested by MartinHjelmare * Force import to connect to hub to retrieve id prior to creating entry * Remove YAML configuration support. * Tidied up config_flow and tests: - removed unnecessary steps - fixed typos * Removed storage of hub in config_flow.
89 lines
2.9 KiB
Python
89 lines
2.9 KiB
Python
"""Base class for Acmeda Roller Blinds."""
|
|
import aiopulse
|
|
|
|
from homeassistant.core import callback
|
|
from homeassistant.helpers import entity
|
|
from homeassistant.helpers.device_registry import async_get_registry as get_dev_reg
|
|
from homeassistant.helpers.dispatcher import async_dispatcher_connect
|
|
from homeassistant.helpers.entity_registry import async_get_registry as get_ent_reg
|
|
|
|
from .const import ACMEDA_ENTITY_REMOVE, DOMAIN, LOGGER
|
|
|
|
|
|
class AcmedaBase(entity.Entity):
|
|
"""Base representation of an Acmeda roller."""
|
|
|
|
def __init__(self, roller: aiopulse.Roller):
|
|
"""Initialize the roller."""
|
|
self.roller = roller
|
|
|
|
async def async_remove_and_unregister(self):
|
|
"""Unregister from entity and device registry and call entity remove function."""
|
|
LOGGER.error("Removing %s %s", self.__class__.__name__, self.unique_id)
|
|
|
|
ent_registry = await get_ent_reg(self.hass)
|
|
if self.entity_id in ent_registry.entities:
|
|
ent_registry.async_remove(self.entity_id)
|
|
|
|
dev_registry = await get_dev_reg(self.hass)
|
|
device = dev_registry.async_get_device(
|
|
identifiers={(DOMAIN, self.unique_id)}, connections=set()
|
|
)
|
|
if device is not None:
|
|
dev_registry.async_update_device(
|
|
device.id, remove_config_entry_id=self.registry_entry.config_entry_id
|
|
)
|
|
|
|
await self.async_remove()
|
|
|
|
async def async_added_to_hass(self):
|
|
"""Entity has been added to hass."""
|
|
self.roller.callback_subscribe(self.notify_update)
|
|
|
|
self.async_on_remove(
|
|
async_dispatcher_connect(
|
|
self.hass,
|
|
ACMEDA_ENTITY_REMOVE.format(self.roller.id),
|
|
self.async_remove_and_unregister,
|
|
)
|
|
)
|
|
|
|
async def async_will_remove_from_hass(self):
|
|
"""Entity being removed from hass."""
|
|
self.roller.callback_unsubscribe(self.notify_update)
|
|
|
|
@callback
|
|
def notify_update(self):
|
|
"""Write updated device state information."""
|
|
LOGGER.debug("Device update notification received: %s", self.name)
|
|
self.async_write_ha_state()
|
|
|
|
@property
|
|
def should_poll(self):
|
|
"""Report that Acmeda entities do not need polling."""
|
|
return False
|
|
|
|
@property
|
|
def unique_id(self):
|
|
"""Return the unique ID of this roller."""
|
|
return self.roller.id
|
|
|
|
@property
|
|
def device_id(self):
|
|
"""Return the ID of this roller."""
|
|
return self.roller.id
|
|
|
|
@property
|
|
def name(self):
|
|
"""Return the name of roller."""
|
|
return self.roller.name
|
|
|
|
@property
|
|
def device_info(self):
|
|
"""Return the device info."""
|
|
return {
|
|
"identifiers": {(DOMAIN, self.unique_id)},
|
|
"name": self.roller.name,
|
|
"manufacturer": "Rollease Acmeda",
|
|
"via_device": (DOMAIN, self.roller.hub.id),
|
|
}
|