mirror of
https://github.com/XiaoMi/ha_xiaomi_home.git
synced 2025-04-04 00:35:33 +08:00
Compare commits
38 Commits
Author | SHA1 | Date | |
---|---|---|---|
|
c6be6be1ec | ||
|
77b0a4531b | ||
|
7d9250914c | ||
|
a09289ef90 | ||
|
b0428dc95a | ||
|
19ed04f2f5 | ||
|
e174a73f52 | ||
|
a1aa1c024f | ||
|
372e635681 | ||
|
3759aa9a1b | ||
|
60d054cf19 | ||
|
6680d9e8cb | ||
|
0ef8cb6370 | ||
|
8f0a69c611 | ||
|
8be0fa5d61 | ||
|
07cb4ed193 | ||
|
5c46504d0e | ||
|
97d89b3a04 | ||
|
4482d257dc | ||
|
d0387be15b | ||
|
27cf1085bd | ||
|
e69448f2eb | ||
|
7901607648 | ||
|
5adcb7ce00 | ||
|
672e5b3f5d | ||
|
417af787c4 | ||
|
6f058bf392 | ||
|
52485d8c7a | ||
|
48554ec0f7 | ||
|
0ce94f7316 | ||
|
20b0004746 | ||
|
57422ddf0d | ||
|
2e60962e94 | ||
|
52fd6371ab | ||
|
8778b00c3a | ||
|
3c16f0ffbb | ||
|
3a5b641ec7 | ||
|
8fb6f9065e |
70
CHANGELOG.md
70
CHANGELOG.md
@ -1,4 +1,74 @@
|
|||||||
# CHANGELOG
|
# CHANGELOG
|
||||||
|
## v0.2.4
|
||||||
|
### Added
|
||||||
|
- Convert the submersion-state, the contact-state and the occupancy-status property to the binary_sensor entity. [#905](https://github.com/XiaoMi/ha_xiaomi_home/pull/905)
|
||||||
|
|
||||||
|
### Changed
|
||||||
|
- suittc.airrtc.wk168 mode descriptions are set to strings of numbers from 1 to 16. [#921](https://github.com/XiaoMi/ha_xiaomi_home/pull/921)
|
||||||
|
- Do not set _attr_suggested_display_precision when the spec.expr is set in spec_modify.yaml [#929](https://github.com/XiaoMi/ha_xiaomi_home/pull/929)
|
||||||
|
- Set "unknown event msg" log to info level.
|
||||||
|
### Fixed
|
||||||
|
- hhcc.plantmonitor.v1 soil moisture and soil ec icon and unit. [#927](https://github.com/XiaoMi/ha_xiaomi_home/pull/27)
|
||||||
|
- cuco.plug.cp2 voltage and power value ratio.
|
||||||
|
- cgllc.airmonitor.s1 unit ppb.
|
||||||
|
- roswan.waterpuri.lte01 tds unit.
|
||||||
|
- lumi.relay.c2acn01 power consumption unit
|
||||||
|
- xiaomi.bhf_light.s1 fan level of ventilation.
|
||||||
|
|
||||||
|
## v0.2.3
|
||||||
|
### Changed
|
||||||
|
- Specify the service name and the property name during the climate entity's on/off feature initialization. [#899](https://github.com/XiaoMi/ha_xiaomi_home/pull/899)
|
||||||
|
- Remove the useless total-battery property from `SPEC_PROP_TRANS_MAP`.
|
||||||
|
### Fixed
|
||||||
|
- Fix the hvac mode setting error when changing the preset mode of the ptc-bath-heater.
|
||||||
|
- Fix the ambiguous descriptions of yeelink.bhf_light.v10 ptc-bath-heater mode value-list.
|
||||||
|
- Fix the power consumption value of chuangmi.plug.212a01. [#910](https://github.com/XiaoMi/ha_xiaomi_home/pull/910)
|
||||||
|
|
||||||
|
## v0.2.2
|
||||||
|
This version has modified the conversion rules of the climate entity, which will have effect on the devices with the ptc-bath-heater, the air-conditioner and the air-fresh service. After updating, you need to restart Home Assistant and check `xiaomi_home > CONFIGURE >
|
||||||
|
Update entity conversion rules > NEXT` to reload the integration.
|
||||||
|
|
||||||
|
这个版本修改了浴霸、空调、新风机的实体转换规则,更新之后需要重启 Home Assistant,并且勾选 `xiaomi_home > 配置 > 更新实体转换规则 > 下一步` 重新加载集成。
|
||||||
|
### Added
|
||||||
|
- Add conversion rules for the air-conditioner service and the air-fresh service. [#879](https://github.com/XiaoMi/ha_xiaomi_home/pull/879)
|
||||||
|
### Changed
|
||||||
|
- Convert the mode of the ptc bath heater to the preset mode of the climate entity. [#874](https://github.com/XiaoMi/ha_xiaomi_home/pull/874)
|
||||||
|
- Use Home Assistant default icon when device_class is set. [#855](https://github.com/XiaoMi/ha_xiaomi_home/pull/855)
|
||||||
|
### Fixed
|
||||||
|
- Fix xiaomi.aircondition.m9 humidity-range unit. [#878](https://github.com/XiaoMi/ha_xiaomi_home/pull/878)
|
||||||
|
- Fix MIoT-Spec-V2 conflicts of xiaomi.fan.p5 and mike.bhf_light.2. [#866](https://github.com/XiaoMi/ha_xiaomi_home/pull/866)
|
||||||
|
|
||||||
|
## v0.2.1
|
||||||
|
### Added
|
||||||
|
- Add the preset mode for the thermostat. [#833](https://github.com/XiaoMi/ha_xiaomi_home/pull/833)
|
||||||
|
### Changed
|
||||||
|
- Change paho-mqtt version to adapt Home Assistant 2025.03. [#839](https://github.com/XiaoMi/ha_xiaomi_home/pull/839)
|
||||||
|
- Revert to use multi_lang.json. [#834](https://github.com/XiaoMi/ha_xiaomi_home/pull/834)
|
||||||
|
### Fixed
|
||||||
|
- Fix the opening and the closing status of linp.wopener.wd1lb. [#826](https://github.com/XiaoMi/ha_xiaomi_home/pull/826)
|
||||||
|
- Fix the format type of the wind-reverse property. [#810](https://github.com/XiaoMi/ha_xiaomi_home/pull/810)
|
||||||
|
- Fix the fan-level property without value-list but with value-range. [#808](https://github.com/XiaoMi/ha_xiaomi_home/pull/808)
|
||||||
|
|
||||||
|
## v0.2.0
|
||||||
|
This version has modified some default units of sensors. After updating, it may cause Home Assistant to pop up some compatibility warnings. You can re-add the integration to resolve it.
|
||||||
|
|
||||||
|
这个版本修改了一些传感器默认单位,更新后会导致 Home Assistant 弹出一些兼容性提示,您可以重新添加集成解决。
|
||||||
|
|
||||||
|
### Added
|
||||||
|
- Add prop trans rule for surge-power. [#595](https://github.com/XiaoMi/ha_xiaomi_home/pull/595)
|
||||||
|
- Support modify spec and value conversion. [#663](https://github.com/XiaoMi/ha_xiaomi_home/pull/663)
|
||||||
|
- Support the electric blanket. [#781](https://github.com/XiaoMi/ha_xiaomi_home/pull/781)
|
||||||
|
- Add device with motor-control service as cover entity. [#688](https://github.com/XiaoMi/ha_xiaomi_home/pull/688)
|
||||||
|
### Changed
|
||||||
|
- Update README file. [#681](https://github.com/XiaoMi/ha_xiaomi_home/pull/681) [#747](https://github.com/XiaoMi/ha_xiaomi_home/pull/747)
|
||||||
|
- Update CONTRIBUTING.md. [#681](https://github.com/XiaoMi/ha_xiaomi_home/pull/681)
|
||||||
|
- Refactor climate.py. [#614](https://github.com/XiaoMi/ha_xiaomi_home/pull/614)
|
||||||
|
### Fixed
|
||||||
|
- Fix variable name or comment errors & fix test_lan error. [#678](https://github.com/XiaoMi/ha_xiaomi_home/pull/678) [#690](https://github.com/XiaoMi/ha_xiaomi_home/pull/690)
|
||||||
|
- Fix water heater error & some type error. [#684](https://github.com/XiaoMi/ha_xiaomi_home/pull/684)
|
||||||
|
- Fix fan level with value-list & fan reverse [#689](https://github.com/XiaoMi/ha_xiaomi_home/pull/689)
|
||||||
|
- Fix sensor display precision [#708](https://github.com/XiaoMi/ha_xiaomi_home/pull/708)
|
||||||
|
- Fix event:motion-detected without arguments [#712](https://github.com/XiaoMi/ha_xiaomi_home/pull/712)
|
||||||
|
|
||||||
## v0.1.5b2
|
## v0.1.5b2
|
||||||
### Added
|
### Added
|
||||||
|
@ -98,6 +98,9 @@ footer: Optional. The footer is the place to reference GitHub issues and PRs tha
|
|||||||
|
|
||||||
When contributing to this project, you agree that your contributions will be licensed under the project's [LICENSE](../LICENSE.md).
|
When contributing to this project, you agree that your contributions will be licensed under the project's [LICENSE](../LICENSE.md).
|
||||||
|
|
||||||
|
|
||||||
|
When you submit your first pull request, GitHub Action will prompt you to sign the Contributor License Agreement (CLA). Only after you sign the CLA, your pull request will be merged.
|
||||||
|
|
||||||
## How to Get Help
|
## How to Get Help
|
||||||
|
|
||||||
If you need help or have questions, feel free to ask in [discussions](https://github.com/XiaoMi/ha_xiaomi_home/discussions/) on GitHub.
|
If you need help or have questions, feel free to ask in [discussions](https://github.com/XiaoMi/ha_xiaomi_home/discussions/) on GitHub.
|
||||||
|
@ -89,4 +89,8 @@ class BinarySensor(MIoTPropertyEntity, BinarySensorEntity):
|
|||||||
@property
|
@property
|
||||||
def is_on(self) -> bool:
|
def is_on(self) -> bool:
|
||||||
"""On/Off state. True if the binary sensor is on, False otherwise."""
|
"""On/Off state. True if the binary sensor is on, False otherwise."""
|
||||||
|
if self.spec.name == 'contact-state':
|
||||||
|
return self._value is False
|
||||||
|
elif self.spec.name == 'occupancy-status':
|
||||||
|
return bool(self._value)
|
||||||
return self._value is True
|
return self._value is True
|
||||||
|
File diff suppressed because it is too large
Load Diff
@ -47,17 +47,14 @@ Cover entities for Xiaomi Home.
|
|||||||
"""
|
"""
|
||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
import logging
|
import logging
|
||||||
from typing import Optional
|
from typing import Any, Optional
|
||||||
|
|
||||||
from homeassistant.config_entries import ConfigEntry
|
from homeassistant.config_entries import ConfigEntry
|
||||||
from homeassistant.core import HomeAssistant
|
from homeassistant.core import HomeAssistant
|
||||||
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
||||||
from homeassistant.components.cover import (
|
from homeassistant.components.cover import (ATTR_POSITION, CoverEntity,
|
||||||
ATTR_POSITION,
|
|
||||||
CoverEntity,
|
|
||||||
CoverEntityFeature,
|
CoverEntityFeature,
|
||||||
CoverDeviceClass
|
CoverDeviceClass)
|
||||||
)
|
|
||||||
|
|
||||||
from .miot.miot_spec import MIoTSpecProperty
|
from .miot.miot_spec import MIoTSpecProperty
|
||||||
from .miot.miot_device import MIoTDevice, MIoTEntityData, MIoTServiceEntity
|
from .miot.miot_device import MIoTDevice, MIoTEntityData, MIoTServiceEntity
|
||||||
@ -66,11 +63,8 @@ from .miot.const import DOMAIN
|
|||||||
_LOGGER = logging.getLogger(__name__)
|
_LOGGER = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
async def async_setup_entry(
|
async def async_setup_entry(hass: HomeAssistant, config_entry: ConfigEntry,
|
||||||
hass: HomeAssistant,
|
async_add_entities: AddEntitiesCallback) -> None:
|
||||||
config_entry: ConfigEntry,
|
|
||||||
async_add_entities: AddEntitiesCallback,
|
|
||||||
) -> None:
|
|
||||||
"""Set up a config entry."""
|
"""Set up a config entry."""
|
||||||
device_list: list[MIoTDevice] = hass.data[DOMAIN]['devices'][
|
device_list: list[MIoTDevice] = hass.data[DOMAIN]['devices'][
|
||||||
config_entry.entry_id]
|
config_entry.entry_id]
|
||||||
@ -82,8 +76,12 @@ async def async_setup_entry(
|
|||||||
data.spec.device_class = CoverDeviceClass.CURTAIN
|
data.spec.device_class = CoverDeviceClass.CURTAIN
|
||||||
elif data.spec.name == 'window-opener':
|
elif data.spec.name == 'window-opener':
|
||||||
data.spec.device_class = CoverDeviceClass.WINDOW
|
data.spec.device_class = CoverDeviceClass.WINDOW
|
||||||
new_entities.append(
|
elif data.spec.name == 'motor-controller':
|
||||||
Cover(miot_device=miot_device, entity_data=data))
|
data.spec.device_class = CoverDeviceClass.SHUTTER
|
||||||
|
elif data.spec.name == 'airer':
|
||||||
|
data.spec.device_class = CoverDeviceClass.BLIND
|
||||||
|
new_entities.append(Cover(miot_device=miot_device,
|
||||||
|
entity_data=data))
|
||||||
|
|
||||||
if new_entities:
|
if new_entities:
|
||||||
async_add_entities(new_entities)
|
async_add_entities(new_entities)
|
||||||
@ -97,18 +95,20 @@ class Cover(MIoTServiceEntity, CoverEntity):
|
|||||||
_prop_motor_value_close: Optional[int]
|
_prop_motor_value_close: Optional[int]
|
||||||
_prop_motor_value_pause: Optional[int]
|
_prop_motor_value_pause: Optional[int]
|
||||||
_prop_status: Optional[MIoTSpecProperty]
|
_prop_status: Optional[MIoTSpecProperty]
|
||||||
_prop_status_opening: Optional[int]
|
_prop_status_opening: Optional[list[int]]
|
||||||
_prop_status_closing: Optional[int]
|
_prop_status_closing: Optional[list[int]]
|
||||||
_prop_status_stop: Optional[int]
|
_prop_status_stop: Optional[list[int]]
|
||||||
|
_prop_status_closed: Optional[list[int]]
|
||||||
_prop_current_position: Optional[MIoTSpecProperty]
|
_prop_current_position: Optional[MIoTSpecProperty]
|
||||||
_prop_target_position: Optional[MIoTSpecProperty]
|
_prop_target_position: Optional[MIoTSpecProperty]
|
||||||
_prop_position_value_min: Optional[int]
|
_prop_position_value_min: Optional[int]
|
||||||
_prop_position_value_max: Optional[int]
|
_prop_position_value_max: Optional[int]
|
||||||
_prop_position_value_range: Optional[int]
|
_prop_position_value_range: Optional[int]
|
||||||
|
_prop_pos_closing: bool
|
||||||
|
_prop_pos_opening: bool
|
||||||
|
|
||||||
def __init__(
|
def __init__(self, miot_device: MIoTDevice,
|
||||||
self, miot_device: MIoTDevice, entity_data: MIoTEntityData
|
entity_data: MIoTEntityData) -> None:
|
||||||
) -> None:
|
|
||||||
"""Initialize the Cover."""
|
"""Initialize the Cover."""
|
||||||
super().__init__(miot_device=miot_device, entity_data=entity_data)
|
super().__init__(miot_device=miot_device, entity_data=entity_data)
|
||||||
self._attr_device_class = entity_data.spec.device_class
|
self._attr_device_class = entity_data.spec.device_class
|
||||||
@ -120,50 +120,64 @@ class Cover(MIoTServiceEntity, CoverEntity):
|
|||||||
self._prop_motor_value_close = None
|
self._prop_motor_value_close = None
|
||||||
self._prop_motor_value_pause = None
|
self._prop_motor_value_pause = None
|
||||||
self._prop_status = None
|
self._prop_status = None
|
||||||
self._prop_status_opening = None
|
self._prop_status_opening = []
|
||||||
self._prop_status_closing = None
|
self._prop_status_closing = []
|
||||||
self._prop_status_stop = None
|
self._prop_status_stop = []
|
||||||
|
self._prop_status_closed = []
|
||||||
self._prop_current_position = None
|
self._prop_current_position = None
|
||||||
self._prop_target_position = None
|
self._prop_target_position = None
|
||||||
self._prop_position_value_min = None
|
self._prop_position_value_min = None
|
||||||
self._prop_position_value_max = None
|
self._prop_position_value_max = None
|
||||||
self._prop_position_value_range = None
|
self._prop_position_value_range = None
|
||||||
|
self._prop_pos_closing = False
|
||||||
|
self._prop_pos_opening = False
|
||||||
|
|
||||||
# properties
|
# properties
|
||||||
for prop in entity_data.props:
|
for prop in entity_data.props:
|
||||||
if prop.name == 'motor-control':
|
if prop.name == 'motor-control':
|
||||||
if not prop.value_list:
|
if not prop.value_list:
|
||||||
_LOGGER.error(
|
_LOGGER.error('motor-control value_list is None, %s',
|
||||||
'motor-control value_list is None, %s', self.entity_id)
|
self.entity_id)
|
||||||
continue
|
continue
|
||||||
for item in prop.value_list.items:
|
for item in prop.value_list.items:
|
||||||
if item.name in {'open'}:
|
if item.name in {'open', 'up'}:
|
||||||
self._attr_supported_features |= (
|
self._attr_supported_features |= (
|
||||||
CoverEntityFeature.OPEN)
|
CoverEntityFeature.OPEN)
|
||||||
self._prop_motor_value_open = item.value
|
self._prop_motor_value_open = item.value
|
||||||
elif item.name in {'close'}:
|
elif item.name in {'close', 'down'}:
|
||||||
self._attr_supported_features |= (
|
self._attr_supported_features |= (
|
||||||
CoverEntityFeature.CLOSE)
|
CoverEntityFeature.CLOSE)
|
||||||
self._prop_motor_value_close = item.value
|
self._prop_motor_value_close = item.value
|
||||||
elif item.name in {'pause'}:
|
elif item.name in {'pause', 'stop'}:
|
||||||
self._attr_supported_features |= (
|
self._attr_supported_features |= (
|
||||||
CoverEntityFeature.STOP)
|
CoverEntityFeature.STOP)
|
||||||
self._prop_motor_value_pause = item.value
|
self._prop_motor_value_pause = item.value
|
||||||
self._prop_motor_control = prop
|
self._prop_motor_control = prop
|
||||||
elif prop.name == 'status':
|
elif prop.name == 'status':
|
||||||
if not prop.value_list:
|
if not prop.value_list:
|
||||||
_LOGGER.error(
|
_LOGGER.error('status value_list is None, %s',
|
||||||
'status value_list is None, %s', self.entity_id)
|
self.entity_id)
|
||||||
continue
|
continue
|
||||||
for item in prop.value_list.items:
|
for item in prop.value_list.items:
|
||||||
if item.name in {'opening', 'open'}:
|
if item.name in {'opening', 'open', 'up'}:
|
||||||
self._prop_status_opening = item.value
|
self._prop_status_opening.append(item.value)
|
||||||
elif item.name in {'closing', 'close'}:
|
elif item.name in {'closing', 'close', 'down'}:
|
||||||
self._prop_status_closing = item.value
|
self._prop_status_closing.append(item.value)
|
||||||
elif item.name in {'stop', 'pause'}:
|
elif item.name in {'stop', 'stopped', 'pause'}:
|
||||||
self._prop_status_stop = item.value
|
self._prop_status_stop.append(item.value)
|
||||||
|
elif item.name in {'closed'}:
|
||||||
|
self._prop_status_closed.append(item.value)
|
||||||
self._prop_status = prop
|
self._prop_status = prop
|
||||||
elif prop.name == 'current-position':
|
elif prop.name == 'current-position':
|
||||||
|
if not prop.value_range:
|
||||||
|
_LOGGER.error(
|
||||||
|
'invalid current-position value_range format, %s',
|
||||||
|
self.entity_id)
|
||||||
|
continue
|
||||||
|
self._prop_position_value_min = prop.value_range.min_
|
||||||
|
self._prop_position_value_max = prop.value_range.max_
|
||||||
|
self._prop_position_value_range = (prop.value_range.max_ -
|
||||||
|
prop.value_range.min_)
|
||||||
self._prop_current_position = prop
|
self._prop_current_position = prop
|
||||||
elif prop.name == 'target-position':
|
elif prop.name == 'target-position':
|
||||||
if not prop.value_range:
|
if not prop.value_range:
|
||||||
@ -173,35 +187,65 @@ class Cover(MIoTServiceEntity, CoverEntity):
|
|||||||
continue
|
continue
|
||||||
self._prop_position_value_min = prop.value_range.min_
|
self._prop_position_value_min = prop.value_range.min_
|
||||||
self._prop_position_value_max = prop.value_range.max_
|
self._prop_position_value_max = prop.value_range.max_
|
||||||
self._prop_position_value_range = (
|
self._prop_position_value_range = (prop.value_range.max_ -
|
||||||
self._prop_position_value_max -
|
prop.value_range.min_)
|
||||||
self._prop_position_value_min)
|
|
||||||
self._attr_supported_features |= CoverEntityFeature.SET_POSITION
|
self._attr_supported_features |= CoverEntityFeature.SET_POSITION
|
||||||
self._prop_target_position = prop
|
self._prop_target_position = prop
|
||||||
|
# For the device that has the current position property but no status
|
||||||
|
# property, the current position property will be used to determine the
|
||||||
|
# opening and the closing status.
|
||||||
|
if (self._prop_status is None) and (self._prop_current_position
|
||||||
|
is not None):
|
||||||
|
self.sub_prop_changed(self._prop_current_position,
|
||||||
|
self._position_changed_handler)
|
||||||
|
|
||||||
|
def _position_changed_handler(self, prop: MIoTSpecProperty,
|
||||||
|
ctx: Any) -> None:
|
||||||
|
self._prop_pos_closing = False
|
||||||
|
self._prop_pos_opening = False
|
||||||
|
self.async_write_ha_state()
|
||||||
|
|
||||||
async def async_open_cover(self, **kwargs) -> None:
|
async def async_open_cover(self, **kwargs) -> None:
|
||||||
"""Open the cover."""
|
"""Open the cover."""
|
||||||
await self.set_property_async(
|
current = None if (self._prop_current_position
|
||||||
self._prop_motor_control, self._prop_motor_value_open)
|
is None) else self.get_prop_value(
|
||||||
|
prop=self._prop_current_position)
|
||||||
|
if (current is not None) and (current < self._prop_position_value_max):
|
||||||
|
self._prop_pos_opening = True
|
||||||
|
self._prop_pos_closing = False
|
||||||
|
await self.set_property_async(self._prop_motor_control,
|
||||||
|
self._prop_motor_value_open)
|
||||||
|
|
||||||
async def async_close_cover(self, **kwargs) -> None:
|
async def async_close_cover(self, **kwargs) -> None:
|
||||||
"""Close the cover."""
|
"""Close the cover."""
|
||||||
await self.set_property_async(
|
current = None if (self._prop_current_position
|
||||||
self._prop_motor_control, self._prop_motor_value_close)
|
is None) else self.get_prop_value(
|
||||||
|
prop=self._prop_current_position)
|
||||||
|
if (current is not None) and (current > self._prop_position_value_min):
|
||||||
|
self._prop_pos_opening = False
|
||||||
|
self._prop_pos_closing = True
|
||||||
|
await self.set_property_async(self._prop_motor_control,
|
||||||
|
self._prop_motor_value_close)
|
||||||
|
|
||||||
async def async_stop_cover(self, **kwargs) -> None:
|
async def async_stop_cover(self, **kwargs) -> None:
|
||||||
"""Stop the cover."""
|
"""Stop the cover."""
|
||||||
await self.set_property_async(
|
self._prop_pos_opening = False
|
||||||
self._prop_motor_control, self._prop_motor_value_pause)
|
self._prop_pos_closing = False
|
||||||
|
await self.set_property_async(self._prop_motor_control,
|
||||||
|
self._prop_motor_value_pause)
|
||||||
|
|
||||||
async def async_set_cover_position(self, **kwargs) -> None:
|
async def async_set_cover_position(self, **kwargs) -> None:
|
||||||
"""Set the position of the cover."""
|
"""Set the position of the cover."""
|
||||||
pos = kwargs.get(ATTR_POSITION, None)
|
pos = kwargs.get(ATTR_POSITION, None)
|
||||||
if pos is None:
|
if pos is None:
|
||||||
return None
|
return None
|
||||||
pos = round(pos*self._prop_position_value_range/100)
|
current = self.current_cover_position
|
||||||
return await self.set_property_async(
|
if current is not None:
|
||||||
prop=self._prop_target_position, value=pos)
|
self._prop_pos_opening = pos > current
|
||||||
|
self._prop_pos_closing = pos < current
|
||||||
|
pos = round(pos * self._prop_position_value_range / 100)
|
||||||
|
await self.set_property_async(prop=self._prop_target_position,
|
||||||
|
value=pos)
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def current_cover_position(self) -> Optional[int]:
|
def current_cover_position(self) -> Optional[int]:
|
||||||
@ -209,28 +253,47 @@ class Cover(MIoTServiceEntity, CoverEntity):
|
|||||||
|
|
||||||
0: the cover is closed, 100: the cover is fully opened, None: unknown.
|
0: the cover is closed, 100: the cover is fully opened, None: unknown.
|
||||||
"""
|
"""
|
||||||
pos = self.get_prop_value(prop=self._prop_current_position)
|
if self._prop_current_position is None:
|
||||||
if pos is None:
|
# Assume that the current position is the same as the target
|
||||||
|
# position when the current position is not defined in the device's
|
||||||
|
# MIoT-Spec-V2.
|
||||||
|
if self._prop_target_position is None:
|
||||||
return None
|
return None
|
||||||
return round(pos*100/self._prop_position_value_range)
|
self._prop_pos_opening = False
|
||||||
|
self._prop_pos_closing = False
|
||||||
|
return self.get_prop_value(prop=self._prop_target_position)
|
||||||
|
pos = self.get_prop_value(prop=self._prop_current_position)
|
||||||
|
return None if pos is None else round(pos * 100 /
|
||||||
|
self._prop_position_value_range)
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def is_opening(self) -> Optional[bool]:
|
def is_opening(self) -> Optional[bool]:
|
||||||
"""Return if the cover is opening."""
|
"""Return if the cover is opening."""
|
||||||
if self._prop_status is None:
|
if self._prop_status and self._prop_status_opening:
|
||||||
return None
|
return (self.get_prop_value(prop=self._prop_status)
|
||||||
return self.get_prop_value(
|
in self._prop_status_opening)
|
||||||
prop=self._prop_status) == self._prop_status_opening
|
# The status has higher priority when determining whether the cover
|
||||||
|
# is opening.
|
||||||
|
return self._prop_pos_opening
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def is_closing(self) -> Optional[bool]:
|
def is_closing(self) -> Optional[bool]:
|
||||||
"""Return if the cover is closing."""
|
"""Return if the cover is closing."""
|
||||||
if self._prop_status is None:
|
if self._prop_status and self._prop_status_closing:
|
||||||
return None
|
return (self.get_prop_value(prop=self._prop_status)
|
||||||
return self.get_prop_value(
|
in self._prop_status_closing)
|
||||||
prop=self._prop_status) == self._prop_status_closing
|
# The status has higher priority when determining whether the cover
|
||||||
|
# is closing.
|
||||||
|
return self._prop_pos_closing
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def is_closed(self) -> Optional[bool]:
|
def is_closed(self) -> Optional[bool]:
|
||||||
"""Return if the cover is closed."""
|
"""Return if the cover is closed."""
|
||||||
return self.get_prop_value(prop=self._prop_current_position) == 0
|
if self.current_cover_position is not None:
|
||||||
|
return self.current_cover_position == 0
|
||||||
|
# The current position is prior to the status when determining
|
||||||
|
# whether the cover is closed.
|
||||||
|
if self._prop_status and self._prop_status_closed:
|
||||||
|
return (self.get_prop_value(prop=self._prop_status)
|
||||||
|
in self._prop_status_closed)
|
||||||
|
return None
|
||||||
|
@ -46,6 +46,7 @@ off Xiaomi or its affiliates' products.
|
|||||||
Event entities for Xiaomi Home.
|
Event entities for Xiaomi Home.
|
||||||
"""
|
"""
|
||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
|
import logging
|
||||||
from typing import Any
|
from typing import Any
|
||||||
|
|
||||||
from homeassistant.config_entries import ConfigEntry
|
from homeassistant.config_entries import ConfigEntry
|
||||||
@ -57,6 +58,8 @@ from .miot.miot_spec import MIoTSpecEvent
|
|||||||
from .miot.miot_device import MIoTDevice, MIoTEventEntity
|
from .miot.miot_device import MIoTDevice, MIoTEventEntity
|
||||||
from .miot.const import DOMAIN
|
from .miot.const import DOMAIN
|
||||||
|
|
||||||
|
_LOGGER = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
async def async_setup_entry(
|
async def async_setup_entry(
|
||||||
hass: HomeAssistant,
|
hass: HomeAssistant,
|
||||||
@ -89,4 +92,5 @@ class Event(MIoTEventEntity, EventEntity):
|
|||||||
self, name: str, arguments: dict[str, Any] | None = None
|
self, name: str, arguments: dict[str, Any] | None = None
|
||||||
) -> None:
|
) -> None:
|
||||||
"""An event is occurred."""
|
"""An event is occurred."""
|
||||||
|
_LOGGER.debug('%s, attributes: %s', name, str(arguments))
|
||||||
self._trigger_event(event_type=name, event_attributes=arguments)
|
self._trigger_event(event_type=name, event_attributes=arguments)
|
||||||
|
@ -52,7 +52,12 @@ import logging
|
|||||||
from homeassistant.config_entries import ConfigEntry
|
from homeassistant.config_entries import ConfigEntry
|
||||||
from homeassistant.core import HomeAssistant
|
from homeassistant.core import HomeAssistant
|
||||||
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
||||||
from homeassistant.components.fan import FanEntity, FanEntityFeature
|
from homeassistant.components.fan import (
|
||||||
|
FanEntity,
|
||||||
|
FanEntityFeature,
|
||||||
|
DIRECTION_FORWARD,
|
||||||
|
DIRECTION_REVERSE
|
||||||
|
)
|
||||||
from homeassistant.util.percentage import (
|
from homeassistant.util.percentage import (
|
||||||
percentage_to_ranged_value,
|
percentage_to_ranged_value,
|
||||||
ranged_value_to_percentage,
|
ranged_value_to_percentage,
|
||||||
@ -167,20 +172,21 @@ class Fan(MIoTServiceEntity, FanEntity):
|
|||||||
self._attr_supported_features |= FanEntityFeature.OSCILLATE
|
self._attr_supported_features |= FanEntityFeature.OSCILLATE
|
||||||
self._prop_horizontal_swing = prop
|
self._prop_horizontal_swing = prop
|
||||||
elif prop.name == 'wind-reverse':
|
elif prop.name == 'wind-reverse':
|
||||||
if prop.format_ == 'bool':
|
if prop.format_ == bool:
|
||||||
self._prop_wind_reverse_forward = False
|
self._prop_wind_reverse_forward = False
|
||||||
self._prop_wind_reverse_reverse = True
|
self._prop_wind_reverse_reverse = True
|
||||||
elif prop.value_list:
|
elif prop.value_list:
|
||||||
for item in prop.value_list.items:
|
for item in prop.value_list.items:
|
||||||
if item.name in {'foreward'}:
|
if item.name in {'foreward', 'forward'}:
|
||||||
self._prop_wind_reverse_forward = item.value
|
self._prop_wind_reverse_forward = item.value
|
||||||
|
elif item.name in {'reversal', 'reverse'}:
|
||||||
self._prop_wind_reverse_reverse = item.value
|
self._prop_wind_reverse_reverse = item.value
|
||||||
if (
|
if (
|
||||||
self._prop_wind_reverse_forward is None
|
self._prop_wind_reverse_forward is None
|
||||||
or self._prop_wind_reverse_reverse is None
|
or self._prop_wind_reverse_reverse is None
|
||||||
):
|
):
|
||||||
# NOTICE: Value may be 0 or False
|
# NOTICE: Value may be 0 or False
|
||||||
_LOGGER.info(
|
_LOGGER.error(
|
||||||
'invalid wind-reverse, %s', self.entity_id)
|
'invalid wind-reverse, %s', self.entity_id)
|
||||||
continue
|
continue
|
||||||
self._attr_supported_features |= FanEntityFeature.DIRECTION
|
self._attr_supported_features |= FanEntityFeature.DIRECTION
|
||||||
@ -202,9 +208,9 @@ class Fan(MIoTServiceEntity, FanEntity):
|
|||||||
if self._speed_names:
|
if self._speed_names:
|
||||||
await self.set_property_async(
|
await self.set_property_async(
|
||||||
prop=self._prop_fan_level,
|
prop=self._prop_fan_level,
|
||||||
value=self.get_map_value(
|
value=self.get_map_key(
|
||||||
map_=self._speed_name_map,
|
map_=self._speed_name_map,
|
||||||
key=percentage_to_ordered_list_item(
|
value=percentage_to_ordered_list_item(
|
||||||
self._speed_names, percentage)))
|
self._speed_names, percentage)))
|
||||||
else:
|
else:
|
||||||
await self.set_property_async(
|
await self.set_property_async(
|
||||||
@ -233,9 +239,9 @@ class Fan(MIoTServiceEntity, FanEntity):
|
|||||||
if self._speed_names:
|
if self._speed_names:
|
||||||
await self.set_property_async(
|
await self.set_property_async(
|
||||||
prop=self._prop_fan_level,
|
prop=self._prop_fan_level,
|
||||||
value=self.get_map_value(
|
value=self.get_map_key(
|
||||||
map_=self._speed_name_map,
|
map_=self._speed_name_map,
|
||||||
key=percentage_to_ordered_list_item(
|
value=percentage_to_ordered_list_item(
|
||||||
self._speed_names, percentage)))
|
self._speed_names, percentage)))
|
||||||
else:
|
else:
|
||||||
await self.set_property_async(
|
await self.set_property_async(
|
||||||
@ -264,7 +270,7 @@ class Fan(MIoTServiceEntity, FanEntity):
|
|||||||
prop=self._prop_wind_reverse,
|
prop=self._prop_wind_reverse,
|
||||||
value=(
|
value=(
|
||||||
self._prop_wind_reverse_reverse
|
self._prop_wind_reverse_reverse
|
||||||
if self.current_direction == 'reverse'
|
if direction == DIRECTION_REVERSE
|
||||||
else self._prop_wind_reverse_forward))
|
else self._prop_wind_reverse_forward))
|
||||||
|
|
||||||
async def async_oscillate(self, oscillating: bool) -> None:
|
async def async_oscillate(self, oscillating: bool) -> None:
|
||||||
@ -293,9 +299,9 @@ class Fan(MIoTServiceEntity, FanEntity):
|
|||||||
"""Return the current direction of the fan."""
|
"""Return the current direction of the fan."""
|
||||||
if not self._prop_wind_reverse:
|
if not self._prop_wind_reverse:
|
||||||
return None
|
return None
|
||||||
return 'reverse' if self.get_prop_value(
|
return DIRECTION_REVERSE if self.get_prop_value(
|
||||||
prop=self._prop_wind_reverse
|
prop=self._prop_wind_reverse
|
||||||
) == self._prop_wind_reverse_reverse else 'forward'
|
) == self._prop_wind_reverse_reverse else DIRECTION_FORWARD
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def percentage(self) -> Optional[int]:
|
def percentage(self) -> Optional[int]:
|
||||||
@ -303,7 +309,7 @@ class Fan(MIoTServiceEntity, FanEntity):
|
|||||||
fan_level = self.get_prop_value(prop=self._prop_fan_level)
|
fan_level = self.get_prop_value(prop=self._prop_fan_level)
|
||||||
if fan_level is None:
|
if fan_level is None:
|
||||||
return None
|
return None
|
||||||
if self._speed_names:
|
if self._speed_names and self._speed_name_map:
|
||||||
return ordered_list_item_to_percentage(
|
return ordered_list_item_to_percentage(
|
||||||
self._speed_names, self._speed_name_map[fan_level])
|
self._speed_names, self._speed_name_map[fan_level])
|
||||||
else:
|
else:
|
||||||
|
@ -96,7 +96,7 @@ class Light(MIoTServiceEntity, LightEntity):
|
|||||||
"""Light entities for Xiaomi Home."""
|
"""Light entities for Xiaomi Home."""
|
||||||
# pylint: disable=unused-argument
|
# pylint: disable=unused-argument
|
||||||
_VALUE_RANGE_MODE_COUNT_MAX = 30
|
_VALUE_RANGE_MODE_COUNT_MAX = 30
|
||||||
_prop_on: MIoTSpecProperty
|
_prop_on: Optional[MIoTSpecProperty]
|
||||||
_prop_brightness: Optional[MIoTSpecProperty]
|
_prop_brightness: Optional[MIoTSpecProperty]
|
||||||
_prop_color_temp: Optional[MIoTSpecProperty]
|
_prop_color_temp: Optional[MIoTSpecProperty]
|
||||||
_prop_color: Optional[MIoTSpecProperty]
|
_prop_color: Optional[MIoTSpecProperty]
|
||||||
@ -179,7 +179,7 @@ class Light(MIoTServiceEntity, LightEntity):
|
|||||||
) / prop.value_range.step)
|
) / prop.value_range.step)
|
||||||
> self._VALUE_RANGE_MODE_COUNT_MAX
|
> self._VALUE_RANGE_MODE_COUNT_MAX
|
||||||
):
|
):
|
||||||
_LOGGER.info(
|
_LOGGER.error(
|
||||||
'too many mode values, %s, %s, %s',
|
'too many mode values, %s, %s, %s',
|
||||||
self.entity_id, prop.name, prop.value_range)
|
self.entity_id, prop.name, prop.value_range)
|
||||||
else:
|
else:
|
||||||
@ -250,23 +250,25 @@ class Light(MIoTServiceEntity, LightEntity):
|
|||||||
|
|
||||||
Shall set attributes in kwargs if applicable.
|
Shall set attributes in kwargs if applicable.
|
||||||
"""
|
"""
|
||||||
result: bool = False
|
|
||||||
# on
|
# on
|
||||||
# Dirty logic for lumi.gateway.mgl03 indicator light
|
# Dirty logic for lumi.gateway.mgl03 indicator light
|
||||||
|
if self._prop_on:
|
||||||
value_on = True if self._prop_on.format_ == bool else 1
|
value_on = True if self._prop_on.format_ == bool else 1
|
||||||
result = await self.set_property_async(
|
await self.set_property_async(
|
||||||
prop=self._prop_on, value=value_on)
|
prop=self._prop_on, value=value_on)
|
||||||
# brightness
|
# brightness
|
||||||
if ATTR_BRIGHTNESS in kwargs:
|
if ATTR_BRIGHTNESS in kwargs:
|
||||||
brightness = brightness_to_value(
|
brightness = brightness_to_value(
|
||||||
self._brightness_scale, kwargs[ATTR_BRIGHTNESS])
|
self._brightness_scale, kwargs[ATTR_BRIGHTNESS])
|
||||||
result = await self.set_property_async(
|
await self.set_property_async(
|
||||||
prop=self._prop_brightness, value=brightness)
|
prop=self._prop_brightness, value=brightness,
|
||||||
|
write_ha_state=False)
|
||||||
# color-temperature
|
# color-temperature
|
||||||
if ATTR_COLOR_TEMP_KELVIN in kwargs:
|
if ATTR_COLOR_TEMP_KELVIN in kwargs:
|
||||||
result = await self.set_property_async(
|
await self.set_property_async(
|
||||||
prop=self._prop_color_temp,
|
prop=self._prop_color_temp,
|
||||||
value=kwargs[ATTR_COLOR_TEMP_KELVIN])
|
value=kwargs[ATTR_COLOR_TEMP_KELVIN],
|
||||||
|
write_ha_state=False)
|
||||||
self._attr_color_mode = ColorMode.COLOR_TEMP
|
self._attr_color_mode = ColorMode.COLOR_TEMP
|
||||||
# rgb color
|
# rgb color
|
||||||
if ATTR_RGB_COLOR in kwargs:
|
if ATTR_RGB_COLOR in kwargs:
|
||||||
@ -274,19 +276,23 @@ class Light(MIoTServiceEntity, LightEntity):
|
|||||||
g = kwargs[ATTR_RGB_COLOR][1]
|
g = kwargs[ATTR_RGB_COLOR][1]
|
||||||
b = kwargs[ATTR_RGB_COLOR][2]
|
b = kwargs[ATTR_RGB_COLOR][2]
|
||||||
rgb = (r << 16) | (g << 8) | b
|
rgb = (r << 16) | (g << 8) | b
|
||||||
result = await self.set_property_async(
|
await self.set_property_async(
|
||||||
prop=self._prop_color, value=rgb)
|
prop=self._prop_color, value=rgb,
|
||||||
|
write_ha_state=False)
|
||||||
self._attr_color_mode = ColorMode.RGB
|
self._attr_color_mode = ColorMode.RGB
|
||||||
# mode
|
# mode
|
||||||
if ATTR_EFFECT in kwargs:
|
if ATTR_EFFECT in kwargs:
|
||||||
result = await self.set_property_async(
|
await self.set_property_async(
|
||||||
prop=self._prop_mode,
|
prop=self._prop_mode,
|
||||||
value=self.get_map_key(
|
value=self.get_map_key(
|
||||||
map_=self._mode_map, value=kwargs[ATTR_EFFECT]))
|
map_=self._mode_map, value=kwargs[ATTR_EFFECT]),
|
||||||
return result
|
write_ha_state=False)
|
||||||
|
self.async_write_ha_state()
|
||||||
|
|
||||||
async def async_turn_off(self, **kwargs) -> None:
|
async def async_turn_off(self, **kwargs) -> None:
|
||||||
"""Turn the light off."""
|
"""Turn the light off."""
|
||||||
|
if not self._prop_on:
|
||||||
|
return
|
||||||
# Dirty logic for lumi.gateway.mgl03 indicator light
|
# Dirty logic for lumi.gateway.mgl03 indicator light
|
||||||
value_on = False if self._prop_on.format_ == bool else 0
|
value_on = False if self._prop_on.format_ == bool else 0
|
||||||
return await self.set_property_async(prop=self._prop_on, value=value_on)
|
await self.set_property_async(prop=self._prop_on, value=value_on)
|
||||||
|
@ -20,12 +20,12 @@
|
|||||||
],
|
],
|
||||||
"requirements": [
|
"requirements": [
|
||||||
"construct>=2.10.56",
|
"construct>=2.10.56",
|
||||||
"paho-mqtt<2.0.0",
|
"paho-mqtt",
|
||||||
"numpy",
|
"numpy",
|
||||||
"cryptography",
|
"cryptography",
|
||||||
"psutil"
|
"psutil"
|
||||||
],
|
],
|
||||||
"version": "v0.1.5b2",
|
"version": "v0.2.4",
|
||||||
"zeroconf": [
|
"zeroconf": [
|
||||||
"_miot-central._tcp.local."
|
"_miot-central._tcp.local."
|
||||||
]
|
]
|
||||||
|
@ -150,7 +150,7 @@ class MIoTClient:
|
|||||||
# Device list update timestamp
|
# Device list update timestamp
|
||||||
_device_list_update_ts: int
|
_device_list_update_ts: int
|
||||||
|
|
||||||
_sub_source_list: dict[str, str]
|
_sub_source_list: dict[str, Optional[str]]
|
||||||
_sub_tree: MIoTMatcher
|
_sub_tree: MIoTMatcher
|
||||||
_sub_device_state: dict[str, MipsDeviceState]
|
_sub_device_state: dict[str, MipsDeviceState]
|
||||||
|
|
||||||
@ -620,7 +620,7 @@ class MIoTClient:
|
|||||||
# Priority local control
|
# Priority local control
|
||||||
if self._ctrl_mode == CtrlMode.AUTO:
|
if self._ctrl_mode == CtrlMode.AUTO:
|
||||||
# Gateway control
|
# Gateway control
|
||||||
device_gw: dict = self._device_list_gateway.get(did, None)
|
device_gw = self._device_list_gateway.get(did, None)
|
||||||
if (
|
if (
|
||||||
device_gw and device_gw.get('online', False)
|
device_gw and device_gw.get('online', False)
|
||||||
and device_gw.get('specv2_access', False)
|
and device_gw.get('specv2_access', False)
|
||||||
@ -641,7 +641,7 @@ class MIoTClient:
|
|||||||
raise MIoTClientError(
|
raise MIoTClientError(
|
||||||
self.__get_exec_error_with_rc(rc=rc))
|
self.__get_exec_error_with_rc(rc=rc))
|
||||||
# Lan control
|
# Lan control
|
||||||
device_lan: dict = self._device_list_lan.get(did, None)
|
device_lan = self._device_list_lan.get(did, None)
|
||||||
if device_lan and device_lan.get('online', False):
|
if device_lan and device_lan.get('online', False):
|
||||||
result = await self._miot_lan.set_prop_async(
|
result = await self._miot_lan.set_prop_async(
|
||||||
did=did, siid=siid, piid=piid, value=value)
|
did=did, siid=siid, piid=piid, value=value)
|
||||||
@ -657,7 +657,7 @@ class MIoTClient:
|
|||||||
# Cloud control
|
# Cloud control
|
||||||
device_cloud = self._device_list_cloud.get(did, None)
|
device_cloud = self._device_list_cloud.get(did, None)
|
||||||
if device_cloud and device_cloud.get('online', False):
|
if device_cloud and device_cloud.get('online', False):
|
||||||
result: list = await self._http.set_prop_async(
|
result = await self._http.set_prop_async(
|
||||||
params=[
|
params=[
|
||||||
{'did': did, 'siid': siid, 'piid': piid, 'value': value}
|
{'did': did, 'siid': siid, 'piid': piid, 'value': value}
|
||||||
])
|
])
|
||||||
@ -746,7 +746,7 @@ class MIoTClient:
|
|||||||
if did not in self._device_list_cache:
|
if did not in self._device_list_cache:
|
||||||
raise MIoTClientError(f'did not exist, {did}')
|
raise MIoTClientError(f'did not exist, {did}')
|
||||||
|
|
||||||
device_gw: dict = self._device_list_gateway.get(did, None)
|
device_gw = self._device_list_gateway.get(did, None)
|
||||||
# Priority local control
|
# Priority local control
|
||||||
if self._ctrl_mode == CtrlMode.AUTO:
|
if self._ctrl_mode == CtrlMode.AUTO:
|
||||||
if (
|
if (
|
||||||
@ -782,7 +782,7 @@ class MIoTClient:
|
|||||||
self.__get_exec_error_with_rc(rc=rc))
|
self.__get_exec_error_with_rc(rc=rc))
|
||||||
# Cloud control
|
# Cloud control
|
||||||
device_cloud = self._device_list_cloud.get(did, None)
|
device_cloud = self._device_list_cloud.get(did, None)
|
||||||
if device_cloud.get('online', False):
|
if device_cloud and device_cloud.get('online', False):
|
||||||
result: dict = await self._http.action_async(
|
result: dict = await self._http.action_async(
|
||||||
did=did, siid=siid, aiid=aiid, in_list=in_list)
|
did=did, siid=siid, aiid=aiid, in_list=in_list)
|
||||||
if result:
|
if result:
|
||||||
@ -798,14 +798,15 @@ class MIoTClient:
|
|||||||
dids=[did]))
|
dids=[did]))
|
||||||
raise MIoTClientError(
|
raise MIoTClientError(
|
||||||
self.__get_exec_error_with_rc(rc=rc))
|
self.__get_exec_error_with_rc(rc=rc))
|
||||||
# Show error message
|
# TODO: Show error message
|
||||||
_LOGGER.error(
|
_LOGGER.error(
|
||||||
'client action failed, %s.%d.%d', did, siid, aiid)
|
'client action failed, %s.%d.%d', did, siid, aiid)
|
||||||
return None
|
return []
|
||||||
|
|
||||||
def sub_prop(
|
def sub_prop(
|
||||||
self, did: str, handler: Callable[[dict, Any], None],
|
self, did: str, handler: Callable[[dict, Any], None],
|
||||||
siid: int = None, piid: int = None, handler_ctx: Any = None
|
siid: Optional[int] = None, piid: Optional[int] = None,
|
||||||
|
handler_ctx: Any = None
|
||||||
) -> bool:
|
) -> bool:
|
||||||
if did not in self._device_list_cache:
|
if did not in self._device_list_cache:
|
||||||
raise MIoTClientError(f'did not exist, {did}')
|
raise MIoTClientError(f'did not exist, {did}')
|
||||||
@ -818,7 +819,9 @@ class MIoTClient:
|
|||||||
_LOGGER.debug('client sub prop, %s', topic)
|
_LOGGER.debug('client sub prop, %s', topic)
|
||||||
return True
|
return True
|
||||||
|
|
||||||
def unsub_prop(self, did: str, siid: int = None, piid: int = None) -> bool:
|
def unsub_prop(
|
||||||
|
self, did: str, siid: Optional[int] = None, piid: Optional[int] = None
|
||||||
|
) -> bool:
|
||||||
topic = (
|
topic = (
|
||||||
f'{did}/p/'
|
f'{did}/p/'
|
||||||
f'{"#" if siid is None or piid is None else f"{siid}/{piid}"}')
|
f'{"#" if siid is None or piid is None else f"{siid}/{piid}"}')
|
||||||
@ -829,7 +832,8 @@ class MIoTClient:
|
|||||||
|
|
||||||
def sub_event(
|
def sub_event(
|
||||||
self, did: str, handler: Callable[[dict, Any], None],
|
self, did: str, handler: Callable[[dict, Any], None],
|
||||||
siid: int = None, eiid: int = None, handler_ctx: Any = None
|
siid: Optional[int] = None, eiid: Optional[int] = None,
|
||||||
|
handler_ctx: Any = None
|
||||||
) -> bool:
|
) -> bool:
|
||||||
if did not in self._device_list_cache:
|
if did not in self._device_list_cache:
|
||||||
raise MIoTClientError(f'did not exist, {did}')
|
raise MIoTClientError(f'did not exist, {did}')
|
||||||
@ -841,7 +845,9 @@ class MIoTClient:
|
|||||||
_LOGGER.debug('client sub event, %s', topic)
|
_LOGGER.debug('client sub event, %s', topic)
|
||||||
return True
|
return True
|
||||||
|
|
||||||
def unsub_event(self, did: str, siid: int = None, eiid: int = None) -> bool:
|
def unsub_event(
|
||||||
|
self, did: str, siid: Optional[int] = None, eiid: Optional[int] = None
|
||||||
|
) -> bool:
|
||||||
topic = (
|
topic = (
|
||||||
f'{did}/e/'
|
f'{did}/e/'
|
||||||
f'{"#" if siid is None or eiid is None else f"{siid}/{eiid}"}')
|
f'{"#" if siid is None or eiid is None else f"{siid}/{eiid}"}')
|
||||||
@ -1081,7 +1087,7 @@ class MIoTClient:
|
|||||||
if state_old == state_new:
|
if state_old == state_new:
|
||||||
continue
|
continue
|
||||||
self._device_list_cache[did]['online'] = state_new
|
self._device_list_cache[did]['online'] = state_new
|
||||||
sub: MipsDeviceState = self._sub_device_state.get(did, None)
|
sub = self._sub_device_state.get(did, None)
|
||||||
if sub and sub.handler:
|
if sub and sub.handler:
|
||||||
sub.handler(did, MIoTDeviceState.OFFLINE, sub.handler_ctx)
|
sub.handler(did, MIoTDeviceState.OFFLINE, sub.handler_ctx)
|
||||||
self.__request_show_devices_changed_notify()
|
self.__request_show_devices_changed_notify()
|
||||||
@ -1091,8 +1097,8 @@ class MIoTClient:
|
|||||||
self, group_id: str, state: bool
|
self, group_id: str, state: bool
|
||||||
) -> None:
|
) -> None:
|
||||||
_LOGGER.info('local mips state changed, %s, %s', group_id, state)
|
_LOGGER.info('local mips state changed, %s, %s', group_id, state)
|
||||||
mips: MipsLocalClient = self._mips_local.get(group_id, None)
|
mips = self._mips_local.get(group_id, None)
|
||||||
if mips is None:
|
if not mips:
|
||||||
_LOGGER.error(
|
_LOGGER.error(
|
||||||
'local mips state changed, mips not exist, %s', group_id)
|
'local mips state changed, mips not exist, %s', group_id)
|
||||||
return
|
return
|
||||||
@ -1124,7 +1130,7 @@ class MIoTClient:
|
|||||||
if state_old == state_new:
|
if state_old == state_new:
|
||||||
continue
|
continue
|
||||||
self._device_list_cache[did]['online'] = state_new
|
self._device_list_cache[did]['online'] = state_new
|
||||||
sub: MipsDeviceState = self._sub_device_state.get(did, None)
|
sub = self._sub_device_state.get(did, None)
|
||||||
if sub and sub.handler:
|
if sub and sub.handler:
|
||||||
sub.handler(did, MIoTDeviceState.OFFLINE, sub.handler_ctx)
|
sub.handler(did, MIoTDeviceState.OFFLINE, sub.handler_ctx)
|
||||||
self.__request_show_devices_changed_notify()
|
self.__request_show_devices_changed_notify()
|
||||||
@ -1171,7 +1177,7 @@ class MIoTClient:
|
|||||||
if state_old == state_new:
|
if state_old == state_new:
|
||||||
continue
|
continue
|
||||||
self._device_list_cache[did]['online'] = state_new
|
self._device_list_cache[did]['online'] = state_new
|
||||||
sub: MipsDeviceState = self._sub_device_state.get(did, None)
|
sub = self._sub_device_state.get(did, None)
|
||||||
if sub and sub.handler:
|
if sub and sub.handler:
|
||||||
sub.handler(did, MIoTDeviceState.OFFLINE, sub.handler_ctx)
|
sub.handler(did, MIoTDeviceState.OFFLINE, sub.handler_ctx)
|
||||||
self._device_list_lan = {}
|
self._device_list_lan = {}
|
||||||
@ -1201,7 +1207,7 @@ class MIoTClient:
|
|||||||
if state_old == state_new:
|
if state_old == state_new:
|
||||||
return
|
return
|
||||||
self._device_list_cache[did]['online'] = state_new
|
self._device_list_cache[did]['online'] = state_new
|
||||||
sub: MipsDeviceState = self._sub_device_state.get(did, None)
|
sub = self._sub_device_state.get(did, None)
|
||||||
if sub and sub.handler:
|
if sub and sub.handler:
|
||||||
sub.handler(
|
sub.handler(
|
||||||
did, MIoTDeviceState.ONLINE if state_new
|
did, MIoTDeviceState.ONLINE if state_new
|
||||||
@ -1257,7 +1263,7 @@ class MIoTClient:
|
|||||||
if state_old == state_new:
|
if state_old == state_new:
|
||||||
return
|
return
|
||||||
self._device_list_cache[did]['online'] = state_new
|
self._device_list_cache[did]['online'] = state_new
|
||||||
sub: MipsDeviceState = self._sub_device_state.get(did, None)
|
sub = self._sub_device_state.get(did, None)
|
||||||
if sub and sub.handler:
|
if sub and sub.handler:
|
||||||
sub.handler(
|
sub.handler(
|
||||||
did, MIoTDeviceState.ONLINE if state_new
|
did, MIoTDeviceState.ONLINE if state_new
|
||||||
@ -1301,9 +1307,8 @@ class MIoTClient:
|
|||||||
async def __load_cache_device_async(self) -> None:
|
async def __load_cache_device_async(self) -> None:
|
||||||
"""Load device list from cache."""
|
"""Load device list from cache."""
|
||||||
cache_list: Optional[dict[str, dict]] = await self._storage.load_async(
|
cache_list: Optional[dict[str, dict]] = await self._storage.load_async(
|
||||||
domain='miot_devices',
|
domain='miot_devices', name=f'{self._uid}_{self._cloud_server}',
|
||||||
name=f'{self._uid}_{self._cloud_server}',
|
type_=dict) # type: ignore
|
||||||
type_=dict)
|
|
||||||
if not cache_list:
|
if not cache_list:
|
||||||
self.__show_client_error_notify(
|
self.__show_client_error_notify(
|
||||||
message=self._i18n.translate(
|
message=self._i18n.translate(
|
||||||
@ -1346,7 +1351,7 @@ class MIoTClient:
|
|||||||
cloud_state_old: Optional[bool] = self._device_list_cloud.get(
|
cloud_state_old: Optional[bool] = self._device_list_cloud.get(
|
||||||
did, {}).get('online', None)
|
did, {}).get('online', None)
|
||||||
cloud_state_new: Optional[bool] = None
|
cloud_state_new: Optional[bool] = None
|
||||||
device_new: dict = cloud_list.pop(did, None)
|
device_new = cloud_list.pop(did, None)
|
||||||
if device_new:
|
if device_new:
|
||||||
cloud_state_new = device_new.get('online', None)
|
cloud_state_new = device_new.get('online', None)
|
||||||
# Update cache device info
|
# Update cache device info
|
||||||
@ -1371,7 +1376,7 @@ class MIoTClient:
|
|||||||
continue
|
continue
|
||||||
info['online'] = state_new
|
info['online'] = state_new
|
||||||
# Call device state changed callback
|
# Call device state changed callback
|
||||||
sub: MipsDeviceState = self._sub_device_state.get(did, None)
|
sub = self._sub_device_state.get(did, None)
|
||||||
if sub and sub.handler:
|
if sub and sub.handler:
|
||||||
sub.handler(
|
sub.handler(
|
||||||
did, MIoTDeviceState.ONLINE if state_new
|
did, MIoTDeviceState.ONLINE if state_new
|
||||||
@ -1426,8 +1431,7 @@ class MIoTClient:
|
|||||||
self, dids: list[str]
|
self, dids: list[str]
|
||||||
) -> None:
|
) -> None:
|
||||||
_LOGGER.debug('refresh cloud device with dids, %s', dids)
|
_LOGGER.debug('refresh cloud device with dids, %s', dids)
|
||||||
cloud_list: dict[str, dict] = (
|
cloud_list = await self._http.get_devices_with_dids_async(dids=dids)
|
||||||
await self._http.get_devices_with_dids_async(dids=dids))
|
|
||||||
if cloud_list is None:
|
if cloud_list is None:
|
||||||
_LOGGER.error('cloud http get_dev_list_async failed, %s', dids)
|
_LOGGER.error('cloud http get_dev_list_async failed, %s', dids)
|
||||||
return
|
return
|
||||||
@ -1466,11 +1470,11 @@ class MIoTClient:
|
|||||||
for did, info in self._device_list_cache.items():
|
for did, info in self._device_list_cache.items():
|
||||||
if did not in filter_dids:
|
if did not in filter_dids:
|
||||||
continue
|
continue
|
||||||
device_old: dict = self._device_list_gateway.get(did, None)
|
device_old = self._device_list_gateway.get(did, None)
|
||||||
gw_state_old = device_old.get(
|
gw_state_old = device_old.get(
|
||||||
'online', False) if device_old else False
|
'online', False) if device_old else False
|
||||||
gw_state_new: bool = False
|
gw_state_new: bool = False
|
||||||
device_new: dict = gw_list.pop(did, None)
|
device_new = gw_list.pop(did, None)
|
||||||
if device_new:
|
if device_new:
|
||||||
# Update gateway device info
|
# Update gateway device info
|
||||||
self._device_list_gateway[did] = {
|
self._device_list_gateway[did] = {
|
||||||
@ -1493,7 +1497,7 @@ class MIoTClient:
|
|||||||
if state_old == state_new:
|
if state_old == state_new:
|
||||||
continue
|
continue
|
||||||
info['online'] = state_new
|
info['online'] = state_new
|
||||||
sub: MipsDeviceState = self._sub_device_state.get(did, None)
|
sub = self._sub_device_state.get(did, None)
|
||||||
if sub and sub.handler:
|
if sub and sub.handler:
|
||||||
sub.handler(
|
sub.handler(
|
||||||
did, MIoTDeviceState.ONLINE if state_new
|
did, MIoTDeviceState.ONLINE if state_new
|
||||||
@ -1518,7 +1522,7 @@ class MIoTClient:
|
|||||||
if state_old == state_new:
|
if state_old == state_new:
|
||||||
continue
|
continue
|
||||||
self._device_list_cache[did]['online'] = state_new
|
self._device_list_cache[did]['online'] = state_new
|
||||||
sub: MipsDeviceState = self._sub_device_state.get(did, None)
|
sub = self._sub_device_state.get(did, None)
|
||||||
if sub and sub.handler:
|
if sub and sub.handler:
|
||||||
sub.handler(
|
sub.handler(
|
||||||
did, MIoTDeviceState.ONLINE if state_new
|
did, MIoTDeviceState.ONLINE if state_new
|
||||||
@ -1533,7 +1537,7 @@ class MIoTClient:
|
|||||||
'refresh gw devices with group_id, %s', group_id)
|
'refresh gw devices with group_id, %s', group_id)
|
||||||
# Remove timer
|
# Remove timer
|
||||||
self._mips_local_state_changed_timers.pop(group_id, None)
|
self._mips_local_state_changed_timers.pop(group_id, None)
|
||||||
mips: MipsLocalClient = self._mips_local.get(group_id, None)
|
mips = self._mips_local.get(group_id, None)
|
||||||
if not mips:
|
if not mips:
|
||||||
_LOGGER.error('mips not exist, %s', group_id)
|
_LOGGER.error('mips not exist, %s', group_id)
|
||||||
return
|
return
|
||||||
@ -1900,16 +1904,16 @@ async def get_miot_instance_async(
|
|||||||
) -> MIoTClient:
|
) -> MIoTClient:
|
||||||
if entry_id is None:
|
if entry_id is None:
|
||||||
raise MIoTClientError('invalid entry_id')
|
raise MIoTClientError('invalid entry_id')
|
||||||
miot_client: MIoTClient = None
|
miot_client = hass.data[DOMAIN].get('miot_clients', {}).get(entry_id, None)
|
||||||
if a := hass.data[DOMAIN].get('miot_clients', {}).get(entry_id, None):
|
if miot_client:
|
||||||
_LOGGER.info('instance exist, %s', entry_id)
|
_LOGGER.info('instance exist, %s', entry_id)
|
||||||
miot_client = a
|
return miot_client
|
||||||
else:
|
# Create new instance
|
||||||
if entry_data is None:
|
if not entry_data:
|
||||||
raise MIoTClientError('entry data is None')
|
raise MIoTClientError('entry data is None')
|
||||||
# Get running loop
|
# Get running loop
|
||||||
loop: asyncio.AbstractEventLoop = asyncio.get_running_loop()
|
loop: asyncio.AbstractEventLoop = asyncio.get_running_loop()
|
||||||
if loop is None:
|
if not loop:
|
||||||
raise MIoTClientError('loop is None')
|
raise MIoTClientError('loop is None')
|
||||||
# MIoT storage
|
# MIoT storage
|
||||||
storage: Optional[MIoTStorage] = hass.data[DOMAIN].get(
|
storage: Optional[MIoTStorage] = hass.data[DOMAIN].get(
|
||||||
@ -1923,8 +1927,7 @@ async def get_miot_instance_async(
|
|||||||
uid='global_config', cloud_server='all',
|
uid='global_config', cloud_server='all',
|
||||||
keys=['network_detect_addr', 'net_interfaces', 'enable_subscribe'])
|
keys=['network_detect_addr', 'net_interfaces', 'enable_subscribe'])
|
||||||
# MIoT network
|
# MIoT network
|
||||||
network_detect_addr: dict = global_config.get(
|
network_detect_addr: dict = global_config.get('network_detect_addr', {})
|
||||||
'network_detect_addr', {})
|
|
||||||
network: Optional[MIoTNetwork] = hass.data[DOMAIN].get(
|
network: Optional[MIoTNetwork] = hass.data[DOMAIN].get(
|
||||||
'miot_network', None)
|
'miot_network', None)
|
||||||
if not network:
|
if not network:
|
||||||
@ -1946,8 +1949,7 @@ async def get_miot_instance_async(
|
|||||||
await mips_service.init_async()
|
await mips_service.init_async()
|
||||||
_LOGGER.info('create mips_service instance')
|
_LOGGER.info('create mips_service instance')
|
||||||
# MIoT lan
|
# MIoT lan
|
||||||
miot_lan: Optional[MIoTLan] = hass.data[DOMAIN].get(
|
miot_lan: Optional[MIoTLan] = hass.data[DOMAIN].get('miot_lan', None)
|
||||||
'miot_lan', None)
|
|
||||||
if not miot_lan:
|
if not miot_lan:
|
||||||
miot_lan = MIoTLan(
|
miot_lan = MIoTLan(
|
||||||
net_ifs=global_config.get('net_interfaces', []),
|
net_ifs=global_config.get('net_interfaces', []),
|
||||||
@ -1969,8 +1971,6 @@ async def get_miot_instance_async(
|
|||||||
)
|
)
|
||||||
miot_client.persistent_notify = persistent_notify
|
miot_client.persistent_notify = persistent_notify
|
||||||
hass.data[DOMAIN]['miot_clients'].setdefault(entry_id, miot_client)
|
hass.data[DOMAIN]['miot_clients'].setdefault(entry_id, miot_client)
|
||||||
_LOGGER.info(
|
_LOGGER.info('new miot_client instance, %s, %s', entry_id, entry_data)
|
||||||
'new miot_client instance, %s, %s', entry_id, entry_data)
|
|
||||||
await miot_client.init_async()
|
await miot_client.init_async()
|
||||||
|
|
||||||
return miot_client
|
return miot_client
|
||||||
|
@ -382,7 +382,7 @@ class MIoTHttpClient:
|
|||||||
|
|
||||||
return res_obj['data']
|
return res_obj['data']
|
||||||
|
|
||||||
async def get_central_cert_async(self, csr: str) -> Optional[str]:
|
async def get_central_cert_async(self, csr: str) -> str:
|
||||||
if not isinstance(csr, str):
|
if not isinstance(csr, str):
|
||||||
raise MIoTHttpError('invalid params')
|
raise MIoTHttpError('invalid params')
|
||||||
|
|
||||||
|
@ -56,6 +56,7 @@ from homeassistant.const import (
|
|||||||
CONCENTRATION_MILLIGRAMS_PER_CUBIC_METER,
|
CONCENTRATION_MILLIGRAMS_PER_CUBIC_METER,
|
||||||
CONCENTRATION_PARTS_PER_BILLION,
|
CONCENTRATION_PARTS_PER_BILLION,
|
||||||
CONCENTRATION_PARTS_PER_MILLION,
|
CONCENTRATION_PARTS_PER_MILLION,
|
||||||
|
DEGREE,
|
||||||
LIGHT_LUX,
|
LIGHT_LUX,
|
||||||
PERCENTAGE,
|
PERCENTAGE,
|
||||||
SIGNAL_STRENGTH_DECIBELS,
|
SIGNAL_STRENGTH_DECIBELS,
|
||||||
@ -72,6 +73,7 @@ from homeassistant.const import (
|
|||||||
UnitOfPower,
|
UnitOfPower,
|
||||||
UnitOfVolume,
|
UnitOfVolume,
|
||||||
UnitOfVolumeFlowRate,
|
UnitOfVolumeFlowRate,
|
||||||
|
UnitOfDataRate
|
||||||
)
|
)
|
||||||
from homeassistant.helpers.entity import DeviceInfo
|
from homeassistant.helpers.entity import DeviceInfo
|
||||||
from homeassistant.components.switch import SwitchDeviceClass
|
from homeassistant.components.switch import SwitchDeviceClass
|
||||||
@ -243,12 +245,12 @@ class MIoTDevice:
|
|||||||
def sub_device_state(
|
def sub_device_state(
|
||||||
self, key: str, handler: Callable[[str, MIoTDeviceState], None]
|
self, key: str, handler: Callable[[str, MIoTDeviceState], None]
|
||||||
) -> int:
|
) -> int:
|
||||||
self._sub_id += 1
|
sub_id = self.__gen_sub_id()
|
||||||
if key in self._device_state_sub_list:
|
if key in self._device_state_sub_list:
|
||||||
self._device_state_sub_list[key][str(self._sub_id)] = handler
|
self._device_state_sub_list[key][str(sub_id)] = handler
|
||||||
else:
|
else:
|
||||||
self._device_state_sub_list[key] = {str(self._sub_id): handler}
|
self._device_state_sub_list[key] = {str(sub_id): handler}
|
||||||
return self._sub_id
|
return sub_id
|
||||||
|
|
||||||
def unsub_device_state(self, key: str, sub_id: int) -> None:
|
def unsub_device_state(self, key: str, sub_id: int) -> None:
|
||||||
sub_list = self._device_state_sub_list.get(key, None)
|
sub_list = self._device_state_sub_list.get(key, None)
|
||||||
@ -266,14 +268,14 @@ class MIoTDevice:
|
|||||||
for handler in self._value_sub_list[key].values():
|
for handler in self._value_sub_list[key].values():
|
||||||
handler(params, ctx)
|
handler(params, ctx)
|
||||||
|
|
||||||
self._sub_id += 1
|
sub_id = self.__gen_sub_id()
|
||||||
if key in self._value_sub_list:
|
if key in self._value_sub_list:
|
||||||
self._value_sub_list[key][str(self._sub_id)] = handler
|
self._value_sub_list[key][str(sub_id)] = handler
|
||||||
else:
|
else:
|
||||||
self._value_sub_list[key] = {str(self._sub_id): handler}
|
self._value_sub_list[key] = {str(sub_id): handler}
|
||||||
self.miot_client.sub_prop(
|
self.miot_client.sub_prop(
|
||||||
did=self._did, handler=_on_prop_changed, siid=siid, piid=piid)
|
did=self._did, handler=_on_prop_changed, siid=siid, piid=piid)
|
||||||
return self._sub_id
|
return sub_id
|
||||||
|
|
||||||
def unsub_property(self, siid: int, piid: int, sub_id: int) -> None:
|
def unsub_property(self, siid: int, piid: int, sub_id: int) -> None:
|
||||||
key: str = f'p.{siid}.{piid}'
|
key: str = f'p.{siid}.{piid}'
|
||||||
@ -294,14 +296,14 @@ class MIoTDevice:
|
|||||||
for handler in self._value_sub_list[key].values():
|
for handler in self._value_sub_list[key].values():
|
||||||
handler(params, ctx)
|
handler(params, ctx)
|
||||||
|
|
||||||
self._sub_id += 1
|
sub_id = self.__gen_sub_id()
|
||||||
if key in self._value_sub_list:
|
if key in self._value_sub_list:
|
||||||
self._value_sub_list[key][str(self._sub_id)] = handler
|
self._value_sub_list[key][str(sub_id)] = handler
|
||||||
else:
|
else:
|
||||||
self._value_sub_list[key] = {str(self._sub_id): handler}
|
self._value_sub_list[key] = {str(sub_id): handler}
|
||||||
self.miot_client.sub_event(
|
self.miot_client.sub_event(
|
||||||
did=self._did, handler=_on_event_occurred, siid=siid, eiid=eiid)
|
did=self._did, handler=_on_event_occurred, siid=siid, eiid=eiid)
|
||||||
return self._sub_id
|
return sub_id
|
||||||
|
|
||||||
def unsub_event(self, siid: int, eiid: int, sub_id: int) -> None:
|
def unsub_event(self, siid: int, eiid: int, sub_id: int) -> None:
|
||||||
key: str = f'e.{siid}.{eiid}'
|
key: str = f'e.{siid}.{eiid}'
|
||||||
@ -414,6 +416,8 @@ class MIoTDevice:
|
|||||||
spec_name: str = spec_instance.name
|
spec_name: str = spec_instance.name
|
||||||
if isinstance(SPEC_DEVICE_TRANS_MAP[spec_name], str):
|
if isinstance(SPEC_DEVICE_TRANS_MAP[spec_name], str):
|
||||||
spec_name = SPEC_DEVICE_TRANS_MAP[spec_name]
|
spec_name = SPEC_DEVICE_TRANS_MAP[spec_name]
|
||||||
|
if 'required' not in SPEC_DEVICE_TRANS_MAP[spec_name]:
|
||||||
|
return None
|
||||||
# 1. The device shall have all required services.
|
# 1. The device shall have all required services.
|
||||||
required_services = SPEC_DEVICE_TRANS_MAP[spec_name]['required'].keys()
|
required_services = SPEC_DEVICE_TRANS_MAP[spec_name]['required'].keys()
|
||||||
if not {
|
if not {
|
||||||
@ -427,9 +431,13 @@ class MIoTDevice:
|
|||||||
for service in spec_instance.services:
|
for service in spec_instance.services:
|
||||||
if service.platform:
|
if service.platform:
|
||||||
continue
|
continue
|
||||||
|
required_properties: dict
|
||||||
|
optional_properties: dict
|
||||||
|
required_actions: set
|
||||||
|
optional_actions: set
|
||||||
# 2. The service shall have all required properties, actions.
|
# 2. The service shall have all required properties, actions.
|
||||||
if service.name in required_services:
|
if service.name in required_services:
|
||||||
required_properties: dict = SPEC_DEVICE_TRANS_MAP[spec_name][
|
required_properties = SPEC_DEVICE_TRANS_MAP[spec_name][
|
||||||
'required'].get(
|
'required'].get(
|
||||||
service.name, {}
|
service.name, {}
|
||||||
).get('required', {}).get('properties', {})
|
).get('required', {}).get('properties', {})
|
||||||
@ -446,7 +454,7 @@ class MIoTDevice:
|
|||||||
service.name, {}
|
service.name, {}
|
||||||
).get('optional', {}).get('actions', set({}))
|
).get('optional', {}).get('actions', set({}))
|
||||||
elif service.name in optional_services:
|
elif service.name in optional_services:
|
||||||
required_properties: dict = SPEC_DEVICE_TRANS_MAP[spec_name][
|
required_properties = SPEC_DEVICE_TRANS_MAP[spec_name][
|
||||||
'optional'].get(
|
'optional'].get(
|
||||||
service.name, {}
|
service.name, {}
|
||||||
).get('required', {}).get('properties', {})
|
).get('required', {}).get('properties', {})
|
||||||
@ -484,7 +492,7 @@ class MIoTDevice:
|
|||||||
set(required_properties.keys()), optional_properties):
|
set(required_properties.keys()), optional_properties):
|
||||||
if prop.unit:
|
if prop.unit:
|
||||||
prop.external_unit = self.unit_convert(prop.unit)
|
prop.external_unit = self.unit_convert(prop.unit)
|
||||||
prop.icon = self.icon_convert(prop.unit)
|
# prop.icon = self.icon_convert(prop.unit)
|
||||||
prop.platform = platform
|
prop.platform = platform
|
||||||
entity_data.props.add(prop)
|
entity_data.props.add(prop)
|
||||||
# action
|
# action
|
||||||
@ -499,85 +507,94 @@ class MIoTDevice:
|
|||||||
return entity_data
|
return entity_data
|
||||||
|
|
||||||
def parse_miot_service_entity(
|
def parse_miot_service_entity(
|
||||||
self, service_instance: MIoTSpecService
|
self, miot_service: MIoTSpecService
|
||||||
) -> Optional[MIoTEntityData]:
|
) -> Optional[MIoTEntityData]:
|
||||||
service = service_instance
|
if (
|
||||||
if service.platform or (service.name not in SPEC_SERVICE_TRANS_MAP):
|
miot_service.platform
|
||||||
|
or miot_service.name not in SPEC_SERVICE_TRANS_MAP
|
||||||
|
):
|
||||||
return None
|
return None
|
||||||
|
service_name = miot_service.name
|
||||||
service_name = service.name
|
|
||||||
if isinstance(SPEC_SERVICE_TRANS_MAP[service_name], str):
|
if isinstance(SPEC_SERVICE_TRANS_MAP[service_name], str):
|
||||||
service_name = SPEC_SERVICE_TRANS_MAP[service_name]
|
service_name = SPEC_SERVICE_TRANS_MAP[service_name]
|
||||||
# 1. The service shall have all required properties.
|
if 'required' not in SPEC_SERVICE_TRANS_MAP[service_name]:
|
||||||
|
return None
|
||||||
|
# Required properties, required access mode
|
||||||
required_properties: dict = SPEC_SERVICE_TRANS_MAP[service_name][
|
required_properties: dict = SPEC_SERVICE_TRANS_MAP[service_name][
|
||||||
'required'].get('properties', {})
|
'required'].get('properties', {})
|
||||||
if not {
|
if not {
|
||||||
prop.name for prop in service.properties if prop.access
|
prop.name for prop in miot_service.properties if prop.access
|
||||||
}.issuperset(set(required_properties.keys())):
|
}.issuperset(set(required_properties.keys())):
|
||||||
return None
|
return None
|
||||||
# 2. The required property shall have all required access mode.
|
for prop in miot_service.properties:
|
||||||
for prop in service.properties:
|
|
||||||
if prop.name in required_properties:
|
if prop.name in required_properties:
|
||||||
if not set(prop.access).issuperset(
|
if not set(prop.access).issuperset(
|
||||||
required_properties[prop.name]):
|
required_properties[prop.name]):
|
||||||
return None
|
return None
|
||||||
|
# Required actions
|
||||||
|
# Required events
|
||||||
platform = SPEC_SERVICE_TRANS_MAP[service_name]['entity']
|
platform = SPEC_SERVICE_TRANS_MAP[service_name]['entity']
|
||||||
entity_data = MIoTEntityData(platform=platform, spec=service_instance)
|
entity_data = MIoTEntityData(platform=platform, spec=miot_service)
|
||||||
|
# Optional properties
|
||||||
optional_properties = SPEC_SERVICE_TRANS_MAP[service_name][
|
optional_properties = SPEC_SERVICE_TRANS_MAP[service_name][
|
||||||
'optional'].get('properties', set({}))
|
'optional'].get('properties', set({}))
|
||||||
for prop in service.properties:
|
for prop in miot_service.properties:
|
||||||
if prop.name in set.union(
|
if prop.name in set.union(
|
||||||
set(required_properties.keys()), optional_properties):
|
set(required_properties.keys()), optional_properties):
|
||||||
if prop.unit:
|
if prop.unit:
|
||||||
prop.external_unit = self.unit_convert(prop.unit)
|
prop.external_unit = self.unit_convert(prop.unit)
|
||||||
prop.icon = self.icon_convert(prop.unit)
|
# prop.icon = self.icon_convert(prop.unit)
|
||||||
prop.platform = platform
|
prop.platform = platform
|
||||||
entity_data.props.add(prop)
|
entity_data.props.add(prop)
|
||||||
# action
|
# Optional actions
|
||||||
# event
|
# Optional events
|
||||||
# No actions or events is in SPEC_SERVICE_TRANS_MAP now.
|
miot_service.platform = platform
|
||||||
service.platform = platform
|
# entity_category
|
||||||
|
if entity_category := SPEC_SERVICE_TRANS_MAP[service_name].get(
|
||||||
|
'entity_category', None):
|
||||||
|
miot_service.entity_category = entity_category
|
||||||
return entity_data
|
return entity_data
|
||||||
|
|
||||||
def parse_miot_property_entity(
|
def parse_miot_property_entity(self, miot_prop: MIoTSpecProperty) -> bool:
|
||||||
self, property_instance: MIoTSpecProperty
|
|
||||||
) -> Optional[dict[str, str]]:
|
|
||||||
prop = property_instance
|
|
||||||
if (
|
if (
|
||||||
prop.platform
|
miot_prop.platform
|
||||||
or (prop.name not in SPEC_PROP_TRANS_MAP['properties'])
|
or miot_prop.name not in SPEC_PROP_TRANS_MAP['properties']
|
||||||
):
|
):
|
||||||
return None
|
return False
|
||||||
|
prop_name = miot_prop.name
|
||||||
prop_name = prop.name
|
|
||||||
if isinstance(SPEC_PROP_TRANS_MAP['properties'][prop_name], str):
|
if isinstance(SPEC_PROP_TRANS_MAP['properties'][prop_name], str):
|
||||||
prop_name = SPEC_PROP_TRANS_MAP['properties'][prop_name]
|
prop_name = SPEC_PROP_TRANS_MAP['properties'][prop_name]
|
||||||
platform = SPEC_PROP_TRANS_MAP['properties'][prop_name]['entity']
|
platform = SPEC_PROP_TRANS_MAP['properties'][prop_name]['entity']
|
||||||
|
# Check
|
||||||
prop_access: set = set({})
|
prop_access: set = set({})
|
||||||
if prop.readable:
|
if miot_prop.readable:
|
||||||
prop_access.add('read')
|
prop_access.add('read')
|
||||||
if prop.writable:
|
if miot_prop.writable:
|
||||||
prop_access.add('write')
|
prop_access.add('write')
|
||||||
if prop_access != (SPEC_PROP_TRANS_MAP[
|
if prop_access != (SPEC_PROP_TRANS_MAP[
|
||||||
'entities'][platform]['access']):
|
'entities'][platform]['access']):
|
||||||
return None
|
return False
|
||||||
if prop.format_.__name__ not in SPEC_PROP_TRANS_MAP[
|
if miot_prop.format_.__name__ not in SPEC_PROP_TRANS_MAP[
|
||||||
'entities'][platform]['format']:
|
'entities'][platform]['format']:
|
||||||
return None
|
return False
|
||||||
if prop.unit:
|
miot_prop.device_class = SPEC_PROP_TRANS_MAP['properties'][prop_name][
|
||||||
prop.external_unit = self.unit_convert(prop.unit)
|
|
||||||
prop.icon = self.icon_convert(prop.unit)
|
|
||||||
device_class = SPEC_PROP_TRANS_MAP['properties'][prop_name][
|
|
||||||
'device_class']
|
'device_class']
|
||||||
result = {'platform': platform, 'device_class': device_class}
|
# Optional params
|
||||||
# optional:
|
if 'state_class' in SPEC_PROP_TRANS_MAP['properties'][prop_name]:
|
||||||
if 'optional' in SPEC_PROP_TRANS_MAP['properties'][prop_name]:
|
miot_prop.state_class = SPEC_PROP_TRANS_MAP['properties'][
|
||||||
optional = SPEC_PROP_TRANS_MAP['properties'][prop_name]['optional']
|
prop_name]['state_class']
|
||||||
if 'state_class' in optional:
|
if (
|
||||||
result['state_class'] = optional['state_class']
|
not miot_prop.external_unit
|
||||||
if not prop.unit and 'unit_of_measurement' in optional:
|
and 'unit_of_measurement' in SPEC_PROP_TRANS_MAP['properties'][
|
||||||
result['unit_of_measurement'] = optional['unit_of_measurement']
|
prop_name]
|
||||||
return result
|
):
|
||||||
|
# Priority: spec_modify.unit > unit_convert > specv2entity.unit
|
||||||
|
miot_prop.external_unit = SPEC_PROP_TRANS_MAP['properties'][
|
||||||
|
prop_name]['unit_of_measurement']
|
||||||
|
# Priority: default.icon when device_class is set > spec_modify.icon
|
||||||
|
# > icon_convert
|
||||||
|
miot_prop.platform = platform
|
||||||
|
return True
|
||||||
|
|
||||||
def spec_transform(self) -> None:
|
def spec_transform(self) -> None:
|
||||||
"""Parse service, property, event, action from device spec."""
|
"""Parse service, property, event, action from device spec."""
|
||||||
@ -589,7 +606,7 @@ class MIoTDevice:
|
|||||||
# STEP 2: service conversion
|
# STEP 2: service conversion
|
||||||
for service in self.spec_instance.services:
|
for service in self.spec_instance.services:
|
||||||
service_entity = self.parse_miot_service_entity(
|
service_entity = self.parse_miot_service_entity(
|
||||||
service_instance=service)
|
miot_service=service)
|
||||||
if service_entity:
|
if service_entity:
|
||||||
self.append_entity(entity_data=service_entity)
|
self.append_entity(entity_data=service_entity)
|
||||||
# STEP 3.1: property conversion
|
# STEP 3.1: property conversion
|
||||||
@ -598,20 +615,11 @@ class MIoTDevice:
|
|||||||
continue
|
continue
|
||||||
if prop.unit:
|
if prop.unit:
|
||||||
prop.external_unit = self.unit_convert(prop.unit)
|
prop.external_unit = self.unit_convert(prop.unit)
|
||||||
|
if not prop.icon:
|
||||||
prop.icon = self.icon_convert(prop.unit)
|
prop.icon = self.icon_convert(prop.unit)
|
||||||
prop_entity = self.parse_miot_property_entity(
|
# Special conversion
|
||||||
property_instance=prop)
|
self.parse_miot_property_entity(miot_prop=prop)
|
||||||
if prop_entity:
|
# General conversion
|
||||||
prop.platform = prop_entity['platform']
|
|
||||||
prop.device_class = prop_entity['device_class']
|
|
||||||
if 'state_class' in prop_entity:
|
|
||||||
prop.state_class = prop_entity['state_class']
|
|
||||||
if 'unit_of_measurement' in prop_entity:
|
|
||||||
prop.external_unit = self.unit_convert(
|
|
||||||
prop_entity['unit_of_measurement'])
|
|
||||||
prop.icon = self.icon_convert(
|
|
||||||
prop_entity['unit_of_measurement'])
|
|
||||||
# general conversion
|
|
||||||
if not prop.platform:
|
if not prop.platform:
|
||||||
if prop.writable:
|
if prop.writable:
|
||||||
if prop.format_ == str:
|
if prop.format_ == str:
|
||||||
@ -625,7 +633,7 @@ class MIoTDevice:
|
|||||||
prop.platform = 'number'
|
prop.platform = 'number'
|
||||||
else:
|
else:
|
||||||
# Irregular property will not be transformed.
|
# Irregular property will not be transformed.
|
||||||
pass
|
continue
|
||||||
elif prop.readable or prop.notifiable:
|
elif prop.readable or prop.notifiable:
|
||||||
if prop.format_ == bool:
|
if prop.format_ == bool:
|
||||||
prop.platform = 'binary_sensor'
|
prop.platform = 'binary_sensor'
|
||||||
@ -653,11 +661,66 @@ class MIoTDevice:
|
|||||||
self.append_action(action=action)
|
self.append_action(action=action)
|
||||||
|
|
||||||
def unit_convert(self, spec_unit: str) -> Optional[str]:
|
def unit_convert(self, spec_unit: str) -> Optional[str]:
|
||||||
"""Convert MIoT unit to Home Assistant unit."""
|
"""Convert MIoT unit to Home Assistant unit.
|
||||||
|
25/01/20: All online prop unit statistical tables: unit, quantity.
|
||||||
|
{
|
||||||
|
"no_unit": 148499,
|
||||||
|
"percentage": 10042,
|
||||||
|
"kelvin": 1895,
|
||||||
|
"rgb": 772, // color
|
||||||
|
"celsius": 5762,
|
||||||
|
"none": 16106,
|
||||||
|
"hours": 1540,
|
||||||
|
"minutes": 5061,
|
||||||
|
"ms": 27,
|
||||||
|
"watt": 216,
|
||||||
|
"arcdegrees": 159,
|
||||||
|
"ppm": 177,
|
||||||
|
"μg/m3": 106,
|
||||||
|
"days": 571,
|
||||||
|
"seconds": 2749,
|
||||||
|
"B/s": 21,
|
||||||
|
"pascal": 110,
|
||||||
|
"mg/m3": 339,
|
||||||
|
"lux": 125,
|
||||||
|
"kWh": 124,
|
||||||
|
"mv": 2,
|
||||||
|
"V": 38,
|
||||||
|
"A": 29,
|
||||||
|
"mV": 4,
|
||||||
|
"L": 352,
|
||||||
|
"m": 37,
|
||||||
|
"毫摩尔每升": 2, // blood-sugar, cholesterol
|
||||||
|
"mmol/L": 1, // urea
|
||||||
|
"weeks": 26,
|
||||||
|
"meter": 3,
|
||||||
|
"dB": 26,
|
||||||
|
"hour": 14,
|
||||||
|
"calorie": 19, // 1 cal = 4.184 J
|
||||||
|
"ppb": 3,
|
||||||
|
"arcdegress": 30,
|
||||||
|
"bpm": 4, // realtime-heartrate
|
||||||
|
"gram": 7,
|
||||||
|
"km/h": 9,
|
||||||
|
"W": 1,
|
||||||
|
"m3/h": 2,
|
||||||
|
"kilopascal": 1,
|
||||||
|
"mL": 4,
|
||||||
|
"mmHg": 4,
|
||||||
|
"w": 1,
|
||||||
|
"liter": 1,
|
||||||
|
"cm": 3,
|
||||||
|
"mA": 2,
|
||||||
|
"kilogram": 2,
|
||||||
|
"kcal/d": 2, // basal-metabolism
|
||||||
|
"times": 1 // exercise-count
|
||||||
|
}
|
||||||
|
"""
|
||||||
unit_map = {
|
unit_map = {
|
||||||
'percentage': PERCENTAGE,
|
'percentage': PERCENTAGE,
|
||||||
'weeks': UnitOfTime.WEEKS,
|
'weeks': UnitOfTime.WEEKS,
|
||||||
'days': UnitOfTime.DAYS,
|
'days': UnitOfTime.DAYS,
|
||||||
|
'hour': UnitOfTime.HOURS,
|
||||||
'hours': UnitOfTime.HOURS,
|
'hours': UnitOfTime.HOURS,
|
||||||
'minutes': UnitOfTime.MINUTES,
|
'minutes': UnitOfTime.MINUTES,
|
||||||
'seconds': UnitOfTime.SECONDS,
|
'seconds': UnitOfTime.SECONDS,
|
||||||
@ -672,30 +735,48 @@ class MIoTDevice:
|
|||||||
'ppb': CONCENTRATION_PARTS_PER_BILLION,
|
'ppb': CONCENTRATION_PARTS_PER_BILLION,
|
||||||
'lux': LIGHT_LUX,
|
'lux': LIGHT_LUX,
|
||||||
'pascal': UnitOfPressure.PA,
|
'pascal': UnitOfPressure.PA,
|
||||||
|
'kilopascal': UnitOfPressure.KPA,
|
||||||
|
'mmHg': UnitOfPressure.MMHG,
|
||||||
'bar': UnitOfPressure.BAR,
|
'bar': UnitOfPressure.BAR,
|
||||||
'watt': UnitOfPower.WATT,
|
|
||||||
'L': UnitOfVolume.LITERS,
|
'L': UnitOfVolume.LITERS,
|
||||||
|
'liter': UnitOfVolume.LITERS,
|
||||||
'mL': UnitOfVolume.MILLILITERS,
|
'mL': UnitOfVolume.MILLILITERS,
|
||||||
'km/h': UnitOfSpeed.KILOMETERS_PER_HOUR,
|
'km/h': UnitOfSpeed.KILOMETERS_PER_HOUR,
|
||||||
'm/s': UnitOfSpeed.METERS_PER_SECOND,
|
'm/s': UnitOfSpeed.METERS_PER_SECOND,
|
||||||
|
'watt': UnitOfPower.WATT,
|
||||||
|
'w': UnitOfPower.WATT,
|
||||||
|
'W': UnitOfPower.WATT,
|
||||||
'kWh': UnitOfEnergy.KILO_WATT_HOUR,
|
'kWh': UnitOfEnergy.KILO_WATT_HOUR,
|
||||||
'A': UnitOfElectricCurrent.AMPERE,
|
'A': UnitOfElectricCurrent.AMPERE,
|
||||||
'mA': UnitOfElectricCurrent.MILLIAMPERE,
|
'mA': UnitOfElectricCurrent.MILLIAMPERE,
|
||||||
'V': UnitOfElectricPotential.VOLT,
|
'V': UnitOfElectricPotential.VOLT,
|
||||||
|
'mv': UnitOfElectricPotential.MILLIVOLT,
|
||||||
'mV': UnitOfElectricPotential.MILLIVOLT,
|
'mV': UnitOfElectricPotential.MILLIVOLT,
|
||||||
|
'cm': UnitOfLength.CENTIMETERS,
|
||||||
'm': UnitOfLength.METERS,
|
'm': UnitOfLength.METERS,
|
||||||
|
'meter': UnitOfLength.METERS,
|
||||||
'km': UnitOfLength.KILOMETERS,
|
'km': UnitOfLength.KILOMETERS,
|
||||||
'm3/h': UnitOfVolumeFlowRate.CUBIC_METERS_PER_HOUR,
|
'm3/h': UnitOfVolumeFlowRate.CUBIC_METERS_PER_HOUR,
|
||||||
'gram': UnitOfMass.GRAMS,
|
'gram': UnitOfMass.GRAMS,
|
||||||
|
'kilogram': UnitOfMass.KILOGRAMS,
|
||||||
'dB': SIGNAL_STRENGTH_DECIBELS,
|
'dB': SIGNAL_STRENGTH_DECIBELS,
|
||||||
|
'arcdegrees': DEGREE,
|
||||||
|
'arcdegress': DEGREE,
|
||||||
'kB': UnitOfInformation.KILOBYTES,
|
'kB': UnitOfInformation.KILOBYTES,
|
||||||
|
'MB': UnitOfInformation.MEGABYTES,
|
||||||
|
'GB': UnitOfInformation.GIGABYTES,
|
||||||
|
'TB': UnitOfInformation.TERABYTES,
|
||||||
|
'B/s': UnitOfDataRate.BYTES_PER_SECOND,
|
||||||
|
'KB/s': UnitOfDataRate.KILOBYTES_PER_SECOND,
|
||||||
|
'MB/s': UnitOfDataRate.MEGABYTES_PER_SECOND,
|
||||||
|
'GB/s': UnitOfDataRate.GIGABYTES_PER_SECOND
|
||||||
}
|
}
|
||||||
|
|
||||||
# Handle UnitOfConductivity separately since
|
# Handle UnitOfConductivity separately since
|
||||||
# it might not be available in all HA versions
|
# it might not be available in all HA versions
|
||||||
try:
|
try:
|
||||||
# pylint: disable=import-outside-toplevel
|
# pylint: disable=import-outside-toplevel
|
||||||
from homeassistant.const import UnitOfConductivity
|
from homeassistant.const import UnitOfConductivity # type: ignore
|
||||||
unit_map['μS/cm'] = UnitOfConductivity.MICROSIEMENS_PER_CM
|
unit_map['μS/cm'] = UnitOfConductivity.MICROSIEMENS_PER_CM
|
||||||
except Exception: # pylint: disable=broad-except
|
except Exception: # pylint: disable=broad-except
|
||||||
unit_map['μS/cm'] = 'μS/cm'
|
unit_map['μS/cm'] = 'μS/cm'
|
||||||
@ -703,59 +784,66 @@ class MIoTDevice:
|
|||||||
return unit_map.get(spec_unit, None)
|
return unit_map.get(spec_unit, None)
|
||||||
|
|
||||||
def icon_convert(self, spec_unit: str) -> Optional[str]:
|
def icon_convert(self, spec_unit: str) -> Optional[str]:
|
||||||
if spec_unit in ['percentage']:
|
if spec_unit in {'percentage'}:
|
||||||
return 'mdi:percent'
|
return 'mdi:percent'
|
||||||
if spec_unit in [
|
if spec_unit in {
|
||||||
'weeks', 'days', 'hours', 'minutes', 'seconds', 'ms', 'μs']:
|
'weeks', 'days', 'hour', 'hours', 'minutes', 'seconds', 'ms', 'μs'
|
||||||
|
}:
|
||||||
return 'mdi:clock'
|
return 'mdi:clock'
|
||||||
if spec_unit in ['celsius']:
|
if spec_unit in {'celsius'}:
|
||||||
return 'mdi:temperature-celsius'
|
return 'mdi:temperature-celsius'
|
||||||
if spec_unit in ['fahrenheit']:
|
if spec_unit in {'fahrenheit'}:
|
||||||
return 'mdi:temperature-fahrenheit'
|
return 'mdi:temperature-fahrenheit'
|
||||||
if spec_unit in ['kelvin']:
|
if spec_unit in {'kelvin'}:
|
||||||
return 'mdi:temperature-kelvin'
|
return 'mdi:temperature-kelvin'
|
||||||
if spec_unit in ['μg/m3', 'mg/m3', 'ppm', 'ppb']:
|
if spec_unit in {'μg/m3', 'mg/m3', 'ppm', 'ppb'}:
|
||||||
return 'mdi:blur'
|
return 'mdi:blur'
|
||||||
if spec_unit in ['lux']:
|
if spec_unit in {'lux'}:
|
||||||
return 'mdi:brightness-6'
|
return 'mdi:brightness-6'
|
||||||
if spec_unit in ['pascal', 'megapascal', 'bar']:
|
if spec_unit in {'pascal', 'kilopascal', 'megapascal', 'mmHg', 'bar'}:
|
||||||
return 'mdi:gauge'
|
return 'mdi:gauge'
|
||||||
if spec_unit in ['watt']:
|
if spec_unit in {'watt', 'w', 'W'}:
|
||||||
return 'mdi:flash-triangle'
|
return 'mdi:flash-triangle'
|
||||||
if spec_unit in ['L', 'mL']:
|
if spec_unit in {'L', 'mL'}:
|
||||||
return 'mdi:gas-cylinder'
|
return 'mdi:gas-cylinder'
|
||||||
if spec_unit in ['km/h', 'm/s']:
|
if spec_unit in {'km/h', 'm/s'}:
|
||||||
return 'mdi:speedometer'
|
return 'mdi:speedometer'
|
||||||
if spec_unit in ['kWh']:
|
if spec_unit in {'kWh'}:
|
||||||
return 'mdi:transmission-tower'
|
return 'mdi:transmission-tower'
|
||||||
if spec_unit in ['A', 'mA']:
|
if spec_unit in {'A', 'mA'}:
|
||||||
return 'mdi:current-ac'
|
return 'mdi:current-ac'
|
||||||
if spec_unit in ['V', 'mV']:
|
if spec_unit in {'V', 'mv', 'mV'}:
|
||||||
return 'mdi:current-dc'
|
return 'mdi:current-dc'
|
||||||
if spec_unit in ['m', 'km']:
|
if spec_unit in {'cm', 'm', 'meter', 'km'}:
|
||||||
return 'mdi:ruler'
|
return 'mdi:ruler'
|
||||||
if spec_unit in ['rgb']:
|
if spec_unit in {'rgb'}:
|
||||||
return 'mdi:palette'
|
return 'mdi:palette'
|
||||||
if spec_unit in ['m3/h', 'L/s']:
|
if spec_unit in {'m3/h', 'L/s'}:
|
||||||
return 'mdi:pipe-leak'
|
return 'mdi:pipe-leak'
|
||||||
if spec_unit in ['μS/cm']:
|
if spec_unit in {'μS/cm'}:
|
||||||
return 'mdi:resistor-nodes'
|
return 'mdi:resistor-nodes'
|
||||||
if spec_unit in ['gram']:
|
if spec_unit in {'gram', 'kilogram'}:
|
||||||
return 'mdi:weight'
|
return 'mdi:weight'
|
||||||
if spec_unit in ['dB']:
|
if spec_unit in {'dB'}:
|
||||||
return 'mdi:signal-distance-variant'
|
return 'mdi:signal-distance-variant'
|
||||||
if spec_unit in ['times']:
|
if spec_unit in {'times'}:
|
||||||
return 'mdi:counter'
|
return 'mdi:counter'
|
||||||
if spec_unit in ['mmol/L']:
|
if spec_unit in {'mmol/L'}:
|
||||||
return 'mdi:dots-hexagon'
|
return 'mdi:dots-hexagon'
|
||||||
if spec_unit in ['arcdegress']:
|
if spec_unit in {'kB', 'MB', 'GB'}:
|
||||||
return 'mdi:angle-obtuse'
|
|
||||||
if spec_unit in ['kB']:
|
|
||||||
return 'mdi:network-pos'
|
return 'mdi:network-pos'
|
||||||
if spec_unit in ['calorie', 'kCal']:
|
if spec_unit in {'arcdegress', 'arcdegrees'}:
|
||||||
|
return 'mdi:angle-obtuse'
|
||||||
|
if spec_unit in {'B/s', 'KB/s', 'MB/s', 'GB/s'}:
|
||||||
|
return 'mdi:network'
|
||||||
|
if spec_unit in {'calorie', 'kCal'}:
|
||||||
return 'mdi:food'
|
return 'mdi:food'
|
||||||
return None
|
return None
|
||||||
|
|
||||||
|
def __gen_sub_id(self) -> int:
|
||||||
|
self._sub_id += 1
|
||||||
|
return self._sub_id
|
||||||
|
|
||||||
def __on_device_state_changed(
|
def __on_device_state_changed(
|
||||||
self, did: str, state: MIoTDeviceState, ctx: Any
|
self, did: str, state: MIoTDeviceState, ctx: Any
|
||||||
) -> None:
|
) -> None:
|
||||||
@ -810,6 +898,7 @@ class MIoTServiceEntity(Entity):
|
|||||||
self._attr_name = (
|
self._attr_name = (
|
||||||
f'{"* "if self.entity_data.spec.proprietary else " "}'
|
f'{"* "if self.entity_data.spec.proprietary else " "}'
|
||||||
f'{self.entity_data.spec.description_trans}')
|
f'{self.entity_data.spec.description_trans}')
|
||||||
|
self._attr_entity_category = entity_data.spec.entity_category
|
||||||
# Set entity attr
|
# Set entity attr
|
||||||
self._attr_unique_id = self.entity_id
|
self._attr_unique_id = self.entity_id
|
||||||
self._attr_should_poll = False
|
self._attr_should_poll = False
|
||||||
@ -903,14 +992,14 @@ class MIoTServiceEntity(Entity):
|
|||||||
siid=event.service.iid, eiid=event.iid, sub_id=sub_id)
|
siid=event.service.iid, eiid=event.iid, sub_id=sub_id)
|
||||||
|
|
||||||
def get_map_value(
|
def get_map_value(
|
||||||
self, map_: dict[int, Any], key: int
|
self, map_: Optional[dict[int, Any]], key: int
|
||||||
) -> Any:
|
) -> Any:
|
||||||
if map_ is None:
|
if map_ is None:
|
||||||
return None
|
return None
|
||||||
return map_.get(key, None)
|
return map_.get(key, None)
|
||||||
|
|
||||||
def get_map_key(
|
def get_map_key(
|
||||||
self, map_: dict[int, Any], value: Any
|
self, map_: Optional[dict[int, Any]], value: Any
|
||||||
) -> Optional[int]:
|
) -> Optional[int]:
|
||||||
if map_ is None:
|
if map_ is None:
|
||||||
return None
|
return None
|
||||||
@ -919,7 +1008,7 @@ class MIoTServiceEntity(Entity):
|
|||||||
return key
|
return key
|
||||||
return None
|
return None
|
||||||
|
|
||||||
def get_prop_value(self, prop: MIoTSpecProperty) -> Any:
|
def get_prop_value(self, prop: Optional[MIoTSpecProperty]) -> Any:
|
||||||
if not prop:
|
if not prop:
|
||||||
_LOGGER.error(
|
_LOGGER.error(
|
||||||
'get_prop_value error, property is None, %s, %s',
|
'get_prop_value error, property is None, %s, %s',
|
||||||
@ -927,7 +1016,9 @@ class MIoTServiceEntity(Entity):
|
|||||||
return None
|
return None
|
||||||
return self._prop_value_map.get(prop, None)
|
return self._prop_value_map.get(prop, None)
|
||||||
|
|
||||||
def set_prop_value(self, prop: MIoTSpecProperty, value: Any) -> None:
|
def set_prop_value(
|
||||||
|
self, prop: Optional[MIoTSpecProperty], value: Any
|
||||||
|
) -> None:
|
||||||
if not prop:
|
if not prop:
|
||||||
_LOGGER.error(
|
_LOGGER.error(
|
||||||
'set_prop_value error, property is None, %s, %s',
|
'set_prop_value error, property is None, %s, %s',
|
||||||
@ -936,13 +1027,14 @@ class MIoTServiceEntity(Entity):
|
|||||||
self._prop_value_map[prop] = value
|
self._prop_value_map[prop] = value
|
||||||
|
|
||||||
async def set_property_async(
|
async def set_property_async(
|
||||||
self, prop: MIoTSpecProperty, value: Any, update: bool = True
|
self, prop: Optional[MIoTSpecProperty], value: Any,
|
||||||
|
update_value: bool = True, write_ha_state: bool = True
|
||||||
) -> bool:
|
) -> bool:
|
||||||
value = prop.value_format(value)
|
|
||||||
if not prop:
|
if not prop:
|
||||||
raise RuntimeError(
|
raise RuntimeError(
|
||||||
f'set property failed, property is None, '
|
f'set property failed, property is None, '
|
||||||
f'{self.entity_id}, {self.name}')
|
f'{self.entity_id}, {self.name}')
|
||||||
|
value = prop.value_format(value)
|
||||||
if prop not in self.entity_data.props:
|
if prop not in self.entity_data.props:
|
||||||
raise RuntimeError(
|
raise RuntimeError(
|
||||||
f'set property failed, unknown property, '
|
f'set property failed, unknown property, '
|
||||||
@ -958,8 +1050,9 @@ class MIoTServiceEntity(Entity):
|
|||||||
except MIoTClientError as e:
|
except MIoTClientError as e:
|
||||||
raise RuntimeError(
|
raise RuntimeError(
|
||||||
f'{e}, {self.entity_id}, {self.name}, {prop.name}') from e
|
f'{e}, {self.entity_id}, {self.name}, {prop.name}') from e
|
||||||
if update:
|
if update_value:
|
||||||
self._prop_value_map[prop] = value
|
self._prop_value_map[prop] = value
|
||||||
|
if write_ha_state:
|
||||||
self.async_write_ha_state()
|
self.async_write_ha_state()
|
||||||
return True
|
return True
|
||||||
|
|
||||||
@ -1184,6 +1277,7 @@ class MIoTPropertyEntity(Entity):
|
|||||||
def __on_value_changed(self, params: dict, ctx: Any) -> None:
|
def __on_value_changed(self, params: dict, ctx: Any) -> None:
|
||||||
_LOGGER.debug('property changed, %s', params)
|
_LOGGER.debug('property changed, %s', params)
|
||||||
self._value = self.spec.value_format(params['value'])
|
self._value = self.spec.value_format(params['value'])
|
||||||
|
self._value = self.spec.eval_expr(self._value)
|
||||||
if not self._pending_write_ha_state_timer:
|
if not self._pending_write_ha_state_timer:
|
||||||
self.async_write_ha_state()
|
self.async_write_ha_state()
|
||||||
|
|
||||||
|
@ -1213,10 +1213,13 @@ class MipsLocalClient(_MipsClient):
|
|||||||
or 'did' not in msg
|
or 'did' not in msg
|
||||||
or 'siid' not in msg
|
or 'siid' not in msg
|
||||||
or 'eiid' not in msg
|
or 'eiid' not in msg
|
||||||
or 'arguments' not in msg
|
# or 'arguments' not in msg
|
||||||
):
|
):
|
||||||
# self.log_error(f'on_event_msg, recv unknown msg, {payload}')
|
self.log_info('unknown event msg, %s', payload)
|
||||||
return
|
return
|
||||||
|
if 'arguments' not in msg:
|
||||||
|
self.log_info('wrong event msg, %s', payload)
|
||||||
|
msg['arguments'] = []
|
||||||
if handler:
|
if handler:
|
||||||
self.log_debug('local, on event_occurred, %s', payload)
|
self.log_debug('local, on event_occurred, %s', payload)
|
||||||
handler(msg, ctx)
|
handler(msg, ctx)
|
||||||
@ -1414,7 +1417,7 @@ class MipsLocalClient(_MipsClient):
|
|||||||
@final
|
@final
|
||||||
@on_dev_list_changed.setter
|
@on_dev_list_changed.setter
|
||||||
def on_dev_list_changed(
|
def on_dev_list_changed(
|
||||||
self, func: Callable[[Any, list[str]], Coroutine]
|
self, func: Optional[Callable[[Any, list[str]], Coroutine]]
|
||||||
) -> None:
|
) -> None:
|
||||||
"""run in main loop."""
|
"""run in main loop."""
|
||||||
self._on_dev_list_changed = func
|
self._on_dev_list_changed = func
|
||||||
|
@ -94,7 +94,7 @@ class MIoTNetwork:
|
|||||||
_main_loop: asyncio.AbstractEventLoop
|
_main_loop: asyncio.AbstractEventLoop
|
||||||
|
|
||||||
_ip_addr_map: dict[str, float]
|
_ip_addr_map: dict[str, float]
|
||||||
_url_addr_list: dict[str, float]
|
_http_addr_map: dict[str, float]
|
||||||
_http_session: aiohttp.ClientSession
|
_http_session: aiohttp.ClientSession
|
||||||
|
|
||||||
_refresh_interval: int
|
_refresh_interval: int
|
||||||
@ -283,8 +283,8 @@ class MIoTNetwork:
|
|||||||
[
|
[
|
||||||
'ping', '-c', '1', '-w',
|
'ping', '-c', '1', '-w',
|
||||||
str(self._DETECT_TIMEOUT), address]),
|
str(self._DETECT_TIMEOUT), address]),
|
||||||
stdout=subprocess.PIPE,
|
stdout=subprocess.DEVNULL,
|
||||||
stderr=subprocess.PIPE
|
stderr=subprocess.DEVNULL
|
||||||
)
|
)
|
||||||
await process.communicate()
|
await process.communicate()
|
||||||
if process.returncode == 0:
|
if process.returncode == 0:
|
||||||
|
@ -56,7 +56,7 @@ from slugify import slugify
|
|||||||
|
|
||||||
# pylint: disable=relative-beyond-top-level
|
# pylint: disable=relative-beyond-top-level
|
||||||
from .const import DEFAULT_INTEGRATION_LANGUAGE, SPEC_STD_LIB_EFFECTIVE_TIME
|
from .const import DEFAULT_INTEGRATION_LANGUAGE, SPEC_STD_LIB_EFFECTIVE_TIME
|
||||||
from .common import MIoTHttp, load_yaml_file
|
from .common import MIoTHttp, load_yaml_file, load_json_file
|
||||||
from .miot_error import MIoTSpecError
|
from .miot_error import MIoTSpecError
|
||||||
from .miot_storage import MIoTStorage
|
from .miot_storage import MIoTStorage
|
||||||
|
|
||||||
@ -465,18 +465,18 @@ class _MIoTSpecBase:
|
|||||||
iid: int
|
iid: int
|
||||||
type_: str
|
type_: str
|
||||||
description: str
|
description: str
|
||||||
description_trans: str
|
description_trans: Optional[str]
|
||||||
proprietary: bool
|
proprietary: bool
|
||||||
need_filter: bool
|
need_filter: bool
|
||||||
name: str
|
name: str
|
||||||
|
icon: Optional[str]
|
||||||
|
|
||||||
# External params
|
# External params
|
||||||
platform: Optional[str]
|
platform: Optional[str]
|
||||||
device_class: Any
|
device_class: Any
|
||||||
state_class: Any
|
state_class: Any
|
||||||
icon: Optional[str]
|
|
||||||
external_unit: Any
|
external_unit: Any
|
||||||
expression: Optional[str]
|
entity_category: Optional[str]
|
||||||
|
|
||||||
spec_id: int
|
spec_id: int
|
||||||
|
|
||||||
@ -489,13 +489,13 @@ class _MIoTSpecBase:
|
|||||||
self.proprietary = spec.get('proprietary', False)
|
self.proprietary = spec.get('proprietary', False)
|
||||||
self.need_filter = spec.get('need_filter', False)
|
self.need_filter = spec.get('need_filter', False)
|
||||||
self.name = spec.get('name', 'xiaomi')
|
self.name = spec.get('name', 'xiaomi')
|
||||||
|
self.icon = spec.get('icon', None)
|
||||||
|
|
||||||
self.platform = None
|
self.platform = None
|
||||||
self.device_class = None
|
self.device_class = None
|
||||||
self.state_class = None
|
self.state_class = None
|
||||||
self.icon = None
|
|
||||||
self.external_unit = None
|
self.external_unit = None
|
||||||
self.expression = None
|
self.entity_category = None
|
||||||
|
|
||||||
self.spec_id = hash(f'{self.type_}.{self.iid}')
|
self.spec_id = hash(f'{self.type_}.{self.iid}')
|
||||||
|
|
||||||
@ -510,6 +510,7 @@ class MIoTSpecProperty(_MIoTSpecBase):
|
|||||||
"""MIoT SPEC property class."""
|
"""MIoT SPEC property class."""
|
||||||
unit: Optional[str]
|
unit: Optional[str]
|
||||||
precision: int
|
precision: int
|
||||||
|
expr: Optional[str]
|
||||||
|
|
||||||
_format_: Type
|
_format_: Type
|
||||||
_value_range: Optional[MIoTSpecValueRange]
|
_value_range: Optional[MIoTSpecValueRange]
|
||||||
@ -531,7 +532,8 @@ class MIoTSpecProperty(_MIoTSpecBase):
|
|||||||
unit: Optional[str] = None,
|
unit: Optional[str] = None,
|
||||||
value_range: Optional[dict] = None,
|
value_range: Optional[dict] = None,
|
||||||
value_list: Optional[list[dict]] = None,
|
value_list: Optional[list[dict]] = None,
|
||||||
precision: Optional[int] = None
|
precision: Optional[int] = None,
|
||||||
|
expr: Optional[str] = None
|
||||||
) -> None:
|
) -> None:
|
||||||
super().__init__(spec=spec)
|
super().__init__(spec=spec)
|
||||||
self.service = service
|
self.service = service
|
||||||
@ -540,7 +542,8 @@ class MIoTSpecProperty(_MIoTSpecBase):
|
|||||||
self.unit = unit
|
self.unit = unit
|
||||||
self.value_range = value_range
|
self.value_range = value_range
|
||||||
self.value_list = value_list
|
self.value_list = value_list
|
||||||
self.precision = precision or 1
|
self.precision = precision if precision is not None else 1
|
||||||
|
self.expr = expr
|
||||||
|
|
||||||
self.spec_id = hash(
|
self.spec_id = hash(
|
||||||
f'p.{self.name}.{self.service.iid}.{self.iid}')
|
f'p.{self.name}.{self.service.iid}.{self.iid}')
|
||||||
@ -613,6 +616,18 @@ class MIoTSpecProperty(_MIoTSpecBase):
|
|||||||
elif isinstance(value, MIoTSpecValueList):
|
elif isinstance(value, MIoTSpecValueList):
|
||||||
self._value_list = value
|
self._value_list = value
|
||||||
|
|
||||||
|
def eval_expr(self, src_value: Any) -> Any:
|
||||||
|
if not self.expr:
|
||||||
|
return src_value
|
||||||
|
try:
|
||||||
|
# pylint: disable=eval-used
|
||||||
|
return eval(self.expr, {'src_value': src_value})
|
||||||
|
except Exception as err: # pylint: disable=broad-exception-caught
|
||||||
|
_LOGGER.error(
|
||||||
|
'eval expression error, %s, %s, %s, %s',
|
||||||
|
self.iid, src_value, self.expr, err)
|
||||||
|
return src_value
|
||||||
|
|
||||||
def value_format(self, value: Any) -> Any:
|
def value_format(self, value: Any) -> Any:
|
||||||
if value is None:
|
if value is None:
|
||||||
return None
|
return None
|
||||||
@ -639,7 +654,9 @@ class MIoTSpecProperty(_MIoTSpecBase):
|
|||||||
'value_range': (
|
'value_range': (
|
||||||
self._value_range.dump() if self._value_range else None),
|
self._value_range.dump() if self._value_range else None),
|
||||||
'value_list': self._value_list.dump() if self._value_list else None,
|
'value_list': self._value_list.dump() if self._value_list else None,
|
||||||
'precision': self.precision
|
'precision': self.precision,
|
||||||
|
'expr': self.expr,
|
||||||
|
'icon': self.icon
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -732,7 +749,6 @@ class MIoTSpecService(_MIoTSpecBase):
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
# @dataclass
|
|
||||||
class MIoTSpecInstance:
|
class MIoTSpecInstance:
|
||||||
"""MIoT SPEC instance class."""
|
"""MIoT SPEC instance class."""
|
||||||
urn: str
|
urn: str
|
||||||
@ -774,7 +790,8 @@ class MIoTSpecInstance:
|
|||||||
unit=prop['unit'],
|
unit=prop['unit'],
|
||||||
value_range=prop['value_range'],
|
value_range=prop['value_range'],
|
||||||
value_list=prop['value_list'],
|
value_list=prop['value_list'],
|
||||||
precision=prop.get('precision', None))
|
precision=prop.get('precision', None),
|
||||||
|
expr=prop.get('expr', None))
|
||||||
spec_service.properties.append(spec_prop)
|
spec_service.properties.append(spec_prop)
|
||||||
for event in service['events']:
|
for event in service['events']:
|
||||||
spec_event = MIoTSpecEvent(
|
spec_event = MIoTSpecEvent(
|
||||||
@ -822,6 +839,7 @@ class _MIoTSpecMultiLang:
|
|||||||
"""MIoT SPEC multi lang class."""
|
"""MIoT SPEC multi lang class."""
|
||||||
# pylint: disable=broad-exception-caught
|
# pylint: disable=broad-exception-caught
|
||||||
_DOMAIN: str = 'miot_specs_multi_lang'
|
_DOMAIN: str = 'miot_specs_multi_lang'
|
||||||
|
_MULTI_LANG_FILE = 'specs/multi_lang.json'
|
||||||
_lang: str
|
_lang: str
|
||||||
_storage: MIoTStorage
|
_storage: MIoTStorage
|
||||||
_main_loop: asyncio.AbstractEventLoop
|
_main_loop: asyncio.AbstractEventLoop
|
||||||
@ -877,6 +895,25 @@ class _MIoTSpecMultiLang:
|
|||||||
except Exception as err:
|
except Exception as err:
|
||||||
trans_local = {}
|
trans_local = {}
|
||||||
_LOGGER.info('get multi lang from local failed, %s, %s', urn, err)
|
_LOGGER.info('get multi lang from local failed, %s, %s', urn, err)
|
||||||
|
# Revert: load multi_lang.json
|
||||||
|
try:
|
||||||
|
trans_local_json = await self._main_loop.run_in_executor(
|
||||||
|
None, load_json_file,
|
||||||
|
os.path.join(
|
||||||
|
os.path.dirname(os.path.abspath(__file__)),
|
||||||
|
self._MULTI_LANG_FILE))
|
||||||
|
urn_strs: list[str] = urn.split(':')
|
||||||
|
urn_key: str = ':'.join(urn_strs[:6])
|
||||||
|
if (
|
||||||
|
isinstance(trans_local_json, dict)
|
||||||
|
and urn_key in trans_local_json
|
||||||
|
and self._lang in trans_local_json[urn_key]
|
||||||
|
):
|
||||||
|
trans_cache.update(trans_local_json[urn_key][self._lang])
|
||||||
|
trans_local = trans_local_json[urn_key]
|
||||||
|
except Exception as err: # pylint: disable=broad-exception-caught
|
||||||
|
_LOGGER.error('multi lang, load json file error, %s', err)
|
||||||
|
# Revert end
|
||||||
# Default language
|
# Default language
|
||||||
if not trans_cache:
|
if not trans_cache:
|
||||||
if trans_cloud and DEFAULT_INTEGRATION_LANGUAGE in trans_cloud:
|
if trans_cloud and DEFAULT_INTEGRATION_LANGUAGE in trans_cloud:
|
||||||
@ -1119,6 +1156,96 @@ class _SpecFilter:
|
|||||||
return False
|
return False
|
||||||
|
|
||||||
|
|
||||||
|
class _SpecModify:
|
||||||
|
"""MIoT-Spec-V2 modify for entity conversion."""
|
||||||
|
_SPEC_MODIFY_FILE = 'specs/spec_modify.yaml'
|
||||||
|
_main_loop: asyncio.AbstractEventLoop
|
||||||
|
_data: Optional[dict]
|
||||||
|
_selected: Optional[dict]
|
||||||
|
|
||||||
|
def __init__(
|
||||||
|
self, loop: Optional[asyncio.AbstractEventLoop] = None
|
||||||
|
) -> None:
|
||||||
|
self._main_loop = loop or asyncio.get_running_loop()
|
||||||
|
self._data = None
|
||||||
|
|
||||||
|
async def init_async(self) -> None:
|
||||||
|
if isinstance(self._data, dict):
|
||||||
|
return
|
||||||
|
modify_data = None
|
||||||
|
self._data = {}
|
||||||
|
self._selected = None
|
||||||
|
try:
|
||||||
|
modify_data = await self._main_loop.run_in_executor(
|
||||||
|
None, load_yaml_file,
|
||||||
|
os.path.join(
|
||||||
|
os.path.dirname(os.path.abspath(__file__)),
|
||||||
|
self._SPEC_MODIFY_FILE))
|
||||||
|
except Exception as err: # pylint: disable=broad-exception-caught
|
||||||
|
_LOGGER.error('spec modify, load file error, %s', err)
|
||||||
|
return
|
||||||
|
if not isinstance(modify_data, dict):
|
||||||
|
_LOGGER.error('spec modify, invalid spec modify content')
|
||||||
|
return
|
||||||
|
for key, value in modify_data.items():
|
||||||
|
if not isinstance(key, str) or not isinstance(value, (dict, str)):
|
||||||
|
_LOGGER.error('spec modify, invalid spec modify data')
|
||||||
|
return
|
||||||
|
|
||||||
|
self._data = modify_data
|
||||||
|
|
||||||
|
async def deinit_async(self) -> None:
|
||||||
|
self._data = None
|
||||||
|
self._selected = None
|
||||||
|
|
||||||
|
async def set_spec_async(self, urn: str) -> None:
|
||||||
|
if not self._data:
|
||||||
|
return
|
||||||
|
self._selected = self._data.get(urn, None)
|
||||||
|
if isinstance(self._selected, str):
|
||||||
|
return await self.set_spec_async(urn=self._selected)
|
||||||
|
|
||||||
|
def get_prop_name(self, siid: int, piid: int) -> Optional[str]:
|
||||||
|
return self.__get_prop_item(siid=siid, piid=piid, key='name')
|
||||||
|
|
||||||
|
def get_prop_unit(self, siid: int, piid: int) -> Optional[str]:
|
||||||
|
return self.__get_prop_item(siid=siid, piid=piid, key='unit')
|
||||||
|
|
||||||
|
def get_prop_expr(self, siid: int, piid: int) -> Optional[str]:
|
||||||
|
return self.__get_prop_item(siid=siid, piid=piid, key='expr')
|
||||||
|
|
||||||
|
def get_prop_icon(self, siid: int, piid: int) -> Optional[str]:
|
||||||
|
return self.__get_prop_item(siid=siid, piid=piid, key='icon')
|
||||||
|
|
||||||
|
def get_prop_access(self, siid: int, piid: int) -> Optional[list]:
|
||||||
|
access = self.__get_prop_item(siid=siid, piid=piid, key='access')
|
||||||
|
if not isinstance(access, list):
|
||||||
|
return None
|
||||||
|
return access
|
||||||
|
|
||||||
|
def get_prop_value_range(self, siid: int, piid: int) -> Optional[list]:
|
||||||
|
value_range = self.__get_prop_item(siid=siid, piid=piid,
|
||||||
|
key='value-range')
|
||||||
|
if not isinstance(value_range, list):
|
||||||
|
return None
|
||||||
|
return value_range
|
||||||
|
|
||||||
|
def get_prop_value_list(self, siid: int, piid: int) -> Optional[list]:
|
||||||
|
value_list = self.__get_prop_item(siid=siid, piid=piid,
|
||||||
|
key='value-list')
|
||||||
|
if not isinstance(value_list, list):
|
||||||
|
return None
|
||||||
|
return value_list
|
||||||
|
|
||||||
|
def __get_prop_item(self, siid: int, piid: int, key: str) -> Optional[str]:
|
||||||
|
if not self._selected:
|
||||||
|
return None
|
||||||
|
prop = self._selected.get(f'prop.{siid}.{piid}', None)
|
||||||
|
if not prop:
|
||||||
|
return None
|
||||||
|
return prop.get(key, None)
|
||||||
|
|
||||||
|
|
||||||
class MIoTSpecParser:
|
class MIoTSpecParser:
|
||||||
"""MIoT SPEC parser."""
|
"""MIoT SPEC parser."""
|
||||||
# pylint: disable=inconsistent-quotes
|
# pylint: disable=inconsistent-quotes
|
||||||
@ -1132,6 +1259,7 @@ class MIoTSpecParser:
|
|||||||
_multi_lang: _MIoTSpecMultiLang
|
_multi_lang: _MIoTSpecMultiLang
|
||||||
_bool_trans: _SpecBoolTranslation
|
_bool_trans: _SpecBoolTranslation
|
||||||
_spec_filter: _SpecFilter
|
_spec_filter: _SpecFilter
|
||||||
|
_spec_modify: _SpecModify
|
||||||
|
|
||||||
_init_done: bool
|
_init_done: bool
|
||||||
|
|
||||||
@ -1149,6 +1277,7 @@ class MIoTSpecParser:
|
|||||||
self._bool_trans = _SpecBoolTranslation(
|
self._bool_trans = _SpecBoolTranslation(
|
||||||
lang=self._lang, loop=self._main_loop)
|
lang=self._lang, loop=self._main_loop)
|
||||||
self._spec_filter = _SpecFilter(loop=self._main_loop)
|
self._spec_filter = _SpecFilter(loop=self._main_loop)
|
||||||
|
self._spec_modify = _SpecModify(loop=self._main_loop)
|
||||||
|
|
||||||
self._init_done = False
|
self._init_done = False
|
||||||
|
|
||||||
@ -1157,6 +1286,7 @@ class MIoTSpecParser:
|
|||||||
return
|
return
|
||||||
await self._bool_trans.init_async()
|
await self._bool_trans.init_async()
|
||||||
await self._spec_filter.init_async()
|
await self._spec_filter.init_async()
|
||||||
|
await self._spec_modify.init_async()
|
||||||
std_lib_cache = await self._storage.load_async(
|
std_lib_cache = await self._storage.load_async(
|
||||||
domain=self._DOMAIN, name='spec_std_lib', type_=dict)
|
domain=self._DOMAIN, name='spec_std_lib', type_=dict)
|
||||||
if (
|
if (
|
||||||
@ -1196,6 +1326,7 @@ class MIoTSpecParser:
|
|||||||
# self._std_lib.deinit()
|
# self._std_lib.deinit()
|
||||||
await self._bool_trans.deinit_async()
|
await self._bool_trans.deinit_async()
|
||||||
await self._spec_filter.deinit_async()
|
await self._spec_filter.deinit_async()
|
||||||
|
await self._spec_modify.deinit_async()
|
||||||
|
|
||||||
async def parse(
|
async def parse(
|
||||||
self, urn: str, skip_cache: bool = False,
|
self, urn: str, skip_cache: bool = False,
|
||||||
@ -1275,6 +1406,8 @@ class MIoTSpecParser:
|
|||||||
await self._multi_lang.set_spec_async(urn=urn)
|
await self._multi_lang.set_spec_async(urn=urn)
|
||||||
# Set spec filter
|
# Set spec filter
|
||||||
await self._spec_filter.set_spec_spec(urn_key=urn_key)
|
await self._spec_filter.set_spec_spec(urn_key=urn_key)
|
||||||
|
# Set spec modify
|
||||||
|
await self._spec_modify.set_spec_async(urn=urn)
|
||||||
# Parse device type
|
# Parse device type
|
||||||
spec_instance: MIoTSpecInstance = MIoTSpecInstance(
|
spec_instance: MIoTSpecInstance = MIoTSpecInstance(
|
||||||
urn=urn, name=urn_strs[3],
|
urn=urn, name=urn_strs[3],
|
||||||
@ -1320,12 +1453,14 @@ class MIoTSpecParser:
|
|||||||
):
|
):
|
||||||
continue
|
continue
|
||||||
p_type_strs: list[str] = property_['type'].split(':')
|
p_type_strs: list[str] = property_['type'].split(':')
|
||||||
|
# Handle special property.unit
|
||||||
|
unit = property_.get('unit', None)
|
||||||
spec_prop: MIoTSpecProperty = MIoTSpecProperty(
|
spec_prop: MIoTSpecProperty = MIoTSpecProperty(
|
||||||
spec=property_,
|
spec=property_,
|
||||||
service=spec_service,
|
service=spec_service,
|
||||||
format_=property_['format'],
|
format_=property_['format'],
|
||||||
access=property_['access'],
|
access=property_['access'],
|
||||||
unit=property_.get('unit', None))
|
unit=unit if unit != 'none' else None)
|
||||||
spec_prop.name = p_type_strs[3]
|
spec_prop.name = p_type_strs[3]
|
||||||
# Filter spec property
|
# Filter spec property
|
||||||
spec_prop.need_filter = (
|
spec_prop.need_filter = (
|
||||||
@ -1341,10 +1476,12 @@ class MIoTSpecParser:
|
|||||||
key=':'.join(p_type_strs[:5]))
|
key=':'.join(p_type_strs[:5]))
|
||||||
or property_['description']
|
or property_['description']
|
||||||
or spec_prop.name)
|
or spec_prop.name)
|
||||||
if 'value-range' in property_:
|
# Modify value-list before translation
|
||||||
spec_prop.value_range = property_['value-range']
|
v_list: list[dict] = self._spec_modify.get_prop_value_list(
|
||||||
elif 'value-list' in property_:
|
siid=service['iid'], piid=property_['iid'])
|
||||||
v_list: list[dict] = property_['value-list']
|
if (v_list is None) and ('value-list' in property_):
|
||||||
|
v_list = property_['value-list']
|
||||||
|
if v_list is not None:
|
||||||
for index, v in enumerate(v_list):
|
for index, v in enumerate(v_list):
|
||||||
if v['description'].strip() == '':
|
if v['description'].strip() == '':
|
||||||
v['description'] = f'v_{v["value"]}'
|
v['description'] = f'v_{v["value"]}'
|
||||||
@ -1358,6 +1495,8 @@ class MIoTSpecParser:
|
|||||||
f'{v["description"]}')
|
f'{v["description"]}')
|
||||||
or v['name'])
|
or v['name'])
|
||||||
spec_prop.value_list = MIoTSpecValueList.from_spec(v_list)
|
spec_prop.value_list = MIoTSpecValueList.from_spec(v_list)
|
||||||
|
if 'value-range' in property_:
|
||||||
|
spec_prop.value_range = property_['value-range']
|
||||||
elif property_['format'] == 'bool':
|
elif property_['format'] == 'bool':
|
||||||
v_tag = ':'.join(p_type_strs[:5])
|
v_tag = ':'.join(p_type_strs[:5])
|
||||||
v_descriptions = (
|
v_descriptions = (
|
||||||
@ -1365,7 +1504,27 @@ class MIoTSpecParser:
|
|||||||
if v_descriptions:
|
if v_descriptions:
|
||||||
# bool without value-list.name
|
# bool without value-list.name
|
||||||
spec_prop.value_list = v_descriptions
|
spec_prop.value_list = v_descriptions
|
||||||
|
# Prop modify
|
||||||
|
spec_prop.unit = self._spec_modify.get_prop_unit(
|
||||||
|
siid=service['iid'], piid=property_['iid']
|
||||||
|
) or spec_prop.unit
|
||||||
|
spec_prop.expr = self._spec_modify.get_prop_expr(
|
||||||
|
siid=service['iid'], piid=property_['iid'])
|
||||||
|
spec_prop.icon = self._spec_modify.get_prop_icon(
|
||||||
|
siid=service['iid'], piid=property_['iid'])
|
||||||
spec_service.properties.append(spec_prop)
|
spec_service.properties.append(spec_prop)
|
||||||
|
custom_access = self._spec_modify.get_prop_access(
|
||||||
|
siid=service['iid'], piid=property_['iid'])
|
||||||
|
if custom_access:
|
||||||
|
spec_prop.access = custom_access
|
||||||
|
custom_range = self._spec_modify.get_prop_value_range(
|
||||||
|
siid=service['iid'], piid=property_['iid'])
|
||||||
|
if custom_range:
|
||||||
|
spec_prop.value_range = custom_range
|
||||||
|
custom_name = self._spec_modify.get_prop_name(
|
||||||
|
siid=service['iid'], piid=property_['iid'])
|
||||||
|
if custom_name:
|
||||||
|
spec_prop.name = custom_name
|
||||||
# Parse service event
|
# Parse service event
|
||||||
for event in service.get('events', []):
|
for event in service.get('events', []):
|
||||||
if (
|
if (
|
||||||
|
@ -59,43 +59,6 @@ data:
|
|||||||
urn:miot-spec-v2:property:wifi-ssid-hidden:000000E3: yes_no
|
urn:miot-spec-v2:property:wifi-ssid-hidden:000000E3: yes_no
|
||||||
urn:miot-spec-v2:property:wind-reverse:00000117: yes_no
|
urn:miot-spec-v2:property:wind-reverse:00000117: yes_no
|
||||||
translate:
|
translate:
|
||||||
contact_state:
|
|
||||||
de:
|
|
||||||
'false': Kein Kontakt
|
|
||||||
'true': Kontakt
|
|
||||||
en:
|
|
||||||
'false': No Contact
|
|
||||||
'true': Contact
|
|
||||||
es:
|
|
||||||
'false': Sin contacto
|
|
||||||
'true': Contacto
|
|
||||||
fr:
|
|
||||||
'false': Pas de contact
|
|
||||||
'true': Contact
|
|
||||||
it:
|
|
||||||
'false': Nessun contatto
|
|
||||||
'true': Contatto
|
|
||||||
ja:
|
|
||||||
'false': 非接触
|
|
||||||
'true': 接触
|
|
||||||
nl:
|
|
||||||
'false': Geen contact
|
|
||||||
'true': Contact
|
|
||||||
pt:
|
|
||||||
'false': Sem contato
|
|
||||||
'true': Contato
|
|
||||||
pt-BR:
|
|
||||||
'false': Sem contato
|
|
||||||
'true': Contato
|
|
||||||
ru:
|
|
||||||
'false': Нет контакта
|
|
||||||
'true': Контакт
|
|
||||||
zh-Hans:
|
|
||||||
'false': 分离
|
|
||||||
'true': 接触
|
|
||||||
zh-Hant:
|
|
||||||
'false': 分離
|
|
||||||
'true': 接觸
|
|
||||||
default:
|
default:
|
||||||
de:
|
de:
|
||||||
'false': Falsch
|
'false': Falsch
|
||||||
@ -133,6 +96,43 @@ translate:
|
|||||||
zh-Hant:
|
zh-Hant:
|
||||||
'false': 假
|
'false': 假
|
||||||
'true': 真
|
'true': 真
|
||||||
|
contact_state:
|
||||||
|
de:
|
||||||
|
'false': Kein Kontakt
|
||||||
|
'true': Kontakt
|
||||||
|
en:
|
||||||
|
'false': No Contact
|
||||||
|
'true': Contact
|
||||||
|
es:
|
||||||
|
'false': Sin contacto
|
||||||
|
'true': Contacto
|
||||||
|
fr:
|
||||||
|
'false': Pas de contact
|
||||||
|
'true': Contact
|
||||||
|
it:
|
||||||
|
'false': Nessun contatto
|
||||||
|
'true': Contatto
|
||||||
|
ja:
|
||||||
|
'false': 非接触
|
||||||
|
'true': 接触
|
||||||
|
nl:
|
||||||
|
'false': Geen contact
|
||||||
|
'true': Contact
|
||||||
|
pt:
|
||||||
|
'false': Sem contato
|
||||||
|
'true': Contato
|
||||||
|
pt-BR:
|
||||||
|
'false': Sem contato
|
||||||
|
'true': Contato
|
||||||
|
ru:
|
||||||
|
'false': Нет контакта
|
||||||
|
'true': Контакт
|
||||||
|
zh-Hans:
|
||||||
|
'false': 分离
|
||||||
|
'true': 接触
|
||||||
|
zh-Hant:
|
||||||
|
'false': 分離
|
||||||
|
'true': 接觸
|
||||||
motion_state:
|
motion_state:
|
||||||
de:
|
de:
|
||||||
'false': Keine Bewegung erkannt
|
'false': Keine Bewegung erkannt
|
||||||
|
187
custom_components/xiaomi_home/miot/specs/multi_lang.json
Normal file
187
custom_components/xiaomi_home/miot/specs/multi_lang.json
Normal file
@ -0,0 +1,187 @@
|
|||||||
|
{
|
||||||
|
"urn:miot-spec-v2:device:gateway:0000A019:xiaomi-hub1": {
|
||||||
|
"de": {
|
||||||
|
"service:001": "Geräteinformationen",
|
||||||
|
"service:001:property:003": "Geräte-ID",
|
||||||
|
"service:001:property:005": "Seriennummer (SN)",
|
||||||
|
"service:002": "Gateway",
|
||||||
|
"service:002:event:001": "Netzwerk geändert",
|
||||||
|
"service:002:event:002": "Netzwerk geändert",
|
||||||
|
"service:002:property:001": "Zugriffsmethode",
|
||||||
|
"service:002:property:001:valuelist:000": "Kabelgebunden",
|
||||||
|
"service:002:property:001:valuelist:001": "5G Drahtlos",
|
||||||
|
"service:002:property:001:valuelist:002": "2.4G Drahtlos",
|
||||||
|
"service:002:property:002": "IP-Adresse",
|
||||||
|
"service:002:property:003": "WiFi-Netzwerkname",
|
||||||
|
"service:002:property:004": "Aktuelle Zeit",
|
||||||
|
"service:002:property:005": "DHCP-Server-MAC-Adresse",
|
||||||
|
"service:003": "Anzeigelampe",
|
||||||
|
"service:003:property:001": "Schalter",
|
||||||
|
"service:004": "Virtueller Dienst",
|
||||||
|
"service:004:action:001": "Virtuelles Ereignis erzeugen",
|
||||||
|
"service:004:event:001": "Virtuelles Ereignis aufgetreten",
|
||||||
|
"service:004:property:001": "Ereignisname"
|
||||||
|
},
|
||||||
|
"en": {
|
||||||
|
"service:001": "Device Information",
|
||||||
|
"service:001:property:003": "Device ID",
|
||||||
|
"service:001:property:005": "Serial Number (SN)",
|
||||||
|
"service:002": "Gateway",
|
||||||
|
"service:002:event:001": "Network Changed",
|
||||||
|
"service:002:event:002": "Network Changed",
|
||||||
|
"service:002:property:001": "Access Method",
|
||||||
|
"service:002:property:001:valuelist:000": "Wired",
|
||||||
|
"service:002:property:001:valuelist:001": "5G Wireless",
|
||||||
|
"service:002:property:001:valuelist:002": "2.4G Wireless",
|
||||||
|
"service:002:property:002": "IP Address",
|
||||||
|
"service:002:property:003": "WiFi Network Name",
|
||||||
|
"service:002:property:004": "Current Time",
|
||||||
|
"service:002:property:005": "DHCP Server MAC Address",
|
||||||
|
"service:003": "Indicator Light",
|
||||||
|
"service:003:property:001": "Switch",
|
||||||
|
"service:004": "Virtual Service",
|
||||||
|
"service:004:action:001": "Generate Virtual Event",
|
||||||
|
"service:004:event:001": "Virtual Event Occurred",
|
||||||
|
"service:004:property:001": "Event Name"
|
||||||
|
},
|
||||||
|
"es": {
|
||||||
|
"service:001": "Información del dispositivo",
|
||||||
|
"service:001:property:003": "ID del dispositivo",
|
||||||
|
"service:001:property:005": "Número de serie (SN)",
|
||||||
|
"service:002": "Puerta de enlace",
|
||||||
|
"service:002:event:001": "Cambio de red",
|
||||||
|
"service:002:event:002": "Cambio de red",
|
||||||
|
"service:002:property:001": "Método de acceso",
|
||||||
|
"service:002:property:001:valuelist:000": "Cableado",
|
||||||
|
"service:002:property:001:valuelist:001": "5G inalámbrico",
|
||||||
|
"service:002:property:001:valuelist:002": "2.4G inalámbrico",
|
||||||
|
"service:002:property:002": "Dirección IP",
|
||||||
|
"service:002:property:003": "Nombre de red WiFi",
|
||||||
|
"service:002:property:004": "Hora actual",
|
||||||
|
"service:002:property:005": "Dirección MAC del servidor DHCP",
|
||||||
|
"service:003": "Luz indicadora",
|
||||||
|
"service:003:property:001": "Interruptor",
|
||||||
|
"service:004": "Servicio virtual",
|
||||||
|
"service:004:action:001": "Generar evento virtual",
|
||||||
|
"service:004:event:001": "Ocurrió un evento virtual",
|
||||||
|
"service:004:property:001": "Nombre del evento"
|
||||||
|
},
|
||||||
|
"fr": {
|
||||||
|
"service:001": "Informations sur l'appareil",
|
||||||
|
"service:001:property:003": "ID de l'appareil",
|
||||||
|
"service:001:property:005": "Numéro de série (SN)",
|
||||||
|
"service:002": "Passerelle",
|
||||||
|
"service:002:event:001": "Changement de réseau",
|
||||||
|
"service:002:event:002": "Changement de réseau",
|
||||||
|
"service:002:property:001": "Méthode d'accès",
|
||||||
|
"service:002:property:001:valuelist:000": "Câblé",
|
||||||
|
"service:002:property:001:valuelist:001": "Sans fil 5G",
|
||||||
|
"service:002:property:001:valuelist:002": "Sans fil 2.4G",
|
||||||
|
"service:002:property:002": "Adresse IP",
|
||||||
|
"service:002:property:003": "Nom du réseau WiFi",
|
||||||
|
"service:002:property:004": "Heure actuelle",
|
||||||
|
"service:002:property:005": "Adresse MAC du serveur DHCP",
|
||||||
|
"service:003": "Voyant lumineux",
|
||||||
|
"service:003:property:001": "Interrupteur",
|
||||||
|
"service:004": "Service virtuel",
|
||||||
|
"service:004:action:001": "Générer un événement virtuel",
|
||||||
|
"service:004:event:001": "Événement virtuel survenu",
|
||||||
|
"service:004:property:001": "Nom de l'événement"
|
||||||
|
},
|
||||||
|
"ja": {
|
||||||
|
"service:001": "デバイス情報",
|
||||||
|
"service:001:property:003": "デバイスID",
|
||||||
|
"service:001:property:005": "シリアル番号 (SN)",
|
||||||
|
"service:002": "ゲートウェイ",
|
||||||
|
"service:002:event:001": "ネットワークが変更されました",
|
||||||
|
"service:002:event:002": "ネットワークが変更されました",
|
||||||
|
"service:002:property:001": "アクセス方法",
|
||||||
|
"service:002:property:001:valuelist:000": "有線",
|
||||||
|
"service:002:property:001:valuelist:001": "5G ワイヤレス",
|
||||||
|
"service:002:property:001:valuelist:002": "2.4G ワイヤレス",
|
||||||
|
"service:002:property:002": "IPアドレス",
|
||||||
|
"service:002:property:003": "WiFiネットワーク名",
|
||||||
|
"service:002:property:004": "現在の時間",
|
||||||
|
"service:002:property:005": "DHCPサーバーMACアドレス",
|
||||||
|
"service:003": "インジケータライト",
|
||||||
|
"service:003:property:001": "スイッチ",
|
||||||
|
"service:004": "バーチャルサービス",
|
||||||
|
"service:004:action:001": "バーチャルイベントを生成",
|
||||||
|
"service:004:event:001": "バーチャルイベントが発生しました",
|
||||||
|
"service:004:property:001": "イベント名"
|
||||||
|
},
|
||||||
|
"ru": {
|
||||||
|
"service:001": "Информация об устройстве",
|
||||||
|
"service:001:property:003": "ID устройства",
|
||||||
|
"service:001:property:005": "Серийный номер (SN)",
|
||||||
|
"service:002": "Шлюз",
|
||||||
|
"service:002:event:001": "Сеть изменена",
|
||||||
|
"service:002:event:002": "Сеть изменена",
|
||||||
|
"service:002:property:001": "Метод доступа",
|
||||||
|
"service:002:property:001:valuelist:000": "Проводной",
|
||||||
|
"service:002:property:001:valuelist:001": "5G Беспроводной",
|
||||||
|
"service:002:property:001:valuelist:002": "2.4G Беспроводной",
|
||||||
|
"service:002:property:002": "IP Адрес",
|
||||||
|
"service:002:property:003": "Название WiFi сети",
|
||||||
|
"service:002:property:004": "Текущее время",
|
||||||
|
"service:002:property:005": "MAC адрес DHCP сервера",
|
||||||
|
"service:003": "Световой индикатор",
|
||||||
|
"service:003:property:001": "Переключатель",
|
||||||
|
"service:004": "Виртуальная служба",
|
||||||
|
"service:004:action:001": "Создать виртуальное событие",
|
||||||
|
"service:004:event:001": "Произошло виртуальное событие",
|
||||||
|
"service:004:property:001": "Название события"
|
||||||
|
},
|
||||||
|
"zh-Hant": {
|
||||||
|
"service:001": "設備信息",
|
||||||
|
"service:001:property:003": "設備ID",
|
||||||
|
"service:001:property:005": "序號 (SN)",
|
||||||
|
"service:002": "網關",
|
||||||
|
"service:002:event:001": "網路發生變化",
|
||||||
|
"service:002:event:002": "網路發生變化",
|
||||||
|
"service:002:property:001": "接入方式",
|
||||||
|
"service:002:property:001:valuelist:000": "有線",
|
||||||
|
"service:002:property:001:valuelist:001": "5G 無線",
|
||||||
|
"service:002:property:001:valuelist:002": "2.4G 無線",
|
||||||
|
"service:002:property:002": "IP地址",
|
||||||
|
"service:002:property:003": "WiFi網路名稱",
|
||||||
|
"service:002:property:004": "當前時間",
|
||||||
|
"service:002:property:005": "DHCP伺服器MAC地址",
|
||||||
|
"service:003": "指示燈",
|
||||||
|
"service:003:property:001": "開關",
|
||||||
|
"service:004": "虛擬服務",
|
||||||
|
"service:004:action:001": "產生虛擬事件",
|
||||||
|
"service:004:event:001": "虛擬事件發生",
|
||||||
|
"service:004:property:001": "事件名稱"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"urn:miot-spec-v2:device:switch:0000A003:lumi-acn040": {
|
||||||
|
"en": {
|
||||||
|
"service:011": "Right Button On and Off",
|
||||||
|
"service:011:property:001": "Right Button On and Off",
|
||||||
|
"service:015:action:001": "Left Button Identify",
|
||||||
|
"service:016:action:001": "Middle Button Identify",
|
||||||
|
"service:017:action:001": "Right Button Identify"
|
||||||
|
},
|
||||||
|
"zh-Hans": {
|
||||||
|
"service:015:action:001": "左键确认",
|
||||||
|
"service:016:action:001": "中键确认",
|
||||||
|
"service:017:action:001": "右键确认"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"urn:miot-spec-v2:device:bath-heater:0000A028:yeelink-v10": {
|
||||||
|
"en": {
|
||||||
|
"service:003:property:001:valuelist:000": "Idle",
|
||||||
|
"service:003:property:001:valuelist:001": "Dry"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"urn:miot-spec-v2:device:plant-monitor:0000A030:hhcc-v1": {
|
||||||
|
"en": {
|
||||||
|
"service:002:property:001": "Soil Moisture"
|
||||||
|
},
|
||||||
|
"zh-Hans": {
|
||||||
|
"service:002:property:001": "土壤湿度",
|
||||||
|
"service:002:property:003": "光照强度"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
184
custom_components/xiaomi_home/miot/specs/spec_modify.yaml
Normal file
184
custom_components/xiaomi_home/miot/specs/spec_modify.yaml
Normal file
@ -0,0 +1,184 @@
|
|||||||
|
urn:miot-spec-v2:device:gateway:0000A019:xiaomi-hub1:1:
|
||||||
|
prop.2.1:
|
||||||
|
name: access-mode
|
||||||
|
access:
|
||||||
|
- read
|
||||||
|
- notify
|
||||||
|
prop.2.2:
|
||||||
|
name: ip-address
|
||||||
|
icon: mdi:ip
|
||||||
|
prop.2.3:
|
||||||
|
name: wifi-ssid
|
||||||
|
access:
|
||||||
|
- read
|
||||||
|
- notify
|
||||||
|
urn:miot-spec-v2:device:gateway:0000A019:xiaomi-hub1:2: urn:miot-spec-v2:device:gateway:0000A019:xiaomi-hub1:1
|
||||||
|
urn:miot-spec-v2:device:gateway:0000A019:xiaomi-hub1:3: urn:miot-spec-v2:device:gateway:0000A019:xiaomi-hub1:1
|
||||||
|
urn:miot-spec-v2:device:gateway:0000A019:lumi-mcn001:1: # lumi.gateway.mcn001
|
||||||
|
prop.2.1: # access-mode
|
||||||
|
access:
|
||||||
|
- read
|
||||||
|
- notify
|
||||||
|
prop.2.2: # ip-address
|
||||||
|
icon: mdi:ip
|
||||||
|
prop.2.3: # wifi-ssid
|
||||||
|
access:
|
||||||
|
- read
|
||||||
|
- notify
|
||||||
|
prop.2.5: # access-mode
|
||||||
|
access:
|
||||||
|
- read
|
||||||
|
- notify
|
||||||
|
urn:miot-spec-v2:device:outlet:0000A002:cuco-cp1md:1:
|
||||||
|
prop.2.2:
|
||||||
|
name: power-consumption
|
||||||
|
expr: round(src_value/1000, 3)
|
||||||
|
urn:miot-spec-v2:device:outlet:0000A002:cuco-v3:1:
|
||||||
|
prop.11.1:
|
||||||
|
name: power-consumption
|
||||||
|
expr: round(src_value/100, 2)
|
||||||
|
urn:miot-spec-v2:device:outlet:0000A002:cuco-v3:2: urn:miot-spec-v2:device:outlet:0000A002:cuco-v3:1
|
||||||
|
urn:miot-spec-v2:device:outlet:0000A002:zimi-zncz01:2:0000C816:
|
||||||
|
prop.3.1:
|
||||||
|
name: electric-power
|
||||||
|
expr: round(src_value/100, 2)
|
||||||
|
urn:miot-spec-v2:device:outlet:0000A002:qmi-psv3:1:0000C816: # qmi.plug.psv3
|
||||||
|
prop.3.3: # voltage
|
||||||
|
unit: mV
|
||||||
|
prop.3.4: # electric-current
|
||||||
|
unit: mA
|
||||||
|
urn:miot-spec-v2:device:motion-sensor:0000A014:lumi-acn001:1: # lumi.motion.acn001
|
||||||
|
prop.3.2: # voltage
|
||||||
|
access:
|
||||||
|
- read
|
||||||
|
- notify
|
||||||
|
unit: mV
|
||||||
|
urn:miot-spec-v2:device:occupancy-sensor:0000A0BF:izq-24:2:0000C824: # izq.sensor_occupy.24
|
||||||
|
prop.2.6: # distance
|
||||||
|
unit: cm
|
||||||
|
urn:miot-spec-v2:device:occupancy-sensor:0000A0BF:linp-hb01:2:0000C824: # linp.sensor_occupy.hb01
|
||||||
|
prop.3.3: # body-distance
|
||||||
|
unit: m
|
||||||
|
urn:miot-spec-v2:device:router:0000A036:xiaomi-rd08:1:
|
||||||
|
prop.2.1:
|
||||||
|
name: download-speed
|
||||||
|
icon: mdi:download
|
||||||
|
unit: B/s
|
||||||
|
prop.2.2:
|
||||||
|
name: upload-speed
|
||||||
|
icon: mdi:upload
|
||||||
|
unit: B/s
|
||||||
|
urn:miot-spec-v2:device:airer:0000A00D:hyd-znlyj5:1:
|
||||||
|
prop.2.3:
|
||||||
|
value-range:
|
||||||
|
- 0
|
||||||
|
- 1
|
||||||
|
- 1
|
||||||
|
urn:miot-spec-v2:device:airer:0000A00D:hyd-znlyj5:2: urn:miot-spec-v2:device:airer:0000A00D:hyd-znlyj5:1
|
||||||
|
urn:miot-spec-v2:device:bath-heater:0000A028:opple-acmoto:1:
|
||||||
|
prop.5.2:
|
||||||
|
value-list:
|
||||||
|
- value: 1
|
||||||
|
description: low
|
||||||
|
- value: 128
|
||||||
|
description: medium
|
||||||
|
- value: 255
|
||||||
|
description: high
|
||||||
|
urn:miot-spec-v2:device:bath-heater:0000A028:mike-2:1:
|
||||||
|
prop.3.1:
|
||||||
|
name: mode-a
|
||||||
|
prop.3.11:
|
||||||
|
name: mode-b
|
||||||
|
prop.3.12:
|
||||||
|
name: mode-c
|
||||||
|
urn:miot-spec-v2:device:fan:0000A005:xiaomi-p51:1:
|
||||||
|
prop.2.2:
|
||||||
|
name: fan-level-a
|
||||||
|
urn:miot-spec-v2:device:air-conditioner:0000A004:xiaomi-m9:6:
|
||||||
|
prop.10.6:
|
||||||
|
unit: none
|
||||||
|
urn:miot-spec-v2:device:air-conditioner:0000A004:xiaomi-m9:1: urn:miot-spec-v2:device:air-conditioner:0000A004:xiaomi-m9:6
|
||||||
|
urn:miot-spec-v2:device:air-conditioner:0000A004:xiaomi-m9:2: urn:miot-spec-v2:device:air-conditioner:0000A004:xiaomi-m9:6
|
||||||
|
urn:miot-spec-v2:device:air-conditioner:0000A004:xiaomi-m9:3: urn:miot-spec-v2:device:air-conditioner:0000A004:xiaomi-m9:6
|
||||||
|
urn:miot-spec-v2:device:air-conditioner:0000A004:xiaomi-m9:4: urn:miot-spec-v2:device:air-conditioner:0000A004:xiaomi-m9:6
|
||||||
|
urn:miot-spec-v2:device:air-conditioner:0000A004:xiaomi-m9:5: urn:miot-spec-v2:device:air-conditioner:0000A004:xiaomi-m9:6
|
||||||
|
urn:miot-spec-v2:device:airer:0000A00D:mrbond-m33a:1:
|
||||||
|
prop.2.3:
|
||||||
|
name: current-position-a
|
||||||
|
prop.2.11:
|
||||||
|
name: current-position-b
|
||||||
|
urn:miot-spec-v2:device:thermostat:0000A031:suittc-wk168:1:
|
||||||
|
prop.2.3:
|
||||||
|
value-list:
|
||||||
|
- value: 1
|
||||||
|
description: '1'
|
||||||
|
- value: 2
|
||||||
|
description: '2'
|
||||||
|
- value: 3
|
||||||
|
description: '3'
|
||||||
|
- value: 4
|
||||||
|
description: '4'
|
||||||
|
- value: 5
|
||||||
|
description: '5'
|
||||||
|
- value: 6
|
||||||
|
description: '6'
|
||||||
|
- value: 7
|
||||||
|
description: '7'
|
||||||
|
- value: 8
|
||||||
|
description: '8'
|
||||||
|
- value: 9
|
||||||
|
description: '9'
|
||||||
|
- value: 10
|
||||||
|
description: '10'
|
||||||
|
- value: 11
|
||||||
|
description: '11'
|
||||||
|
- value: 12
|
||||||
|
description: '12'
|
||||||
|
- value: 13
|
||||||
|
description: '13'
|
||||||
|
- value: 14
|
||||||
|
description: '14'
|
||||||
|
- value: 15
|
||||||
|
description: '15'
|
||||||
|
- value: 16
|
||||||
|
description: '16'
|
||||||
|
urn:miot-spec-v2:device:outlet:0000A002:chuangmi-212a01:3:
|
||||||
|
prop.5.1:
|
||||||
|
expr: round(src_value*6/1000000, 3)
|
||||||
|
urn:miot-spec-v2:device:outlet:0000A002:chuangmi-212a01:1: urn:miot-spec-v2:device:outlet:0000A002:chuangmi-212a01:3
|
||||||
|
urn:miot-spec-v2:device:outlet:0000A002:chuangmi-212a01:2: urn:miot-spec-v2:device:outlet:0000A002:chuangmi-212a01:3
|
||||||
|
urn:miot-spec-v2:device:outlet:0000A002:cuco-cp2:2:
|
||||||
|
prop.2.3:
|
||||||
|
expr: round(src_value/10, 1)
|
||||||
|
prop.2.4:
|
||||||
|
unit: mA
|
||||||
|
prop.3.2:
|
||||||
|
expr: round(src_value/10, 1)
|
||||||
|
urn:miot-spec-v2:device:outlet:0000A002:cuco-cp2:1: urn:miot-spec-v2:device:outlet:0000A002:cuco-cp2:2
|
||||||
|
urn:miot-spec-v2:device:plant-monitor:0000A030:hhcc-v1:1:
|
||||||
|
prop.2.1:
|
||||||
|
name: soil-moisture
|
||||||
|
icon: mdi:watering-can
|
||||||
|
prop.2.2:
|
||||||
|
name: soil-ec
|
||||||
|
icon: mdi:sprout-outline
|
||||||
|
unit: μS/cm
|
||||||
|
urn:miot-spec-v2:device:air-monitor:0000A008:cgllc-s1:1:
|
||||||
|
prop.2.5:
|
||||||
|
name: voc-density
|
||||||
|
urn:miot-spec-v2:device:water-purifier:0000A013:roswan-lte01:1:0000D05A:
|
||||||
|
prop.4.1:
|
||||||
|
unit: ppm
|
||||||
|
prop.4.2:
|
||||||
|
unit: ppm
|
||||||
|
urn:miot-spec-v2:device:water-purifier:0000A013:yunmi-s20:1: # yunmi.waterpuri.s20
|
||||||
|
prop.4.1: # tds-in
|
||||||
|
unit: ppm
|
||||||
|
prop.4.2: # tds-out
|
||||||
|
unit: ppm
|
||||||
|
urn:miot-spec-v2:device:relay:0000A03D:lumi-c2acn01:1:
|
||||||
|
prop.4.1:
|
||||||
|
unit: kWh
|
||||||
|
urn:miot-spec-v2:device:bath-heater:0000A028:xiaomi-s1:1:
|
||||||
|
prop.4.4:
|
||||||
|
name: fan-level-ventilation
|
@ -48,12 +48,19 @@ Conversion rules of MIoT-Spec-V2 instance to Home Assistant entity.
|
|||||||
from homeassistant.components.sensor import SensorDeviceClass
|
from homeassistant.components.sensor import SensorDeviceClass
|
||||||
from homeassistant.components.sensor import SensorStateClass
|
from homeassistant.components.sensor import SensorStateClass
|
||||||
from homeassistant.components.event import EventDeviceClass
|
from homeassistant.components.event import EventDeviceClass
|
||||||
|
from homeassistant.components.binary_sensor import BinarySensorDeviceClass
|
||||||
|
|
||||||
from homeassistant.const import (
|
from homeassistant.const import (
|
||||||
|
CONCENTRATION_MICROGRAMS_PER_CUBIC_METER,
|
||||||
|
EntityCategory,
|
||||||
|
LIGHT_LUX,
|
||||||
UnitOfEnergy,
|
UnitOfEnergy,
|
||||||
UnitOfPower,
|
UnitOfPower,
|
||||||
UnitOfElectricCurrent,
|
UnitOfElectricCurrent,
|
||||||
UnitOfElectricPotential,
|
UnitOfElectricPotential,
|
||||||
|
UnitOfTemperature,
|
||||||
|
UnitOfPressure,
|
||||||
|
PERCENTAGE
|
||||||
)
|
)
|
||||||
|
|
||||||
# pylint: disable=pointless-string-statement
|
# pylint: disable=pointless-string-statement
|
||||||
@ -96,7 +103,7 @@ from homeassistant.const import (
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
"""
|
"""
|
||||||
SPEC_DEVICE_TRANS_MAP: dict[str, dict | str] = {
|
SPEC_DEVICE_TRANS_MAP: dict = {
|
||||||
'humidifier': {
|
'humidifier': {
|
||||||
'required': {
|
'required': {
|
||||||
'humidifier': {
|
'humidifier': {
|
||||||
@ -132,7 +139,7 @@ SPEC_DEVICE_TRANS_MAP: dict[str, dict | str] = {
|
|||||||
'optional': {
|
'optional': {
|
||||||
'properties': {'mode', 'target-humidity'}
|
'properties': {'mode', 'target-humidity'}
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
},
|
},
|
||||||
'optional': {
|
'optional': {
|
||||||
'environment': {
|
'environment': {
|
||||||
@ -158,8 +165,7 @@ SPEC_DEVICE_TRANS_MAP: dict[str, dict | str] = {
|
|||||||
'continue-sweep',
|
'continue-sweep',
|
||||||
'stop-and-gocharge'
|
'stop-and-gocharge'
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
'optional': {
|
'optional': {
|
||||||
@ -172,9 +178,9 @@ SPEC_DEVICE_TRANS_MAP: dict[str, dict | str] = {
|
|||||||
'required': {
|
'required': {
|
||||||
'properties': {
|
'properties': {
|
||||||
'battery-level': {'read'}
|
'battery-level': {'read'}
|
||||||
},
|
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
|
}
|
||||||
},
|
},
|
||||||
'entity': 'vacuum'
|
'entity': 'vacuum'
|
||||||
},
|
},
|
||||||
@ -190,7 +196,7 @@ SPEC_DEVICE_TRANS_MAP: dict[str, dict | str] = {
|
|||||||
},
|
},
|
||||||
'optional': {
|
'optional': {
|
||||||
'properties': {'target-humidity'}
|
'properties': {'target-humidity'}
|
||||||
},
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
'optional': {
|
'optional': {
|
||||||
@ -219,6 +225,31 @@ SPEC_DEVICE_TRANS_MAP: dict[str, dict | str] = {
|
|||||||
'entity': 'air-conditioner'
|
'entity': 'air-conditioner'
|
||||||
},
|
},
|
||||||
'air-condition-outlet': 'air-conditioner',
|
'air-condition-outlet': 'air-conditioner',
|
||||||
|
'thermostat': {
|
||||||
|
'required': {
|
||||||
|
'thermostat': {
|
||||||
|
'required': {
|
||||||
|
'properties': {
|
||||||
|
'on': {'read', 'write'}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
'optional': {
|
||||||
|
'properties': {
|
||||||
|
'target-temperature', 'mode', 'fan-level',
|
||||||
|
'temperature'}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
'optional': {
|
||||||
|
'environment': {
|
||||||
|
'required': {},
|
||||||
|
'optional': {
|
||||||
|
'properties': {'temperature', 'relative-humidity'}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
'entity': 'thermostat'
|
||||||
|
},
|
||||||
'heater': {
|
'heater': {
|
||||||
'required': {
|
'required': {
|
||||||
'heater': {
|
'heater': {
|
||||||
@ -229,7 +260,7 @@ SPEC_DEVICE_TRANS_MAP: dict[str, dict | str] = {
|
|||||||
},
|
},
|
||||||
'optional': {
|
'optional': {
|
||||||
'properties': {'target-temperature', 'heat-level'}
|
'properties': {'target-temperature', 'heat-level'}
|
||||||
},
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
'optional': {
|
'optional': {
|
||||||
@ -238,10 +269,58 @@ SPEC_DEVICE_TRANS_MAP: dict[str, dict | str] = {
|
|||||||
'optional': {
|
'optional': {
|
||||||
'properties': {'temperature', 'relative-humidity'}
|
'properties': {'temperature', 'relative-humidity'}
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
},
|
},
|
||||||
'entity': 'heater'
|
'entity': 'heater'
|
||||||
|
},
|
||||||
|
'bath-heater': {
|
||||||
|
'required': {
|
||||||
|
'ptc-bath-heater': {
|
||||||
|
'required': {
|
||||||
|
'properties': {
|
||||||
|
'mode':{'read', 'write'}
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
'optional': {
|
||||||
|
'properties': {'target-temperature', 'temperature'}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
'optional': {
|
||||||
|
'fan-control': {
|
||||||
|
'required': {},
|
||||||
|
'optional': {
|
||||||
|
'properties': {
|
||||||
|
'on', 'fan-level', 'horizontal-swing', 'vertical-swing'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
'environment': {
|
||||||
|
'required': {},
|
||||||
|
'optional': {
|
||||||
|
'properties': {'temperature'}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
'entity': 'bath-heater',
|
||||||
|
},
|
||||||
|
'electric-blanket': {
|
||||||
|
'required': {
|
||||||
|
'electric-blanket': {
|
||||||
|
'required': {
|
||||||
|
'properties': {
|
||||||
|
'on': {'read', 'write'},
|
||||||
|
'target-temperature': {'read', 'write'}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
'optional': {
|
||||||
|
'properties': {'mode', 'temperature'}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
'optional': {},
|
||||||
|
'entity': 'electric-blanket'
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
"""SPEC_SERVICE_TRANS_MAP
|
"""SPEC_SERVICE_TRANS_MAP
|
||||||
@ -259,11 +338,12 @@ SPEC_DEVICE_TRANS_MAP: dict[str, dict | str] = {
|
|||||||
'events': set<event instance name: str>,
|
'events': set<event instance name: str>,
|
||||||
'actions': set<action instance name: str>
|
'actions': set<action instance name: str>
|
||||||
},
|
},
|
||||||
'entity': str
|
'entity': str,
|
||||||
|
'entity_category'?: str
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
"""
|
"""
|
||||||
SPEC_SERVICE_TRANS_MAP: dict[str, dict | str] = {
|
SPEC_SERVICE_TRANS_MAP: dict = {
|
||||||
'light': {
|
'light': {
|
||||||
'required': {
|
'required': {
|
||||||
'properties': {
|
'properties': {
|
||||||
@ -277,10 +357,23 @@ SPEC_SERVICE_TRANS_MAP: dict[str, dict | str] = {
|
|||||||
},
|
},
|
||||||
'entity': 'light'
|
'entity': 'light'
|
||||||
},
|
},
|
||||||
'indicator-light': 'light',
|
|
||||||
'ambient-light': 'light',
|
'ambient-light': 'light',
|
||||||
'night-light': 'light',
|
'night-light': 'light',
|
||||||
'white-light': 'light',
|
'white-light': 'light',
|
||||||
|
'indicator-light': {
|
||||||
|
'required': {
|
||||||
|
'properties': {
|
||||||
|
'on': {'read', 'write'}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
'optional': {
|
||||||
|
'properties': {
|
||||||
|
'mode', 'brightness',
|
||||||
|
}
|
||||||
|
},
|
||||||
|
'entity': 'light',
|
||||||
|
'entity_category': EntityCategory.CONFIG
|
||||||
|
},
|
||||||
'fan': {
|
'fan': {
|
||||||
'required': {
|
'required': {
|
||||||
'properties': {
|
'properties': {
|
||||||
@ -295,6 +388,7 @@ SPEC_SERVICE_TRANS_MAP: dict[str, dict | str] = {
|
|||||||
},
|
},
|
||||||
'fan-control': 'fan',
|
'fan-control': 'fan',
|
||||||
'ceiling-fan': 'fan',
|
'ceiling-fan': 'fan',
|
||||||
|
'air-fresh': 'fan',
|
||||||
'water-heater': {
|
'water-heater': {
|
||||||
'required': {
|
'required': {
|
||||||
'properties': {
|
'properties': {
|
||||||
@ -314,12 +408,27 @@ SPEC_SERVICE_TRANS_MAP: dict[str, dict | str] = {
|
|||||||
},
|
},
|
||||||
'optional': {
|
'optional': {
|
||||||
'properties': {
|
'properties': {
|
||||||
'motor-control', 'status', 'current-position', 'target-position'
|
'status', 'current-position', 'target-position'
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
'entity': 'cover'
|
'entity': 'cover'
|
||||||
},
|
},
|
||||||
'window-opener': 'curtain'
|
'window-opener': 'curtain',
|
||||||
|
'motor-controller': 'curtain',
|
||||||
|
'airer': 'curtain',
|
||||||
|
'air-conditioner': {
|
||||||
|
'required': {
|
||||||
|
'properties': {
|
||||||
|
'on': {'read', 'write'},
|
||||||
|
'mode': {'read', 'write'},
|
||||||
|
'target-temperature': {'read', 'write'}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
'optional': {
|
||||||
|
'properties': {'target-humidity'}
|
||||||
|
},
|
||||||
|
'entity': 'air-conditioner'
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
"""SPEC_PROP_TRANS_MAP
|
"""SPEC_PROP_TRANS_MAP
|
||||||
@ -334,121 +443,145 @@ SPEC_SERVICE_TRANS_MAP: dict[str, dict | str] = {
|
|||||||
'<property instance name>':{
|
'<property instance name>':{
|
||||||
'device_class': str,
|
'device_class': str,
|
||||||
'entity': str,
|
'entity': str,
|
||||||
'optional':{
|
'state_class'?: str,
|
||||||
'state_class': str,
|
'unit_of_measurement'?: str
|
||||||
'unit_of_measurement': str
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
"""
|
"""
|
||||||
SPEC_PROP_TRANS_MAP: dict[str, dict | str] = {
|
SPEC_PROP_TRANS_MAP: dict = {
|
||||||
'entities': {
|
'entities': {
|
||||||
'sensor': {
|
'sensor': {
|
||||||
'format': {'int', 'float'},
|
'format': {'int', 'float'},
|
||||||
'access': {'read'}
|
'access': {'read'}
|
||||||
},
|
},
|
||||||
|
'binary_sensor': {
|
||||||
|
'format': {'bool', 'int'},
|
||||||
|
'access': {'read'}
|
||||||
|
},
|
||||||
'switch': {
|
'switch': {
|
||||||
'format': {'bool'},
|
'format': {'bool'},
|
||||||
'access': {'read', 'write'}
|
'access': {'read', 'write'}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
'properties': {
|
'properties': {
|
||||||
|
'submersion-state': {
|
||||||
|
'device_class': BinarySensorDeviceClass.MOISTURE,
|
||||||
|
'entity': 'binary_sensor'
|
||||||
|
},
|
||||||
|
'contact-state': {
|
||||||
|
'device_class': BinarySensorDeviceClass.DOOR,
|
||||||
|
'entity': 'binary_sensor'
|
||||||
|
},
|
||||||
|
'occupancy-status': {
|
||||||
|
'device_class': BinarySensorDeviceClass.OCCUPANCY,
|
||||||
|
'entity': 'binary_sensor',
|
||||||
|
},
|
||||||
'temperature': {
|
'temperature': {
|
||||||
'device_class': SensorDeviceClass.TEMPERATURE,
|
'device_class': SensorDeviceClass.TEMPERATURE,
|
||||||
'entity': 'sensor'
|
'entity': 'sensor',
|
||||||
|
'state_class': SensorStateClass.MEASUREMENT,
|
||||||
|
'unit_of_measurement': UnitOfTemperature.CELSIUS
|
||||||
},
|
},
|
||||||
'relative-humidity': {
|
'relative-humidity': {
|
||||||
'device_class': SensorDeviceClass.HUMIDITY,
|
'device_class': SensorDeviceClass.HUMIDITY,
|
||||||
'entity': 'sensor'
|
'entity': 'sensor',
|
||||||
|
'state_class': SensorStateClass.MEASUREMENT,
|
||||||
|
'unit_of_measurement': PERCENTAGE
|
||||||
},
|
},
|
||||||
'air-quality-index': {
|
'air-quality-index': {
|
||||||
'device_class': SensorDeviceClass.AQI,
|
'device_class': SensorDeviceClass.AQI,
|
||||||
'entity': 'sensor'
|
'entity': 'sensor',
|
||||||
|
'state_class': SensorStateClass.MEASUREMENT,
|
||||||
},
|
},
|
||||||
'pm2.5-density': {
|
'pm2.5-density': {
|
||||||
'device_class': SensorDeviceClass.PM25,
|
'device_class': SensorDeviceClass.PM25,
|
||||||
'entity': 'sensor'
|
'entity': 'sensor',
|
||||||
|
'state_class': SensorStateClass.MEASUREMENT,
|
||||||
|
'unit_of_measurement': CONCENTRATION_MICROGRAMS_PER_CUBIC_METER
|
||||||
},
|
},
|
||||||
'pm10-density': {
|
'pm10-density': {
|
||||||
'device_class': SensorDeviceClass.PM10,
|
'device_class': SensorDeviceClass.PM10,
|
||||||
'entity': 'sensor'
|
'entity': 'sensor',
|
||||||
|
'state_class': SensorStateClass.MEASUREMENT,
|
||||||
|
'unit_of_measurement': CONCENTRATION_MICROGRAMS_PER_CUBIC_METER
|
||||||
},
|
},
|
||||||
'pm1': {
|
'pm1': {
|
||||||
'device_class': SensorDeviceClass.PM1,
|
'device_class': SensorDeviceClass.PM1,
|
||||||
'entity': 'sensor'
|
'entity': 'sensor',
|
||||||
|
'state_class': SensorStateClass.MEASUREMENT,
|
||||||
|
'unit_of_measurement': CONCENTRATION_MICROGRAMS_PER_CUBIC_METER
|
||||||
},
|
},
|
||||||
'atmospheric-pressure': {
|
'atmospheric-pressure': {
|
||||||
'device_class': SensorDeviceClass.ATMOSPHERIC_PRESSURE,
|
'device_class': SensorDeviceClass.ATMOSPHERIC_PRESSURE,
|
||||||
'entity': 'sensor'
|
'entity': 'sensor',
|
||||||
|
'state_class': SensorStateClass.MEASUREMENT,
|
||||||
|
'unit_of_measurement': UnitOfPressure.PA
|
||||||
},
|
},
|
||||||
'tvoc-density': {
|
'tvoc-density': {
|
||||||
'device_class': SensorDeviceClass.VOLATILE_ORGANIC_COMPOUNDS,
|
'device_class': SensorDeviceClass.VOLATILE_ORGANIC_COMPOUNDS,
|
||||||
'entity': 'sensor'
|
'entity': 'sensor',
|
||||||
|
'state_class': SensorStateClass.MEASUREMENT
|
||||||
|
},
|
||||||
|
'voc-density': {
|
||||||
|
'device_class': SensorDeviceClass.VOLATILE_ORGANIC_COMPOUNDS_PARTS,
|
||||||
|
'entity': 'sensor',
|
||||||
|
'state_class': SensorStateClass.MEASUREMENT
|
||||||
},
|
},
|
||||||
'voc-density': 'tvoc-density',
|
|
||||||
'battery-level': {
|
'battery-level': {
|
||||||
'device_class': SensorDeviceClass.BATTERY,
|
'device_class': SensorDeviceClass.BATTERY,
|
||||||
'entity': 'sensor'
|
'entity': 'sensor',
|
||||||
|
'state_class': SensorStateClass.MEASUREMENT,
|
||||||
|
'unit_of_measurement': PERCENTAGE
|
||||||
},
|
},
|
||||||
'voltage': {
|
'voltage': {
|
||||||
'device_class': SensorDeviceClass.VOLTAGE,
|
'device_class': SensorDeviceClass.VOLTAGE,
|
||||||
'entity': 'sensor',
|
'entity': 'sensor',
|
||||||
'optional': {
|
|
||||||
'state_class': SensorStateClass.MEASUREMENT,
|
'state_class': SensorStateClass.MEASUREMENT,
|
||||||
'unit_of_measurement': UnitOfElectricPotential.VOLT
|
'unit_of_measurement': UnitOfElectricPotential.VOLT
|
||||||
}
|
},
|
||||||
|
'electric-current': {
|
||||||
|
'device_class': SensorDeviceClass.CURRENT,
|
||||||
|
'entity': 'sensor',
|
||||||
|
'state_class': SensorStateClass.MEASUREMENT,
|
||||||
|
'unit_of_measurement': UnitOfElectricCurrent.AMPERE
|
||||||
},
|
},
|
||||||
'illumination': {
|
'illumination': {
|
||||||
'device_class': SensorDeviceClass.ILLUMINANCE,
|
'device_class': SensorDeviceClass.ILLUMINANCE,
|
||||||
'entity': 'sensor'
|
'entity': 'sensor',
|
||||||
|
'state_class': SensorStateClass.MEASUREMENT,
|
||||||
|
'unit_of_measurement': LIGHT_LUX
|
||||||
},
|
},
|
||||||
'no-one-determine-time': {
|
'no-one-determine-time': {
|
||||||
'device_class': SensorDeviceClass.DURATION,
|
'device_class': SensorDeviceClass.DURATION,
|
||||||
'entity': 'sensor'
|
'entity': 'sensor'
|
||||||
},
|
},
|
||||||
|
'has-someone-duration': 'no-one-determine-time',
|
||||||
|
'no-one-duration': 'no-one-determine-time',
|
||||||
'electric-power': {
|
'electric-power': {
|
||||||
'device_class': SensorDeviceClass.POWER,
|
'device_class': SensorDeviceClass.POWER,
|
||||||
'entity': 'sensor',
|
'entity': 'sensor',
|
||||||
'optional': {
|
|
||||||
'state_class': SensorStateClass.MEASUREMENT,
|
'state_class': SensorStateClass.MEASUREMENT,
|
||||||
'unit_of_measurement': UnitOfPower.WATT
|
'unit_of_measurement': UnitOfPower.WATT
|
||||||
}
|
|
||||||
},
|
},
|
||||||
'electric-current': {
|
'surge-power': {
|
||||||
'device_class': SensorDeviceClass.CURRENT,
|
'device_class': SensorDeviceClass.POWER,
|
||||||
'entity': 'sensor',
|
'entity': 'sensor',
|
||||||
'optional': {
|
|
||||||
'state_class': SensorStateClass.MEASUREMENT,
|
'state_class': SensorStateClass.MEASUREMENT,
|
||||||
'unit_of_measurement': UnitOfElectricCurrent.AMPERE
|
'unit_of_measurement': UnitOfPower.WATT
|
||||||
}
|
|
||||||
},
|
},
|
||||||
'power-consumption': {
|
'power-consumption': {
|
||||||
'device_class': SensorDeviceClass.ENERGY,
|
'device_class': SensorDeviceClass.ENERGY,
|
||||||
'entity': 'sensor',
|
'entity': 'sensor',
|
||||||
'optional': {
|
|
||||||
'state_class': SensorStateClass.TOTAL_INCREASING,
|
'state_class': SensorStateClass.TOTAL_INCREASING,
|
||||||
'unit_of_measurement': UnitOfEnergy.KILO_WATT_HOUR
|
'unit_of_measurement': UnitOfEnergy.KILO_WATT_HOUR
|
||||||
}
|
|
||||||
},
|
},
|
||||||
'power': {
|
'power': {
|
||||||
'device_class': SensorDeviceClass.POWER,
|
'device_class': SensorDeviceClass.POWER,
|
||||||
'entity': 'sensor',
|
'entity': 'sensor',
|
||||||
'optional': {
|
|
||||||
'state_class': SensorStateClass.MEASUREMENT,
|
'state_class': SensorStateClass.MEASUREMENT,
|
||||||
'unit_of_measurement': UnitOfPower.WATT
|
'unit_of_measurement': UnitOfPower.WATT
|
||||||
}
|
}
|
||||||
},
|
|
||||||
'total-battery': {
|
|
||||||
'device_class': SensorDeviceClass.ENERGY,
|
|
||||||
'entity': 'sensor',
|
|
||||||
'optional': {
|
|
||||||
'state_class': SensorStateClass.TOTAL_INCREASING,
|
|
||||||
'unit_of_measurement': UnitOfEnergy.KILO_WATT_HOUR
|
|
||||||
}
|
|
||||||
},
|
|
||||||
'has-someone-duration': 'no-one-determine-time',
|
|
||||||
'no-one-duration': 'no-one-determine-time'
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -88,7 +88,7 @@ class Number(MIoTPropertyEntity, NumberEntity):
|
|||||||
if self.spec.external_unit:
|
if self.spec.external_unit:
|
||||||
self._attr_native_unit_of_measurement = self.spec.external_unit
|
self._attr_native_unit_of_measurement = self.spec.external_unit
|
||||||
# Set icon
|
# Set icon
|
||||||
if self.spec.icon:
|
if self.spec.icon and not self.device_class:
|
||||||
self._attr_icon = self.spec.icon
|
self._attr_icon = self.spec.icon
|
||||||
# Set value range
|
# Set value range
|
||||||
if self._value_range:
|
if self._value_range:
|
||||||
|
@ -95,7 +95,7 @@ class Sensor(MIoTPropertyEntity, SensorEntity):
|
|||||||
# Set device_class
|
# Set device_class
|
||||||
if self._value_list:
|
if self._value_list:
|
||||||
self._attr_device_class = SensorDeviceClass.ENUM
|
self._attr_device_class = SensorDeviceClass.ENUM
|
||||||
self._attr_icon = 'mdi:message-text'
|
self._attr_icon = 'mdi:format-text'
|
||||||
self._attr_native_unit_of_measurement = None
|
self._attr_native_unit_of_measurement = None
|
||||||
self._attr_options = self._value_list.descriptions
|
self._attr_options = self._value_list.descriptions
|
||||||
else:
|
else:
|
||||||
@ -109,11 +109,14 @@ class Sensor(MIoTPropertyEntity, SensorEntity):
|
|||||||
self._attr_device_class, None) # type: ignore
|
self._attr_device_class, None) # type: ignore
|
||||||
self._attr_native_unit_of_measurement = list(
|
self._attr_native_unit_of_measurement = list(
|
||||||
unit_sets)[0] if unit_sets else None
|
unit_sets)[0] if unit_sets else None
|
||||||
|
# Set suggested precision
|
||||||
|
if spec.format_ in {int, float} and spec.expr is None:
|
||||||
|
self._attr_suggested_display_precision = spec.precision
|
||||||
# Set state_class
|
# Set state_class
|
||||||
if spec.state_class:
|
if spec.state_class:
|
||||||
self._attr_state_class = spec.state_class
|
self._attr_state_class = spec.state_class
|
||||||
# Set icon
|
# Set icon
|
||||||
if spec.icon:
|
if spec.icon and not self.device_class:
|
||||||
self._attr_icon = spec.icon
|
self._attr_icon = spec.icon
|
||||||
|
|
||||||
@property
|
@property
|
||||||
|
@ -100,7 +100,7 @@ class WaterHeater(MIoTServiceEntity, WaterHeaterEntity):
|
|||||||
) -> None:
|
) -> None:
|
||||||
"""Initialize the Water heater."""
|
"""Initialize the Water heater."""
|
||||||
super().__init__(miot_device=miot_device, entity_data=entity_data)
|
super().__init__(miot_device=miot_device, entity_data=entity_data)
|
||||||
self._attr_temperature_unit = None
|
self._attr_temperature_unit = None # type: ignore
|
||||||
self._attr_supported_features = WaterHeaterEntityFeature(0)
|
self._attr_supported_features = WaterHeaterEntityFeature(0)
|
||||||
self._prop_on = None
|
self._prop_on = None
|
||||||
self._prop_temp = None
|
self._prop_temp = None
|
||||||
@ -112,20 +112,20 @@ class WaterHeater(MIoTServiceEntity, WaterHeaterEntity):
|
|||||||
for prop in entity_data.props:
|
for prop in entity_data.props:
|
||||||
# on
|
# on
|
||||||
if prop.name == 'on':
|
if prop.name == 'on':
|
||||||
|
self._attr_supported_features |= WaterHeaterEntityFeature.ON_OFF
|
||||||
self._prop_on = prop
|
self._prop_on = prop
|
||||||
# temperature
|
# temperature
|
||||||
if prop.name == 'temperature':
|
if prop.name == 'temperature':
|
||||||
if prop.value_range:
|
if not prop.value_range:
|
||||||
if (
|
|
||||||
self._attr_temperature_unit is None
|
|
||||||
and prop.external_unit
|
|
||||||
):
|
|
||||||
self._attr_temperature_unit = prop.external_unit
|
|
||||||
self._prop_temp = prop
|
|
||||||
else:
|
|
||||||
_LOGGER.error(
|
_LOGGER.error(
|
||||||
'invalid temperature value_range format, %s',
|
'invalid temperature value_range format, %s',
|
||||||
self.entity_id)
|
self.entity_id)
|
||||||
|
continue
|
||||||
|
if prop.external_unit:
|
||||||
|
self._attr_temperature_unit = prop.external_unit
|
||||||
|
self._attr_min_temp = prop.value_range.min_
|
||||||
|
self._attr_max_temp = prop.value_range.max_
|
||||||
|
self._prop_temp = prop
|
||||||
# target-temperature
|
# target-temperature
|
||||||
if prop.name == 'target-temperature':
|
if prop.name == 'target-temperature':
|
||||||
if not prop.value_range:
|
if not prop.value_range:
|
||||||
@ -133,8 +133,8 @@ class WaterHeater(MIoTServiceEntity, WaterHeaterEntity):
|
|||||||
'invalid target-temperature value_range format, %s',
|
'invalid target-temperature value_range format, %s',
|
||||||
self.entity_id)
|
self.entity_id)
|
||||||
continue
|
continue
|
||||||
self._attr_min_temp = prop.value_range.min_
|
self._attr_target_temperature_low = prop.value_range.min_
|
||||||
self._attr_max_temp = prop.value_range.max_
|
self._attr_target_temperature_high = prop.value_range.max_
|
||||||
self._attr_precision = prop.value_range.step
|
self._attr_precision = prop.value_range.step
|
||||||
if self._attr_temperature_unit is None and prop.external_unit:
|
if self._attr_temperature_unit is None and prop.external_unit:
|
||||||
self._attr_temperature_unit = prop.external_unit
|
self._attr_temperature_unit = prop.external_unit
|
||||||
@ -166,6 +166,8 @@ class WaterHeater(MIoTServiceEntity, WaterHeaterEntity):
|
|||||||
|
|
||||||
async def async_set_temperature(self, **kwargs: Any) -> None:
|
async def async_set_temperature(self, **kwargs: Any) -> None:
|
||||||
"""Set the temperature the water heater should heat water to."""
|
"""Set the temperature the water heater should heat water to."""
|
||||||
|
if not self._prop_target_temp:
|
||||||
|
return
|
||||||
await self.set_property_async(
|
await self.set_property_async(
|
||||||
prop=self._prop_target_temp, value=kwargs[ATTR_TEMPERATURE])
|
prop=self._prop_target_temp, value=kwargs[ATTR_TEMPERATURE])
|
||||||
|
|
||||||
@ -181,16 +183,11 @@ class WaterHeater(MIoTServiceEntity, WaterHeaterEntity):
|
|||||||
return
|
return
|
||||||
if self.get_prop_value(prop=self._prop_on) is False:
|
if self.get_prop_value(prop=self._prop_on) is False:
|
||||||
await self.set_property_async(
|
await self.set_property_async(
|
||||||
prop=self._prop_on, value=True, update=False)
|
prop=self._prop_on, value=True, write_ha_state=False)
|
||||||
await self.set_property_async(
|
await self.set_property_async(
|
||||||
prop=self._prop_mode,
|
prop=self._prop_mode,
|
||||||
value=self.get_map_key(
|
value=self.get_map_key(
|
||||||
map_=self._mode_map,
|
map_=self._mode_map, value=operation_mode))
|
||||||
value=operation_mode))
|
|
||||||
|
|
||||||
async def async_turn_away_mode_on(self) -> None:
|
|
||||||
"""Set the water heater to away mode."""
|
|
||||||
await self.hass.async_add_executor_job(self.turn_away_mode_on)
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def current_temperature(self) -> Optional[float]:
|
def current_temperature(self) -> Optional[float]:
|
||||||
@ -200,6 +197,8 @@ class WaterHeater(MIoTServiceEntity, WaterHeaterEntity):
|
|||||||
@property
|
@property
|
||||||
def target_temperature(self) -> Optional[float]:
|
def target_temperature(self) -> Optional[float]:
|
||||||
"""Return the target temperature."""
|
"""Return the target temperature."""
|
||||||
|
if not self._prop_target_temp:
|
||||||
|
return None
|
||||||
return self.get_prop_value(prop=self._prop_target_temp)
|
return self.get_prop_value(prop=self._prop_target_temp)
|
||||||
|
|
||||||
@property
|
@property
|
||||||
|
@ -98,6 +98,8 @@ footer :(可选)关联的 issue 或 pull request 编号。
|
|||||||
|
|
||||||
在为本项目做出贡献时,您同意您的贡献遵循本项目的[许可证](../LICENSE.md) 。
|
在为本项目做出贡献时,您同意您的贡献遵循本项目的[许可证](../LICENSE.md) 。
|
||||||
|
|
||||||
|
当您第一次提交拉取请求时,GitHub Action 会提示您签署贡献者许可协议(Contributor License Agreement,CLA)。只有签署了 CLA ,本项目才会合入您的拉取请求。
|
||||||
|
|
||||||
## 获取帮助
|
## 获取帮助
|
||||||
|
|
||||||
如果您需要帮助或有疑问,可在 GitHub 的[讨论区](https://github.com/XiaoMi/ha_xiaomi_home/discussions/)询问。
|
如果您需要帮助或有疑问,可在 GitHub 的[讨论区](https://github.com/XiaoMi/ha_xiaomi_home/discussions/)询问。
|
||||||
|
@ -376,7 +376,7 @@ siid、piid、eiid、aiid、value 均为十进制三位整数。
|
|||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
> 在 Home Assistant 中修改了 `custom_components/xiaomi_home/translations/` 路径下的 `specv2entity.py`、`spec_filter.json`、`multi_lang.json` 文件的内容,需要在集成配置中更新实体转换规则才能生效。方法:[设置 > 设备与服务 > 已配置 > Xiaomi Home](https://my.home-assistant.io/redirect/integration/?domain=xiaomi_home) > 配置 > 更新实体转换规则
|
> 在 Home Assistant 中修改了 `custom_components/xiaomi_home/miot/specs` 路径下的 `specv2entity.py`、`spec_filter.json`、`multi_lang.json` 文件的内容,需要在集成配置中更新实体转换规则才能生效。方法:[设置 > 设备与服务 > 已配置 > Xiaomi Home](https://my.home-assistant.io/redirect/integration/?domain=xiaomi_home) > 配置 > 更新实体转换规则
|
||||||
|
|
||||||
## 文档
|
## 文档
|
||||||
|
|
||||||
|
@ -20,6 +20,9 @@ SPEC_BOOL_TRANS_FILE = path.join(
|
|||||||
SPEC_FILTER_FILE = path.join(
|
SPEC_FILTER_FILE = path.join(
|
||||||
ROOT_PATH,
|
ROOT_PATH,
|
||||||
'../custom_components/xiaomi_home/miot/specs/spec_filter.yaml')
|
'../custom_components/xiaomi_home/miot/specs/spec_filter.yaml')
|
||||||
|
SPEC_MODIFY_FILE = path.join(
|
||||||
|
ROOT_PATH,
|
||||||
|
'../custom_components/xiaomi_home/miot/specs/spec_modify.yaml')
|
||||||
|
|
||||||
|
|
||||||
def load_json_file(file_path: str) -> Optional[dict]:
|
def load_json_file(file_path: str) -> Optional[dict]:
|
||||||
@ -54,7 +57,8 @@ def load_yaml_file(file_path: str) -> Optional[dict]:
|
|||||||
def save_yaml_file(file_path: str, data: dict) -> None:
|
def save_yaml_file(file_path: str, data: dict) -> None:
|
||||||
with open(file_path, 'w', encoding='utf-8') as file:
|
with open(file_path, 'w', encoding='utf-8') as file:
|
||||||
yaml.safe_dump(
|
yaml.safe_dump(
|
||||||
data, file, default_flow_style=False, allow_unicode=True, indent=2)
|
data, file, default_flow_style=False,
|
||||||
|
allow_unicode=True, indent=2, sort_keys=False)
|
||||||
|
|
||||||
|
|
||||||
def dict_str_str(d: dict) -> bool:
|
def dict_str_str(d: dict) -> bool:
|
||||||
@ -135,6 +139,21 @@ def bool_trans(d: dict) -> bool:
|
|||||||
return True
|
return True
|
||||||
|
|
||||||
|
|
||||||
|
def spec_modify(data: dict) -> bool:
|
||||||
|
"""dict[str, str | dict[str, dict]]"""
|
||||||
|
if not isinstance(data, dict):
|
||||||
|
return False
|
||||||
|
for urn, content in data.items():
|
||||||
|
if not isinstance(urn, str) or not isinstance(content, (dict, str)):
|
||||||
|
return False
|
||||||
|
if isinstance(content, str):
|
||||||
|
continue
|
||||||
|
for key, value in content.items():
|
||||||
|
if not isinstance(key, str) or not isinstance(value, dict):
|
||||||
|
return False
|
||||||
|
return True
|
||||||
|
|
||||||
|
|
||||||
def compare_dict_structure(dict1: dict, dict2: dict) -> bool:
|
def compare_dict_structure(dict1: dict, dict2: dict) -> bool:
|
||||||
if not isinstance(dict1, dict) or not isinstance(dict2, dict):
|
if not isinstance(dict1, dict) or not isinstance(dict2, dict):
|
||||||
_LOGGER.info('invalid type')
|
_LOGGER.info('invalid type')
|
||||||
@ -181,6 +200,12 @@ def sort_spec_filter(file_path: str):
|
|||||||
return filter_data
|
return filter_data
|
||||||
|
|
||||||
|
|
||||||
|
def sort_spec_modify(file_path: str):
|
||||||
|
filter_data = load_yaml_file(file_path=file_path)
|
||||||
|
assert isinstance(filter_data, dict), f'{file_path} format error'
|
||||||
|
return dict(sorted(filter_data.items()))
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.github
|
@pytest.mark.github
|
||||||
def test_bool_trans():
|
def test_bool_trans():
|
||||||
data = load_yaml_file(SPEC_BOOL_TRANS_FILE)
|
data = load_yaml_file(SPEC_BOOL_TRANS_FILE)
|
||||||
@ -197,6 +222,14 @@ def test_spec_filter():
|
|||||||
assert spec_filter(data), f'{SPEC_FILTER_FILE} format error'
|
assert spec_filter(data), f'{SPEC_FILTER_FILE} format error'
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.github
|
||||||
|
def test_spec_modify():
|
||||||
|
data = load_yaml_file(SPEC_MODIFY_FILE)
|
||||||
|
assert isinstance(data, dict)
|
||||||
|
assert data, f'load {SPEC_MODIFY_FILE} failed'
|
||||||
|
assert spec_modify(data), f'{SPEC_MODIFY_FILE} format error'
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.github
|
@pytest.mark.github
|
||||||
def test_miot_i18n():
|
def test_miot_i18n():
|
||||||
for file_name in listdir(MIOT_I18N_RELATIVE_PATH):
|
for file_name in listdir(MIOT_I18N_RELATIVE_PATH):
|
||||||
@ -286,3 +319,6 @@ def test_sort_spec_data():
|
|||||||
sort_data = sort_spec_filter(file_path=SPEC_FILTER_FILE)
|
sort_data = sort_spec_filter(file_path=SPEC_FILTER_FILE)
|
||||||
save_yaml_file(file_path=SPEC_FILTER_FILE, data=sort_data)
|
save_yaml_file(file_path=SPEC_FILTER_FILE, data=sort_data)
|
||||||
_LOGGER.info('%s formatted.', SPEC_FILTER_FILE)
|
_LOGGER.info('%s formatted.', SPEC_FILTER_FILE)
|
||||||
|
sort_data = sort_spec_modify(file_path=SPEC_MODIFY_FILE)
|
||||||
|
save_yaml_file(file_path=SPEC_MODIFY_FILE, data=sort_data)
|
||||||
|
_LOGGER.info('%s formatted.', SPEC_MODIFY_FILE)
|
||||||
|
@ -56,7 +56,7 @@ async def test_lan_async(test_devices: dict):
|
|||||||
|
|
||||||
# Your central hub gateway did
|
# Your central hub gateway did
|
||||||
test_did = '111111'
|
test_did = '111111'
|
||||||
# Your central hub gateway did
|
# Your central hub gateway token
|
||||||
test_token = '11223344556677d9a03d43936fc384205'
|
test_token = '11223344556677d9a03d43936fc384205'
|
||||||
test_model = 'xiaomi.gateway.hub1'
|
test_model = 'xiaomi.gateway.hub1'
|
||||||
# Your computer interface list, such as enp3s0, wlp5s0
|
# Your computer interface list, such as enp3s0, wlp5s0
|
||||||
@ -152,3 +152,5 @@ async def test_lan_async(test_devices: dict):
|
|||||||
await asyncio.sleep(0.2)
|
await asyncio.sleep(0.2)
|
||||||
|
|
||||||
await miot_lan.deinit_async()
|
await miot_lan.deinit_async()
|
||||||
|
await mips_service.deinit_async()
|
||||||
|
await miot_network.deinit_async()
|
||||||
|
Loading…
x
Reference in New Issue
Block a user