Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: user defined custom service #284

Draft
wants to merge 11 commits into
base: main
Choose a base branch
from
3 changes: 2 additions & 1 deletion custom_components/xiaomi_home/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -155,7 +155,8 @@ def ha_persistent_notify(
for entity in filter_entities:
device.entity_list[platform].remove(entity)
entity_id = device.gen_service_entity_id(
ha_domain=platform, siid=entity.spec.iid)
ha_domain=platform, siid=entity.spec.iid,
description=entity.spec.description)
if er.async_get(entity_id_or_uuid=entity_id):
er.async_remove(entity_id=entity_id)
if platform in device.prop_list:
Expand Down
8 changes: 5 additions & 3 deletions custom_components/xiaomi_home/miot/miot_device.py
Original file line number Diff line number Diff line change
Expand Up @@ -298,10 +298,11 @@ def gen_device_entity_id(self, ha_domain: str) -> str:
f'{ha_domain}.{self._model_strs[0][:9]}_{self.did_tag}_'
f'{self._model_strs[-1][:20]}')

def gen_service_entity_id(self, ha_domain: str, siid: int) -> str:
def gen_service_entity_id(self, ha_domain: str, siid: int,
description: str) -> str:
return (
f'{ha_domain}.{self._model_strs[0][:9]}_{self.did_tag}_'
f'{self._model_strs[-1][:20]}_s_{siid}')
f'{self._model_strs[-1][:20]}_s_{siid}_{description}')

def gen_prop_entity_id(
self, ha_domain: str, spec_name: str, siid: int, piid: int
Expand Down Expand Up @@ -731,7 +732,8 @@ def __init__(
self._attr_name = f' {self.entity_data.spec.description_trans}'
elif isinstance(entity_data.spec, MIoTSpecService):
self.entity_id = miot_device.gen_service_entity_id(
DOMAIN, siid=entity_data.spec.iid)
DOMAIN, siid=entity_data.spec.iid,
description=entity_data.spec.description)
self._attr_name = (
f'{"* "if self.entity_data.spec.proprietary else " "}'
f'{self.entity_data.spec.description_trans}')
Expand Down
13 changes: 11 additions & 2 deletions custom_components/xiaomi_home/miot/miot_spec.py
Original file line number Diff line number Diff line change
Expand Up @@ -61,6 +61,7 @@
MIoTStorage,
SpecBoolTranslation,
SpecFilter,
SpecCustomService,
SpecMultiLang)

_LOGGER = logging.getLogger(__name__)
Expand Down Expand Up @@ -466,6 +467,7 @@ class MIoTSpecParser:
_bool_trans: SpecBoolTranslation
_multi_lang: SpecMultiLang
_spec_filter: SpecFilter
_custom_service: SpecCustomService

