2019-06-13 15:43:57 +00:00
|
|
|
"""Alexa capabilities."""
|
2021-03-17 22:34:25 +00:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2019-06-13 15:43:57 +00:00
|
|
|
import logging
|
|
|
|
|
2020-01-15 17:15:31 +00:00
|
|
|
from homeassistant.components import (
|
|
|
|
cover,
|
|
|
|
fan,
|
|
|
|
image_processing,
|
|
|
|
input_number,
|
|
|
|
light,
|
2020-03-10 04:59:06 +00:00
|
|
|
timer,
|
2020-01-15 17:15:31 +00:00
|
|
|
vacuum,
|
|
|
|
)
|
2019-12-08 13:56:42 +00:00
|
|
|
from homeassistant.components.alarm_control_panel import ATTR_CODE_FORMAT, FORMAT_NUMBER
|
2020-01-25 01:57:58 +00:00
|
|
|
from homeassistant.components.alarm_control_panel.const import (
|
|
|
|
SUPPORT_ALARM_ARM_AWAY,
|
|
|
|
SUPPORT_ALARM_ARM_HOME,
|
|
|
|
SUPPORT_ALARM_ARM_NIGHT,
|
|
|
|
)
|
2019-12-08 13:56:42 +00:00
|
|
|
import homeassistant.components.climate.const as climate
|
2021-07-21 04:21:05 +00:00
|
|
|
from homeassistant.components.lock import STATE_LOCKING, STATE_UNLOCKING
|
2019-12-08 13:56:42 +00:00
|
|
|
import homeassistant.components.media_player.const as media_player
|
2019-06-13 15:43:57 +00:00
|
|
|
from homeassistant.const import (
|
|
|
|
ATTR_SUPPORTED_FEATURES,
|
|
|
|
ATTR_TEMPERATURE,
|
|
|
|
ATTR_UNIT_OF_MEASUREMENT,
|
2019-10-04 15:41:47 +00:00
|
|
|
STATE_ALARM_ARMED_AWAY,
|
|
|
|
STATE_ALARM_ARMED_CUSTOM_BYPASS,
|
|
|
|
STATE_ALARM_ARMED_HOME,
|
|
|
|
STATE_ALARM_ARMED_NIGHT,
|
2020-03-10 04:59:06 +00:00
|
|
|
STATE_IDLE,
|
2019-06-13 15:43:57 +00:00
|
|
|
STATE_LOCKED,
|
|
|
|
STATE_OFF,
|
|
|
|
STATE_ON,
|
2019-10-25 21:53:33 +00:00
|
|
|
STATE_PAUSED,
|
|
|
|
STATE_PLAYING,
|
2019-06-13 15:43:57 +00:00
|
|
|
STATE_UNAVAILABLE,
|
2019-08-31 14:46:26 +00:00
|
|
|
STATE_UNKNOWN,
|
2019-10-25 21:53:33 +00:00
|
|
|
STATE_UNLOCKED,
|
2019-06-13 15:43:57 +00:00
|
|
|
)
|
2020-08-30 12:36:00 +00:00
|
|
|
from homeassistant.core import State
|
2019-06-13 15:43:57 +00:00
|
|
|
import homeassistant.util.color as color_util
|
2019-09-19 16:19:27 +00:00
|
|
|
import homeassistant.util.dt as dt_util
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
from .const import (
|
|
|
|
API_TEMP_UNITS,
|
|
|
|
API_THERMOSTAT_MODES,
|
Climate 1.0 (#23899)
* Climate 1.0 / part 1/2/3
* fix flake
* Lint
* Update Google Assistant
* ambiclimate to climate 1.0 (#24911)
* Fix Alexa
* Lint
* Migrate zhong_hong
* Migrate tuya
* Migrate honeywell to new climate schema (#24257)
* Update one
* Fix model climate v2
* Cleanup p4
* Add comfort hold mode
* Fix old code
* Update homeassistant/components/climate/__init__.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* Update homeassistant/components/climate/const.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* First renaming
* Rename operation to hvac for paulus
* Rename hold mode to preset mode
* Cleanup & update comments
* Remove on/off
* Fix supported feature count
* Update services
* Update demo
* Fix tests & use current_hvac
* Update comment
* Fix tests & add typing
* Add more typing
* Update modes
* Fix tests
* Cleanup low/high with range
* Update homematic part 1
* Finish homematic
* Fix lint
* fix hm mapping
* Support simple devices
* convert lcn
* migrate oem
* Fix xs1
* update hive
* update mil
* Update toon
* migrate deconz
* cleanup
* update tesla
* Fix lint
* Fix vera
* Migrate zwave
* Migrate velbus
* Cleanup humity feature
* Cleanup
* Migrate wink
* migrate dyson
* Fix current hvac
* Renaming
* Fix lint
* Migrate tfiac
* migrate tado
* Fix PRESET can be None
* apply PR#23913 from dev
* remove EU component, etc.
* remove EU component, etc.
* ready to test now
* de-linted
* some tweaks
* de-lint
* better handling of edge cases
* delint
* fix set_mode typos
* apply PR#23913 from dev
* remove EU component, etc.
* ready to test now
* de-linted
* some tweaks
* de-lint
* better handling of edge cases
* delint
* fix set_mode typos
* delint, move debug code
* away preset now working
* code tidy-up
* code tidy-up 2
* code tidy-up 3
* address issues #18932, #15063
* address issues #18932, #15063 - 2/2
* refactor MODE_AUTO to MODE_HEAT_COOL and use F not C
* add low/high to set_temp
* add low/high to set_temp 2
* add low/high to set_temp - delint
* run HA scripts
* port changes from PR #24402
* manual rebase
* manual rebase 2
* delint
* minor change
* remove SUPPORT_HVAC_ACTION
* Migrate radiotherm
* Convert touchline
* Migrate flexit
* Migrate nuheat
* Migrate maxcube
* Fix names maxcube const
* Migrate proliphix
* Migrate heatmiser
* Migrate fritzbox
* Migrate opentherm_gw
* Migrate venstar
* Migrate daikin
* Migrate modbus
* Fix elif
* Migrate Homematic IP Cloud to climate-1.0 (#24913)
* hmip climate fix
* Update hvac_mode and preset_mode
* fix lint
* Fix lint
* Migrate generic_thermostat
* Migrate incomfort to new climate schema (#24915)
* initial commit
* Update climate.py
* Migrate eq3btsmart
* Lint
* cleanup PRESET_MANUAL
* Migrate ecobee
* No conditional features
* KNX: Migrate climate component to new climate platform (#24931)
* Migrate climate component
* Remove unused code
* Corrected line length
* Lint
* Lint
* fix tests
* Fix value
* Migrate geniushub to new climate schema (#24191)
* Update one
* Fix model climate v2
* Cleanup p4
* Add comfort hold mode
* Fix old code
* Update homeassistant/components/climate/__init__.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* Update homeassistant/components/climate/const.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* First renaming
* Rename operation to hvac for paulus
* Rename hold mode to preset mode
* Cleanup & update comments
* Remove on/off
* Fix supported feature count
* Update services
* Update demo
* Fix tests & use current_hvac
* Update comment
* Fix tests & add typing
* Add more typing
* Update modes
* Fix tests
* Cleanup low/high with range
* Update homematic part 1
* Finish homematic
* Fix lint
* fix hm mapping
* Support simple devices
* convert lcn
* migrate oem
* Fix xs1
* update hive
* update mil
* Update toon
* migrate deconz
* cleanup
* update tesla
* Fix lint
* Fix vera
* Migrate zwave
* Migrate velbus
* Cleanup humity feature
* Cleanup
* Migrate wink
* migrate dyson
* Fix current hvac
* Renaming
* Fix lint
* Migrate tfiac
* migrate tado
* delinted
* delinted
* use latest client
* clean up mappings
* clean up mappings
* add duration to set_temperature
* add duration to set_temperature
* manual rebase
* tweak
* fix regression
* small fix
* fix rebase mixup
* address comments
* finish refactor
* fix regression
* tweak type hints
* delint
* manual rebase
* WIP: Fixes for honeywell migration to climate-1.0 (#24938)
* add type hints
* code tidy-up
* Fixes for incomfort migration to climate-1.0 (#24936)
* delint type hints
* no async unless await
* revert: no async unless await
* revert: no async unless await 2
* delint
* fix typo
* Fix homekit_controller on climate-1.0 (#24948)
* Fix tests on climate-1.0 branch
* As part of climate-1.0, make state return the heating-cooling.current characteristic
* Fixes from review
* lint
* Fix imports
* Migrate stibel_eltron
* Fix lint
* Migrate coolmaster to climate 1.0 (#24967)
* Migrate coolmaster to climate 1.0
* fix lint errors
* More lint fixes
* Fix demo to work with UI
* Migrate spider
* Demo update
* Updated frontend to 20190705.0
* Fix boost mode (#24980)
* Prepare Netatmo for climate 1.0 (#24973)
* Migration Netatmo
* Address comments
* Update climate.py
* Migrate ephember
* Migrate Sensibo
* Implemented review comments (#24942)
* Migrate ESPHome
* Migrate MQTT
* Migrate Nest
* Migrate melissa
* Initial/partial migration of ST
* Migrate ST
* Remove Away mode (#24995)
* Migrate evohome, cache access tokens (#24491)
* add water_heater, add storage - initial commit
* add water_heater, add storage - initial commit
delint
add missing code
desiderata
update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
delint
* Add Broker, Water Heater & Refactor
add missing code
desiderata
* update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
* bugfix - loc_idx may not be 0
more refactor - ensure pure async
more refactoring
appears all r/o attributes are working
tweak precsion, DHW & delint
remove unused code
remove unused code 2
remove unused code, refactor _save_auth_tokens()
* support RoundThermostat
bugfix opmode, switch to util.dt, add until=1h
revert breaking change
* store at_expires as naive UTC
remove debug code
delint
tidy up exception handling
delint
add water_heater, add storage - initial commit
delint
add missing code
desiderata
update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
add water_heater, add storage - initial commit
delint
add missing code
desiderata
update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
delint
bugfix - loc_idx may not be 0
more refactor - ensure pure async
more refactoring
appears all r/o attributes are working
tweak precsion, DHW & delint
remove unused code
remove unused code 2
remove unused code, refactor _save_auth_tokens()
support RoundThermostat
bugfix opmode, switch to util.dt, add until=1h
revert breaking change
store at_expires as naive UTC
remove debug code
delint
tidy up exception handling
delint
* update CODEOWNERS
* fix regression
* fix requirements
* migrate to climate-1.0
* tweaking
* de-lint
* TCS working? & delint
* tweaking
* TCS code finalised
* remove available() logic
* refactor _switchpoints()
* tidy up switchpoint code
* tweak
* teaking device_state_attributes
* some refactoring
* move PRESET_CUSTOM back to evohome
* move CONF_ACCESS_TOKEN_EXPIRES CONF_REFRESH_TOKEN back to evohome
* refactor SP code and dt conversion
* delinted
* delinted
* remove water_heater
* fix regression
* Migrate homekit
* Cleanup away mode
* Fix tests
* add helpers
* fix tests melissa
* Fix nehueat
* fix zwave
* add more tests
* fix deconz
* Fix climate test emulate_hue
* fix tests
* fix dyson tests
* fix demo with new layout
* fix honeywell
* Switch homekit_controller to use HVAC_MODE_HEAT_COOL instead of HVAC_MODE_AUTO (#25009)
* Lint
* PyLint
* Pylint
* fix fritzbox tests
* Fix google
* Fix all tests
* Fix lint
* Fix auto for homekit like controler
* Fix lint
* fix lint
2019-07-08 12:00:24 +00:00
|
|
|
API_THERMOSTAT_PRESETS,
|
2019-06-13 15:43:57 +00:00
|
|
|
DATE_FORMAT,
|
2021-09-24 06:14:45 +00:00
|
|
|
PRESET_MODE_NA,
|
2019-11-25 23:17:12 +00:00
|
|
|
Inputs,
|
2019-06-13 15:43:57 +00:00
|
|
|
)
|
|
|
|
from .errors import UnsupportedProperty
|
2019-12-19 11:44:17 +00:00
|
|
|
from .resources import (
|
|
|
|
AlexaCapabilityResource,
|
|
|
|
AlexaGlobalCatalog,
|
|
|
|
AlexaModeResource,
|
|
|
|
AlexaPresetResource,
|
|
|
|
AlexaSemantics,
|
|
|
|
)
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
2019-10-23 05:01:03 +00:00
|
|
|
class AlexaCapability:
|
2019-06-13 15:43:57 +00:00
|
|
|
"""Base class for Alexa capability interfaces.
|
|
|
|
|
|
|
|
The Smart Home Skills API defines a number of "capability interfaces",
|
|
|
|
roughly analogous to domains in Home Assistant. The supported interfaces
|
|
|
|
describe what actions can be performed on a particular device.
|
|
|
|
|
|
|
|
https://developer.amazon.com/docs/device-apis/message-guide.html
|
|
|
|
"""
|
|
|
|
|
2020-01-03 20:23:22 +00:00
|
|
|
supported_locales = {"en-US"}
|
|
|
|
|
2021-05-20 15:51:39 +00:00
|
|
|
def __init__(self, entity: State, instance: str | None = None) -> None:
|
2019-10-23 05:01:03 +00:00
|
|
|
"""Initialize an Alexa capability."""
|
2019-06-13 15:43:57 +00:00
|
|
|
self.entity = entity
|
2019-10-23 05:01:03 +00:00
|
|
|
self.instance = instance
|
2019-06-13 15:43:57 +00:00
|
|
|
|
2020-08-30 12:36:00 +00:00
|
|
|
def name(self) -> str:
|
2019-06-13 15:43:57 +00:00
|
|
|
"""Return the Alexa API name of this interface."""
|
|
|
|
raise NotImplementedError
|
|
|
|
|
|
|
|
@staticmethod
|
2021-03-17 22:34:25 +00:00
|
|
|
def properties_supported() -> list[dict]:
|
2019-06-13 15:43:57 +00:00
|
|
|
"""Return what properties this entity supports."""
|
|
|
|
return []
|
|
|
|
|
|
|
|
@staticmethod
|
2020-08-30 12:36:00 +00:00
|
|
|
def properties_proactively_reported() -> bool:
|
2019-06-13 15:43:57 +00:00
|
|
|
"""Return True if properties asynchronously reported."""
|
|
|
|
return False
|
|
|
|
|
|
|
|
@staticmethod
|
2020-08-30 12:36:00 +00:00
|
|
|
def properties_retrievable() -> bool:
|
2019-06-13 15:43:57 +00:00
|
|
|
"""Return True if properties can be retrieved."""
|
|
|
|
return False
|
|
|
|
|
2019-10-23 05:01:03 +00:00
|
|
|
@staticmethod
|
2021-08-10 08:46:33 +00:00
|
|
|
def properties_non_controllable() -> bool | None:
|
2019-10-23 05:01:03 +00:00
|
|
|
"""Return True if non controllable."""
|
|
|
|
return None
|
|
|
|
|
2019-06-13 15:43:57 +00:00
|
|
|
@staticmethod
|
|
|
|
def get_property(name):
|
|
|
|
"""Read and return a property.
|
|
|
|
|
|
|
|
Return value should be a dict, or raise UnsupportedProperty.
|
|
|
|
|
|
|
|
Properties can also have a timeOfSample and uncertaintyInMilliseconds,
|
|
|
|
but returning those metadata is not yet implemented.
|
|
|
|
"""
|
|
|
|
raise UnsupportedProperty(name)
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def supports_deactivation():
|
|
|
|
"""Applicable only to scenes."""
|
|
|
|
return None
|
|
|
|
|
2019-10-23 18:41:26 +00:00
|
|
|
@staticmethod
|
|
|
|
def capability_proactively_reported():
|
|
|
|
"""Return True if the capability is proactively reported.
|
|
|
|
|
|
|
|
Set properties_proactively_reported() for proactively reported properties.
|
|
|
|
Applicable to DoorbellEventSource.
|
|
|
|
"""
|
|
|
|
return None
|
|
|
|
|
2019-10-23 05:01:03 +00:00
|
|
|
@staticmethod
|
|
|
|
def capability_resources():
|
2019-12-19 11:44:17 +00:00
|
|
|
"""Return the capability object.
|
|
|
|
|
|
|
|
Applicable to ToggleController, RangeController, and ModeController interfaces.
|
|
|
|
"""
|
2019-10-23 05:01:03 +00:00
|
|
|
return []
|
|
|
|
|
2019-10-04 15:41:47 +00:00
|
|
|
@staticmethod
|
|
|
|
def configuration():
|
2019-12-24 22:06:39 +00:00
|
|
|
"""Return the configuration object.
|
|
|
|
|
|
|
|
Applicable to the ThermostatController, SecurityControlPanel, ModeController, RangeController,
|
|
|
|
and EventDetectionSensor.
|
|
|
|
"""
|
|
|
|
return []
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def configurations():
|
|
|
|
"""Return the configurations object.
|
|
|
|
|
|
|
|
The plural configurations object is different that the singular configuration object.
|
|
|
|
Applicable to EqualizerController interface.
|
|
|
|
"""
|
2019-10-04 15:41:47 +00:00
|
|
|
return []
|
|
|
|
|
2019-11-25 23:17:12 +00:00
|
|
|
@staticmethod
|
|
|
|
def inputs():
|
|
|
|
"""Applicable only to media players."""
|
|
|
|
return []
|
|
|
|
|
2019-12-19 11:44:17 +00:00
|
|
|
@staticmethod
|
|
|
|
def semantics():
|
|
|
|
"""Return the semantics object.
|
|
|
|
|
|
|
|
Applicable to ToggleController, RangeController, and ModeController interfaces.
|
|
|
|
"""
|
|
|
|
return []
|
|
|
|
|
2019-10-25 20:34:52 +00:00
|
|
|
@staticmethod
|
|
|
|
def supported_operations():
|
|
|
|
"""Return the supportedOperations object."""
|
|
|
|
return []
|
|
|
|
|
2020-03-28 04:19:11 +00:00
|
|
|
@staticmethod
|
|
|
|
def camera_stream_configurations():
|
|
|
|
"""Applicable only to CameraStreamController."""
|
|
|
|
return None
|
|
|
|
|
2019-06-13 15:43:57 +00:00
|
|
|
def serialize_discovery(self):
|
|
|
|
"""Serialize according to the Discovery API."""
|
2019-10-23 18:41:26 +00:00
|
|
|
result = {"type": "AlexaInterface", "interface": self.name(), "version": "3"}
|
|
|
|
|
2021-10-17 17:56:00 +00:00
|
|
|
if (instance := self.instance) is not None:
|
2019-12-19 11:44:17 +00:00
|
|
|
result["instance"] = instance
|
|
|
|
|
2019-10-23 18:41:26 +00:00
|
|
|
properties_supported = self.properties_supported()
|
|
|
|
if properties_supported:
|
|
|
|
result["properties"] = {
|
2019-07-31 19:25:30 +00:00
|
|
|
"supported": self.properties_supported(),
|
|
|
|
"proactivelyReported": self.properties_proactively_reported(),
|
|
|
|
"retrievable": self.properties_retrievable(),
|
2019-10-23 18:41:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
proactively_reported = self.capability_proactively_reported()
|
|
|
|
if proactively_reported is not None:
|
|
|
|
result["proactivelyReported"] = proactively_reported
|
2019-06-13 15:43:57 +00:00
|
|
|
|
2019-10-23 05:01:03 +00:00
|
|
|
non_controllable = self.properties_non_controllable()
|
|
|
|
if non_controllable is not None:
|
|
|
|
result["properties"]["nonControllable"] = non_controllable
|
|
|
|
|
2019-06-13 15:43:57 +00:00
|
|
|
supports_deactivation = self.supports_deactivation()
|
|
|
|
if supports_deactivation is not None:
|
2019-07-31 19:25:30 +00:00
|
|
|
result["supportsDeactivation"] = supports_deactivation
|
2019-10-04 15:41:47 +00:00
|
|
|
|
2019-12-19 11:44:17 +00:00
|
|
|
capability_resources = self.capability_resources()
|
2019-10-23 05:01:03 +00:00
|
|
|
if capability_resources:
|
|
|
|
result["capabilityResources"] = capability_resources
|
|
|
|
|
2019-10-04 15:41:47 +00:00
|
|
|
configuration = self.configuration()
|
|
|
|
if configuration:
|
|
|
|
result["configuration"] = configuration
|
|
|
|
|
2019-12-24 22:06:39 +00:00
|
|
|
# The plural configurations object is different than the singular configuration object above.
|
|
|
|
configurations = self.configurations()
|
|
|
|
if configurations:
|
|
|
|
result["configurations"] = configurations
|
|
|
|
|
2019-12-19 11:44:17 +00:00
|
|
|
semantics = self.semantics()
|
|
|
|
if semantics:
|
|
|
|
result["semantics"] = semantics
|
2019-10-23 05:01:03 +00:00
|
|
|
|
2019-10-25 20:34:52 +00:00
|
|
|
supported_operations = self.supported_operations()
|
|
|
|
if supported_operations:
|
|
|
|
result["supportedOperations"] = supported_operations
|
|
|
|
|
2019-11-25 23:17:12 +00:00
|
|
|
inputs = self.inputs()
|
|
|
|
if inputs:
|
|
|
|
result["inputs"] = inputs
|
|
|
|
|
2020-03-28 04:19:11 +00:00
|
|
|
camera_stream_configurations = self.camera_stream_configurations()
|
|
|
|
if camera_stream_configurations:
|
|
|
|
result["cameraStreamConfigurations"] = camera_stream_configurations
|
|
|
|
|
2019-06-13 15:43:57 +00:00
|
|
|
return result
|
|
|
|
|
|
|
|
def serialize_properties(self):
|
|
|
|
"""Return properties serialized for an API response."""
|
|
|
|
for prop in self.properties_supported():
|
2019-07-31 19:25:30 +00:00
|
|
|
prop_name = prop["name"]
|
2020-08-30 12:36:00 +00:00
|
|
|
try:
|
|
|
|
prop_value = self.get_property(prop_name)
|
|
|
|
except UnsupportedProperty:
|
|
|
|
raise
|
|
|
|
except Exception: # pylint: disable=broad-except
|
|
|
|
_LOGGER.exception(
|
|
|
|
"Unexpected error getting %s.%s property from %s",
|
|
|
|
self.name(),
|
|
|
|
prop_name,
|
|
|
|
self.entity,
|
|
|
|
)
|
|
|
|
prop_value = None
|
|
|
|
|
|
|
|
if prop_value is None:
|
|
|
|
continue
|
|
|
|
|
|
|
|
result = {
|
|
|
|
"name": prop_name,
|
|
|
|
"namespace": self.name(),
|
|
|
|
"value": prop_value,
|
|
|
|
"timeOfSample": dt_util.utcnow().strftime(DATE_FORMAT),
|
|
|
|
"uncertaintyInMilliseconds": 0,
|
|
|
|
}
|
2021-10-17 17:56:00 +00:00
|
|
|
if (instance := self.instance) is not None:
|
2020-08-30 12:36:00 +00:00
|
|
|
result["instance"] = instance
|
|
|
|
|
|
|
|
yield result
|
2019-10-23 05:01:03 +00:00
|
|
|
|
|
|
|
|
2019-11-26 07:05:10 +00:00
|
|
|
class Alexa(AlexaCapability):
|
|
|
|
"""Implements Alexa Interface.
|
|
|
|
|
|
|
|
Although endpoints implement this interface implicitly,
|
|
|
|
The API suggests you should explicitly include this interface.
|
|
|
|
|
|
|
|
https://developer.amazon.com/docs/device-apis/alexa-interface.html
|
|
|
|
"""
|
|
|
|
|
2020-01-03 20:23:22 +00:00
|
|
|
supported_locales = {
|
|
|
|
"de-DE",
|
|
|
|
"en-AU",
|
|
|
|
"en-CA",
|
|
|
|
"en-GB",
|
|
|
|
"en-IN",
|
|
|
|
"en-US",
|
|
|
|
"es-ES",
|
|
|
|
"es-MX",
|
|
|
|
"fr-CA",
|
|
|
|
"fr-FR",
|
|
|
|
"it-IT",
|
|
|
|
"ja-JP",
|
|
|
|
}
|
|
|
|
|
2019-11-26 07:05:10 +00:00
|
|
|
def name(self):
|
|
|
|
"""Return the Alexa API name of this interface."""
|
|
|
|
return "Alexa"
|
|
|
|
|
|
|
|
|
2019-10-23 05:01:03 +00:00
|
|
|
class AlexaEndpointHealth(AlexaCapability):
|
2019-06-13 15:43:57 +00:00
|
|
|
"""Implements Alexa.EndpointHealth.
|
|
|
|
|
|
|
|
https://developer.amazon.com/docs/smarthome/state-reporting-for-a-smart-home-skill.html#report-state-when-alexa-requests-it
|
|
|
|
"""
|
|
|
|
|
2020-01-03 20:23:22 +00:00
|
|
|
supported_locales = {
|
|
|
|
"de-DE",
|
|
|
|
"en-AU",
|
|
|
|
"en-CA",
|
|
|
|
"en-GB",
|
|
|
|
"en-IN",
|
|
|
|
"en-US",
|
|
|
|
"es-ES",
|
|
|
|
"fr-FR",
|
|
|
|
"it-IT",
|
|
|
|
"ja-JP",
|
|
|
|
}
|
|
|
|
|
2019-06-13 15:43:57 +00:00
|
|
|
def __init__(self, hass, entity):
|
|
|
|
"""Initialize the entity."""
|
|
|
|
super().__init__(entity)
|
|
|
|
self.hass = hass
|
|
|
|
|
|
|
|
def name(self):
|
|
|
|
"""Return the Alexa API name of this interface."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return "Alexa.EndpointHealth"
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
def properties_supported(self):
|
|
|
|
"""Return what properties this entity supports."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return [{"name": "connectivity"}]
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
def properties_proactively_reported(self):
|
|
|
|
"""Return True if properties asynchronously reported."""
|
2019-12-09 08:37:15 +00:00
|
|
|
return True
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
def properties_retrievable(self):
|
|
|
|
"""Return True if properties can be retrieved."""
|
|
|
|
return True
|
|
|
|
|
|
|
|
def get_property(self, name):
|
|
|
|
"""Read and return a property."""
|
2019-07-31 19:25:30 +00:00
|
|
|
if name != "connectivity":
|
2019-06-13 15:43:57 +00:00
|
|
|
raise UnsupportedProperty(name)
|
|
|
|
|
|
|
|
if self.entity.state == STATE_UNAVAILABLE:
|
2019-07-31 19:25:30 +00:00
|
|
|
return {"value": "UNREACHABLE"}
|
|
|
|
return {"value": "OK"}
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
|
2019-10-23 05:01:03 +00:00
|
|
|
class AlexaPowerController(AlexaCapability):
|
2019-06-13 15:43:57 +00:00
|
|
|
"""Implements Alexa.PowerController.
|
|
|
|
|
|
|
|
https://developer.amazon.com/docs/device-apis/alexa-powercontroller.html
|
|
|
|
"""
|
|
|
|
|
2020-01-03 20:23:22 +00:00
|
|
|
supported_locales = {
|
|
|
|
"de-DE",
|
|
|
|
"en-AU",
|
|
|
|
"en-CA",
|
|
|
|
"en-GB",
|
|
|
|
"en-IN",
|
|
|
|
"en-US",
|
|
|
|
"es-ES",
|
|
|
|
"fr-FR",
|
|
|
|
"it-IT",
|
|
|
|
"ja-JP",
|
|
|
|
}
|
|
|
|
|
2019-06-13 15:43:57 +00:00
|
|
|
def name(self):
|
|
|
|
"""Return the Alexa API name of this interface."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return "Alexa.PowerController"
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
def properties_supported(self):
|
|
|
|
"""Return what properties this entity supports."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return [{"name": "powerState"}]
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
def properties_proactively_reported(self):
|
|
|
|
"""Return True if properties asynchronously reported."""
|
|
|
|
return True
|
|
|
|
|
|
|
|
def properties_retrievable(self):
|
|
|
|
"""Return True if properties can be retrieved."""
|
|
|
|
return True
|
|
|
|
|
|
|
|
def get_property(self, name):
|
|
|
|
"""Read and return a property."""
|
2019-07-31 19:25:30 +00:00
|
|
|
if name != "powerState":
|
2019-06-13 15:43:57 +00:00
|
|
|
raise UnsupportedProperty(name)
|
|
|
|
|
Climate 1.0 (#23899)
* Climate 1.0 / part 1/2/3
* fix flake
* Lint
* Update Google Assistant
* ambiclimate to climate 1.0 (#24911)
* Fix Alexa
* Lint
* Migrate zhong_hong
* Migrate tuya
* Migrate honeywell to new climate schema (#24257)
* Update one
* Fix model climate v2
* Cleanup p4
* Add comfort hold mode
* Fix old code
* Update homeassistant/components/climate/__init__.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* Update homeassistant/components/climate/const.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* First renaming
* Rename operation to hvac for paulus
* Rename hold mode to preset mode
* Cleanup & update comments
* Remove on/off
* Fix supported feature count
* Update services
* Update demo
* Fix tests & use current_hvac
* Update comment
* Fix tests & add typing
* Add more typing
* Update modes
* Fix tests
* Cleanup low/high with range
* Update homematic part 1
* Finish homematic
* Fix lint
* fix hm mapping
* Support simple devices
* convert lcn
* migrate oem
* Fix xs1
* update hive
* update mil
* Update toon
* migrate deconz
* cleanup
* update tesla
* Fix lint
* Fix vera
* Migrate zwave
* Migrate velbus
* Cleanup humity feature
* Cleanup
* Migrate wink
* migrate dyson
* Fix current hvac
* Renaming
* Fix lint
* Migrate tfiac
* migrate tado
* Fix PRESET can be None
* apply PR#23913 from dev
* remove EU component, etc.
* remove EU component, etc.
* ready to test now
* de-linted
* some tweaks
* de-lint
* better handling of edge cases
* delint
* fix set_mode typos
* apply PR#23913 from dev
* remove EU component, etc.
* ready to test now
* de-linted
* some tweaks
* de-lint
* better handling of edge cases
* delint
* fix set_mode typos
* delint, move debug code
* away preset now working
* code tidy-up
* code tidy-up 2
* code tidy-up 3
* address issues #18932, #15063
* address issues #18932, #15063 - 2/2
* refactor MODE_AUTO to MODE_HEAT_COOL and use F not C
* add low/high to set_temp
* add low/high to set_temp 2
* add low/high to set_temp - delint
* run HA scripts
* port changes from PR #24402
* manual rebase
* manual rebase 2
* delint
* minor change
* remove SUPPORT_HVAC_ACTION
* Migrate radiotherm
* Convert touchline
* Migrate flexit
* Migrate nuheat
* Migrate maxcube
* Fix names maxcube const
* Migrate proliphix
* Migrate heatmiser
* Migrate fritzbox
* Migrate opentherm_gw
* Migrate venstar
* Migrate daikin
* Migrate modbus
* Fix elif
* Migrate Homematic IP Cloud to climate-1.0 (#24913)
* hmip climate fix
* Update hvac_mode and preset_mode
* fix lint
* Fix lint
* Migrate generic_thermostat
* Migrate incomfort to new climate schema (#24915)
* initial commit
* Update climate.py
* Migrate eq3btsmart
* Lint
* cleanup PRESET_MANUAL
* Migrate ecobee
* No conditional features
* KNX: Migrate climate component to new climate platform (#24931)
* Migrate climate component
* Remove unused code
* Corrected line length
* Lint
* Lint
* fix tests
* Fix value
* Migrate geniushub to new climate schema (#24191)
* Update one
* Fix model climate v2
* Cleanup p4
* Add comfort hold mode
* Fix old code
* Update homeassistant/components/climate/__init__.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* Update homeassistant/components/climate/const.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* First renaming
* Rename operation to hvac for paulus
* Rename hold mode to preset mode
* Cleanup & update comments
* Remove on/off
* Fix supported feature count
* Update services
* Update demo
* Fix tests & use current_hvac
* Update comment
* Fix tests & add typing
* Add more typing
* Update modes
* Fix tests
* Cleanup low/high with range
* Update homematic part 1
* Finish homematic
* Fix lint
* fix hm mapping
* Support simple devices
* convert lcn
* migrate oem
* Fix xs1
* update hive
* update mil
* Update toon
* migrate deconz
* cleanup
* update tesla
* Fix lint
* Fix vera
* Migrate zwave
* Migrate velbus
* Cleanup humity feature
* Cleanup
* Migrate wink
* migrate dyson
* Fix current hvac
* Renaming
* Fix lint
* Migrate tfiac
* migrate tado
* delinted
* delinted
* use latest client
* clean up mappings
* clean up mappings
* add duration to set_temperature
* add duration to set_temperature
* manual rebase
* tweak
* fix regression
* small fix
* fix rebase mixup
* address comments
* finish refactor
* fix regression
* tweak type hints
* delint
* manual rebase
* WIP: Fixes for honeywell migration to climate-1.0 (#24938)
* add type hints
* code tidy-up
* Fixes for incomfort migration to climate-1.0 (#24936)
* delint type hints
* no async unless await
* revert: no async unless await
* revert: no async unless await 2
* delint
* fix typo
* Fix homekit_controller on climate-1.0 (#24948)
* Fix tests on climate-1.0 branch
* As part of climate-1.0, make state return the heating-cooling.current characteristic
* Fixes from review
* lint
* Fix imports
* Migrate stibel_eltron
* Fix lint
* Migrate coolmaster to climate 1.0 (#24967)
* Migrate coolmaster to climate 1.0
* fix lint errors
* More lint fixes
* Fix demo to work with UI
* Migrate spider
* Demo update
* Updated frontend to 20190705.0
* Fix boost mode (#24980)
* Prepare Netatmo for climate 1.0 (#24973)
* Migration Netatmo
* Address comments
* Update climate.py
* Migrate ephember
* Migrate Sensibo
* Implemented review comments (#24942)
* Migrate ESPHome
* Migrate MQTT
* Migrate Nest
* Migrate melissa
* Initial/partial migration of ST
* Migrate ST
* Remove Away mode (#24995)
* Migrate evohome, cache access tokens (#24491)
* add water_heater, add storage - initial commit
* add water_heater, add storage - initial commit
delint
add missing code
desiderata
update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
delint
* Add Broker, Water Heater & Refactor
add missing code
desiderata
* update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
* bugfix - loc_idx may not be 0
more refactor - ensure pure async
more refactoring
appears all r/o attributes are working
tweak precsion, DHW & delint
remove unused code
remove unused code 2
remove unused code, refactor _save_auth_tokens()
* support RoundThermostat
bugfix opmode, switch to util.dt, add until=1h
revert breaking change
* store at_expires as naive UTC
remove debug code
delint
tidy up exception handling
delint
add water_heater, add storage - initial commit
delint
add missing code
desiderata
update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
add water_heater, add storage - initial commit
delint
add missing code
desiderata
update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
delint
bugfix - loc_idx may not be 0
more refactor - ensure pure async
more refactoring
appears all r/o attributes are working
tweak precsion, DHW & delint
remove unused code
remove unused code 2
remove unused code, refactor _save_auth_tokens()
support RoundThermostat
bugfix opmode, switch to util.dt, add until=1h
revert breaking change
store at_expires as naive UTC
remove debug code
delint
tidy up exception handling
delint
* update CODEOWNERS
* fix regression
* fix requirements
* migrate to climate-1.0
* tweaking
* de-lint
* TCS working? & delint
* tweaking
* TCS code finalised
* remove available() logic
* refactor _switchpoints()
* tidy up switchpoint code
* tweak
* teaking device_state_attributes
* some refactoring
* move PRESET_CUSTOM back to evohome
* move CONF_ACCESS_TOKEN_EXPIRES CONF_REFRESH_TOKEN back to evohome
* refactor SP code and dt conversion
* delinted
* delinted
* remove water_heater
* fix regression
* Migrate homekit
* Cleanup away mode
* Fix tests
* add helpers
* fix tests melissa
* Fix nehueat
* fix zwave
* add more tests
* fix deconz
* Fix climate test emulate_hue
* fix tests
* fix dyson tests
* fix demo with new layout
* fix honeywell
* Switch homekit_controller to use HVAC_MODE_HEAT_COOL instead of HVAC_MODE_AUTO (#25009)
* Lint
* PyLint
* Pylint
* fix fritzbox tests
* Fix google
* Fix all tests
* Fix lint
* Fix auto for homekit like controler
* Fix lint
* fix lint
2019-07-08 12:00:24 +00:00
|
|
|
if self.entity.domain == climate.DOMAIN:
|
|
|
|
is_on = self.entity.state != climate.HVAC_MODE_OFF
|
2021-09-24 06:14:45 +00:00
|
|
|
elif self.entity.domain == fan.DOMAIN:
|
|
|
|
is_on = self.entity.state == fan.STATE_ON
|
2020-03-09 15:58:47 +00:00
|
|
|
elif self.entity.domain == vacuum.DOMAIN:
|
|
|
|
is_on = self.entity.state == vacuum.STATE_CLEANING
|
2020-03-10 04:59:06 +00:00
|
|
|
elif self.entity.domain == timer.DOMAIN:
|
|
|
|
is_on = self.entity.state != STATE_IDLE
|
Climate 1.0 (#23899)
* Climate 1.0 / part 1/2/3
* fix flake
* Lint
* Update Google Assistant
* ambiclimate to climate 1.0 (#24911)
* Fix Alexa
* Lint
* Migrate zhong_hong
* Migrate tuya
* Migrate honeywell to new climate schema (#24257)
* Update one
* Fix model climate v2
* Cleanup p4
* Add comfort hold mode
* Fix old code
* Update homeassistant/components/climate/__init__.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* Update homeassistant/components/climate/const.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* First renaming
* Rename operation to hvac for paulus
* Rename hold mode to preset mode
* Cleanup & update comments
* Remove on/off
* Fix supported feature count
* Update services
* Update demo
* Fix tests & use current_hvac
* Update comment
* Fix tests & add typing
* Add more typing
* Update modes
* Fix tests
* Cleanup low/high with range
* Update homematic part 1
* Finish homematic
* Fix lint
* fix hm mapping
* Support simple devices
* convert lcn
* migrate oem
* Fix xs1
* update hive
* update mil
* Update toon
* migrate deconz
* cleanup
* update tesla
* Fix lint
* Fix vera
* Migrate zwave
* Migrate velbus
* Cleanup humity feature
* Cleanup
* Migrate wink
* migrate dyson
* Fix current hvac
* Renaming
* Fix lint
* Migrate tfiac
* migrate tado
* Fix PRESET can be None
* apply PR#23913 from dev
* remove EU component, etc.
* remove EU component, etc.
* ready to test now
* de-linted
* some tweaks
* de-lint
* better handling of edge cases
* delint
* fix set_mode typos
* apply PR#23913 from dev
* remove EU component, etc.
* ready to test now
* de-linted
* some tweaks
* de-lint
* better handling of edge cases
* delint
* fix set_mode typos
* delint, move debug code
* away preset now working
* code tidy-up
* code tidy-up 2
* code tidy-up 3
* address issues #18932, #15063
* address issues #18932, #15063 - 2/2
* refactor MODE_AUTO to MODE_HEAT_COOL and use F not C
* add low/high to set_temp
* add low/high to set_temp 2
* add low/high to set_temp - delint
* run HA scripts
* port changes from PR #24402
* manual rebase
* manual rebase 2
* delint
* minor change
* remove SUPPORT_HVAC_ACTION
* Migrate radiotherm
* Convert touchline
* Migrate flexit
* Migrate nuheat
* Migrate maxcube
* Fix names maxcube const
* Migrate proliphix
* Migrate heatmiser
* Migrate fritzbox
* Migrate opentherm_gw
* Migrate venstar
* Migrate daikin
* Migrate modbus
* Fix elif
* Migrate Homematic IP Cloud to climate-1.0 (#24913)
* hmip climate fix
* Update hvac_mode and preset_mode
* fix lint
* Fix lint
* Migrate generic_thermostat
* Migrate incomfort to new climate schema (#24915)
* initial commit
* Update climate.py
* Migrate eq3btsmart
* Lint
* cleanup PRESET_MANUAL
* Migrate ecobee
* No conditional features
* KNX: Migrate climate component to new climate platform (#24931)
* Migrate climate component
* Remove unused code
* Corrected line length
* Lint
* Lint
* fix tests
* Fix value
* Migrate geniushub to new climate schema (#24191)
* Update one
* Fix model climate v2
* Cleanup p4
* Add comfort hold mode
* Fix old code
* Update homeassistant/components/climate/__init__.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* Update homeassistant/components/climate/const.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* First renaming
* Rename operation to hvac for paulus
* Rename hold mode to preset mode
* Cleanup & update comments
* Remove on/off
* Fix supported feature count
* Update services
* Update demo
* Fix tests & use current_hvac
* Update comment
* Fix tests & add typing
* Add more typing
* Update modes
* Fix tests
* Cleanup low/high with range
* Update homematic part 1
* Finish homematic
* Fix lint
* fix hm mapping
* Support simple devices
* convert lcn
* migrate oem
* Fix xs1
* update hive
* update mil
* Update toon
* migrate deconz
* cleanup
* update tesla
* Fix lint
* Fix vera
* Migrate zwave
* Migrate velbus
* Cleanup humity feature
* Cleanup
* Migrate wink
* migrate dyson
* Fix current hvac
* Renaming
* Fix lint
* Migrate tfiac
* migrate tado
* delinted
* delinted
* use latest client
* clean up mappings
* clean up mappings
* add duration to set_temperature
* add duration to set_temperature
* manual rebase
* tweak
* fix regression
* small fix
* fix rebase mixup
* address comments
* finish refactor
* fix regression
* tweak type hints
* delint
* manual rebase
* WIP: Fixes for honeywell migration to climate-1.0 (#24938)
* add type hints
* code tidy-up
* Fixes for incomfort migration to climate-1.0 (#24936)
* delint type hints
* no async unless await
* revert: no async unless await
* revert: no async unless await 2
* delint
* fix typo
* Fix homekit_controller on climate-1.0 (#24948)
* Fix tests on climate-1.0 branch
* As part of climate-1.0, make state return the heating-cooling.current characteristic
* Fixes from review
* lint
* Fix imports
* Migrate stibel_eltron
* Fix lint
* Migrate coolmaster to climate 1.0 (#24967)
* Migrate coolmaster to climate 1.0
* fix lint errors
* More lint fixes
* Fix demo to work with UI
* Migrate spider
* Demo update
* Updated frontend to 20190705.0
* Fix boost mode (#24980)
* Prepare Netatmo for climate 1.0 (#24973)
* Migration Netatmo
* Address comments
* Update climate.py
* Migrate ephember
* Migrate Sensibo
* Implemented review comments (#24942)
* Migrate ESPHome
* Migrate MQTT
* Migrate Nest
* Migrate melissa
* Initial/partial migration of ST
* Migrate ST
* Remove Away mode (#24995)
* Migrate evohome, cache access tokens (#24491)
* add water_heater, add storage - initial commit
* add water_heater, add storage - initial commit
delint
add missing code
desiderata
update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
delint
* Add Broker, Water Heater & Refactor
add missing code
desiderata
* update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
* bugfix - loc_idx may not be 0
more refactor - ensure pure async
more refactoring
appears all r/o attributes are working
tweak precsion, DHW & delint
remove unused code
remove unused code 2
remove unused code, refactor _save_auth_tokens()
* support RoundThermostat
bugfix opmode, switch to util.dt, add until=1h
revert breaking change
* store at_expires as naive UTC
remove debug code
delint
tidy up exception handling
delint
add water_heater, add storage - initial commit
delint
add missing code
desiderata
update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
add water_heater, add storage - initial commit
delint
add missing code
desiderata
update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
delint
bugfix - loc_idx may not be 0
more refactor - ensure pure async
more refactoring
appears all r/o attributes are working
tweak precsion, DHW & delint
remove unused code
remove unused code 2
remove unused code, refactor _save_auth_tokens()
support RoundThermostat
bugfix opmode, switch to util.dt, add until=1h
revert breaking change
store at_expires as naive UTC
remove debug code
delint
tidy up exception handling
delint
* update CODEOWNERS
* fix regression
* fix requirements
* migrate to climate-1.0
* tweaking
* de-lint
* TCS working? & delint
* tweaking
* TCS code finalised
* remove available() logic
* refactor _switchpoints()
* tidy up switchpoint code
* tweak
* teaking device_state_attributes
* some refactoring
* move PRESET_CUSTOM back to evohome
* move CONF_ACCESS_TOKEN_EXPIRES CONF_REFRESH_TOKEN back to evohome
* refactor SP code and dt conversion
* delinted
* delinted
* remove water_heater
* fix regression
* Migrate homekit
* Cleanup away mode
* Fix tests
* add helpers
* fix tests melissa
* Fix nehueat
* fix zwave
* add more tests
* fix deconz
* Fix climate test emulate_hue
* fix tests
* fix dyson tests
* fix demo with new layout
* fix honeywell
* Switch homekit_controller to use HVAC_MODE_HEAT_COOL instead of HVAC_MODE_AUTO (#25009)
* Lint
* PyLint
* Pylint
* fix fritzbox tests
* Fix google
* Fix all tests
* Fix lint
* Fix auto for homekit like controler
* Fix lint
* fix lint
2019-07-08 12:00:24 +00:00
|
|
|
|
|
|
|
else:
|
|
|
|
is_on = self.entity.state != STATE_OFF
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
return "ON" if is_on else "OFF"
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
|
2019-10-23 05:01:03 +00:00
|
|
|
class AlexaLockController(AlexaCapability):
|
2019-06-13 15:43:57 +00:00
|
|
|
"""Implements Alexa.LockController.
|
|
|
|
|
|
|
|
https://developer.amazon.com/docs/device-apis/alexa-lockcontroller.html
|
|
|
|
"""
|
|
|
|
|
2020-01-03 20:23:22 +00:00
|
|
|
supported_locales = {
|
|
|
|
"de-DE",
|
|
|
|
"en-AU",
|
|
|
|
"en-CA",
|
|
|
|
"en-GB",
|
2020-11-20 13:47:17 +00:00
|
|
|
"en-IN",
|
2020-01-03 20:23:22 +00:00
|
|
|
"en-US",
|
|
|
|
"es-ES",
|
2020-11-20 13:47:17 +00:00
|
|
|
"es-MX",
|
|
|
|
"es-US",
|
|
|
|
"fr-CA",
|
|
|
|
"fr-FR",
|
|
|
|
"hi-IN",
|
2020-01-03 20:23:22 +00:00
|
|
|
"it-IT",
|
|
|
|
"ja-JP",
|
2020-11-20 13:47:17 +00:00
|
|
|
"pt-BR",
|
2020-01-03 20:23:22 +00:00
|
|
|
}
|
|
|
|
|
2019-06-13 15:43:57 +00:00
|
|
|
def name(self):
|
|
|
|
"""Return the Alexa API name of this interface."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return "Alexa.LockController"
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
def properties_supported(self):
|
|
|
|
"""Return what properties this entity supports."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return [{"name": "lockState"}]
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
def properties_retrievable(self):
|
|
|
|
"""Return True if properties can be retrieved."""
|
|
|
|
return True
|
|
|
|
|
|
|
|
def properties_proactively_reported(self):
|
|
|
|
"""Return True if properties asynchronously reported."""
|
|
|
|
return True
|
|
|
|
|
|
|
|
def get_property(self, name):
|
|
|
|
"""Read and return a property."""
|
2019-07-31 19:25:30 +00:00
|
|
|
if name != "lockState":
|
2019-06-13 15:43:57 +00:00
|
|
|
raise UnsupportedProperty(name)
|
|
|
|
|
2021-07-21 04:21:05 +00:00
|
|
|
# If its unlocking its still locked and not unlocked yet
|
|
|
|
if self.entity.state in (STATE_UNLOCKING, STATE_LOCKED):
|
2019-07-31 19:25:30 +00:00
|
|
|
return "LOCKED"
|
2021-07-21 04:21:05 +00:00
|
|
|
# If its locking its still unlocked and not locked yet
|
|
|
|
if self.entity.state in (STATE_LOCKING, STATE_UNLOCKED):
|
2019-07-31 19:25:30 +00:00
|
|
|
return "UNLOCKED"
|
|
|
|
return "JAMMED"
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
|
2019-10-23 05:01:03 +00:00
|
|
|
class AlexaSceneController(AlexaCapability):
|
2019-06-13 15:43:57 +00:00
|
|
|
"""Implements Alexa.SceneController.
|
|
|
|
|
|
|
|
https://developer.amazon.com/docs/device-apis/alexa-scenecontroller.html
|
|
|
|
"""
|
|
|
|
|
2020-01-03 20:23:22 +00:00
|
|
|
supported_locales = {
|
|
|
|
"de-DE",
|
2020-11-20 13:47:17 +00:00
|
|
|
"en-AU",
|
2020-01-03 20:23:22 +00:00
|
|
|
"en-CA",
|
|
|
|
"en-GB",
|
|
|
|
"en-IN",
|
|
|
|
"en-US",
|
|
|
|
"es-ES",
|
|
|
|
"fr-FR",
|
|
|
|
"it-IT",
|
2020-11-20 13:47:17 +00:00
|
|
|
"ja-JP",
|
2020-01-03 20:23:22 +00:00
|
|
|
}
|
|
|
|
|
2019-06-13 15:43:57 +00:00
|
|
|
def __init__(self, entity, supports_deactivation):
|
|
|
|
"""Initialize the entity."""
|
|
|
|
super().__init__(entity)
|
|
|
|
self.supports_deactivation = lambda: supports_deactivation
|
|
|
|
|
|
|
|
def name(self):
|
|
|
|
"""Return the Alexa API name of this interface."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return "Alexa.SceneController"
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
|
2019-10-23 05:01:03 +00:00
|
|
|
class AlexaBrightnessController(AlexaCapability):
|
2019-06-13 15:43:57 +00:00
|
|
|
"""Implements Alexa.BrightnessController.
|
|
|
|
|
|
|
|
https://developer.amazon.com/docs/device-apis/alexa-brightnesscontroller.html
|
|
|
|
"""
|
|
|
|
|
2020-01-03 20:23:22 +00:00
|
|
|
supported_locales = {
|
|
|
|
"de-DE",
|
|
|
|
"en-AU",
|
|
|
|
"en-CA",
|
|
|
|
"en-GB",
|
|
|
|
"en-IN",
|
|
|
|
"en-US",
|
|
|
|
"es-ES",
|
|
|
|
"fr-FR",
|
2020-11-20 13:47:17 +00:00
|
|
|
"hi-IN",
|
2020-01-03 20:23:22 +00:00
|
|
|
"it-IT",
|
|
|
|
"ja-JP",
|
2020-11-20 13:47:17 +00:00
|
|
|
"pt-BR",
|
2020-01-03 20:23:22 +00:00
|
|
|
}
|
|
|
|
|
2019-06-13 15:43:57 +00:00
|
|
|
def name(self):
|
|
|
|
"""Return the Alexa API name of this interface."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return "Alexa.BrightnessController"
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
def properties_supported(self):
|
|
|
|
"""Return what properties this entity supports."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return [{"name": "brightness"}]
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
def properties_proactively_reported(self):
|
|
|
|
"""Return True if properties asynchronously reported."""
|
|
|
|
return True
|
|
|
|
|
|
|
|
def properties_retrievable(self):
|
|
|
|
"""Return True if properties can be retrieved."""
|
|
|
|
return True
|
|
|
|
|
|
|
|
def get_property(self, name):
|
|
|
|
"""Read and return a property."""
|
2019-07-31 19:25:30 +00:00
|
|
|
if name != "brightness":
|
2019-06-13 15:43:57 +00:00
|
|
|
raise UnsupportedProperty(name)
|
2019-07-31 19:25:30 +00:00
|
|
|
if "brightness" in self.entity.attributes:
|
|
|
|
return round(self.entity.attributes["brightness"] / 255.0 * 100)
|
2019-06-13 15:43:57 +00:00
|
|
|
return 0
|
|
|
|
|
|
|
|
|
2019-10-23 05:01:03 +00:00
|
|
|
class AlexaColorController(AlexaCapability):
|
2019-06-13 15:43:57 +00:00
|
|
|
"""Implements Alexa.ColorController.
|
|
|
|
|
|
|
|
https://developer.amazon.com/docs/device-apis/alexa-colorcontroller.html
|
|
|
|
"""
|
|
|
|
|
2020-01-03 20:23:22 +00:00
|
|
|
supported_locales = {
|
|
|
|
"de-DE",
|
|
|
|
"en-AU",
|
|
|
|
"en-CA",
|
|
|
|
"en-GB",
|
|
|
|
"en-IN",
|
|
|
|
"en-US",
|
|
|
|
"es-ES",
|
|
|
|
"fr-FR",
|
2020-11-20 13:47:17 +00:00
|
|
|
"hi-IN",
|
2020-01-03 20:23:22 +00:00
|
|
|
"it-IT",
|
|
|
|
"ja-JP",
|
2020-11-20 13:47:17 +00:00
|
|
|
"pt-BR",
|
2020-01-03 20:23:22 +00:00
|
|
|
}
|
|
|
|
|
2019-06-13 15:43:57 +00:00
|
|
|
def name(self):
|
|
|
|
"""Return the Alexa API name of this interface."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return "Alexa.ColorController"
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
def properties_supported(self):
|
|
|
|
"""Return what properties this entity supports."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return [{"name": "color"}]
|
2019-06-13 15:43:57 +00:00
|
|
|
|
2020-01-10 21:34:50 +00:00
|
|
|
def properties_proactively_reported(self):
|
|
|
|
"""Return True if properties asynchronously reported."""
|
|
|
|
return True
|
|
|
|
|
2019-06-13 15:43:57 +00:00
|
|
|
def properties_retrievable(self):
|
|
|
|
"""Return True if properties can be retrieved."""
|
|
|
|
return True
|
|
|
|
|
|
|
|
def get_property(self, name):
|
|
|
|
"""Read and return a property."""
|
2019-07-31 19:25:30 +00:00
|
|
|
if name != "color":
|
2019-06-13 15:43:57 +00:00
|
|
|
raise UnsupportedProperty(name)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
hue, saturation = self.entity.attributes.get(light.ATTR_HS_COLOR, (0, 0))
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
return {
|
2019-07-31 19:25:30 +00:00
|
|
|
"hue": hue,
|
|
|
|
"saturation": saturation / 100.0,
|
|
|
|
"brightness": self.entity.attributes.get(light.ATTR_BRIGHTNESS, 0) / 255.0,
|
2019-06-13 15:43:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-10-23 05:01:03 +00:00
|
|
|
class AlexaColorTemperatureController(AlexaCapability):
|
2019-06-13 15:43:57 +00:00
|
|
|
"""Implements Alexa.ColorTemperatureController.
|
|
|
|
|
|
|
|
https://developer.amazon.com/docs/device-apis/alexa-colortemperaturecontroller.html
|
|
|
|
"""
|
|
|
|
|
2020-01-03 20:23:22 +00:00
|
|
|
supported_locales = {
|
|
|
|
"de-DE",
|
|
|
|
"en-AU",
|
|
|
|
"en-CA",
|
|
|
|
"en-GB",
|
|
|
|
"en-IN",
|
|
|
|
"en-US",
|
|
|
|
"es-ES",
|
|
|
|
"fr-FR",
|
2020-11-20 13:47:17 +00:00
|
|
|
"hi-IN",
|
2020-01-03 20:23:22 +00:00
|
|
|
"it-IT",
|
|
|
|
"ja-JP",
|
2020-11-20 13:47:17 +00:00
|
|
|
"pt-BR",
|
2020-01-03 20:23:22 +00:00
|
|
|
}
|
|
|
|
|
2019-06-13 15:43:57 +00:00
|
|
|
def name(self):
|
|
|
|
"""Return the Alexa API name of this interface."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return "Alexa.ColorTemperatureController"
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
def properties_supported(self):
|
|
|
|
"""Return what properties this entity supports."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return [{"name": "colorTemperatureInKelvin"}]
|
2019-06-13 15:43:57 +00:00
|
|
|
|
2020-01-10 21:34:50 +00:00
|
|
|
def properties_proactively_reported(self):
|
|
|
|
"""Return True if properties asynchronously reported."""
|
|
|
|
return True
|
|
|
|
|
2019-06-13 15:43:57 +00:00
|
|
|
def properties_retrievable(self):
|
|
|
|
"""Return True if properties can be retrieved."""
|
|
|
|
return True
|
|
|
|
|
|
|
|
def get_property(self, name):
|
|
|
|
"""Read and return a property."""
|
2019-07-31 19:25:30 +00:00
|
|
|
if name != "colorTemperatureInKelvin":
|
2019-06-13 15:43:57 +00:00
|
|
|
raise UnsupportedProperty(name)
|
2019-07-31 19:25:30 +00:00
|
|
|
if "color_temp" in self.entity.attributes:
|
2019-06-13 15:43:57 +00:00
|
|
|
return color_util.color_temperature_mired_to_kelvin(
|
2019-07-31 19:25:30 +00:00
|
|
|
self.entity.attributes["color_temp"]
|
|
|
|
)
|
2019-10-02 22:59:21 +00:00
|
|
|
return None
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
|
2019-10-23 05:01:03 +00:00
|
|
|
class AlexaPercentageController(AlexaCapability):
|
2019-06-13 15:43:57 +00:00
|
|
|
"""Implements Alexa.PercentageController.
|
|
|
|
|
|
|
|
https://developer.amazon.com/docs/device-apis/alexa-percentagecontroller.html
|
|
|
|
"""
|
|
|
|
|
2020-01-03 20:23:22 +00:00
|
|
|
supported_locales = {
|
|
|
|
"de-DE",
|
|
|
|
"en-AU",
|
|
|
|
"en-CA",
|
|
|
|
"en-GB",
|
|
|
|
"en-IN",
|
|
|
|
"en-US",
|
|
|
|
"es-ES",
|
|
|
|
"fr-FR",
|
|
|
|
"it-IT",
|
|
|
|
"ja-JP",
|
|
|
|
}
|
|
|
|
|
2019-06-13 15:43:57 +00:00
|
|
|
def name(self):
|
|
|
|
"""Return the Alexa API name of this interface."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return "Alexa.PercentageController"
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
def properties_supported(self):
|
|
|
|
"""Return what properties this entity supports."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return [{"name": "percentage"}]
|
2019-06-13 15:43:57 +00:00
|
|
|
|
2020-01-10 21:34:50 +00:00
|
|
|
def properties_proactively_reported(self):
|
|
|
|
"""Return True if properties asynchronously reported."""
|
|
|
|
return True
|
|
|
|
|
2019-06-13 15:43:57 +00:00
|
|
|
def properties_retrievable(self):
|
|
|
|
"""Return True if properties can be retrieved."""
|
|
|
|
return True
|
|
|
|
|
|
|
|
def get_property(self, name):
|
|
|
|
"""Read and return a property."""
|
2019-07-31 19:25:30 +00:00
|
|
|
if name != "percentage":
|
2019-06-13 15:43:57 +00:00
|
|
|
raise UnsupportedProperty(name)
|
|
|
|
|
|
|
|
if self.entity.domain == fan.DOMAIN:
|
2021-02-20 06:45:14 +00:00
|
|
|
return self.entity.attributes.get(fan.ATTR_PERCENTAGE) or 0
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
if self.entity.domain == cover.DOMAIN:
|
|
|
|
return self.entity.attributes.get(cover.ATTR_CURRENT_POSITION, 0)
|
|
|
|
|
|
|
|
return 0
|
|
|
|
|
|
|
|
|
2019-10-23 05:01:03 +00:00
|
|
|
class AlexaSpeaker(AlexaCapability):
|
2019-06-13 15:43:57 +00:00
|
|
|
"""Implements Alexa.Speaker.
|
|
|
|
|
|
|
|
https://developer.amazon.com/docs/device-apis/alexa-speaker.html
|
|
|
|
"""
|
|
|
|
|
2020-11-20 13:47:17 +00:00
|
|
|
supported_locales = {
|
|
|
|
"de-DE",
|
|
|
|
"en-AU",
|
|
|
|
"en-CA",
|
|
|
|
"en-GB",
|
|
|
|
"en-IN",
|
|
|
|
"en-US",
|
|
|
|
"es-ES",
|
|
|
|
"es-MX",
|
|
|
|
"it-IT",
|
|
|
|
"ja-JP",
|
|
|
|
}
|
2020-01-03 20:23:22 +00:00
|
|
|
|
2019-06-13 15:43:57 +00:00
|
|
|
def name(self):
|
|
|
|
"""Return the Alexa API name of this interface."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return "Alexa.Speaker"
|
2019-06-13 15:43:57 +00:00
|
|
|
|
2020-02-03 22:20:39 +00:00
|
|
|
def properties_supported(self):
|
|
|
|
"""Return what properties this entity supports."""
|
|
|
|
properties = [{"name": "volume"}]
|
|
|
|
|
|
|
|
supported = self.entity.attributes.get(ATTR_SUPPORTED_FEATURES, 0)
|
|
|
|
if supported & media_player.SUPPORT_VOLUME_MUTE:
|
|
|
|
properties.append({"name": "muted"})
|
|
|
|
|
|
|
|
return properties
|
|
|
|
|
|
|
|
def properties_proactively_reported(self):
|
|
|
|
"""Return True if properties asynchronously reported."""
|
|
|
|
return True
|
|
|
|
|
|
|
|
def properties_retrievable(self):
|
|
|
|
"""Return True if properties can be retrieved."""
|
|
|
|
return True
|
|
|
|
|
|
|
|
def get_property(self, name):
|
|
|
|
"""Read and return a property."""
|
|
|
|
if name == "volume":
|
|
|
|
current_level = self.entity.attributes.get(
|
|
|
|
media_player.ATTR_MEDIA_VOLUME_LEVEL
|
|
|
|
)
|
2020-03-09 16:38:31 +00:00
|
|
|
if current_level is not None:
|
|
|
|
return round(float(current_level) * 100)
|
2020-02-03 22:20:39 +00:00
|
|
|
|
|
|
|
if name == "muted":
|
|
|
|
return bool(
|
|
|
|
self.entity.attributes.get(media_player.ATTR_MEDIA_VOLUME_MUTED)
|
|
|
|
)
|
|
|
|
|
|
|
|
return None
|
|
|
|
|
2019-06-13 15:43:57 +00:00
|
|
|
|
2019-10-23 05:01:03 +00:00
|
|
|
class AlexaStepSpeaker(AlexaCapability):
|
2019-06-13 15:43:57 +00:00
|
|
|
"""Implements Alexa.StepSpeaker.
|
|
|
|
|
|
|
|
https://developer.amazon.com/docs/device-apis/alexa-stepspeaker.html
|
|
|
|
"""
|
|
|
|
|
2020-11-20 13:47:17 +00:00
|
|
|
supported_locales = {
|
|
|
|
"de-DE",
|
|
|
|
"en-AU",
|
|
|
|
"en-CA",
|
|
|
|
"en-GB",
|
|
|
|
"en-IN",
|
|
|
|
"en-US",
|
|
|
|
"es-ES",
|
|
|
|
"it-IT",
|
|
|
|
}
|
2020-01-03 20:23:22 +00:00
|
|
|
|
2019-06-13 15:43:57 +00:00
|
|
|
def name(self):
|
|
|
|
"""Return the Alexa API name of this interface."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return "Alexa.StepSpeaker"
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
|
2019-10-23 05:01:03 +00:00
|
|
|
class AlexaPlaybackController(AlexaCapability):
|
2019-06-13 15:43:57 +00:00
|
|
|
"""Implements Alexa.PlaybackController.
|
|
|
|
|
|
|
|
https://developer.amazon.com/docs/device-apis/alexa-playbackcontroller.html
|
|
|
|
"""
|
|
|
|
|
2020-01-03 20:23:22 +00:00
|
|
|
supported_locales = {"de-DE", "en-AU", "en-CA", "en-GB", "en-IN", "en-US", "fr-FR"}
|
|
|
|
|
2019-06-13 15:43:57 +00:00
|
|
|
def name(self):
|
|
|
|
"""Return the Alexa API name of this interface."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return "Alexa.PlaybackController"
|
2019-06-13 15:43:57 +00:00
|
|
|
|
2019-10-25 20:34:52 +00:00
|
|
|
def supported_operations(self):
|
|
|
|
"""Return the supportedOperations object.
|
|
|
|
|
|
|
|
Supported Operations: FastForward, Next, Pause, Play, Previous, Rewind, StartOver, Stop
|
|
|
|
"""
|
|
|
|
supported_features = self.entity.attributes.get(ATTR_SUPPORTED_FEATURES, 0)
|
|
|
|
|
|
|
|
operations = {
|
|
|
|
media_player.SUPPORT_NEXT_TRACK: "Next",
|
|
|
|
media_player.SUPPORT_PAUSE: "Pause",
|
|
|
|
media_player.SUPPORT_PLAY: "Play",
|
|
|
|
media_player.SUPPORT_PREVIOUS_TRACK: "Previous",
|
|
|
|
media_player.SUPPORT_STOP: "Stop",
|
|
|
|
}
|
|
|
|
|
2020-04-04 21:20:48 +00:00
|
|
|
return [
|
|
|
|
value
|
|
|
|
for operation, value in operations.items()
|
|
|
|
if operation & supported_features
|
|
|
|
]
|
2019-10-25 20:34:52 +00:00
|
|
|
|
2019-06-13 15:43:57 +00:00
|
|
|
|
2019-10-23 05:01:03 +00:00
|
|
|
class AlexaInputController(AlexaCapability):
|
2019-06-13 15:43:57 +00:00
|
|
|
"""Implements Alexa.InputController.
|
|
|
|
|
|
|
|
https://developer.amazon.com/docs/device-apis/alexa-inputcontroller.html
|
|
|
|
"""
|
|
|
|
|
2020-01-03 20:23:22 +00:00
|
|
|
supported_locales = {"de-DE", "en-AU", "en-CA", "en-GB", "en-IN", "en-US"}
|
|
|
|
|
2019-06-13 15:43:57 +00:00
|
|
|
def name(self):
|
|
|
|
"""Return the Alexa API name of this interface."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return "Alexa.InputController"
|
2019-06-13 15:43:57 +00:00
|
|
|
|
2019-11-25 23:17:12 +00:00
|
|
|
def inputs(self):
|
|
|
|
"""Return the list of valid supported inputs."""
|
|
|
|
source_list = self.entity.attributes.get(
|
|
|
|
media_player.ATTR_INPUT_SOURCE_LIST, []
|
|
|
|
)
|
2020-04-04 21:20:48 +00:00
|
|
|
return AlexaInputController.get_valid_inputs(source_list)
|
2020-02-04 00:19:40 +00:00
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def get_valid_inputs(source_list):
|
|
|
|
"""Return list of supported inputs."""
|
2019-11-25 23:17:12 +00:00
|
|
|
input_list = []
|
|
|
|
for source in source_list:
|
|
|
|
formatted_source = (
|
|
|
|
source.lower().replace("-", "").replace("_", "").replace(" ", "")
|
|
|
|
)
|
2020-10-28 19:43:48 +00:00
|
|
|
if formatted_source in Inputs.VALID_SOURCE_NAME_MAP:
|
2019-11-25 23:17:12 +00:00
|
|
|
input_list.append(
|
|
|
|
{"name": Inputs.VALID_SOURCE_NAME_MAP[formatted_source]}
|
|
|
|
)
|
|
|
|
|
|
|
|
return input_list
|
|
|
|
|
2019-06-13 15:43:57 +00:00
|
|
|
|
2019-10-23 05:01:03 +00:00
|
|
|
class AlexaTemperatureSensor(AlexaCapability):
|
2019-06-13 15:43:57 +00:00
|
|
|
"""Implements Alexa.TemperatureSensor.
|
|
|
|
|
|
|
|
https://developer.amazon.com/docs/device-apis/alexa-temperaturesensor.html
|
|
|
|
"""
|
|
|
|
|
2020-01-03 20:23:22 +00:00
|
|
|
supported_locales = {
|
|
|
|
"de-DE",
|
|
|
|
"en-AU",
|
|
|
|
"en-CA",
|
|
|
|
"en-GB",
|
|
|
|
"en-IN",
|
|
|
|
"en-US",
|
|
|
|
"es-ES",
|
|
|
|
"fr-FR",
|
|
|
|
"it-IT",
|
|
|
|
"ja-JP",
|
|
|
|
}
|
|
|
|
|
2019-06-13 15:43:57 +00:00
|
|
|
def __init__(self, hass, entity):
|
|
|
|
"""Initialize the entity."""
|
|
|
|
super().__init__(entity)
|
|
|
|
self.hass = hass
|
|
|
|
|
|
|
|
def name(self):
|
|
|
|
"""Return the Alexa API name of this interface."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return "Alexa.TemperatureSensor"
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
def properties_supported(self):
|
|
|
|
"""Return what properties this entity supports."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return [{"name": "temperature"}]
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
def properties_proactively_reported(self):
|
|
|
|
"""Return True if properties asynchronously reported."""
|
|
|
|
return True
|
|
|
|
|
|
|
|
def properties_retrievable(self):
|
|
|
|
"""Return True if properties can be retrieved."""
|
|
|
|
return True
|
|
|
|
|
|
|
|
def get_property(self, name):
|
|
|
|
"""Read and return a property."""
|
2019-07-31 19:25:30 +00:00
|
|
|
if name != "temperature":
|
2019-06-13 15:43:57 +00:00
|
|
|
raise UnsupportedProperty(name)
|
|
|
|
|
|
|
|
unit = self.entity.attributes.get(ATTR_UNIT_OF_MEASUREMENT)
|
|
|
|
temp = self.entity.state
|
|
|
|
if self.entity.domain == climate.DOMAIN:
|
|
|
|
unit = self.hass.config.units.temperature_unit
|
2019-07-31 19:25:30 +00:00
|
|
|
temp = self.entity.attributes.get(climate.ATTR_CURRENT_TEMPERATURE)
|
2019-08-31 14:46:26 +00:00
|
|
|
|
2019-10-02 22:53:04 +00:00
|
|
|
if temp in (STATE_UNAVAILABLE, STATE_UNKNOWN, None):
|
2019-08-31 14:46:26 +00:00
|
|
|
return None
|
|
|
|
|
|
|
|
try:
|
|
|
|
temp = float(temp)
|
|
|
|
except ValueError:
|
|
|
|
_LOGGER.warning("Invalid temp value %s for %s", temp, self.entity.entity_id)
|
|
|
|
return None
|
|
|
|
|
|
|
|
return {"value": temp, "scale": API_TEMP_UNITS[unit]}
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
|
2019-10-23 05:01:03 +00:00
|
|
|
class AlexaContactSensor(AlexaCapability):
|
2019-06-13 15:43:57 +00:00
|
|
|
"""Implements Alexa.ContactSensor.
|
|
|
|
|
|
|
|
The Alexa.ContactSensor interface describes the properties and events used
|
|
|
|
to report the state of an endpoint that detects contact between two
|
|
|
|
surfaces. For example, a contact sensor can report whether a door or window
|
|
|
|
is open.
|
|
|
|
|
|
|
|
https://developer.amazon.com/docs/device-apis/alexa-contactsensor.html
|
|
|
|
"""
|
|
|
|
|
2020-11-20 13:47:17 +00:00
|
|
|
supported_locales = {
|
|
|
|
"de-DE",
|
|
|
|
"en-AU",
|
|
|
|
"en-CA",
|
|
|
|
"en-IN",
|
|
|
|
"en-US",
|
|
|
|
"es-ES",
|
|
|
|
"it-IT",
|
|
|
|
"ja-JP",
|
|
|
|
}
|
2020-01-03 20:23:22 +00:00
|
|
|
|
2019-06-13 15:43:57 +00:00
|
|
|
def __init__(self, hass, entity):
|
|
|
|
"""Initialize the entity."""
|
|
|
|
super().__init__(entity)
|
|
|
|
self.hass = hass
|
|
|
|
|
|
|
|
def name(self):
|
|
|
|
"""Return the Alexa API name of this interface."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return "Alexa.ContactSensor"
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
def properties_supported(self):
|
|
|
|
"""Return what properties this entity supports."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return [{"name": "detectionState"}]
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
def properties_proactively_reported(self):
|
|
|
|
"""Return True if properties asynchronously reported."""
|
|
|
|
return True
|
|
|
|
|
|
|
|
def properties_retrievable(self):
|
|
|
|
"""Return True if properties can be retrieved."""
|
|
|
|
return True
|
|
|
|
|
|
|
|
def get_property(self, name):
|
|
|
|
"""Read and return a property."""
|
2019-07-31 19:25:30 +00:00
|
|
|
if name != "detectionState":
|
2019-06-13 15:43:57 +00:00
|
|
|
raise UnsupportedProperty(name)
|
|
|
|
|
|
|
|
if self.entity.state == STATE_ON:
|
2019-07-31 19:25:30 +00:00
|
|
|
return "DETECTED"
|
|
|
|
return "NOT_DETECTED"
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
|
2019-10-23 05:01:03 +00:00
|
|
|
class AlexaMotionSensor(AlexaCapability):
|
2019-06-13 15:43:57 +00:00
|
|
|
"""Implements Alexa.MotionSensor.
|
|
|
|
|
|
|
|
https://developer.amazon.com/docs/device-apis/alexa-motionsensor.html
|
|
|
|
"""
|
|
|
|
|
2020-11-20 13:47:17 +00:00
|
|
|
supported_locales = {
|
|
|
|
"de-DE",
|
|
|
|
"en-AU",
|
|
|
|
"en-CA",
|
|
|
|
"en-IN",
|
|
|
|
"en-US",
|
|
|
|
"es-ES",
|
|
|
|
"it-IT",
|
|
|
|
"ja-JP",
|
|
|
|
"pt-BR",
|
|
|
|
}
|
2020-01-03 20:23:22 +00:00
|
|
|
|
2019-06-13 15:43:57 +00:00
|
|
|
def __init__(self, hass, entity):
|
|
|
|
"""Initialize the entity."""
|
|
|
|
super().__init__(entity)
|
|
|
|
self.hass = hass
|
|
|
|
|
|
|
|
def name(self):
|
|
|
|
"""Return the Alexa API name of this interface."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return "Alexa.MotionSensor"
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
def properties_supported(self):
|
|
|
|
"""Return what properties this entity supports."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return [{"name": "detectionState"}]
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
def properties_proactively_reported(self):
|
|
|
|
"""Return True if properties asynchronously reported."""
|
|
|
|
return True
|
|
|
|
|
|
|
|
def properties_retrievable(self):
|
|
|
|
"""Return True if properties can be retrieved."""
|
|
|
|
return True
|
|
|
|
|
|
|
|
def get_property(self, name):
|
|
|
|
"""Read and return a property."""
|
2019-07-31 19:25:30 +00:00
|
|
|
if name != "detectionState":
|
2019-06-13 15:43:57 +00:00
|
|
|
raise UnsupportedProperty(name)
|
|
|
|
|
|
|
|
if self.entity.state == STATE_ON:
|
2019-07-31 19:25:30 +00:00
|
|
|
return "DETECTED"
|
|
|
|
return "NOT_DETECTED"
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
|
2019-10-23 05:01:03 +00:00
|
|
|
class AlexaThermostatController(AlexaCapability):
|
2019-06-13 15:43:57 +00:00
|
|
|
"""Implements Alexa.ThermostatController.
|
|
|
|
|
|
|
|
https://developer.amazon.com/docs/device-apis/alexa-thermostatcontroller.html
|
|
|
|
"""
|
|
|
|
|
2020-01-03 20:23:22 +00:00
|
|
|
supported_locales = {
|
|
|
|
"de-DE",
|
|
|
|
"en-AU",
|
|
|
|
"en-CA",
|
|
|
|
"en-GB",
|
|
|
|
"en-IN",
|
|
|
|
"en-US",
|
|
|
|
"es-ES",
|
|
|
|
"fr-FR",
|
|
|
|
"it-IT",
|
|
|
|
"ja-JP",
|
2020-11-20 13:47:17 +00:00
|
|
|
"pt-BR",
|
2020-01-03 20:23:22 +00:00
|
|
|
}
|
|
|
|
|
2019-06-13 15:43:57 +00:00
|
|
|
def __init__(self, hass, entity):
|
|
|
|
"""Initialize the entity."""
|
|
|
|
super().__init__(entity)
|
|
|
|
self.hass = hass
|
|
|
|
|
|
|
|
def name(self):
|
|
|
|
"""Return the Alexa API name of this interface."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return "Alexa.ThermostatController"
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
def properties_supported(self):
|
|
|
|
"""Return what properties this entity supports."""
|
2019-07-31 19:25:30 +00:00
|
|
|
properties = [{"name": "thermostatMode"}]
|
2019-06-13 15:43:57 +00:00
|
|
|
supported = self.entity.attributes.get(ATTR_SUPPORTED_FEATURES, 0)
|
|
|
|
if supported & climate.SUPPORT_TARGET_TEMPERATURE:
|
2019-07-31 19:25:30 +00:00
|
|
|
properties.append({"name": "targetSetpoint"})
|
Climate 1.0 (#23899)
* Climate 1.0 / part 1/2/3
* fix flake
* Lint
* Update Google Assistant
* ambiclimate to climate 1.0 (#24911)
* Fix Alexa
* Lint
* Migrate zhong_hong
* Migrate tuya
* Migrate honeywell to new climate schema (#24257)
* Update one
* Fix model climate v2
* Cleanup p4
* Add comfort hold mode
* Fix old code
* Update homeassistant/components/climate/__init__.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* Update homeassistant/components/climate/const.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* First renaming
* Rename operation to hvac for paulus
* Rename hold mode to preset mode
* Cleanup & update comments
* Remove on/off
* Fix supported feature count
* Update services
* Update demo
* Fix tests & use current_hvac
* Update comment
* Fix tests & add typing
* Add more typing
* Update modes
* Fix tests
* Cleanup low/high with range
* Update homematic part 1
* Finish homematic
* Fix lint
* fix hm mapping
* Support simple devices
* convert lcn
* migrate oem
* Fix xs1
* update hive
* update mil
* Update toon
* migrate deconz
* cleanup
* update tesla
* Fix lint
* Fix vera
* Migrate zwave
* Migrate velbus
* Cleanup humity feature
* Cleanup
* Migrate wink
* migrate dyson
* Fix current hvac
* Renaming
* Fix lint
* Migrate tfiac
* migrate tado
* Fix PRESET can be None
* apply PR#23913 from dev
* remove EU component, etc.
* remove EU component, etc.
* ready to test now
* de-linted
* some tweaks
* de-lint
* better handling of edge cases
* delint
* fix set_mode typos
* apply PR#23913 from dev
* remove EU component, etc.
* ready to test now
* de-linted
* some tweaks
* de-lint
* better handling of edge cases
* delint
* fix set_mode typos
* delint, move debug code
* away preset now working
* code tidy-up
* code tidy-up 2
* code tidy-up 3
* address issues #18932, #15063
* address issues #18932, #15063 - 2/2
* refactor MODE_AUTO to MODE_HEAT_COOL and use F not C
* add low/high to set_temp
* add low/high to set_temp 2
* add low/high to set_temp - delint
* run HA scripts
* port changes from PR #24402
* manual rebase
* manual rebase 2
* delint
* minor change
* remove SUPPORT_HVAC_ACTION
* Migrate radiotherm
* Convert touchline
* Migrate flexit
* Migrate nuheat
* Migrate maxcube
* Fix names maxcube const
* Migrate proliphix
* Migrate heatmiser
* Migrate fritzbox
* Migrate opentherm_gw
* Migrate venstar
* Migrate daikin
* Migrate modbus
* Fix elif
* Migrate Homematic IP Cloud to climate-1.0 (#24913)
* hmip climate fix
* Update hvac_mode and preset_mode
* fix lint
* Fix lint
* Migrate generic_thermostat
* Migrate incomfort to new climate schema (#24915)
* initial commit
* Update climate.py
* Migrate eq3btsmart
* Lint
* cleanup PRESET_MANUAL
* Migrate ecobee
* No conditional features
* KNX: Migrate climate component to new climate platform (#24931)
* Migrate climate component
* Remove unused code
* Corrected line length
* Lint
* Lint
* fix tests
* Fix value
* Migrate geniushub to new climate schema (#24191)
* Update one
* Fix model climate v2
* Cleanup p4
* Add comfort hold mode
* Fix old code
* Update homeassistant/components/climate/__init__.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* Update homeassistant/components/climate/const.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* First renaming
* Rename operation to hvac for paulus
* Rename hold mode to preset mode
* Cleanup & update comments
* Remove on/off
* Fix supported feature count
* Update services
* Update demo
* Fix tests & use current_hvac
* Update comment
* Fix tests & add typing
* Add more typing
* Update modes
* Fix tests
* Cleanup low/high with range
* Update homematic part 1
* Finish homematic
* Fix lint
* fix hm mapping
* Support simple devices
* convert lcn
* migrate oem
* Fix xs1
* update hive
* update mil
* Update toon
* migrate deconz
* cleanup
* update tesla
* Fix lint
* Fix vera
* Migrate zwave
* Migrate velbus
* Cleanup humity feature
* Cleanup
* Migrate wink
* migrate dyson
* Fix current hvac
* Renaming
* Fix lint
* Migrate tfiac
* migrate tado
* delinted
* delinted
* use latest client
* clean up mappings
* clean up mappings
* add duration to set_temperature
* add duration to set_temperature
* manual rebase
* tweak
* fix regression
* small fix
* fix rebase mixup
* address comments
* finish refactor
* fix regression
* tweak type hints
* delint
* manual rebase
* WIP: Fixes for honeywell migration to climate-1.0 (#24938)
* add type hints
* code tidy-up
* Fixes for incomfort migration to climate-1.0 (#24936)
* delint type hints
* no async unless await
* revert: no async unless await
* revert: no async unless await 2
* delint
* fix typo
* Fix homekit_controller on climate-1.0 (#24948)
* Fix tests on climate-1.0 branch
* As part of climate-1.0, make state return the heating-cooling.current characteristic
* Fixes from review
* lint
* Fix imports
* Migrate stibel_eltron
* Fix lint
* Migrate coolmaster to climate 1.0 (#24967)
* Migrate coolmaster to climate 1.0
* fix lint errors
* More lint fixes
* Fix demo to work with UI
* Migrate spider
* Demo update
* Updated frontend to 20190705.0
* Fix boost mode (#24980)
* Prepare Netatmo for climate 1.0 (#24973)
* Migration Netatmo
* Address comments
* Update climate.py
* Migrate ephember
* Migrate Sensibo
* Implemented review comments (#24942)
* Migrate ESPHome
* Migrate MQTT
* Migrate Nest
* Migrate melissa
* Initial/partial migration of ST
* Migrate ST
* Remove Away mode (#24995)
* Migrate evohome, cache access tokens (#24491)
* add water_heater, add storage - initial commit
* add water_heater, add storage - initial commit
delint
add missing code
desiderata
update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
delint
* Add Broker, Water Heater & Refactor
add missing code
desiderata
* update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
* bugfix - loc_idx may not be 0
more refactor - ensure pure async
more refactoring
appears all r/o attributes are working
tweak precsion, DHW & delint
remove unused code
remove unused code 2
remove unused code, refactor _save_auth_tokens()
* support RoundThermostat
bugfix opmode, switch to util.dt, add until=1h
revert breaking change
* store at_expires as naive UTC
remove debug code
delint
tidy up exception handling
delint
add water_heater, add storage - initial commit
delint
add missing code
desiderata
update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
add water_heater, add storage - initial commit
delint
add missing code
desiderata
update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
delint
bugfix - loc_idx may not be 0
more refactor - ensure pure async
more refactoring
appears all r/o attributes are working
tweak precsion, DHW & delint
remove unused code
remove unused code 2
remove unused code, refactor _save_auth_tokens()
support RoundThermostat
bugfix opmode, switch to util.dt, add until=1h
revert breaking change
store at_expires as naive UTC
remove debug code
delint
tidy up exception handling
delint
* update CODEOWNERS
* fix regression
* fix requirements
* migrate to climate-1.0
* tweaking
* de-lint
* TCS working? & delint
* tweaking
* TCS code finalised
* remove available() logic
* refactor _switchpoints()
* tidy up switchpoint code
* tweak
* teaking device_state_attributes
* some refactoring
* move PRESET_CUSTOM back to evohome
* move CONF_ACCESS_TOKEN_EXPIRES CONF_REFRESH_TOKEN back to evohome
* refactor SP code and dt conversion
* delinted
* delinted
* remove water_heater
* fix regression
* Migrate homekit
* Cleanup away mode
* Fix tests
* add helpers
* fix tests melissa
* Fix nehueat
* fix zwave
* add more tests
* fix deconz
* Fix climate test emulate_hue
* fix tests
* fix dyson tests
* fix demo with new layout
* fix honeywell
* Switch homekit_controller to use HVAC_MODE_HEAT_COOL instead of HVAC_MODE_AUTO (#25009)
* Lint
* PyLint
* Pylint
* fix fritzbox tests
* Fix google
* Fix all tests
* Fix lint
* Fix auto for homekit like controler
* Fix lint
* fix lint
2019-07-08 12:00:24 +00:00
|
|
|
if supported & climate.SUPPORT_TARGET_TEMPERATURE_RANGE:
|
2019-07-31 19:25:30 +00:00
|
|
|
properties.append({"name": "lowerSetpoint"})
|
|
|
|
properties.append({"name": "upperSetpoint"})
|
2019-06-13 15:43:57 +00:00
|
|
|
return properties
|
|
|
|
|
|
|
|
def properties_proactively_reported(self):
|
|
|
|
"""Return True if properties asynchronously reported."""
|
|
|
|
return True
|
|
|
|
|
|
|
|
def properties_retrievable(self):
|
|
|
|
"""Return True if properties can be retrieved."""
|
|
|
|
return True
|
|
|
|
|
|
|
|
def get_property(self, name):
|
|
|
|
"""Read and return a property."""
|
2019-10-02 22:53:04 +00:00
|
|
|
if self.entity.state == STATE_UNAVAILABLE:
|
|
|
|
return None
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
if name == "thermostatMode":
|
Climate 1.0 (#23899)
* Climate 1.0 / part 1/2/3
* fix flake
* Lint
* Update Google Assistant
* ambiclimate to climate 1.0 (#24911)
* Fix Alexa
* Lint
* Migrate zhong_hong
* Migrate tuya
* Migrate honeywell to new climate schema (#24257)
* Update one
* Fix model climate v2
* Cleanup p4
* Add comfort hold mode
* Fix old code
* Update homeassistant/components/climate/__init__.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* Update homeassistant/components/climate/const.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* First renaming
* Rename operation to hvac for paulus
* Rename hold mode to preset mode
* Cleanup & update comments
* Remove on/off
* Fix supported feature count
* Update services
* Update demo
* Fix tests & use current_hvac
* Update comment
* Fix tests & add typing
* Add more typing
* Update modes
* Fix tests
* Cleanup low/high with range
* Update homematic part 1
* Finish homematic
* Fix lint
* fix hm mapping
* Support simple devices
* convert lcn
* migrate oem
* Fix xs1
* update hive
* update mil
* Update toon
* migrate deconz
* cleanup
* update tesla
* Fix lint
* Fix vera
* Migrate zwave
* Migrate velbus
* Cleanup humity feature
* Cleanup
* Migrate wink
* migrate dyson
* Fix current hvac
* Renaming
* Fix lint
* Migrate tfiac
* migrate tado
* Fix PRESET can be None
* apply PR#23913 from dev
* remove EU component, etc.
* remove EU component, etc.
* ready to test now
* de-linted
* some tweaks
* de-lint
* better handling of edge cases
* delint
* fix set_mode typos
* apply PR#23913 from dev
* remove EU component, etc.
* ready to test now
* de-linted
* some tweaks
* de-lint
* better handling of edge cases
* delint
* fix set_mode typos
* delint, move debug code
* away preset now working
* code tidy-up
* code tidy-up 2
* code tidy-up 3
* address issues #18932, #15063
* address issues #18932, #15063 - 2/2
* refactor MODE_AUTO to MODE_HEAT_COOL and use F not C
* add low/high to set_temp
* add low/high to set_temp 2
* add low/high to set_temp - delint
* run HA scripts
* port changes from PR #24402
* manual rebase
* manual rebase 2
* delint
* minor change
* remove SUPPORT_HVAC_ACTION
* Migrate radiotherm
* Convert touchline
* Migrate flexit
* Migrate nuheat
* Migrate maxcube
* Fix names maxcube const
* Migrate proliphix
* Migrate heatmiser
* Migrate fritzbox
* Migrate opentherm_gw
* Migrate venstar
* Migrate daikin
* Migrate modbus
* Fix elif
* Migrate Homematic IP Cloud to climate-1.0 (#24913)
* hmip climate fix
* Update hvac_mode and preset_mode
* fix lint
* Fix lint
* Migrate generic_thermostat
* Migrate incomfort to new climate schema (#24915)
* initial commit
* Update climate.py
* Migrate eq3btsmart
* Lint
* cleanup PRESET_MANUAL
* Migrate ecobee
* No conditional features
* KNX: Migrate climate component to new climate platform (#24931)
* Migrate climate component
* Remove unused code
* Corrected line length
* Lint
* Lint
* fix tests
* Fix value
* Migrate geniushub to new climate schema (#24191)
* Update one
* Fix model climate v2
* Cleanup p4
* Add comfort hold mode
* Fix old code
* Update homeassistant/components/climate/__init__.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* Update homeassistant/components/climate/const.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* First renaming
* Rename operation to hvac for paulus
* Rename hold mode to preset mode
* Cleanup & update comments
* Remove on/off
* Fix supported feature count
* Update services
* Update demo
* Fix tests & use current_hvac
* Update comment
* Fix tests & add typing
* Add more typing
* Update modes
* Fix tests
* Cleanup low/high with range
* Update homematic part 1
* Finish homematic
* Fix lint
* fix hm mapping
* Support simple devices
* convert lcn
* migrate oem
* Fix xs1
* update hive
* update mil
* Update toon
* migrate deconz
* cleanup
* update tesla
* Fix lint
* Fix vera
* Migrate zwave
* Migrate velbus
* Cleanup humity feature
* Cleanup
* Migrate wink
* migrate dyson
* Fix current hvac
* Renaming
* Fix lint
* Migrate tfiac
* migrate tado
* delinted
* delinted
* use latest client
* clean up mappings
* clean up mappings
* add duration to set_temperature
* add duration to set_temperature
* manual rebase
* tweak
* fix regression
* small fix
* fix rebase mixup
* address comments
* finish refactor
* fix regression
* tweak type hints
* delint
* manual rebase
* WIP: Fixes for honeywell migration to climate-1.0 (#24938)
* add type hints
* code tidy-up
* Fixes for incomfort migration to climate-1.0 (#24936)
* delint type hints
* no async unless await
* revert: no async unless await
* revert: no async unless await 2
* delint
* fix typo
* Fix homekit_controller on climate-1.0 (#24948)
* Fix tests on climate-1.0 branch
* As part of climate-1.0, make state return the heating-cooling.current characteristic
* Fixes from review
* lint
* Fix imports
* Migrate stibel_eltron
* Fix lint
* Migrate coolmaster to climate 1.0 (#24967)
* Migrate coolmaster to climate 1.0
* fix lint errors
* More lint fixes
* Fix demo to work with UI
* Migrate spider
* Demo update
* Updated frontend to 20190705.0
* Fix boost mode (#24980)
* Prepare Netatmo for climate 1.0 (#24973)
* Migration Netatmo
* Address comments
* Update climate.py
* Migrate ephember
* Migrate Sensibo
* Implemented review comments (#24942)
* Migrate ESPHome
* Migrate MQTT
* Migrate Nest
* Migrate melissa
* Initial/partial migration of ST
* Migrate ST
* Remove Away mode (#24995)
* Migrate evohome, cache access tokens (#24491)
* add water_heater, add storage - initial commit
* add water_heater, add storage - initial commit
delint
add missing code
desiderata
update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
delint
* Add Broker, Water Heater & Refactor
add missing code
desiderata
* update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
* bugfix - loc_idx may not be 0
more refactor - ensure pure async
more refactoring
appears all r/o attributes are working
tweak precsion, DHW & delint
remove unused code
remove unused code 2
remove unused code, refactor _save_auth_tokens()
* support RoundThermostat
bugfix opmode, switch to util.dt, add until=1h
revert breaking change
* store at_expires as naive UTC
remove debug code
delint
tidy up exception handling
delint
add water_heater, add storage - initial commit
delint
add missing code
desiderata
update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
add water_heater, add storage - initial commit
delint
add missing code
desiderata
update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
delint
bugfix - loc_idx may not be 0
more refactor - ensure pure async
more refactoring
appears all r/o attributes are working
tweak precsion, DHW & delint
remove unused code
remove unused code 2
remove unused code, refactor _save_auth_tokens()
support RoundThermostat
bugfix opmode, switch to util.dt, add until=1h
revert breaking change
store at_expires as naive UTC
remove debug code
delint
tidy up exception handling
delint
* update CODEOWNERS
* fix regression
* fix requirements
* migrate to climate-1.0
* tweaking
* de-lint
* TCS working? & delint
* tweaking
* TCS code finalised
* remove available() logic
* refactor _switchpoints()
* tidy up switchpoint code
* tweak
* teaking device_state_attributes
* some refactoring
* move PRESET_CUSTOM back to evohome
* move CONF_ACCESS_TOKEN_EXPIRES CONF_REFRESH_TOKEN back to evohome
* refactor SP code and dt conversion
* delinted
* delinted
* remove water_heater
* fix regression
* Migrate homekit
* Cleanup away mode
* Fix tests
* add helpers
* fix tests melissa
* Fix nehueat
* fix zwave
* add more tests
* fix deconz
* Fix climate test emulate_hue
* fix tests
* fix dyson tests
* fix demo with new layout
* fix honeywell
* Switch homekit_controller to use HVAC_MODE_HEAT_COOL instead of HVAC_MODE_AUTO (#25009)
* Lint
* PyLint
* Pylint
* fix fritzbox tests
* Fix google
* Fix all tests
* Fix lint
* Fix auto for homekit like controler
* Fix lint
* fix lint
2019-07-08 12:00:24 +00:00
|
|
|
preset = self.entity.attributes.get(climate.ATTR_PRESET_MODE)
|
|
|
|
|
|
|
|
if preset in API_THERMOSTAT_PRESETS:
|
|
|
|
mode = API_THERMOSTAT_PRESETS[preset]
|
|
|
|
else:
|
|
|
|
mode = API_THERMOSTAT_MODES.get(self.entity.state)
|
|
|
|
if mode is None:
|
|
|
|
_LOGGER.error(
|
|
|
|
"%s (%s) has unsupported state value '%s'",
|
2019-07-31 19:25:30 +00:00
|
|
|
self.entity.entity_id,
|
|
|
|
type(self.entity),
|
|
|
|
self.entity.state,
|
|
|
|
)
|
Climate 1.0 (#23899)
* Climate 1.0 / part 1/2/3
* fix flake
* Lint
* Update Google Assistant
* ambiclimate to climate 1.0 (#24911)
* Fix Alexa
* Lint
* Migrate zhong_hong
* Migrate tuya
* Migrate honeywell to new climate schema (#24257)
* Update one
* Fix model climate v2
* Cleanup p4
* Add comfort hold mode
* Fix old code
* Update homeassistant/components/climate/__init__.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* Update homeassistant/components/climate/const.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* First renaming
* Rename operation to hvac for paulus
* Rename hold mode to preset mode
* Cleanup & update comments
* Remove on/off
* Fix supported feature count
* Update services
* Update demo
* Fix tests & use current_hvac
* Update comment
* Fix tests & add typing
* Add more typing
* Update modes
* Fix tests
* Cleanup low/high with range
* Update homematic part 1
* Finish homematic
* Fix lint
* fix hm mapping
* Support simple devices
* convert lcn
* migrate oem
* Fix xs1
* update hive
* update mil
* Update toon
* migrate deconz
* cleanup
* update tesla
* Fix lint
* Fix vera
* Migrate zwave
* Migrate velbus
* Cleanup humity feature
* Cleanup
* Migrate wink
* migrate dyson
* Fix current hvac
* Renaming
* Fix lint
* Migrate tfiac
* migrate tado
* Fix PRESET can be None
* apply PR#23913 from dev
* remove EU component, etc.
* remove EU component, etc.
* ready to test now
* de-linted
* some tweaks
* de-lint
* better handling of edge cases
* delint
* fix set_mode typos
* apply PR#23913 from dev
* remove EU component, etc.
* ready to test now
* de-linted
* some tweaks
* de-lint
* better handling of edge cases
* delint
* fix set_mode typos
* delint, move debug code
* away preset now working
* code tidy-up
* code tidy-up 2
* code tidy-up 3
* address issues #18932, #15063
* address issues #18932, #15063 - 2/2
* refactor MODE_AUTO to MODE_HEAT_COOL and use F not C
* add low/high to set_temp
* add low/high to set_temp 2
* add low/high to set_temp - delint
* run HA scripts
* port changes from PR #24402
* manual rebase
* manual rebase 2
* delint
* minor change
* remove SUPPORT_HVAC_ACTION
* Migrate radiotherm
* Convert touchline
* Migrate flexit
* Migrate nuheat
* Migrate maxcube
* Fix names maxcube const
* Migrate proliphix
* Migrate heatmiser
* Migrate fritzbox
* Migrate opentherm_gw
* Migrate venstar
* Migrate daikin
* Migrate modbus
* Fix elif
* Migrate Homematic IP Cloud to climate-1.0 (#24913)
* hmip climate fix
* Update hvac_mode and preset_mode
* fix lint
* Fix lint
* Migrate generic_thermostat
* Migrate incomfort to new climate schema (#24915)
* initial commit
* Update climate.py
* Migrate eq3btsmart
* Lint
* cleanup PRESET_MANUAL
* Migrate ecobee
* No conditional features
* KNX: Migrate climate component to new climate platform (#24931)
* Migrate climate component
* Remove unused code
* Corrected line length
* Lint
* Lint
* fix tests
* Fix value
* Migrate geniushub to new climate schema (#24191)
* Update one
* Fix model climate v2
* Cleanup p4
* Add comfort hold mode
* Fix old code
* Update homeassistant/components/climate/__init__.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* Update homeassistant/components/climate/const.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* First renaming
* Rename operation to hvac for paulus
* Rename hold mode to preset mode
* Cleanup & update comments
* Remove on/off
* Fix supported feature count
* Update services
* Update demo
* Fix tests & use current_hvac
* Update comment
* Fix tests & add typing
* Add more typing
* Update modes
* Fix tests
* Cleanup low/high with range
* Update homematic part 1
* Finish homematic
* Fix lint
* fix hm mapping
* Support simple devices
* convert lcn
* migrate oem
* Fix xs1
* update hive
* update mil
* Update toon
* migrate deconz
* cleanup
* update tesla
* Fix lint
* Fix vera
* Migrate zwave
* Migrate velbus
* Cleanup humity feature
* Cleanup
* Migrate wink
* migrate dyson
* Fix current hvac
* Renaming
* Fix lint
* Migrate tfiac
* migrate tado
* delinted
* delinted
* use latest client
* clean up mappings
* clean up mappings
* add duration to set_temperature
* add duration to set_temperature
* manual rebase
* tweak
* fix regression
* small fix
* fix rebase mixup
* address comments
* finish refactor
* fix regression
* tweak type hints
* delint
* manual rebase
* WIP: Fixes for honeywell migration to climate-1.0 (#24938)
* add type hints
* code tidy-up
* Fixes for incomfort migration to climate-1.0 (#24936)
* delint type hints
* no async unless await
* revert: no async unless await
* revert: no async unless await 2
* delint
* fix typo
* Fix homekit_controller on climate-1.0 (#24948)
* Fix tests on climate-1.0 branch
* As part of climate-1.0, make state return the heating-cooling.current characteristic
* Fixes from review
* lint
* Fix imports
* Migrate stibel_eltron
* Fix lint
* Migrate coolmaster to climate 1.0 (#24967)
* Migrate coolmaster to climate 1.0
* fix lint errors
* More lint fixes
* Fix demo to work with UI
* Migrate spider
* Demo update
* Updated frontend to 20190705.0
* Fix boost mode (#24980)
* Prepare Netatmo for climate 1.0 (#24973)
* Migration Netatmo
* Address comments
* Update climate.py
* Migrate ephember
* Migrate Sensibo
* Implemented review comments (#24942)
* Migrate ESPHome
* Migrate MQTT
* Migrate Nest
* Migrate melissa
* Initial/partial migration of ST
* Migrate ST
* Remove Away mode (#24995)
* Migrate evohome, cache access tokens (#24491)
* add water_heater, add storage - initial commit
* add water_heater, add storage - initial commit
delint
add missing code
desiderata
update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
delint
* Add Broker, Water Heater & Refactor
add missing code
desiderata
* update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
* bugfix - loc_idx may not be 0
more refactor - ensure pure async
more refactoring
appears all r/o attributes are working
tweak precsion, DHW & delint
remove unused code
remove unused code 2
remove unused code, refactor _save_auth_tokens()
* support RoundThermostat
bugfix opmode, switch to util.dt, add until=1h
revert breaking change
* store at_expires as naive UTC
remove debug code
delint
tidy up exception handling
delint
add water_heater, add storage - initial commit
delint
add missing code
desiderata
update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
add water_heater, add storage - initial commit
delint
add missing code
desiderata
update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
delint
bugfix - loc_idx may not be 0
more refactor - ensure pure async
more refactoring
appears all r/o attributes are working
tweak precsion, DHW & delint
remove unused code
remove unused code 2
remove unused code, refactor _save_auth_tokens()
support RoundThermostat
bugfix opmode, switch to util.dt, add until=1h
revert breaking change
store at_expires as naive UTC
remove debug code
delint
tidy up exception handling
delint
* update CODEOWNERS
* fix regression
* fix requirements
* migrate to climate-1.0
* tweaking
* de-lint
* TCS working? & delint
* tweaking
* TCS code finalised
* remove available() logic
* refactor _switchpoints()
* tidy up switchpoint code
* tweak
* teaking device_state_attributes
* some refactoring
* move PRESET_CUSTOM back to evohome
* move CONF_ACCESS_TOKEN_EXPIRES CONF_REFRESH_TOKEN back to evohome
* refactor SP code and dt conversion
* delinted
* delinted
* remove water_heater
* fix regression
* Migrate homekit
* Cleanup away mode
* Fix tests
* add helpers
* fix tests melissa
* Fix nehueat
* fix zwave
* add more tests
* fix deconz
* Fix climate test emulate_hue
* fix tests
* fix dyson tests
* fix demo with new layout
* fix honeywell
* Switch homekit_controller to use HVAC_MODE_HEAT_COOL instead of HVAC_MODE_AUTO (#25009)
* Lint
* PyLint
* Pylint
* fix fritzbox tests
* Fix google
* Fix all tests
* Fix lint
* Fix auto for homekit like controler
* Fix lint
* fix lint
2019-07-08 12:00:24 +00:00
|
|
|
raise UnsupportedProperty(name)
|
2019-06-13 15:43:57 +00:00
|
|
|
return mode
|
|
|
|
|
|
|
|
unit = self.hass.config.units.temperature_unit
|
2019-07-31 19:25:30 +00:00
|
|
|
if name == "targetSetpoint":
|
2019-06-13 15:43:57 +00:00
|
|
|
temp = self.entity.attributes.get(ATTR_TEMPERATURE)
|
2019-07-31 19:25:30 +00:00
|
|
|
elif name == "lowerSetpoint":
|
2019-06-13 15:43:57 +00:00
|
|
|
temp = self.entity.attributes.get(climate.ATTR_TARGET_TEMP_LOW)
|
2019-07-31 19:25:30 +00:00
|
|
|
elif name == "upperSetpoint":
|
2019-06-13 15:43:57 +00:00
|
|
|
temp = self.entity.attributes.get(climate.ATTR_TARGET_TEMP_HIGH)
|
|
|
|
else:
|
|
|
|
raise UnsupportedProperty(name)
|
|
|
|
|
|
|
|
if temp is None:
|
|
|
|
return None
|
|
|
|
|
2019-08-31 14:46:26 +00:00
|
|
|
try:
|
|
|
|
temp = float(temp)
|
|
|
|
except ValueError:
|
|
|
|
_LOGGER.warning(
|
|
|
|
"Invalid temp value %s for %s in %s", temp, name, self.entity.entity_id
|
|
|
|
)
|
|
|
|
return None
|
|
|
|
|
|
|
|
return {"value": temp, "scale": API_TEMP_UNITS[unit]}
|
2019-10-03 20:28:02 +00:00
|
|
|
|
2019-10-25 20:42:21 +00:00
|
|
|
def configuration(self):
|
|
|
|
"""Return configuration object.
|
|
|
|
|
|
|
|
Translates climate HVAC_MODES and PRESETS to supported Alexa ThermostatMode Values.
|
|
|
|
ThermostatMode Value must be AUTO, COOL, HEAT, ECO, OFF, or CUSTOM.
|
|
|
|
"""
|
|
|
|
supported_modes = []
|
|
|
|
hvac_modes = self.entity.attributes.get(climate.ATTR_HVAC_MODES)
|
|
|
|
for mode in hvac_modes:
|
2021-10-17 18:19:56 +00:00
|
|
|
if thermostat_mode := API_THERMOSTAT_MODES.get(mode):
|
2019-10-25 20:42:21 +00:00
|
|
|
supported_modes.append(thermostat_mode)
|
|
|
|
|
|
|
|
preset_modes = self.entity.attributes.get(climate.ATTR_PRESET_MODES)
|
2019-11-25 03:57:18 +00:00
|
|
|
if preset_modes:
|
|
|
|
for mode in preset_modes:
|
|
|
|
thermostat_mode = API_THERMOSTAT_PRESETS.get(mode)
|
|
|
|
if thermostat_mode:
|
|
|
|
supported_modes.append(thermostat_mode)
|
2019-10-25 20:42:21 +00:00
|
|
|
|
|
|
|
# Return False for supportsScheduling until supported with event listener in handler.
|
|
|
|
configuration = {"supportsScheduling": False}
|
|
|
|
|
|
|
|
if supported_modes:
|
|
|
|
configuration["supportedModes"] = supported_modes
|
|
|
|
|
|
|
|
return configuration
|
|
|
|
|
2019-10-03 20:28:02 +00:00
|
|
|
|
2019-10-23 05:01:03 +00:00
|
|
|
class AlexaPowerLevelController(AlexaCapability):
|
2019-10-03 20:28:02 +00:00
|
|
|
"""Implements Alexa.PowerLevelController.
|
|
|
|
|
|
|
|
https://developer.amazon.com/docs/device-apis/alexa-powerlevelcontroller.html
|
|
|
|
"""
|
|
|
|
|
2020-01-03 20:23:22 +00:00
|
|
|
supported_locales = {
|
|
|
|
"de-DE",
|
|
|
|
"en-AU",
|
|
|
|
"en-CA",
|
|
|
|
"en-GB",
|
|
|
|
"en-IN",
|
|
|
|
"en-US",
|
|
|
|
"es-ES",
|
|
|
|
"fr-FR",
|
|
|
|
"it-IT",
|
|
|
|
"ja-JP",
|
|
|
|
}
|
|
|
|
|
2019-10-03 20:28:02 +00:00
|
|
|
def name(self):
|
|
|
|
"""Return the Alexa API name of this interface."""
|
|
|
|
return "Alexa.PowerLevelController"
|
|
|
|
|
|
|
|
def properties_supported(self):
|
|
|
|
"""Return what properties this entity supports."""
|
|
|
|
return [{"name": "powerLevel"}]
|
|
|
|
|
|
|
|
def properties_proactively_reported(self):
|
|
|
|
"""Return True if properties asynchronously reported."""
|
|
|
|
return True
|
|
|
|
|
|
|
|
def properties_retrievable(self):
|
|
|
|
"""Return True if properties can be retrieved."""
|
|
|
|
return True
|
|
|
|
|
|
|
|
def get_property(self, name):
|
|
|
|
"""Read and return a property."""
|
|
|
|
if name != "powerLevel":
|
|
|
|
raise UnsupportedProperty(name)
|
|
|
|
|
2019-10-04 15:41:47 +00:00
|
|
|
|
2019-10-23 05:01:03 +00:00
|
|
|
class AlexaSecurityPanelController(AlexaCapability):
|
2019-10-04 15:41:47 +00:00
|
|
|
"""Implements Alexa.SecurityPanelController.
|
|
|
|
|
|
|
|
https://developer.amazon.com/docs/device-apis/alexa-securitypanelcontroller.html
|
|
|
|
"""
|
|
|
|
|
2020-09-02 07:56:11 +00:00
|
|
|
supported_locales = {
|
|
|
|
"de-DE",
|
|
|
|
"en-AU",
|
|
|
|
"en-CA",
|
|
|
|
"en-GB",
|
|
|
|
"en-IN",
|
|
|
|
"en-US",
|
|
|
|
"es-ES",
|
|
|
|
"es-MX",
|
|
|
|
"es-US",
|
|
|
|
"fr-CA",
|
|
|
|
"fr-FR",
|
|
|
|
"it-IT",
|
|
|
|
"ja-JP",
|
2020-11-20 13:47:17 +00:00
|
|
|
"pt-BR",
|
2020-09-02 07:56:11 +00:00
|
|
|
}
|
2020-01-03 20:23:22 +00:00
|
|
|
|
2019-10-04 15:41:47 +00:00
|
|
|
def __init__(self, hass, entity):
|
|
|
|
"""Initialize the entity."""
|
|
|
|
super().__init__(entity)
|
|
|
|
self.hass = hass
|
|
|
|
|
|
|
|
def name(self):
|
|
|
|
"""Return the Alexa API name of this interface."""
|
|
|
|
return "Alexa.SecurityPanelController"
|
|
|
|
|
|
|
|
def properties_supported(self):
|
|
|
|
"""Return what properties this entity supports."""
|
|
|
|
return [{"name": "armState"}]
|
|
|
|
|
|
|
|
def properties_proactively_reported(self):
|
|
|
|
"""Return True if properties asynchronously reported."""
|
|
|
|
return True
|
|
|
|
|
|
|
|
def properties_retrievable(self):
|
|
|
|
"""Return True if properties can be retrieved."""
|
|
|
|
return True
|
|
|
|
|
|
|
|
def get_property(self, name):
|
|
|
|
"""Read and return a property."""
|
|
|
|
if name != "armState":
|
|
|
|
raise UnsupportedProperty(name)
|
|
|
|
|
|
|
|
arm_state = self.entity.state
|
|
|
|
if arm_state == STATE_ALARM_ARMED_HOME:
|
|
|
|
return "ARMED_STAY"
|
|
|
|
if arm_state == STATE_ALARM_ARMED_AWAY:
|
|
|
|
return "ARMED_AWAY"
|
|
|
|
if arm_state == STATE_ALARM_ARMED_NIGHT:
|
|
|
|
return "ARMED_NIGHT"
|
|
|
|
if arm_state == STATE_ALARM_ARMED_CUSTOM_BYPASS:
|
|
|
|
return "ARMED_STAY"
|
|
|
|
return "DISARMED"
|
|
|
|
|
|
|
|
def configuration(self):
|
2019-10-23 05:01:03 +00:00
|
|
|
"""Return configuration object with supported authorization types."""
|
2019-10-04 15:41:47 +00:00
|
|
|
code_format = self.entity.attributes.get(ATTR_CODE_FORMAT)
|
2020-01-25 01:57:58 +00:00
|
|
|
supported = self.entity.attributes[ATTR_SUPPORTED_FEATURES]
|
|
|
|
configuration = {}
|
|
|
|
|
|
|
|
supported_arm_states = [{"value": "DISARMED"}]
|
|
|
|
if supported & SUPPORT_ALARM_ARM_AWAY:
|
|
|
|
supported_arm_states.append({"value": "ARMED_AWAY"})
|
|
|
|
if supported & SUPPORT_ALARM_ARM_HOME:
|
|
|
|
supported_arm_states.append({"value": "ARMED_STAY"})
|
|
|
|
if supported & SUPPORT_ALARM_ARM_NIGHT:
|
|
|
|
supported_arm_states.append({"value": "ARMED_NIGHT"})
|
|
|
|
|
|
|
|
configuration["supportedArmStates"] = supported_arm_states
|
2019-10-04 15:41:47 +00:00
|
|
|
|
|
|
|
if code_format == FORMAT_NUMBER:
|
2020-01-25 01:57:58 +00:00
|
|
|
configuration["supportedAuthorizationTypes"] = [{"type": "FOUR_DIGIT_PIN"}]
|
|
|
|
|
|
|
|
return configuration
|
2019-10-23 05:01:03 +00:00
|
|
|
|
|
|
|
|
|
|
|
class AlexaModeController(AlexaCapability):
|
|
|
|
"""Implements Alexa.ModeController.
|
|
|
|
|
2020-01-17 23:04:46 +00:00
|
|
|
The instance property must be unique across ModeController, RangeController, ToggleController within the same device.
|
|
|
|
The instance property should be a concatenated string of device domain period and single word.
|
|
|
|
e.g. fan.speed & fan.direction.
|
|
|
|
|
|
|
|
The instance property must not contain words from other instance property strings within the same device.
|
|
|
|
e.g. Instance property cover.position & cover.tilt_position will cause the Alexa.Discovery directive to fail.
|
|
|
|
|
|
|
|
An instance property string value may be reused for different devices.
|
|
|
|
|
2019-10-23 05:01:03 +00:00
|
|
|
https://developer.amazon.com/docs/device-apis/alexa-modecontroller.html
|
|
|
|
"""
|
|
|
|
|
2020-01-03 20:23:22 +00:00
|
|
|
supported_locales = {
|
|
|
|
"de-DE",
|
|
|
|
"en-AU",
|
|
|
|
"en-CA",
|
|
|
|
"en-GB",
|
|
|
|
"en-IN",
|
|
|
|
"en-US",
|
|
|
|
"es-ES",
|
|
|
|
"es-MX",
|
|
|
|
"fr-CA",
|
|
|
|
"fr-FR",
|
|
|
|
"it-IT",
|
|
|
|
"ja-JP",
|
|
|
|
}
|
|
|
|
|
2019-10-23 05:01:03 +00:00
|
|
|
def __init__(self, entity, instance, non_controllable=False):
|
|
|
|
"""Initialize the entity."""
|
|
|
|
super().__init__(entity, instance)
|
2019-12-19 11:44:17 +00:00
|
|
|
self._resource = None
|
|
|
|
self._semantics = None
|
2019-10-23 05:01:03 +00:00
|
|
|
self.properties_non_controllable = lambda: non_controllable
|
|
|
|
|
|
|
|
def name(self):
|
|
|
|
"""Return the Alexa API name of this interface."""
|
|
|
|
return "Alexa.ModeController"
|
|
|
|
|
|
|
|
def properties_supported(self):
|
|
|
|
"""Return what properties this entity supports."""
|
|
|
|
return [{"name": "mode"}]
|
|
|
|
|
|
|
|
def properties_proactively_reported(self):
|
|
|
|
"""Return True if properties asynchronously reported."""
|
|
|
|
return True
|
|
|
|
|
|
|
|
def properties_retrievable(self):
|
|
|
|
"""Return True if properties can be retrieved."""
|
2019-12-19 11:44:17 +00:00
|
|
|
return True
|
2019-10-23 05:01:03 +00:00
|
|
|
|
|
|
|
def get_property(self, name):
|
|
|
|
"""Read and return a property."""
|
|
|
|
if name != "mode":
|
|
|
|
raise UnsupportedProperty(name)
|
|
|
|
|
2019-12-19 11:44:17 +00:00
|
|
|
# Fan Direction
|
2019-10-23 05:01:03 +00:00
|
|
|
if self.instance == f"{fan.DOMAIN}.{fan.ATTR_DIRECTION}":
|
2019-12-19 11:44:17 +00:00
|
|
|
mode = self.entity.attributes.get(fan.ATTR_DIRECTION, None)
|
|
|
|
if mode in (fan.DIRECTION_FORWARD, fan.DIRECTION_REVERSE, STATE_UNKNOWN):
|
|
|
|
return f"{fan.ATTR_DIRECTION}.{mode}"
|
2019-10-23 05:01:03 +00:00
|
|
|
|
2021-05-31 18:58:01 +00:00
|
|
|
# Fan preset_mode
|
|
|
|
if self.instance == f"{fan.DOMAIN}.{fan.ATTR_PRESET_MODE}":
|
|
|
|
mode = self.entity.attributes.get(fan.ATTR_PRESET_MODE, None)
|
|
|
|
if mode in self.entity.attributes.get(fan.ATTR_PRESET_MODES, None):
|
|
|
|
return f"{fan.ATTR_PRESET_MODE}.{mode}"
|
|
|
|
|
2019-12-19 11:44:17 +00:00
|
|
|
# Cover Position
|
2019-11-25 23:07:33 +00:00
|
|
|
if self.instance == f"{cover.DOMAIN}.{cover.ATTR_POSITION}":
|
2019-12-19 11:44:17 +00:00
|
|
|
# Return state instead of position when using ModeController.
|
|
|
|
mode = self.entity.state
|
|
|
|
if mode in (
|
|
|
|
cover.STATE_OPEN,
|
|
|
|
cover.STATE_OPENING,
|
|
|
|
cover.STATE_CLOSED,
|
|
|
|
cover.STATE_CLOSING,
|
|
|
|
STATE_UNKNOWN,
|
|
|
|
):
|
|
|
|
return f"{cover.ATTR_POSITION}.{mode}"
|
2019-11-25 23:07:33 +00:00
|
|
|
|
2019-10-23 05:01:03 +00:00
|
|
|
return None
|
|
|
|
|
|
|
|
def configuration(self):
|
|
|
|
"""Return configuration with modeResources."""
|
2019-12-19 11:44:17 +00:00
|
|
|
if isinstance(self._resource, AlexaCapabilityResource):
|
|
|
|
return self._resource.serialize_configuration()
|
|
|
|
|
|
|
|
return None
|
2019-10-23 05:01:03 +00:00
|
|
|
|
|
|
|
def capability_resources(self):
|
|
|
|
"""Return capabilityResources object."""
|
|
|
|
|
2019-12-19 11:44:17 +00:00
|
|
|
# Fan Direction Resource
|
2019-10-23 05:01:03 +00:00
|
|
|
if self.instance == f"{fan.DOMAIN}.{fan.ATTR_DIRECTION}":
|
2019-12-19 11:44:17 +00:00
|
|
|
self._resource = AlexaModeResource(
|
|
|
|
[AlexaGlobalCatalog.SETTING_DIRECTION], False
|
|
|
|
)
|
|
|
|
self._resource.add_mode(
|
|
|
|
f"{fan.ATTR_DIRECTION}.{fan.DIRECTION_FORWARD}", [fan.DIRECTION_FORWARD]
|
|
|
|
)
|
|
|
|
self._resource.add_mode(
|
|
|
|
f"{fan.ATTR_DIRECTION}.{fan.DIRECTION_REVERSE}", [fan.DIRECTION_REVERSE]
|
|
|
|
)
|
|
|
|
return self._resource.serialize_capability_resources()
|
2019-10-23 05:01:03 +00:00
|
|
|
|
2021-05-31 18:58:01 +00:00
|
|
|
# Fan preset_mode
|
|
|
|
if self.instance == f"{fan.DOMAIN}.{fan.ATTR_PRESET_MODE}":
|
|
|
|
self._resource = AlexaModeResource(
|
|
|
|
[AlexaGlobalCatalog.SETTING_PRESET], False
|
|
|
|
)
|
2021-09-24 06:14:45 +00:00
|
|
|
preset_modes = self.entity.attributes.get(fan.ATTR_PRESET_MODES, [])
|
|
|
|
for preset_mode in preset_modes:
|
2021-05-31 18:58:01 +00:00
|
|
|
self._resource.add_mode(
|
|
|
|
f"{fan.ATTR_PRESET_MODE}.{preset_mode}", [preset_mode]
|
|
|
|
)
|
2021-09-24 06:14:45 +00:00
|
|
|
# Fans with a single preset_mode completely break Alexa discovery, add a
|
|
|
|
# fake preset (see issue #53832).
|
|
|
|
if len(preset_modes) == 1:
|
|
|
|
self._resource.add_mode(
|
|
|
|
f"{fan.ATTR_PRESET_MODE}.{PRESET_MODE_NA}", [PRESET_MODE_NA]
|
|
|
|
)
|
2021-05-31 18:58:01 +00:00
|
|
|
return self._resource.serialize_capability_resources()
|
|
|
|
|
2019-12-19 11:44:17 +00:00
|
|
|
# Cover Position Resources
|
2019-11-25 23:07:33 +00:00
|
|
|
if self.instance == f"{cover.DOMAIN}.{cover.ATTR_POSITION}":
|
2019-12-19 11:44:17 +00:00
|
|
|
self._resource = AlexaModeResource(
|
|
|
|
["Position", AlexaGlobalCatalog.SETTING_OPENING], False
|
|
|
|
)
|
|
|
|
self._resource.add_mode(
|
|
|
|
f"{cover.ATTR_POSITION}.{cover.STATE_OPEN}",
|
|
|
|
[AlexaGlobalCatalog.VALUE_OPEN],
|
|
|
|
)
|
|
|
|
self._resource.add_mode(
|
|
|
|
f"{cover.ATTR_POSITION}.{cover.STATE_CLOSED}",
|
|
|
|
[AlexaGlobalCatalog.VALUE_CLOSE],
|
|
|
|
)
|
2020-01-22 19:04:31 +00:00
|
|
|
self._resource.add_mode(
|
|
|
|
f"{cover.ATTR_POSITION}.custom",
|
|
|
|
["Custom", AlexaGlobalCatalog.SETTING_PRESET],
|
|
|
|
)
|
2019-12-19 11:44:17 +00:00
|
|
|
return self._resource.serialize_capability_resources()
|
2019-11-25 23:07:33 +00:00
|
|
|
|
2019-12-19 11:44:17 +00:00
|
|
|
return None
|
2019-10-23 05:01:03 +00:00
|
|
|
|
2019-12-19 11:44:17 +00:00
|
|
|
def semantics(self):
|
|
|
|
"""Build and return semantics object."""
|
2020-01-17 23:04:46 +00:00
|
|
|
supported = self.entity.attributes.get(ATTR_SUPPORTED_FEATURES, 0)
|
2019-10-23 05:01:03 +00:00
|
|
|
|
2019-12-19 11:44:17 +00:00
|
|
|
# Cover Position
|
2019-11-25 23:07:33 +00:00
|
|
|
if self.instance == f"{cover.DOMAIN}.{cover.ATTR_POSITION}":
|
2020-01-17 23:04:46 +00:00
|
|
|
lower_labels = [AlexaSemantics.ACTION_LOWER]
|
|
|
|
raise_labels = [AlexaSemantics.ACTION_RAISE]
|
2019-12-19 11:44:17 +00:00
|
|
|
self._semantics = AlexaSemantics()
|
2020-01-17 23:04:46 +00:00
|
|
|
|
|
|
|
# Add open/close semantics if tilt is not supported.
|
|
|
|
if not supported & cover.SUPPORT_SET_TILT_POSITION:
|
|
|
|
lower_labels.append(AlexaSemantics.ACTION_CLOSE)
|
|
|
|
raise_labels.append(AlexaSemantics.ACTION_OPEN)
|
|
|
|
self._semantics.add_states_to_value(
|
|
|
|
[AlexaSemantics.STATES_CLOSED],
|
|
|
|
f"{cover.ATTR_POSITION}.{cover.STATE_CLOSED}",
|
|
|
|
)
|
|
|
|
self._semantics.add_states_to_value(
|
|
|
|
[AlexaSemantics.STATES_OPEN],
|
|
|
|
f"{cover.ATTR_POSITION}.{cover.STATE_OPEN}",
|
|
|
|
)
|
|
|
|
|
2019-12-19 11:44:17 +00:00
|
|
|
self._semantics.add_action_to_directive(
|
2020-01-17 23:04:46 +00:00
|
|
|
lower_labels,
|
2019-12-19 11:44:17 +00:00
|
|
|
"SetMode",
|
|
|
|
{"mode": f"{cover.ATTR_POSITION}.{cover.STATE_CLOSED}"},
|
|
|
|
)
|
|
|
|
self._semantics.add_action_to_directive(
|
2020-01-17 23:04:46 +00:00
|
|
|
raise_labels,
|
2019-12-19 11:44:17 +00:00
|
|
|
"SetMode",
|
|
|
|
{"mode": f"{cover.ATTR_POSITION}.{cover.STATE_OPEN}"},
|
|
|
|
)
|
2020-01-17 23:04:46 +00:00
|
|
|
|
2019-12-19 11:44:17 +00:00
|
|
|
return self._semantics.serialize_semantics()
|
2019-10-23 05:01:03 +00:00
|
|
|
|
2019-12-19 11:44:17 +00:00
|
|
|
return None
|
2019-10-23 05:01:03 +00:00
|
|
|
|
|
|
|
|
|
|
|
class AlexaRangeController(AlexaCapability):
|
|
|
|
"""Implements Alexa.RangeController.
|
|
|
|
|
2020-01-17 23:04:46 +00:00
|
|
|
The instance property must be unique across ModeController, RangeController, ToggleController within the same device.
|
|
|
|
The instance property should be a concatenated string of device domain period and single word.
|
|
|
|
e.g. fan.speed & fan.direction.
|
|
|
|
|
|
|
|
The instance property must not contain words from other instance property strings within the same device.
|
|
|
|
e.g. Instance property cover.position & cover.tilt_position will cause the Alexa.Discovery directive to fail.
|
|
|
|
|
|
|
|
An instance property string value may be reused for different devices.
|
|
|
|
|
2019-10-23 05:01:03 +00:00
|
|
|
https://developer.amazon.com/docs/device-apis/alexa-rangecontroller.html
|
|
|
|
"""
|
|
|
|
|
2020-01-03 20:23:22 +00:00
|
|
|
supported_locales = {
|
|
|
|
"de-DE",
|
|
|
|
"en-AU",
|
|
|
|
"en-CA",
|
|
|
|
"en-GB",
|
|
|
|
"en-IN",
|
|
|
|
"en-US",
|
|
|
|
"es-ES",
|
|
|
|
"es-MX",
|
|
|
|
"fr-CA",
|
|
|
|
"fr-FR",
|
|
|
|
"it-IT",
|
|
|
|
"ja-JP",
|
|
|
|
}
|
|
|
|
|
2019-10-23 05:01:03 +00:00
|
|
|
def __init__(self, entity, instance, non_controllable=False):
|
|
|
|
"""Initialize the entity."""
|
|
|
|
super().__init__(entity, instance)
|
2019-12-19 11:44:17 +00:00
|
|
|
self._resource = None
|
|
|
|
self._semantics = None
|
2019-10-23 05:01:03 +00:00
|
|
|
self.properties_non_controllable = lambda: non_controllable
|
|
|
|
|
|
|
|
def name(self):
|
|
|
|
"""Return the Alexa API name of this interface."""
|
|
|
|
return "Alexa.RangeController"
|
|
|
|
|
|
|
|
def properties_supported(self):
|
|
|
|
"""Return what properties this entity supports."""
|
|
|
|
return [{"name": "rangeValue"}]
|
|
|
|
|
|
|
|
def properties_proactively_reported(self):
|
|
|
|
"""Return True if properties asynchronously reported."""
|
|
|
|
return True
|
|
|
|
|
|
|
|
def properties_retrievable(self):
|
|
|
|
"""Return True if properties can be retrieved."""
|
|
|
|
return True
|
|
|
|
|
|
|
|
def get_property(self, name):
|
|
|
|
"""Read and return a property."""
|
|
|
|
if name != "rangeValue":
|
|
|
|
raise UnsupportedProperty(name)
|
|
|
|
|
2020-01-31 22:14:43 +00:00
|
|
|
# Return None for unavailable and unknown states.
|
|
|
|
# Allows the Alexa.EndpointHealth Interface to handle the unavailable state in a stateReport.
|
|
|
|
if self.entity.state in (STATE_UNAVAILABLE, STATE_UNKNOWN, None):
|
|
|
|
return None
|
|
|
|
|
2019-12-19 11:44:17 +00:00
|
|
|
# Cover Position
|
|
|
|
if self.instance == f"{cover.DOMAIN}.{cover.ATTR_POSITION}":
|
|
|
|
return self.entity.attributes.get(cover.ATTR_CURRENT_POSITION)
|
|
|
|
|
2020-01-17 23:04:46 +00:00
|
|
|
# Cover Tilt
|
|
|
|
if self.instance == f"{cover.DOMAIN}.tilt":
|
2019-12-19 11:44:17 +00:00
|
|
|
return self.entity.attributes.get(cover.ATTR_CURRENT_TILT_POSITION)
|
|
|
|
|
2021-09-24 06:14:45 +00:00
|
|
|
# Fan speed percentage
|
|
|
|
if self.instance == f"{fan.DOMAIN}.{fan.ATTR_PERCENTAGE}":
|
|
|
|
supported = self.entity.attributes.get(ATTR_SUPPORTED_FEATURES, 0)
|
|
|
|
if supported and fan.SUPPORT_SET_SPEED:
|
|
|
|
return self.entity.attributes.get(fan.ATTR_PERCENTAGE)
|
|
|
|
return 100 if self.entity.state == fan.STATE_ON else 0
|
|
|
|
|
2019-12-23 14:17:37 +00:00
|
|
|
# Input Number Value
|
|
|
|
if self.instance == f"{input_number.DOMAIN}.{input_number.ATTR_VALUE}":
|
|
|
|
return float(self.entity.state)
|
|
|
|
|
2020-01-15 17:15:31 +00:00
|
|
|
# Vacuum Fan Speed
|
|
|
|
if self.instance == f"{vacuum.DOMAIN}.{vacuum.ATTR_FAN_SPEED}":
|
2020-01-31 22:14:43 +00:00
|
|
|
speed_list = self.entity.attributes.get(vacuum.ATTR_FAN_SPEED_LIST)
|
|
|
|
speed = self.entity.attributes.get(vacuum.ATTR_FAN_SPEED)
|
|
|
|
if speed_list is not None and speed is not None:
|
|
|
|
speed_index = next(
|
|
|
|
(i for i, v in enumerate(speed_list) if v == speed), None
|
|
|
|
)
|
|
|
|
return speed_index
|
2020-01-15 17:15:31 +00:00
|
|
|
|
2019-10-23 05:01:03 +00:00
|
|
|
return None
|
|
|
|
|
|
|
|
def configuration(self):
|
|
|
|
"""Return configuration with presetResources."""
|
2019-12-19 11:44:17 +00:00
|
|
|
if isinstance(self._resource, AlexaCapabilityResource):
|
|
|
|
return self._resource.serialize_configuration()
|
|
|
|
|
|
|
|
return None
|
2019-10-23 05:01:03 +00:00
|
|
|
|
|
|
|
def capability_resources(self):
|
|
|
|
"""Return capabilityResources object."""
|
|
|
|
|
2021-09-24 06:14:45 +00:00
|
|
|
# Fan Speed Percentage Resources
|
|
|
|
if self.instance == f"{fan.DOMAIN}.{fan.ATTR_PERCENTAGE}":
|
|
|
|
percentage_step = self.entity.attributes.get(fan.ATTR_PERCENTAGE_STEP)
|
2019-12-19 11:44:17 +00:00
|
|
|
self._resource = AlexaPresetResource(
|
2021-09-24 06:14:45 +00:00
|
|
|
labels=["Percentage", AlexaGlobalCatalog.SETTING_FAN_SPEED],
|
2020-01-11 01:26:37 +00:00
|
|
|
min_value=0,
|
2021-09-24 06:14:45 +00:00
|
|
|
max_value=100,
|
2021-10-20 11:05:11 +00:00
|
|
|
# precision must be a divider of 100 and must be an integer; set step
|
|
|
|
# size to 1 for a consistent behavior except for on/off fans
|
|
|
|
precision=1 if percentage_step else 100,
|
2021-09-24 06:14:45 +00:00
|
|
|
unit=AlexaGlobalCatalog.UNIT_PERCENT,
|
2019-12-19 11:44:17 +00:00
|
|
|
)
|
|
|
|
return self._resource.serialize_capability_resources()
|
2019-10-23 05:01:03 +00:00
|
|
|
|
2019-12-19 11:44:17 +00:00
|
|
|
# Cover Position Resources
|
|
|
|
if self.instance == f"{cover.DOMAIN}.{cover.ATTR_POSITION}":
|
|
|
|
self._resource = AlexaPresetResource(
|
|
|
|
["Position", AlexaGlobalCatalog.SETTING_OPENING],
|
|
|
|
min_value=0,
|
|
|
|
max_value=100,
|
|
|
|
precision=1,
|
|
|
|
unit=AlexaGlobalCatalog.UNIT_PERCENT,
|
|
|
|
)
|
|
|
|
return self._resource.serialize_capability_resources()
|
|
|
|
|
2020-01-17 23:04:46 +00:00
|
|
|
# Cover Tilt Resources
|
|
|
|
if self.instance == f"{cover.DOMAIN}.tilt":
|
2019-12-19 11:44:17 +00:00
|
|
|
self._resource = AlexaPresetResource(
|
2020-01-17 23:04:46 +00:00
|
|
|
["Tilt", "Angle", AlexaGlobalCatalog.SETTING_DIRECTION],
|
2019-12-19 11:44:17 +00:00
|
|
|
min_value=0,
|
|
|
|
max_value=100,
|
|
|
|
precision=1,
|
|
|
|
unit=AlexaGlobalCatalog.UNIT_PERCENT,
|
|
|
|
)
|
|
|
|
return self._resource.serialize_capability_resources()
|
2019-10-23 05:01:03 +00:00
|
|
|
|
2019-12-23 14:17:37 +00:00
|
|
|
# Input Number Value
|
|
|
|
if self.instance == f"{input_number.DOMAIN}.{input_number.ATTR_VALUE}":
|
|
|
|
min_value = float(self.entity.attributes[input_number.ATTR_MIN])
|
|
|
|
max_value = float(self.entity.attributes[input_number.ATTR_MAX])
|
|
|
|
precision = float(self.entity.attributes.get(input_number.ATTR_STEP, 1))
|
2020-11-25 19:03:56 +00:00
|
|
|
unit = self.entity.attributes.get(ATTR_UNIT_OF_MEASUREMENT)
|
2019-12-23 14:17:37 +00:00
|
|
|
|
|
|
|
self._resource = AlexaPresetResource(
|
2020-01-22 19:04:31 +00:00
|
|
|
["Value", AlexaGlobalCatalog.SETTING_PRESET],
|
2019-12-23 14:17:37 +00:00
|
|
|
min_value=min_value,
|
|
|
|
max_value=max_value,
|
|
|
|
precision=precision,
|
|
|
|
unit=unit,
|
|
|
|
)
|
|
|
|
self._resource.add_preset(
|
|
|
|
value=min_value, labels=[AlexaGlobalCatalog.VALUE_MINIMUM]
|
|
|
|
)
|
|
|
|
self._resource.add_preset(
|
|
|
|
value=max_value, labels=[AlexaGlobalCatalog.VALUE_MAXIMUM]
|
|
|
|
)
|
|
|
|
return self._resource.serialize_capability_resources()
|
|
|
|
|
2020-01-15 17:15:31 +00:00
|
|
|
# Vacuum Fan Speed Resources
|
|
|
|
if self.instance == f"{vacuum.DOMAIN}.{vacuum.ATTR_FAN_SPEED}":
|
|
|
|
speed_list = self.entity.attributes[vacuum.ATTR_FAN_SPEED_LIST]
|
|
|
|
max_value = len(speed_list) - 1
|
|
|
|
self._resource = AlexaPresetResource(
|
|
|
|
labels=[AlexaGlobalCatalog.SETTING_FAN_SPEED],
|
|
|
|
min_value=0,
|
|
|
|
max_value=max_value,
|
|
|
|
precision=1,
|
|
|
|
)
|
|
|
|
for index, speed in enumerate(speed_list):
|
|
|
|
labels = [speed.replace("_", " ")]
|
|
|
|
if index == 1:
|
|
|
|
labels.append(AlexaGlobalCatalog.VALUE_MINIMUM)
|
|
|
|
if index == max_value:
|
|
|
|
labels.append(AlexaGlobalCatalog.VALUE_MAXIMUM)
|
|
|
|
self._resource.add_preset(value=index, labels=labels)
|
|
|
|
|
|
|
|
return self._resource.serialize_capability_resources()
|
|
|
|
|
2019-12-19 11:44:17 +00:00
|
|
|
return None
|
2019-10-23 05:01:03 +00:00
|
|
|
|
2019-12-19 11:44:17 +00:00
|
|
|
def semantics(self):
|
|
|
|
"""Build and return semantics object."""
|
2020-01-17 23:04:46 +00:00
|
|
|
supported = self.entity.attributes.get(ATTR_SUPPORTED_FEATURES, 0)
|
2019-10-23 05:01:03 +00:00
|
|
|
|
2019-12-19 11:44:17 +00:00
|
|
|
# Cover Position
|
|
|
|
if self.instance == f"{cover.DOMAIN}.{cover.ATTR_POSITION}":
|
2020-01-17 23:04:46 +00:00
|
|
|
lower_labels = [AlexaSemantics.ACTION_LOWER]
|
|
|
|
raise_labels = [AlexaSemantics.ACTION_RAISE]
|
2019-12-19 11:44:17 +00:00
|
|
|
self._semantics = AlexaSemantics()
|
2020-01-17 23:04:46 +00:00
|
|
|
|
|
|
|
# Add open/close semantics if tilt is not supported.
|
|
|
|
if not supported & cover.SUPPORT_SET_TILT_POSITION:
|
|
|
|
lower_labels.append(AlexaSemantics.ACTION_CLOSE)
|
|
|
|
raise_labels.append(AlexaSemantics.ACTION_OPEN)
|
|
|
|
self._semantics.add_states_to_value(
|
|
|
|
[AlexaSemantics.STATES_CLOSED], value=0
|
|
|
|
)
|
|
|
|
self._semantics.add_states_to_range(
|
|
|
|
[AlexaSemantics.STATES_OPEN], min_value=1, max_value=100
|
|
|
|
)
|
|
|
|
|
2019-12-19 11:44:17 +00:00
|
|
|
self._semantics.add_action_to_directive(
|
2020-01-17 23:04:46 +00:00
|
|
|
lower_labels, "SetRangeValue", {"rangeValue": 0}
|
2019-12-19 11:44:17 +00:00
|
|
|
)
|
|
|
|
self._semantics.add_action_to_directive(
|
2020-01-17 23:04:46 +00:00
|
|
|
raise_labels, "SetRangeValue", {"rangeValue": 100}
|
2019-10-23 05:01:03 +00:00
|
|
|
)
|
2019-12-19 11:44:17 +00:00
|
|
|
return self._semantics.serialize_semantics()
|
2019-10-23 05:01:03 +00:00
|
|
|
|
2020-01-17 23:04:46 +00:00
|
|
|
# Cover Tilt
|
|
|
|
if self.instance == f"{cover.DOMAIN}.tilt":
|
2019-12-19 11:44:17 +00:00
|
|
|
self._semantics = AlexaSemantics()
|
|
|
|
self._semantics.add_action_to_directive(
|
|
|
|
[AlexaSemantics.ACTION_CLOSE], "SetRangeValue", {"rangeValue": 0}
|
|
|
|
)
|
|
|
|
self._semantics.add_action_to_directive(
|
|
|
|
[AlexaSemantics.ACTION_OPEN], "SetRangeValue", {"rangeValue": 100}
|
|
|
|
)
|
|
|
|
self._semantics.add_states_to_value([AlexaSemantics.STATES_CLOSED], value=0)
|
|
|
|
self._semantics.add_states_to_range(
|
|
|
|
[AlexaSemantics.STATES_OPEN], min_value=1, max_value=100
|
|
|
|
)
|
|
|
|
return self._semantics.serialize_semantics()
|
|
|
|
|
2021-09-24 06:14:45 +00:00
|
|
|
# Fan Speed Percentage
|
|
|
|
if self.instance == f"{fan.DOMAIN}.{fan.ATTR_PERCENTAGE}":
|
|
|
|
lower_labels = [AlexaSemantics.ACTION_LOWER]
|
|
|
|
raise_labels = [AlexaSemantics.ACTION_RAISE]
|
|
|
|
self._semantics = AlexaSemantics()
|
|
|
|
|
|
|
|
self._semantics.add_action_to_directive(
|
|
|
|
lower_labels, "SetRangeValue", {"rangeValue": 0}
|
|
|
|
)
|
|
|
|
self._semantics.add_action_to_directive(
|
|
|
|
raise_labels, "SetRangeValue", {"rangeValue": 100}
|
|
|
|
)
|
|
|
|
return self._semantics.serialize_semantics()
|
|
|
|
|
2019-12-19 11:44:17 +00:00
|
|
|
return None
|
2019-10-23 05:01:03 +00:00
|
|
|
|
|
|
|
|
|
|
|
class AlexaToggleController(AlexaCapability):
|
|
|
|
"""Implements Alexa.ToggleController.
|
|
|
|
|
2020-01-17 23:04:46 +00:00
|
|
|
The instance property must be unique across ModeController, RangeController, ToggleController within the same device.
|
|
|
|
The instance property should be a concatenated string of device domain period and single word.
|
|
|
|
e.g. fan.speed & fan.direction.
|
|
|
|
|
|
|
|
The instance property must not contain words from other instance property strings within the same device.
|
|
|
|
e.g. Instance property cover.position & cover.tilt_position will cause the Alexa.Discovery directive to fail.
|
|
|
|
|
|
|
|
An instance property string value may be reused for different devices.
|
|
|
|
|
2019-10-23 05:01:03 +00:00
|
|
|
https://developer.amazon.com/docs/device-apis/alexa-togglecontroller.html
|
|
|
|
"""
|
|
|
|
|
2020-01-03 20:23:22 +00:00
|
|
|
supported_locales = {
|
|
|
|
"de-DE",
|
|
|
|
"en-AU",
|
|
|
|
"en-CA",
|
|
|
|
"en-GB",
|
|
|
|
"en-IN",
|
|
|
|
"en-US",
|
|
|
|
"es-ES",
|
|
|
|
"es-MX",
|
|
|
|
"fr-CA",
|
|
|
|
"fr-FR",
|
|
|
|
"it-IT",
|
|
|
|
"ja-JP",
|
2020-11-20 13:47:17 +00:00
|
|
|
"pt-BR",
|
2020-01-03 20:23:22 +00:00
|
|
|
}
|
|
|
|
|
2019-10-23 05:01:03 +00:00
|
|
|
def __init__(self, entity, instance, non_controllable=False):
|
|
|
|
"""Initialize the entity."""
|
|
|
|
super().__init__(entity, instance)
|
2019-12-19 11:44:17 +00:00
|
|
|
self._resource = None
|
|
|
|
self._semantics = None
|
2019-10-23 05:01:03 +00:00
|
|
|
self.properties_non_controllable = lambda: non_controllable
|
|
|
|
|
|
|
|
def name(self):
|
|
|
|
"""Return the Alexa API name of this interface."""
|
|
|
|
return "Alexa.ToggleController"
|
|
|
|
|
|
|
|
def properties_supported(self):
|
|
|
|
"""Return what properties this entity supports."""
|
|
|
|
return [{"name": "toggleState"}]
|
|
|
|
|
|
|
|
def properties_proactively_reported(self):
|
|
|
|
"""Return True if properties asynchronously reported."""
|
|
|
|
return True
|
|
|
|
|
|
|
|
def properties_retrievable(self):
|
|
|
|
"""Return True if properties can be retrieved."""
|
|
|
|
return True
|
|
|
|
|
|
|
|
def get_property(self, name):
|
|
|
|
"""Read and return a property."""
|
|
|
|
if name != "toggleState":
|
|
|
|
raise UnsupportedProperty(name)
|
|
|
|
|
2019-12-19 11:44:17 +00:00
|
|
|
# Fan Oscillating
|
2019-10-23 05:01:03 +00:00
|
|
|
if self.instance == f"{fan.DOMAIN}.{fan.ATTR_OSCILLATING}":
|
|
|
|
is_on = bool(self.entity.attributes.get(fan.ATTR_OSCILLATING))
|
|
|
|
return "ON" if is_on else "OFF"
|
|
|
|
|
|
|
|
return None
|
|
|
|
|
|
|
|
def capability_resources(self):
|
|
|
|
"""Return capabilityResources object."""
|
|
|
|
|
2019-12-19 11:44:17 +00:00
|
|
|
# Fan Oscillating Resource
|
2019-10-23 05:01:03 +00:00
|
|
|
if self.instance == f"{fan.DOMAIN}.{fan.ATTR_OSCILLATING}":
|
2019-12-19 11:44:17 +00:00
|
|
|
self._resource = AlexaCapabilityResource(
|
|
|
|
[AlexaGlobalCatalog.SETTING_OSCILLATE, "Rotate", "Rotation"]
|
|
|
|
)
|
|
|
|
return self._resource.serialize_capability_resources()
|
2019-10-23 05:01:03 +00:00
|
|
|
|
2019-12-19 11:44:17 +00:00
|
|
|
return None
|
2019-10-23 15:28:23 +00:00
|
|
|
|
|
|
|
|
|
|
|
class AlexaChannelController(AlexaCapability):
|
|
|
|
"""Implements Alexa.ChannelController.
|
|
|
|
|
|
|
|
https://developer.amazon.com/docs/device-apis/alexa-channelcontroller.html
|
|
|
|
"""
|
|
|
|
|
2020-11-20 13:47:17 +00:00
|
|
|
supported_locales = {
|
|
|
|
"de-DE",
|
|
|
|
"en-AU",
|
|
|
|
"en-CA",
|
|
|
|
"en-GB",
|
|
|
|
"en-IN",
|
|
|
|
"en-US",
|
|
|
|
"es-ES",
|
|
|
|
"es-MX",
|
|
|
|
"fr-FR",
|
|
|
|
"hi-IN",
|
|
|
|
"it-IT",
|
|
|
|
"ja-JP",
|
|
|
|
"pt-BR",
|
|
|
|
}
|
2020-01-03 20:23:22 +00:00
|
|
|
|
2019-10-23 15:28:23 +00:00
|
|
|
def name(self):
|
|
|
|
"""Return the Alexa API name of this interface."""
|
|
|
|
return "Alexa.ChannelController"
|
2019-10-23 18:41:26 +00:00
|
|
|
|
|
|
|
|
|
|
|
class AlexaDoorbellEventSource(AlexaCapability):
|
|
|
|
"""Implements Alexa.DoorbellEventSource.
|
|
|
|
|
|
|
|
https://developer.amazon.com/docs/device-apis/alexa-doorbelleventsource.html
|
|
|
|
"""
|
|
|
|
|
2020-05-29 21:39:17 +00:00
|
|
|
supported_locales = {
|
|
|
|
"de-DE",
|
|
|
|
"en-AU",
|
|
|
|
"en-CA",
|
|
|
|
"en-GB",
|
|
|
|
"en-IN",
|
|
|
|
"en-US",
|
|
|
|
"es-ES",
|
|
|
|
"es-MX",
|
2020-11-20 13:47:17 +00:00
|
|
|
"es-US",
|
2020-05-29 21:39:17 +00:00
|
|
|
"fr-CA",
|
|
|
|
"fr-FR",
|
2020-11-20 13:47:17 +00:00
|
|
|
"hi-IN",
|
2020-05-29 21:39:17 +00:00
|
|
|
"it-IT",
|
|
|
|
"ja-JP",
|
|
|
|
}
|
2020-01-03 20:23:22 +00:00
|
|
|
|
2019-10-23 18:41:26 +00:00
|
|
|
def name(self):
|
|
|
|
"""Return the Alexa API name of this interface."""
|
|
|
|
return "Alexa.DoorbellEventSource"
|
|
|
|
|
|
|
|
def capability_proactively_reported(self):
|
|
|
|
"""Return True for proactively reported capability."""
|
|
|
|
return True
|
2019-10-25 21:53:33 +00:00
|
|
|
|
|
|
|
|
|
|
|
class AlexaPlaybackStateReporter(AlexaCapability):
|
|
|
|
"""Implements Alexa.PlaybackStateReporter.
|
|
|
|
|
|
|
|
https://developer.amazon.com/docs/device-apis/alexa-playbackstatereporter.html
|
|
|
|
"""
|
|
|
|
|
2020-11-20 13:47:17 +00:00
|
|
|
supported_locales = {"de-DE", "en-GB", "en-US", "es-MX", "fr-FR"}
|
2020-01-03 20:23:22 +00:00
|
|
|
|
2019-10-25 21:53:33 +00:00
|
|
|
def name(self):
|
|
|
|
"""Return the Alexa API name of this interface."""
|
|
|
|
return "Alexa.PlaybackStateReporter"
|
|
|
|
|
|
|
|
def properties_supported(self):
|
|
|
|
"""Return what properties this entity supports."""
|
|
|
|
return [{"name": "playbackState"}]
|
|
|
|
|
|
|
|
def properties_proactively_reported(self):
|
|
|
|
"""Return True if properties asynchronously reported."""
|
|
|
|
return True
|
|
|
|
|
|
|
|
def properties_retrievable(self):
|
|
|
|
"""Return True if properties can be retrieved."""
|
|
|
|
return True
|
|
|
|
|
|
|
|
def get_property(self, name):
|
|
|
|
"""Read and return a property."""
|
|
|
|
if name != "playbackState":
|
|
|
|
raise UnsupportedProperty(name)
|
|
|
|
|
|
|
|
playback_state = self.entity.state
|
|
|
|
if playback_state == STATE_PLAYING:
|
|
|
|
return {"state": "PLAYING"}
|
|
|
|
if playback_state == STATE_PAUSED:
|
|
|
|
return {"state": "PAUSED"}
|
|
|
|
|
|
|
|
return {"state": "STOPPED"}
|
2019-10-31 09:38:44 +00:00
|
|
|
|
|
|
|
|
|
|
|
class AlexaSeekController(AlexaCapability):
|
|
|
|
"""Implements Alexa.SeekController.
|
|
|
|
|
|
|
|
https://developer.amazon.com/docs/device-apis/alexa-seekcontroller.html
|
|
|
|
"""
|
|
|
|
|
2020-11-20 13:47:17 +00:00
|
|
|
supported_locales = {"de-DE", "en-GB", "en-US", "es-MX"}
|
2020-01-03 20:23:22 +00:00
|
|
|
|
2019-10-31 09:38:44 +00:00
|
|
|
def name(self):
|
|
|
|
"""Return the Alexa API name of this interface."""
|
|
|
|
return "Alexa.SeekController"
|
2019-12-14 07:47:45 +00:00
|
|
|
|
|
|
|
|
|
|
|
class AlexaEventDetectionSensor(AlexaCapability):
|
|
|
|
"""Implements Alexa.EventDetectionSensor.
|
|
|
|
|
|
|
|
https://developer.amazon.com/docs/device-apis/alexa-eventdetectionsensor.html
|
|
|
|
"""
|
|
|
|
|
2020-01-03 20:23:22 +00:00
|
|
|
supported_locales = {"en-US"}
|
|
|
|
|
2019-12-14 07:47:45 +00:00
|
|
|
def __init__(self, hass, entity):
|
|
|
|
"""Initialize the entity."""
|
|
|
|
super().__init__(entity)
|
|
|
|
self.hass = hass
|
|
|
|
|
|
|
|
def name(self):
|
|
|
|
"""Return the Alexa API name of this interface."""
|
|
|
|
return "Alexa.EventDetectionSensor"
|
|
|
|
|
|
|
|
def properties_supported(self):
|
|
|
|
"""Return what properties this entity supports."""
|
|
|
|
return [{"name": "humanPresenceDetectionState"}]
|
|
|
|
|
|
|
|
def properties_proactively_reported(self):
|
|
|
|
"""Return True if properties asynchronously reported."""
|
|
|
|
return True
|
|
|
|
|
|
|
|
def get_property(self, name):
|
|
|
|
"""Read and return a property."""
|
|
|
|
if name != "humanPresenceDetectionState":
|
|
|
|
raise UnsupportedProperty(name)
|
|
|
|
|
|
|
|
human_presence = "NOT_DETECTED"
|
|
|
|
state = self.entity.state
|
|
|
|
|
|
|
|
# Return None for unavailable and unknown states.
|
|
|
|
# Allows the Alexa.EndpointHealth Interface to handle the unavailable state in a stateReport.
|
|
|
|
if state in (STATE_UNAVAILABLE, STATE_UNKNOWN, None):
|
|
|
|
return None
|
|
|
|
|
|
|
|
if self.entity.domain == image_processing.DOMAIN:
|
|
|
|
if int(state):
|
|
|
|
human_presence = "DETECTED"
|
|
|
|
elif state == STATE_ON:
|
|
|
|
human_presence = "DETECTED"
|
|
|
|
|
|
|
|
return {"value": human_presence}
|
|
|
|
|
|
|
|
def configuration(self):
|
|
|
|
"""Return supported detection types."""
|
|
|
|
return {
|
|
|
|
"detectionMethods": ["AUDIO", "VIDEO"],
|
|
|
|
"detectionModes": {
|
|
|
|
"humanPresence": {
|
|
|
|
"featureAvailability": "ENABLED",
|
|
|
|
"supportsNotDetected": True,
|
|
|
|
}
|
|
|
|
},
|
|
|
|
}
|
2019-12-24 22:06:39 +00:00
|
|
|
|
|
|
|
|
|
|
|
class AlexaEqualizerController(AlexaCapability):
|
|
|
|
"""Implements Alexa.EqualizerController.
|
|
|
|
|
|
|
|
https://developer.amazon.com/en-US/docs/alexa/device-apis/alexa-equalizercontroller.html
|
|
|
|
"""
|
|
|
|
|
2020-11-20 13:47:17 +00:00
|
|
|
supported_locales = {"de-DE", "en-IN", "en-US", "es-ES", "it-IT", "ja-JP", "pt-BR"}
|
2020-05-31 04:42:40 +00:00
|
|
|
VALID_SOUND_MODES = {
|
|
|
|
"MOVIE",
|
|
|
|
"MUSIC",
|
|
|
|
"NIGHT",
|
|
|
|
"SPORT",
|
|
|
|
"TV",
|
|
|
|
}
|
2020-01-03 20:23:22 +00:00
|
|
|
|
2019-12-24 22:06:39 +00:00
|
|
|
def name(self):
|
|
|
|
"""Return the Alexa API name of this interface."""
|
|
|
|
return "Alexa.EqualizerController"
|
|
|
|
|
|
|
|
def properties_supported(self):
|
|
|
|
"""Return what properties this entity supports.
|
|
|
|
|
|
|
|
Either bands, mode or both can be specified. Only mode is supported at this time.
|
|
|
|
"""
|
|
|
|
return [{"name": "mode"}]
|
|
|
|
|
2021-05-17 07:26:37 +00:00
|
|
|
def properties_retrievable(self):
|
|
|
|
"""Return True if properties can be retrieved."""
|
|
|
|
return True
|
|
|
|
|
2019-12-24 22:06:39 +00:00
|
|
|
def get_property(self, name):
|
|
|
|
"""Read and return a property."""
|
|
|
|
if name != "mode":
|
|
|
|
raise UnsupportedProperty(name)
|
|
|
|
|
|
|
|
sound_mode = self.entity.attributes.get(media_player.ATTR_SOUND_MODE)
|
2020-05-31 04:42:40 +00:00
|
|
|
if sound_mode and sound_mode.upper() in self.VALID_SOUND_MODES:
|
2019-12-24 22:06:39 +00:00
|
|
|
return sound_mode.upper()
|
|
|
|
|
|
|
|
return None
|
|
|
|
|
|
|
|
def configurations(self):
|
2020-05-31 04:42:40 +00:00
|
|
|
"""Return the sound modes supported in the configurations object."""
|
2019-12-24 22:06:39 +00:00
|
|
|
configurations = None
|
2020-05-31 04:42:40 +00:00
|
|
|
supported_sound_modes = self.get_valid_inputs(
|
|
|
|
self.entity.attributes.get(media_player.ATTR_SOUND_MODE_LIST, [])
|
|
|
|
)
|
|
|
|
if supported_sound_modes:
|
2019-12-24 22:06:39 +00:00
|
|
|
configurations = {"modes": {"supported": supported_sound_modes}}
|
|
|
|
|
|
|
|
return configurations
|
2020-01-10 22:11:50 +00:00
|
|
|
|
2020-05-31 04:42:40 +00:00
|
|
|
@classmethod
|
|
|
|
def get_valid_inputs(cls, sound_mode_list):
|
|
|
|
"""Return list of supported inputs."""
|
|
|
|
input_list = []
|
|
|
|
for sound_mode in sound_mode_list:
|
|
|
|
sound_mode = sound_mode.upper()
|
|
|
|
|
|
|
|
if sound_mode in cls.VALID_SOUND_MODES:
|
|
|
|
input_list.append({"name": sound_mode})
|
|
|
|
|
|
|
|
return input_list
|
|
|
|
|
2020-01-10 22:11:50 +00:00
|
|
|
|
|
|
|
class AlexaTimeHoldController(AlexaCapability):
|
|
|
|
"""Implements Alexa.TimeHoldController.
|
|
|
|
|
|
|
|
https://developer.amazon.com/docs/device-apis/alexa-timeholdcontroller.html
|
|
|
|
"""
|
|
|
|
|
|
|
|
supported_locales = {"en-US"}
|
|
|
|
|
|
|
|
def __init__(self, entity, allow_remote_resume=False):
|
|
|
|
"""Initialize the entity."""
|
|
|
|
super().__init__(entity)
|
|
|
|
self._allow_remote_resume = allow_remote_resume
|
|
|
|
|
|
|
|
def name(self):
|
|
|
|
"""Return the Alexa API name of this interface."""
|
|
|
|
return "Alexa.TimeHoldController"
|
|
|
|
|
|
|
|
def configuration(self):
|
|
|
|
"""Return configuration object.
|
|
|
|
|
|
|
|
Set allowRemoteResume to True if Alexa can restart the operation on the device.
|
|
|
|
When false, Alexa does not send the Resume directive.
|
|
|
|
"""
|
|
|
|
return {"allowRemoteResume": self._allow_remote_resume}
|
2020-03-28 04:19:11 +00:00
|
|
|
|
|
|
|
|
|
|
|
class AlexaCameraStreamController(AlexaCapability):
|
|
|
|
"""Implements Alexa.CameraStreamController.
|
|
|
|
|
|
|
|
https://developer.amazon.com/docs/device-apis/alexa-camerastreamcontroller.html
|
|
|
|
"""
|
|
|
|
|
|
|
|
supported_locales = {
|
|
|
|
"de-DE",
|
|
|
|
"en-AU",
|
|
|
|
"en-CA",
|
|
|
|
"en-GB",
|
|
|
|
"en-IN",
|
|
|
|
"en-US",
|
|
|
|
"es-ES",
|
|
|
|
"fr-FR",
|
2020-11-20 13:47:17 +00:00
|
|
|
"hi-IN",
|
2020-03-28 04:19:11 +00:00
|
|
|
"it-IT",
|
|
|
|
"ja-JP",
|
2020-11-20 13:47:17 +00:00
|
|
|
"pt-BR",
|
2020-03-28 04:19:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
def name(self):
|
|
|
|
"""Return the Alexa API name of this interface."""
|
|
|
|
return "Alexa.CameraStreamController"
|
|
|
|
|
|
|
|
def camera_stream_configurations(self):
|
|
|
|
"""Return cameraStreamConfigurations object."""
|
2020-04-04 21:20:48 +00:00
|
|
|
return [
|
2020-03-28 04:19:11 +00:00
|
|
|
{
|
|
|
|
"protocols": ["HLS"],
|
|
|
|
"resolutions": [{"width": 1280, "height": 720}],
|
|
|
|
"authorizationTypes": ["NONE"],
|
|
|
|
"videoCodecs": ["H264"],
|
|
|
|
"audioCodecs": ["AAC"],
|
|
|
|
}
|
|
|
|
]
|