diff --git a/.experiment.py b/.experiment.py index 0ae89f6..cdbe35f 100644 --- a/.experiment.py +++ b/.experiment.py @@ -1,35 +1,43 @@ -import asyncio -from asyncio import Event, FIRST_COMPLETED -import signal +import re -stop = Event() -loop = asyncio.get_event_loop() - - -async def handler( - reader: asyncio.StreamReader, - writer: asyncio.StreamWriter, -): - await reader.read(100) - ans = b'HTTP/1.1 200 OK\r\nContent-Length:1\r\n\r\nd' - writer.write(ans) - await writer.drain() - writer.close() - await writer.wait_closed() - - -async def serve(): - server = await asyncio.start_server( - handler, - host='0.0.0.0', - port=8888, - ) - async with server: - await asyncio.wait((server.serve_forever(), stop.wait()), return_when=FIRST_COMPLETED) - -if __name__ == '__main__': - loop.add_signal_handler( - signal.SIGINT, stop.set - ) - loop.run_until_complete(serve()) +PATT_FW = re.compile(r'fw:\s(.+)\)') +data = """ +MegaD-2561 by ab-log.ru (fw: 4.48b7)
Config
-- MODS --
XP1
XP2
-- XT2 --
P30 - OUT
P31 - OUT
P32 - IN
P33 - I2C/SCL
P34 - DS
P35 - NC
-- XP5/6 --
P36 - ADC
P37 - NC + +MegaD-2561 by ab-log.ru (fw: 4.48b7)
Config
-- MODS --
XP1
XP2
-- XT2 --
P30 - OUT
P31 - OUT
P32 - IN
P33 - I2C/SCL
P34 - DS
P35 - NC
-- XP5/6 --
P36 - ADC
P37 - NC +MegaD-2561 by +ab-log.ru + (fw: 4.48b7) +
+Config +
+-- MODS -- +
+XP1 +
+XP2 +
+-- XT2 -- +
+P30 - OUT +
+P31 - OUT +
+P32 - IN +
+P33 - I2C/SCL +
+P34 - DS +
+P35 - NC +
+-- XP5/6 -- +
+P36 - ADC +
+P37 - NC +MegaD-2561 by ab-log.ru (fw: 4.48b7)
Config
-- MODS --
XP1
XP2
-- XT2 --
P30 - OUT
P31 - OUT
P32 - IN
P33 - I2C/SCL
P34 - DS
P35 - NC
-- XP5/6 --
P36 - ADC
P37 - NC +MegaD-2561 by ab-log.ru (fw: 4.48b7)
Config
-- MODS --
XP1
XP2
-- XT2 --
P30 - OUT
P31 - OUT
P32 - IN
P33 - I2C/SCL
P34 - DS
P35 - NC
-- XP5/6 --
P36 - ADC
P37 - NC +""" +print(PATT_FW.search(data).groups()[0]) \ No newline at end of file diff --git a/custom_components/mega/__init__.py b/custom_components/mega/__init__.py index 3757f3e..57b53e1 100644 --- a/custom_components/mega/__init__.py +++ b/custom_components/mega/__init__.py @@ -47,6 +47,14 @@ CUSTOMIZE_DS2413 = { vol.Optional(str.lower, description='адрес и индекс устройства'): CUSTOMIZE_PORT } + +def extender(x): + if isinstance(x, str) and 'e' in x: + return x + else: + raise ValueError('must has "e" in port name') + + CONFIG_SCHEMA = vol.Schema( { DOMAIN: { @@ -58,7 +66,7 @@ CONFIG_SCHEMA = vol.Schema( description='Ответ по умолчанию', default=None ): vol.Any(cv.template, None), - vol.Optional(int, description='номер порта'): vol.Any( + vol.Optional(vol.Any(int, extender), description='номер порта'): vol.Any( CUSTOMIZE_PORT, CUSTOMIZE_DS2413, ) @@ -128,6 +136,7 @@ async def get_hub(hass, entry): async def _add_mega(hass: HomeAssistant, entry: ConfigEntry): id = entry.data.get('id', entry.entry_id) hub = await get_hub(hass, entry) + hub.fw = await hub.get_fw() hass.data[DOMAIN][id] = hub hass.data[DOMAIN][CONF_ALL][id] = hub if not await hub.authenticate(): @@ -173,7 +182,7 @@ async def updater(hass: HomeAssistant, entry: ConfigEntry): async def async_remove_entry(hass, entry) -> None: """Handle removal of an entry.""" id = entry.data.get('id', entry.entry_id) - hub: MegaD = hass.data[DOMAIN][id] + hub: MegaD = hass.data[DOMAIN].get(id) if hub is None: return _LOGGER.debug(f'remove {id}') diff --git a/custom_components/mega/config_parser.py b/custom_components/mega/config_parser.py new file mode 100644 index 0000000..b0f90b4 --- /dev/null +++ b/custom_components/mega/config_parser.py @@ -0,0 +1,52 @@ +from dataclasses import dataclass, field +from bs4 import BeautifulSoup + + +@dataclass(frozen=True, eq=True) +class Config: + pty: str = None + m: str = None + gr: str = None + d: str = None + inta: str = field(compare=False, hash=False, default=None) + ety: str = None + misc: str = field(compare=False, hash=False, default=None) + + +def parse_config(page: str): + page = BeautifulSoup(page, features="lxml") + ret = {} + for x in [ + 'pty', + 'm', + 'gr', + 'd', + 'ety', + ]: + v = page.find('select', attrs={'name': x}) + if v is None: + continue + else: + v = v.find(selected=True) + if v: + v = v['value'] + ret[x] = v + v = page.find('input', attrs={'name': 'inta'}) + if v: + ret['inta'] = v['value'] + v = page.find('input', attrs={'name': 'misc'}) + if v: + ret['misc'] = v.get('checked', False) + return Config(**ret) + + +DIGITAL_IN = Config(pty="0") +RELAY_OUT = Config(pty="1", m="0") +PWM_OUT = Config(pty="1", m="1") +DS2413 = Config(pty="1", m="2") +MCP230 = Config(pty="4", m="1", gr="3", d="20") +MCP230_OUT = Config(ety="1") +MCP230_IN = Config(ety="0") +PCA9685 = Config(pty="4", m="1", gr="3", d="21") +OWIRE_BUS = Config(pty="3", d="5") + diff --git a/custom_components/mega/const.py b/custom_components/mega/const.py index 478ae88..317660c 100644 --- a/custom_components/mega/const.py +++ b/custom_components/mega/const.py @@ -50,3 +50,5 @@ PRESS = 'press' LUX = 'lux' SINGLE_CLICK = 'single' DOUBLE_CLICK = 'double' + +PATT_FW = re.compile(r'fw:\s(.+)\)') \ No newline at end of file diff --git a/custom_components/mega/entities.py b/custom_components/mega/entities.py index d4ab7cf..5a7893a 100644 --- a/custom_components/mega/entities.py +++ b/custom_components/mega/entities.py @@ -96,7 +96,7 @@ class BaseMegaEntity(CoordinatorEntity, RestoreEntity): "name": f'{self._mega_id} port {self.port}', "manufacturer": 'ab-log.ru', # "model": self.light.productname, - # "sw_version": self.light.swversion, + "sw_version": self.mega.fw, "via_device": (DOMAIN, self._mega_id), } @@ -214,6 +214,7 @@ class MegaOutPort(MegaPushEntity): def __init__( self, dimmer=False, + dimmer_scale=1, *args, **kwargs ): super().__init__( @@ -222,6 +223,7 @@ class MegaOutPort(MegaPushEntity): self._brightness = None self._is_on = None self.dimmer = dimmer + self.dimmer_scale = dimmer_scale # @property # def assumed_state(self) -> bool: @@ -235,10 +237,22 @@ class MegaOutPort(MegaPushEntity): def brightness(self): if not self.dimmer: return - val = self.mega.values.get(self.port, {}).get("value") - if val is None and self._state is not None: + val = self.mega.values.get(self.port, {}) + if isinstance(val, dict) and len(val) == 0 and self._state is not None: return self._state.attributes.get("brightness") + elif isinstance(self.port, str) and 'e' in self.port: + if isinstance(val, str): + val = safe_int(val) + else: + val = 0 + if val == 0: + return self._brightness + else: + return val elif val is not None: + val = val.get("value") + if val is None: + return try: val = int(val) return val @@ -248,9 +262,16 @@ class MegaOutPort(MegaPushEntity): @property def is_on(self) -> bool: val = self.mega.values.get(self.port, {}) - - if val is None and self._state is not None: + if isinstance(val, dict) and len(val) == 0 and self._state is not None: return self._state == 'ON' + elif isinstance(self.port, str) and 'e' in self.port and val: + if val is None: + return + if self.dimmer: + val = safe_int(val) + return val > 0 if not self.invert else val == 0 + else: + return val == 'ON' if not self.invert else val == 'OFF' elif val is not None: val = val.get("value") if not isinstance(val, str) and self.index is not None and self.addr is not None: @@ -286,11 +307,11 @@ class MegaOutPort(MegaPushEntity): async def async_turn_on(self, brightness=None, **kwargs) -> None: brightness = brightness or self.brightness or 255 - + self._brightness = brightness if self.dimmer and brightness == 0: - cmd = 255 + cmd = 255 * self.dimmer_scale elif self.dimmer: - cmd = brightness + cmd = brightness * self.dimmer_scale else: cmd = 1 if not self.invert else 0 _cmd = {"cmd": f"{self.cmd_port}:{cmd}"} @@ -305,6 +326,11 @@ class MegaOutPort(MegaPushEntity): conv=False, http_cmd='list', ) + elif isinstance(self.port, str) and 'e' in self.port: + if not self.dimmer: + self.mega.values[self.port] = 'ON' if not self.invert else 'OFF' + else: + self.mega.values[self.port] = cmd else: self.mega.values[self.port] = {'value': cmd} await self.get_state() @@ -324,6 +350,8 @@ class MegaOutPort(MegaPushEntity): conv=False, http_cmd='list', ) + elif isinstance(self.port, str) and 'e' in self.port: + self.mega.values[self.port] = 'OFF' if not self.invert else 'ON' else: self.mega.values[self.port] = {'value': cmd} await self.get_state() diff --git a/custom_components/mega/http.py b/custom_components/mega/http.py index fa18c56..c6da5e3 100644 --- a/custom_components/mega/http.py +++ b/custom_components/mega/http.py @@ -15,6 +15,8 @@ from .tools import make_ints from . import hub as h _LOGGER = logging.getLogger(__name__).getChild('http') +ext = {f'ext{x}' for x in range(16)} + class MegaView(HomeAssistantView): @@ -93,15 +95,27 @@ class MegaView(HomeAssistantView): data['mega_id'] = hub.id ret = 'd' if hub.force_d else '' if port is not None: - hub.values[port] = data - for cb in self.callbacks[hub.id][port]: - cb(data) - template: Template = self.templates.get(hub.id, {}).get(port, hub.def_response) + if set(data).issubset(ext): + ret = '' # пока ответ всегда пустой, неясно какая будет реакция на непустой ответ + for e in ext: + if e in data: + idx = e[-1] + pt = f'{port}e{idx}' + data['value'] = 'ON' if data[e] == '1' else 'OFF' + data['m'] = 1 if data[e] == '0' else 0 # имитация поведения обычного входа, чтобы события обрабатывались аналогично + hub.values[pt] = data + for cb in self.callbacks[hub.id][pt]: + cb(data) + else: + hub.values[port] = data + for cb in self.callbacks[hub.id][port]: + cb(data) + template: Template = self.templates.get(hub.id, {}).get(port, hub.def_response) + if template is not None: + template.hass = hass + ret = template.async_render(data) if hub.update_all and update_all: asyncio.create_task(self.later_update(hub)) - if template is not None: - template.hass = hass - ret = template.async_render(data) _LOGGER.debug('response %s', ret) Response(body='' if hub.fake_response else ret, content_type='text/plain') diff --git a/custom_components/mega/hub.py b/custom_components/mega/hub.py index b63d33a..6dfc392 100644 --- a/custom_components/mega/hub.py +++ b/custom_components/mega/hub.py @@ -16,10 +16,11 @@ from homeassistant.const import ( ) from homeassistant.core import HomeAssistant from homeassistant.helpers.update_coordinator import DataUpdateCoordinator +from .config_parser import parse_config, DS2413, MCP230, MCP230_OUT, MCP230_IN, PCA9685 from .const import ( TEMP, HUM, PRESS, LUX, PATT_SPLIT, DOMAIN, - CONF_HTTP, EVENT_BINARY_SENSOR, CONF_CUSTOM, CONF_FORCE_D, CONF_DEF_RESPONSE + CONF_HTTP, EVENT_BINARY_SENSOR, CONF_CUSTOM, CONF_FORCE_D, CONF_DEF_RESPONSE, PATT_FW ) from .entities import set_events_off, BaseMegaEntity, MegaOutPort from .exceptions import CannotConnect, NoPort @@ -79,6 +80,7 @@ class MegaD: allow_hosts: str=None, protected=True, restore_on_restart=False, + extenders=None, **kwargs, ): """Initialize.""" @@ -93,6 +95,7 @@ class MegaD: self.http.hubs[mqtt_id] = self else: self.http = None + self.extenders = extenders or [] self.poll_outs = poll_outs self.update_all = update_all if update_all is not None else True self.nports = nports @@ -127,10 +130,12 @@ class MegaD: self.updater = DataUpdateCoordinator( hass, self.lg, - name="sensors", + name="megad", update_method=self.poll, update_interval=timedelta(seconds=self.poll_interval) if self.poll_interval else None, ) + self.updaters = [] + self.fw = '' self.notifiers = defaultdict(asyncio.Condition) if not mqtt_id: _id = host.split(".")[-1] @@ -236,6 +241,12 @@ class MegaD: Polling ports """ self.lg.debug('poll') + for x in self.extenders: + ret = await self._update_extender(x) + if not isinstance(ret, dict): + self.lg.warning(f'wrong updater result: {ret} from extender {x}') + continue + self.values.update(ret) if self.mqtt is None: await self.get_all_ports() await self.get_sensors(only_list=True) @@ -259,12 +270,18 @@ class MegaD: _id = _id['value'] return _id or 'megad/' + self.host.split('.')[-1] + async def get_fw(self): + data = await self.request() + return PATT_FW.search(data).groups()[0] + async def send_command(self, port=None, cmd=None): return await self.request(pt=port, cmd=cmd) async def request(self, **kwargs): cmd = '&'.join([f'{k}={v}' for k, v in kwargs.items() if v is not None]) - url = f"http://{self.host}/{self.sec}/?{cmd}" + url = f"http://{self.host}/{self.sec}" + if cmd: + url = f"{url}/?{cmd}" self.lg.debug('request: %s', url) async with self._http_lck: async with aiohttp.request("get", url=url) as req: @@ -432,62 +449,77 @@ class MegaD: return await req.text() async def scan_port(self, port): - async with self.lck: - if port in self._scanned: - return self._scanned[port] - url = f'http://{self.host}/{self.sec}/?pt={port}' - self.lg.debug( - f'scan port %s: %s', port, url - ) - async with aiohttp.request('get', url) as req: - html = await req.text() - if req.status != 200: - return - tree = BeautifulSoup(html, features="lxml") - pty = tree.find('select', attrs={'name': 'pty'}) - if pty is None: - return - else: - pty = pty.find(selected=True) - if pty: - pty = pty['value'] - else: - return - if pty in ['0', '1']: - m = tree.find('select', attrs={'name': 'm'}) - if m: - m = m.find(selected=True)['value'] - self._scanned[port] = (pty, m) - return pty, m - elif pty == '3': - m = tree.find('select', attrs={'name': 'd'}) - if m: - m = m.find(selected=True)['value'] - self._scanned[port] = (pty, m) - return pty, m - elif pty in ('2', '4'): # эта часть не очень проработана, тут есть i2c который может работать неправильно - m = tree.find('select', attrs={'name': 'd'}) - if m: - m = m.find(selected=True)['value'] - self._scanned[port] = (pty, m or '0') - return pty, m or '0' + data = await self.request(pt=port) + return parse_config(data) + # async with self.lck: + # if port in self._scanned: + # return self._scanned[port] + # url = f'http://{self.host}/{self.sec}/?pt={port}' + # self.lg.debug( + # f'scan port %s: %s', port, url + # ) + # async with aiohttp.request('get', url) as req: + # html = await req.text() + # if req.status != 200: + # return + # tree = BeautifulSoup(html, features="lxml") + # pty = tree.find('select', attrs={'name': 'pty'}) + # if pty is None: + # return + # else: + # pty = pty.find(selected=True) + # if pty: + # pty = pty['value'] + # else: + # return + # if pty in ['0', '1']: + # m = tree.find('select', attrs={'name': 'm'}) + # if m: + # m = m.find(selected=True)['value'] + # self._scanned[port] = (pty, m) + # return pty, m + # elif pty == '3': + # m = tree.find('select', attrs={'name': 'd'}) + # if m: + # m = m.find(selected=True)['value'] + # self._scanned[port] = (pty, m) + # return pty, m + # elif pty in ('2', '4'): # эта часть не очень проработана, тут есть i2c который может работать неправильно + # m = tree.find('select', attrs={'name': 'd'}) + # if m: + # m = m.find(selected=True)['value'] + # self._scanned[port] = (pty, m or '0') + # return pty, m or '0' async def scan_ports(self, nports=37): for x in range(0, nports+1): ret = await self.scan_port(x) if ret: - yield [x, *ret] + yield x, ret self.nports = nports+1 + async def _update_extender(self, port): + """ + Обновление mcp230, так же подходит для PCA9685 + :param port: + :return: + """ + values = await self.request(pt=port, cmd='get') + ret = {} + for i, x in enumerate(values.split(';')): + ret[f'{port}e{i}'] = x + return ret + async def get_config(self, nports=37): ret = defaultdict(lambda: defaultdict(list)) ret['mqtt_id'] = await self.get_mqtt_id() - async for port, pty, m in self.scan_ports(nports): - if pty == "0": + ret['extenders'] = extenders = [] + async for port, cfg in self.scan_ports(nports): + if cfg.pty == "0": ret['binary_sensor'][port].append({}) - elif pty == "1" and (m in ['0', '1', '3'] or m is None): - ret['light'][port].append({'dimmer': m == '1'}) - elif pty == "1" and m == "2": + elif cfg.pty == "1" and (cfg.m in ['0', '1', '3'] or cfg.m is None): + ret['light'][port].append({'dimmer': cfg.m == '1'}) + elif cfg == DS2413: # ds2413 _data = await self.get_port(port=port, force_http=True, http_cmd='list', conv=False) data = _data.get('value', {}) @@ -499,21 +531,34 @@ class MegaD: {"index": 0, "addr": addr, "id_suffix": f'{addr}_a', "http_cmd": 'ds2413'}, {"index": 1, "addr": addr, "id_suffix": f'{addr}_b', "http_cmd": 'ds2413'}, ]) - elif pty in ('3', '2', '4'): - try: - http_cmd = 'get' - if m == '5' and pty == '3': - # 1-wire bus + elif cfg == MCP230: + extenders.append(port) + values = await self.request(pt=port, cmd='get') + values = values.split(';') + for n in range(len(values)): + ext_page = await self.request(pt=port, ext=n) + ext_cfg = parse_config(ext_page) + if ext_cfg.ety == '1': + ret['light'][f'{port}e{n}'].append({}) + elif ext_cfg.ety == '0': + ret['binary_sensor'][f'{port}e{n}'].append({}) + elif cfg == PCA9685: + extenders.append(port) + values = await self.request(pt=port, cmd='get') + values = values.split(';') + for n in range(len(values)): + ret['light'][f'{port}e{n}'].append({'dimmer': True, 'dimmer_scale': 16}) + elif cfg.pty in ('3', '2', '4'): + http_cmd = 'get' + if cfg.d == '5' and cfg.pty == '3': + # 1-wire bus + values = await self.get_port(port, force_http=True, http_cmd='list') + http_cmd = 'list' + else: + values = await self.get_port(port, force_http=True) + if values is None or (isinstance(values, dict) and str(values.get('value')) in ('', 'None')): values = await self.get_port(port, force_http=True, http_cmd='list') http_cmd = 'list' - else: - values = await self.get_port(port, force_http=True) - if values is None or (isinstance(values, dict) and str(values.get('value')) in ('', 'None')): - values = await self.get_port(port, force_http=True, http_cmd='list') - http_cmd = 'list' - except asyncio.TimeoutError: - self.lg.warning(f'timout on port {port}') - continue self.lg.debug(f'values: %s', values) if values is None: self.lg.warning(f'port {port} is of type sensor but response is None, skipping it') @@ -523,8 +568,8 @@ class MegaD: if isinstance(values, str) and TEMP_PATT.search(values): values = {TEMP: values} elif not isinstance(values, dict): - if pty == '4' and m in I2C_DEVICE_TYPES: - values = {I2C_DEVICE_TYPES[m]: values} + if cfg.pty == '4' and cfg.d in I2C_DEVICE_TYPES: + values = {I2C_DEVICE_TYPES.get(cfg.m): values} else: values = {None: values} for key in values: diff --git a/custom_components/mega/manifest.json b/custom_components/mega/manifest.json index af5e967..862b7e5 100644 --- a/custom_components/mega/manifest.json +++ b/custom_components/mega/manifest.json @@ -15,5 +15,5 @@ "@andvikt" ], "issue_tracker": "https://github.com/andvikt/mega_hacs/issues", - "version": "v0.4.2b1" + "version": "v0.5.1b1" } \ No newline at end of file diff --git a/readme.md b/readme.md index fa54f85..89d85f1 100644 --- a/readme.md +++ b/readme.md @@ -21,6 +21,7 @@ `light`, для шим - `light` с поддержкой яркости, для цифровых входов `binary_sensor`, для датчиков `sensor`) - Возможность работы с несколькими megad +- Автоматическое восстановление состояний выходов после перезагрузки контроллера - Обратная связь по [http](https://github.com/andvikt/mega_hacs/wiki/http) или mqtt (`deprecated`, поддержка mqtt будет выключена в версиях >= 1.0.0, тк в нем нет необходимости) - [События](https://github.com/andvikt/mega_hacs/wiki/События) на двойные/долгие нажатия @@ -28,6 +29,7 @@ большого кол-ва команд (например в сценах). Каждая следующая команда отправляется только после получения ответа о выполнении предыдущей. - поддержка [ds2413](https://www.ab-log.ru/smart-house/ethernet/megad-2w) (начиная с версии 0.4.1) +- поддержка MCP23008/MCP23017/PCA9685 (начиная с версии 0.5.1) ## Установка Рекомендованный способ с поддержкой обновлений - [HACS](https://hacs.xyz/docs/installation/installation):