def __init__(
self, lang: str = DEFAULT_INTEGRATION_LANGUAGE,
Expand All @@ -484,13 +486,15 @@ def __init__(
lang=self._lang, loop=self._main_loop)
self._multi_lang = SpecMultiLang(lang=self._lang, loop=self._main_loop)
self._spec_filter = SpecFilter(loop=self._main_loop)
self._custom_service = SpecCustomService(loop=self._main_loop)

async def init_async(self) -> None:
if self._init_done is True:
return
await self._bool_trans.init_async()
await self._multi_lang.init_async()
await self._spec_filter.init_async()
await self._custom_service.init_async()
std_lib_cache: dict = None
if self._storage:
std_lib_cache: dict = await self._storage.load_async(
Expand Down Expand Up @@ -536,6 +540,7 @@ async def deinit_async(self) -> None:
await self._bool_trans.deinit_async()
await self._multi_lang.deinit_async()
await self._spec_filter.deinit_async()
await self._custom_service.deinit_async()
self._ram_cache.clear()

async def parse(
Expand Down Expand Up @@ -779,6 +784,12 @@ async def __parse(self, urn: str) -> MIoTSpecInstance:
_LOGGER.debug('parse urn, %s', urn)
# Load spec instance
instance: dict = await self.__get_instance(urn=urn)
urn_strs: list[str] = urn.split(':')
urn_key: str = ':'.join(urn_strs[:6])
# Modify the spec instance by custom spec
instance = self._custom_service.modify_spec(urn_key=urn_key,
spec=instance)
# Check required fields in the device instance
if (
not isinstance(instance, dict)
or 'type' not in instance
Expand All @@ -796,8 +807,6 @@ async def __parse(self, urn: str) -> MIoTSpecInstance:
or not isinstance(res_trans['data'], dict)
):
raise MIoTSpecError('invalid translation data')
urn_strs: list[str] = urn.split(':')
urn_key: str = ':'.join(urn_strs[:6])
trans_data: dict[str, str] = None
if self._lang == 'zh-Hans':
# Simplified Chinese
Expand Down
62 changes: 62 additions & 0 deletions custom_components/xiaomi_home/miot/miot_storage.py
Original file line number Diff line number Diff line change
Expand Up @@ -1033,3 +1033,65 @@ def __get_manufacturer_data(self) -> dict:
except Exception as err: # pylint: disable=broad-exception-caught
_LOGGER.error('get manufacturer info failed, %s', err)
return None


class SpecCustomService:
"""Custom MIoT-Spec-V2 service defined by the user."""
CUSTOM_SPEC_FILE = 'specs/custom_service.json'
_main_loop: asyncio.AbstractEventLoop
_data: dict[str, dict[str, any]]

def __init__(self, loop: Optional[asyncio.AbstractEventLoop]) -> None:
self._main_loop = loop or asyncio.get_event_loop()
self._data = None

async def init_async(self) -> None:
if isinstance(self._data, dict):
return
custom_data = None
self._data = {}
try:
custom_data = await self._main_loop.run_in_executor(
None, load_json_file,
os.path.join(
os.path.dirname(os.path.abspath(__file__)),
self.CUSTOM_SPEC_FILE))
except Exception as err: # pylint: disable=broad-exception-caught
_LOGGER.error('custom service, load file error, %s', err)
return
if not isinstance(custom_data, dict):
_LOGGER.error('custom service, invalid spec content')
return
for values in list(custom_data.values()):
if not isinstance(values, dict):
_LOGGER.error('custom service, invalid spec data')
return
self._data = custom_data

async def deinit_async(self) -> None:
self._data = None

def modify_spec(self, urn_key: str, spec: dict) -> dict | None:
"""MUST call init_async() first."""
if not self._data:
_LOGGER.error('self._data is None')
return spec
if urn_key not in self._data:
return spec
if 'services' not in spec:
return spec
if isinstance(self._data[urn_key], str):
urn_key = self._data[urn_key]
spec_services = spec['services']
custom_spec = self._data.get(urn_key, None)
# Replace services by custom defined spec
for i, service in enumerate(spec_services):
siid = str(service['iid'])
if siid in custom_spec:
spec_services[i] = custom_spec[siid]
# Add new services
if 'new' in custom_spec:
for service in custom_spec['new']:
spec_services.append(service)

return spec
152 changes: 152 additions & 0 deletions custom_components/xiaomi_home/miot/specs/custom_service.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,152 @@
{
"urn:miot-spec-v2:device:airer:0000A00D:hyd-lyjpro": {
"3": {
"iid": 3,
"type": "urn:miot-spec-v2:service:light:00007802:hyd-lyjpro:1",
"description": "Light",
"properties": [
{
"iid": 1,
"type": "urn:miot-spec-v2:property:on:00000006:hyd-lyjpro:1",
"description": "Sunlight",
"format": "bool",
"access": [
"read",
"write",
"notify"
]
},
{
"iid": 3,
"type": "urn:miot-spec-v2:property:flex-switch:000000EC:hyd-lyjpro:1",
"description": "Flex Switch",
"format": "uint8",
"access": [
"read",
"write",
"notify"
],
"value-list": [
{
"value": 1,
"description": "Overturn"
}
]
}
]
},
"new": [
{
"iid": 3,
"type": "urn:miot-spec-v2:service:light:00007802:hyd-lyjpro:1",
"description": "Moonlight",
"properties": [
{
"iid": 2,
"type": "urn:miot-spec-v2:property:on:00000006:hyd-lyjpro:1",
"description": "Switch Status",
"format": "bool",
"access": [
"read",
"write",
"notify"
]
}
]
}
]
},
"urn:miot-spec-v2:device:light:0000A001:yeelink-ceiling19": "urn:miot-spec-v2:device:light:0000A001:yeelink-ceiling4",
"urn:miot-spec-v2:device:light:0000A001:yeelink-ceiling20": "urn:miot-spec-v2:device:light:0000A001:yeelink-ceiling4",
"urn:miot-spec-v2:device:light:0000A001:yeelink-ceiling4": {
"new": [
{
"iid": 200,
"type": "urn:miot-spec-v2:service:ambient-light:0000789D:yeelink-ceiling4:1",
"description": "Ambient Light",
"properties": [
{
"iid": 201,
"type": "urn:miot-spec-v2:property:on:00000006:yeelink-ceiling4:1",
"description": "Switch Status",
"format": "bool",
"access": [
"read",
"write"
]
},
{
"iid": 202,
"type": "urn:miot-spec-v2:property:brightness:0000000D:yeelink-ceiling4:1",
"description": "Brightness",
"format": "uint8",
"access": [
"read",
"write"
],
"unit": "percentage",
"value-range": [
1,
100,
1
]
},
{
"iid": 203,
"type": "urn:miot-spec-v2:property:color-temperature:0000000F:yeelink-ceiling4:1",
"description": "Color Temperature",
"format": "uint32",
"access": [
"read",
"write"
],
"unit": "kelvin",
"value-range": [
1700,
6500,
1
]
},
{
"iid": 204,
"type": "urn:miot-spec-v2:property:color:0000000E:yeelink-ceiling4:1",
"description": "Color",
"format": "uint32",
"access": [
"read",
"write"
],
"unit": "rgb",
"value-range": [
1,
16777215,
1
]
}
]
}
]
},
"urn:miot-spec-v2:device:water-heater:0000A02A:zimi-h03": {
"new": [
{
"iid": 2,
"type": "urn:miot-spec-v2:service:switch:0000780C:zimi-h03:1",
"description": "Heat Water",
"properties": [
{
"iid": 6,
"type": "urn:miot-spec-v2:property:on:00000006:zimi-h03:1",
"description": "Switch Status",
"format": "bool",
"access": [
"read",
"write",
"notify"
]
}
]
}
]
}
}
2 changes: 1 addition & 1 deletion custom_components/xiaomi_home/miot/specs/multi_lang.json
Original file line number Diff line number Diff line change
Expand Up @@ -155,7 +155,7 @@
"service:004:property:001": "事件名稱"
}
},
"urn:miot-spec-v2:device:switch:0000A003:lumi-acn040:1": {
"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",
Expand Down
Loading
Loading