mirror of
https://github.com/andvikt/mega_hacs.git
synced 2025-12-12 17:44:28 +05:00
Compare commits
12 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
4fccb23c39 | ||
|
|
03e4d3ff7e | ||
|
|
7907e0cd85 | ||
|
|
5211ee5330 | ||
|
|
4e0f1dddcb | ||
|
|
742a0a9a09 | ||
|
|
59443989a0 | ||
|
|
5b86ceefe4 | ||
|
|
8cf000beae | ||
|
|
99317da9f6 | ||
|
|
97911d0241 | ||
|
|
d9925a2de0 |
13
.github/ISSUE_TEMPLATE/bug-report.md
vendored
13
.github/ISSUE_TEMPLATE/bug-report.md
vendored
@@ -8,19 +8,26 @@ assignees: ''
|
|||||||
---
|
---
|
||||||
|
|
||||||
**Описание**
|
**Описание**
|
||||||
A clear and concise description of what the bug is.
|
Описание проблемы
|
||||||
|
|
||||||
**Версии систем**
|
**Версии систем**
|
||||||
Enviroment: raspberry/linux/windows/macos/docker
|
Enviroment: raspberry/linux/windows/macos/docker
|
||||||
HA version:
|
HA version:
|
||||||
mega_hacs version:
|
mega_hacs version:
|
||||||
megad firmware version:
|
megad firmware version:
|
||||||
|
используется mqtt: true/false
|
||||||
|
|
||||||
**Ожидаемое поведение**
|
**Ожидаемое поведение**
|
||||||
A clear and concise description of what you expected to happen.
|
Описание правильного поведения
|
||||||
|
|
||||||
**Screenshots**
|
**Screenshots**
|
||||||
If applicable, add screenshots to help explain your problem.
|
If applicable, add screenshots to help explain your problem.
|
||||||
|
|
||||||
**LOG**
|
**LOG**
|
||||||
Прочитайте в документации как включить подробный лог интеграции и приложите его здесь
|
Просьба прикладывать детальный лог, который можно включить в конфиге так:
|
||||||
|
```yaml
|
||||||
|
logger:
|
||||||
|
default: info
|
||||||
|
logs:
|
||||||
|
custom_components.mega: debug
|
||||||
|
```
|
||||||
|
|||||||
@@ -17,7 +17,7 @@ from homeassistant.components import mqtt
|
|||||||
from homeassistant.config_entries import ConfigEntry
|
from homeassistant.config_entries import ConfigEntry
|
||||||
from .const import DOMAIN, CONF_INVERT, CONF_RELOAD, PLATFORMS, CONF_PORTS, CONF_CUSTOM, CONF_SKIP, CONF_PORT_TO_SCAN, \
|
from .const import DOMAIN, CONF_INVERT, CONF_RELOAD, PLATFORMS, CONF_PORTS, CONF_CUSTOM, CONF_SKIP, CONF_PORT_TO_SCAN, \
|
||||||
CONF_MQTT_INPUTS, CONF_HTTP, CONF_RESPONSE_TEMPLATE, CONF_ACTION, CONF_GET_VALUE, CONF_ALLOW_HOSTS, \
|
CONF_MQTT_INPUTS, CONF_HTTP, CONF_RESPONSE_TEMPLATE, CONF_ACTION, CONF_GET_VALUE, CONF_ALLOW_HOSTS, \
|
||||||
CONF_CONV_TEMPLATE
|
CONF_CONV_TEMPLATE, CONF_ALL
|
||||||
from .hub import MegaD
|
from .hub import MegaD
|
||||||
from .config_flow import ConfigFlow
|
from .config_flow import ConfigFlow
|
||||||
from .http import MegaView
|
from .http import MegaView
|
||||||
@@ -65,6 +65,7 @@ async def async_setup(hass: HomeAssistant, config: dict):
|
|||||||
"""YAML-конфигурация содержит только кастомизации портов"""
|
"""YAML-конфигурация содержит только кастомизации портов"""
|
||||||
hass.data[DOMAIN] = {CONF_CUSTOM: config.get(DOMAIN, {})}
|
hass.data[DOMAIN] = {CONF_CUSTOM: config.get(DOMAIN, {})}
|
||||||
hass.data[DOMAIN][CONF_HTTP] = view = MegaView(cfg=config.get(DOMAIN, {}))
|
hass.data[DOMAIN][CONF_HTTP] = view = MegaView(cfg=config.get(DOMAIN, {}))
|
||||||
|
hass.data[DOMAIN][CONF_ALL] = {}
|
||||||
view.allowed_hosts |= set(config.get(DOMAIN, {}).get(CONF_ALLOW_HOSTS, []))
|
view.allowed_hosts |= set(config.get(DOMAIN, {}).get(CONF_ALLOW_HOSTS, []))
|
||||||
hass.http.register_view(view)
|
hass.http.register_view(view)
|
||||||
hass.services.async_register(
|
hass.services.async_register(
|
||||||
@@ -75,12 +76,12 @@ async def async_setup(hass: HomeAssistant, config: dict):
|
|||||||
hass.services.async_register(
|
hass.services.async_register(
|
||||||
DOMAIN, 'get_port', partial(_get_port, hass), schema=vol.Schema({
|
DOMAIN, 'get_port', partial(_get_port, hass), schema=vol.Schema({
|
||||||
vol.Optional('mega_id'): str,
|
vol.Optional('mega_id'): str,
|
||||||
vol.Optional('port'): int,
|
vol.Optional('port'): vol.Any(int, [int]),
|
||||||
})
|
})
|
||||||
)
|
)
|
||||||
hass.services.async_register(
|
hass.services.async_register(
|
||||||
DOMAIN, 'run_cmd', partial(_run_cmd, hass), schema=vol.Schema({
|
DOMAIN, 'run_cmd', partial(_run_cmd, hass), schema=vol.Schema({
|
||||||
vol.Required('port'): int,
|
vol.Optional('port'): int,
|
||||||
vol.Required('cmd'): str,
|
vol.Required('cmd'): str,
|
||||||
vol.Optional('mega_id'): str,
|
vol.Optional('mega_id'): str,
|
||||||
})
|
})
|
||||||
@@ -115,6 +116,7 @@ async def _add_mega(hass: HomeAssistant, entry: ConfigEntry):
|
|||||||
hub = await get_hub(hass, entry)
|
hub = await get_hub(hass, entry)
|
||||||
hass.data[DOMAIN][id] = hass.data[DOMAIN]['__def'] = hub
|
hass.data[DOMAIN][id] = hass.data[DOMAIN]['__def'] = hub
|
||||||
hass.data[DOMAIN][entry.data.get(CONF_HOST)] = hub
|
hass.data[DOMAIN][entry.data.get(CONF_HOST)] = hub
|
||||||
|
hass.data[DOMAIN][CONF_ALL][id] = hub
|
||||||
if not await hub.authenticate():
|
if not await hub.authenticate():
|
||||||
raise Exception("not authentificated")
|
raise Exception("not authentificated")
|
||||||
mid = await hub.get_mqtt_id()
|
mid = await hub.get_mqtt_id()
|
||||||
@@ -164,6 +166,7 @@ async def async_remove_entry(hass, entry) -> None:
|
|||||||
_LOGGER.debug(f'remove {id}')
|
_LOGGER.debug(f'remove {id}')
|
||||||
_hubs.pop(id, None)
|
_hubs.pop(id, None)
|
||||||
hass.data[DOMAIN].pop(id, None)
|
hass.data[DOMAIN].pop(id, None)
|
||||||
|
hass.data[DOMAIN][CONF_ALL].pop(id, None)
|
||||||
task: asyncio.Task = _POLL_TASKS.pop(id, None)
|
task: asyncio.Task = _POLL_TASKS.pop(id, None)
|
||||||
if task is not None:
|
if task is not None:
|
||||||
task.cancel()
|
task.cancel()
|
||||||
@@ -208,27 +211,32 @@ async def _get_port(hass: HomeAssistant, call: ServiceCall):
|
|||||||
if mega_id:
|
if mega_id:
|
||||||
hub: MegaD = hass.data[DOMAIN][mega_id]
|
hub: MegaD = hass.data[DOMAIN][mega_id]
|
||||||
if port is None:
|
if port is None:
|
||||||
await hub.get_all_ports()
|
await hub.get_all_ports(check_skip=True)
|
||||||
else:
|
elif isinstance(port, int):
|
||||||
await hub.get_port(port)
|
await hub.get_port(port)
|
||||||
|
elif isinstance(port, list):
|
||||||
|
for x in port:
|
||||||
|
await hub.get_port(x)
|
||||||
else:
|
else:
|
||||||
for hub in hass.data[DOMAIN].values():
|
for hub in hass.data[DOMAIN][CONF_ALL].values():
|
||||||
if not isinstance(hub, MegaD):
|
if not isinstance(hub, MegaD):
|
||||||
continue
|
continue
|
||||||
if port is None:
|
if port is None:
|
||||||
await hub.get_all_ports()
|
await hub.get_all_ports(check_skip=True)
|
||||||
else:
|
elif isinstance(port, int):
|
||||||
await hub.get_port(port)
|
await hub.get_port(port)
|
||||||
|
elif isinstance(port, list):
|
||||||
|
for x in port:
|
||||||
|
await hub.get_port(x)
|
||||||
|
|
||||||
|
|
||||||
@bind_hass
|
@bind_hass
|
||||||
async def _run_cmd(hass: HomeAssistant, call: ServiceCall):
|
async def _run_cmd(hass: HomeAssistant, call: ServiceCall):
|
||||||
port = call.data.get('port')
|
|
||||||
mega_id = call.data.get('mega_id')
|
mega_id = call.data.get('mega_id')
|
||||||
cmd = call.data.get('cmd')
|
cmd = call.data.get('cmd')
|
||||||
if mega_id:
|
if mega_id:
|
||||||
hub: MegaD = hass.data[DOMAIN][mega_id]
|
hub: MegaD = hass.data[DOMAIN][mega_id]
|
||||||
await hub.send_command(port=port, cmd=cmd)
|
await hub.request(cmd=cmd)
|
||||||
else:
|
else:
|
||||||
for hub in hass.data[DOMAIN].values():
|
for hub in hass.data[DOMAIN].values():
|
||||||
await hub.send_command(port=port, cmd=cmd)
|
await hub.request(cmd=cmd)
|
||||||
|
|||||||
@@ -10,7 +10,7 @@ from homeassistant.config_entries import ConfigEntry
|
|||||||
from homeassistant.const import CONF_HOST, CONF_ID, CONF_PASSWORD, CONF_SCAN_INTERVAL
|
from homeassistant.const import CONF_HOST, CONF_ID, CONF_PASSWORD, CONF_SCAN_INTERVAL
|
||||||
from homeassistant.core import callback, HomeAssistant
|
from homeassistant.core import callback, HomeAssistant
|
||||||
from .const import DOMAIN, CONF_PORT_TO_SCAN, CONF_RELOAD, PLATFORMS, CONF_MQTT_INPUTS, \
|
from .const import DOMAIN, CONF_PORT_TO_SCAN, CONF_RELOAD, PLATFORMS, CONF_MQTT_INPUTS, \
|
||||||
CONF_NPORTS, CONF_UPDATE_ALL # pylint:disable=unused-import
|
CONF_NPORTS, CONF_UPDATE_ALL, CONF_POLL_OUTS # pylint:disable=unused-import
|
||||||
from .hub import MegaD
|
from .hub import MegaD
|
||||||
from . import exceptions
|
from . import exceptions
|
||||||
|
|
||||||
@@ -18,10 +18,11 @@ _LOGGER = logging.getLogger(__name__)
|
|||||||
|
|
||||||
STEP_USER_DATA_SCHEMA = vol.Schema(
|
STEP_USER_DATA_SCHEMA = vol.Schema(
|
||||||
{
|
{
|
||||||
vol.Required(CONF_ID, default='def'): str,
|
vol.Required(CONF_ID, default='mega'): str,
|
||||||
vol.Required(CONF_HOST, default="192.168.0.14"): str,
|
vol.Required(CONF_HOST, default="192.168.0.14"): str,
|
||||||
vol.Required(CONF_PASSWORD, default="sec"): str,
|
vol.Required(CONF_PASSWORD, default="sec"): str,
|
||||||
vol.Optional(CONF_SCAN_INTERVAL, default=0): int,
|
vol.Optional(CONF_SCAN_INTERVAL, default=0): int,
|
||||||
|
vol.Optional(CONF_POLL_OUTS, default=False): bool,
|
||||||
vol.Optional(CONF_PORT_TO_SCAN, default=0): int,
|
vol.Optional(CONF_PORT_TO_SCAN, default=0): int,
|
||||||
vol.Optional(CONF_MQTT_INPUTS, default=True): bool,
|
vol.Optional(CONF_MQTT_INPUTS, default=True): bool,
|
||||||
vol.Optional(CONF_NPORTS, default=37): int,
|
vol.Optional(CONF_NPORTS, default=37): int,
|
||||||
@@ -131,6 +132,7 @@ class OptionsFlowHandler(config_entries.OptionsFlow):
|
|||||||
step_id="init",
|
step_id="init",
|
||||||
data_schema=vol.Schema({
|
data_schema=vol.Schema({
|
||||||
vol.Optional(CONF_SCAN_INTERVAL, default=e.get(CONF_SCAN_INTERVAL, 0)): int,
|
vol.Optional(CONF_SCAN_INTERVAL, default=e.get(CONF_SCAN_INTERVAL, 0)): int,
|
||||||
|
vol.Optional(CONF_POLL_OUTS, default=e.get(CONF_POLL_OUTS, False)): bool,
|
||||||
vol.Optional(CONF_PORT_TO_SCAN, default=e.get(CONF_PORT_TO_SCAN, 0)): int,
|
vol.Optional(CONF_PORT_TO_SCAN, default=e.get(CONF_PORT_TO_SCAN, 0)): int,
|
||||||
vol.Optional(CONF_MQTT_INPUTS, default=e.get(CONF_MQTT_INPUTS, True)): bool,
|
vol.Optional(CONF_MQTT_INPUTS, default=e.get(CONF_MQTT_INPUTS, True)): bool,
|
||||||
vol.Optional(CONF_NPORTS, default=e.get(CONF_NPORTS, 37)): int,
|
vol.Optional(CONF_NPORTS, default=e.get(CONF_NPORTS, 37)): int,
|
||||||
|
|||||||
@@ -16,6 +16,7 @@ CONF_INVERT = 'invert'
|
|||||||
CONF_PORTS = 'ports'
|
CONF_PORTS = 'ports'
|
||||||
CONF_CUSTOM = '__custom'
|
CONF_CUSTOM = '__custom'
|
||||||
CONF_HTTP = '__http'
|
CONF_HTTP = '__http'
|
||||||
|
CONF_ALL = '__all'
|
||||||
CONF_SKIP = 'skip'
|
CONF_SKIP = 'skip'
|
||||||
CONF_MQTT_INPUTS = 'mqtt_inputs'
|
CONF_MQTT_INPUTS = 'mqtt_inputs'
|
||||||
CONF_NPORTS = 'nports'
|
CONF_NPORTS = 'nports'
|
||||||
@@ -25,6 +26,7 @@ CONF_UPDATE_ALL = 'update_all'
|
|||||||
CONF_GET_VALUE = 'get_value'
|
CONF_GET_VALUE = 'get_value'
|
||||||
CONF_ALLOW_HOSTS = 'allow_hosts'
|
CONF_ALLOW_HOSTS = 'allow_hosts'
|
||||||
CONF_CONV_TEMPLATE = 'conv_template'
|
CONF_CONV_TEMPLATE = 'conv_template'
|
||||||
|
CONF_POLL_OUTS = 'poll_outs'
|
||||||
PLATFORMS = [
|
PLATFORMS = [
|
||||||
"light",
|
"light",
|
||||||
"switch",
|
"switch",
|
||||||
@@ -40,5 +42,6 @@ LONG = 'long'
|
|||||||
RELEASE = 'release'
|
RELEASE = 'release'
|
||||||
LONG_RELEASE = 'long_release'
|
LONG_RELEASE = 'long_release'
|
||||||
PRESS = 'press'
|
PRESS = 'press'
|
||||||
|
LUX = 'lux'
|
||||||
SINGLE_CLICK = 'single'
|
SINGLE_CLICK = 'single'
|
||||||
DOUBLE_CLICK = 'double'
|
DOUBLE_CLICK = 'double'
|
||||||
|
|||||||
@@ -246,16 +246,15 @@ class MegaOutPort(MegaPushEntity):
|
|||||||
cmd = brightness
|
cmd = brightness
|
||||||
else:
|
else:
|
||||||
cmd = 1 if not self.invert else 0
|
cmd = 1 if not self.invert else 0
|
||||||
await self.mega.send_command(self.port, f"{self.port}:{cmd}")
|
await self.mega.request(cmd=f"{self.port}:{cmd}")
|
||||||
self.mega.values[self.port] = {'value': cmd}
|
self.mega.values[self.port] = {'value': cmd}
|
||||||
await self.get_state()
|
await self.get_state()
|
||||||
|
|
||||||
|
|
||||||
async def async_turn_off(self, **kwargs) -> None:
|
async def async_turn_off(self, **kwargs) -> None:
|
||||||
|
|
||||||
cmd = "0" if not self.invert else "1"
|
cmd = "0" if not self.invert else "1"
|
||||||
|
|
||||||
await self.mega.send_command(self.port, f"{self.port}:{cmd}")
|
await self.mega.request(cmd=f"{self.port}:{cmd}")
|
||||||
self.mega.values[self.port] = {'value': cmd}
|
self.mega.values[self.port] = {'value': cmd}
|
||||||
await self.get_state()
|
await self.get_state()
|
||||||
|
|
||||||
|
|||||||
@@ -10,30 +10,44 @@ import json
|
|||||||
|
|
||||||
from bs4 import BeautifulSoup
|
from bs4 import BeautifulSoup
|
||||||
from homeassistant.components import mqtt
|
from homeassistant.components import mqtt
|
||||||
from homeassistant.const import DEVICE_CLASS_TEMPERATURE, DEVICE_CLASS_HUMIDITY
|
from homeassistant.const import (DEVICE_CLASS_TEMPERATURE, DEVICE_CLASS_HUMIDITY, DEVICE_CLASS_PRESSURE,
|
||||||
|
DEVICE_CLASS_ILLUMINANCE, TEMP_CELSIUS, PERCENTAGE, LIGHT_LUX)
|
||||||
from homeassistant.core import HomeAssistant
|
from homeassistant.core import HomeAssistant
|
||||||
from homeassistant.helpers.entity import Entity
|
from homeassistant.helpers.entity import Entity
|
||||||
from homeassistant.helpers.update_coordinator import DataUpdateCoordinator
|
from homeassistant.helpers.update_coordinator import DataUpdateCoordinator
|
||||||
from .const import TEMP, HUM, PATT_SPLIT, DOMAIN, CONF_HTTP, EVENT_BINARY_SENSOR
|
from .const import TEMP, HUM, PRESS, LUX, PATT_SPLIT, DOMAIN, CONF_HTTP, EVENT_BINARY_SENSOR, CONF_CUSTOM, CONF_SKIP
|
||||||
from .entities import set_events_off
|
from .entities import set_events_off, BaseMegaEntity
|
||||||
from .exceptions import CannotConnect
|
from .exceptions import CannotConnect
|
||||||
from .tools import make_ints
|
from .tools import make_ints
|
||||||
|
|
||||||
TEMP_PATT = re.compile(r'temp:([01234567890\.]+)')
|
TEMP_PATT = re.compile(r'temp:([01234567890\.]+)')
|
||||||
HUM_PATT = re.compile(r'hum:([01234567890\.]+)')
|
HUM_PATT = re.compile(r'hum:([01234567890\.]+)')
|
||||||
|
PRESS_PATT = re.compile(r'press:([01234567890\.]+)')
|
||||||
|
LUX_PATT = re.compile(r'lux:([01234567890\.]+)')
|
||||||
PATTERNS = {
|
PATTERNS = {
|
||||||
TEMP: TEMP_PATT,
|
TEMP: TEMP_PATT,
|
||||||
HUM: HUM_PATT,
|
HUM: HUM_PATT,
|
||||||
|
PRESS: PRESS_PATT,
|
||||||
|
LUX: LUX_PATT
|
||||||
}
|
}
|
||||||
UNITS = {
|
UNITS = {
|
||||||
TEMP: '°C',
|
TEMP: TEMP_CELSIUS,
|
||||||
HUM: '%'
|
HUM: PERCENTAGE,
|
||||||
|
PRESS: 'mmHg',
|
||||||
|
LUX: LIGHT_LUX
|
||||||
}
|
}
|
||||||
CLASSES = {
|
CLASSES = {
|
||||||
TEMP: DEVICE_CLASS_TEMPERATURE,
|
TEMP: DEVICE_CLASS_TEMPERATURE,
|
||||||
HUM: DEVICE_CLASS_HUMIDITY
|
HUM: DEVICE_CLASS_HUMIDITY,
|
||||||
|
PRESS: DEVICE_CLASS_PRESSURE,
|
||||||
|
LUX: DEVICE_CLASS_ILLUMINANCE
|
||||||
|
}
|
||||||
|
I2C_DEVICE_TYPES = {
|
||||||
|
"2": LUX, # BH1750
|
||||||
|
"3": LUX, # TSL2591
|
||||||
|
"7": LUX, # MAX44009
|
||||||
|
"70": LUX, # OPT3001
|
||||||
}
|
}
|
||||||
|
|
||||||
class NoPort(Exception):
|
class NoPort(Exception):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
@@ -57,6 +71,7 @@ class MegaD:
|
|||||||
nports=38,
|
nports=38,
|
||||||
inverted: typing.List[int] = None,
|
inverted: typing.List[int] = None,
|
||||||
update_all=True,
|
update_all=True,
|
||||||
|
poll_outs=False,
|
||||||
**kwargs,
|
**kwargs,
|
||||||
):
|
):
|
||||||
"""Initialize."""
|
"""Initialize."""
|
||||||
@@ -66,6 +81,7 @@ class MegaD:
|
|||||||
self.http.allowed_hosts |= {host}
|
self.http.allowed_hosts |= {host}
|
||||||
else:
|
else:
|
||||||
self.http = None
|
self.http = None
|
||||||
|
self.poll_outs = poll_outs
|
||||||
self.update_all = update_all if update_all is not None else True
|
self.update_all = update_all if update_all is not None else True
|
||||||
self.nports = nports
|
self.nports = nports
|
||||||
self.mqtt_inputs = mqtt_inputs
|
self.mqtt_inputs = mqtt_inputs
|
||||||
@@ -81,7 +97,7 @@ class MegaD:
|
|||||||
self._notif_lck = asyncio.Lock()
|
self._notif_lck = asyncio.Lock()
|
||||||
self.cnd = asyncio.Condition()
|
self.cnd = asyncio.Condition()
|
||||||
self.online = True
|
self.online = True
|
||||||
self.entities: typing.List[Entity] = []
|
self.entities: typing.List[BaseMegaEntity] = []
|
||||||
self.poll_interval = scan_interval
|
self.poll_interval = scan_interval
|
||||||
self.subs = None
|
self.subs = None
|
||||||
self.lg: logging.Logger = lg.getChild(self.id)
|
self.lg: logging.Logger = lg.getChild(self.id)
|
||||||
@@ -91,6 +107,7 @@ class MegaD:
|
|||||||
self.last_update = datetime.now()
|
self.last_update = datetime.now()
|
||||||
self._callbacks: typing.DefaultDict[int, typing.List[typing.Callable[[dict], typing.Coroutine]]] = defaultdict(list)
|
self._callbacks: typing.DefaultDict[int, typing.List[typing.Callable[[dict], typing.Coroutine]]] = defaultdict(list)
|
||||||
self._loop = loop
|
self._loop = loop
|
||||||
|
self._customize = None
|
||||||
self.values = {}
|
self.values = {}
|
||||||
self.last_port = None
|
self.last_port = None
|
||||||
self.updater = DataUpdateCoordinator(
|
self.updater = DataUpdateCoordinator(
|
||||||
@@ -138,6 +155,14 @@ class MegaD:
|
|||||||
await self.get_port(x.port, force_http=True, http_cmd=x.http_cmd)
|
await self.get_port(x.port, force_http=True, http_cmd=x.http_cmd)
|
||||||
ports.append(x.port)
|
ports.append(x.port)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def customize(self):
|
||||||
|
if self._customize is None:
|
||||||
|
c = self.hass.data.get(DOMAIN, {}).get(CONF_CUSTOM) or {}
|
||||||
|
c = c.get(self.id) or {}
|
||||||
|
self._customize = c
|
||||||
|
return self._customize
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def is_online(self):
|
def is_online(self):
|
||||||
return (datetime.now() - self.last_update).total_seconds() < (self.poll_interval + 10)
|
return (datetime.now() - self.last_update).total_seconds() < (self.poll_interval + 10)
|
||||||
@@ -168,8 +193,9 @@ class MegaD:
|
|||||||
if self.mqtt is None:
|
if self.mqtt is None:
|
||||||
await self.get_all_ports()
|
await self.get_all_ports()
|
||||||
await self.get_sensors(only_list=True)
|
await self.get_sensors(only_list=True)
|
||||||
return
|
elif self.poll_outs:
|
||||||
if len(self.sensors) > 0:
|
await self.get_all_ports(check_skip=True)
|
||||||
|
elif len(self.sensors) > 0:
|
||||||
await self.get_sensors()
|
await self.get_sensors()
|
||||||
else:
|
else:
|
||||||
await self.get_port(self.port_to_scan)
|
await self.get_port(self.port_to_scan)
|
||||||
@@ -259,15 +285,24 @@ class MegaD:
|
|||||||
except asyncio.TimeoutError:
|
except asyncio.TimeoutError:
|
||||||
self.lg.error(f'timeout when getting port {port}')
|
self.lg.error(f'timeout when getting port {port}')
|
||||||
|
|
||||||
async def get_all_ports(self, only_out=False):
|
@property
|
||||||
|
def ports(self):
|
||||||
|
return {e.port for e in self.entities}
|
||||||
|
|
||||||
|
async def get_all_ports(self, only_out=False, check_skip=False):
|
||||||
if not self.mqtt_inputs:
|
if not self.mqtt_inputs:
|
||||||
ret = await self.request(cmd='all')
|
ret = await self.request(cmd='all')
|
||||||
for port, x in enumerate(ret.split(';')):
|
for port, x in enumerate(ret.split(';')):
|
||||||
|
if check_skip and not port in self.ports:
|
||||||
|
continue
|
||||||
ret = self.parse_response(x)
|
ret = self.parse_response(x)
|
||||||
self.values[port] = ret
|
self.values[port] = ret
|
||||||
else:
|
elif not check_skip:
|
||||||
for x in range(self.nports + 1):
|
for x in range(self.nports + 1):
|
||||||
await self.get_port(x)
|
await self.get_port(x)
|
||||||
|
else:
|
||||||
|
for x in self.ports:
|
||||||
|
await self.get_port(x)
|
||||||
|
|
||||||
async def reboot(self, save=True):
|
async def reboot(self, save=True):
|
||||||
await self.save()
|
await self.save()
|
||||||
@@ -376,9 +411,12 @@ class MegaD:
|
|||||||
m = m.find(selected=True)['value']
|
m = m.find(selected=True)['value']
|
||||||
self._scanned[port] = (pty, m)
|
self._scanned[port] = (pty, m)
|
||||||
return pty, m
|
return pty, m
|
||||||
elif pty == '2':
|
elif pty in ('2', '4'): # эта часть не очень проработана, тут есть i2c который может работать неправильно
|
||||||
self._scanned[port] = (pty, '0')
|
m = tree.find('select', attrs={'name': 'd'})
|
||||||
return pty, '0'
|
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):
|
async def scan_ports(self, nports=37):
|
||||||
for x in range(0, nports+1):
|
for x in range(0, nports+1):
|
||||||
@@ -394,7 +432,7 @@ class MegaD:
|
|||||||
ret['binary_sensor'][port].append({})
|
ret['binary_sensor'][port].append({})
|
||||||
elif pty == "1" and (m in ['0', '1', '3'] or m is None):
|
elif pty == "1" and (m in ['0', '1', '3'] or m is None):
|
||||||
ret['light'][port].append({'dimmer': m == '1'})
|
ret['light'][port].append({'dimmer': m == '1'})
|
||||||
elif pty in ('3', '2'):
|
elif pty in ('3', '2', '4'):
|
||||||
try:
|
try:
|
||||||
http_cmd = 'get'
|
http_cmd = 'get'
|
||||||
values = await self.get_port(port, force_http=True)
|
values = await self.get_port(port, force_http=True)
|
||||||
@@ -413,7 +451,10 @@ class MegaD:
|
|||||||
if isinstance(values, str) and TEMP_PATT.search(values):
|
if isinstance(values, str) and TEMP_PATT.search(values):
|
||||||
values = {TEMP: values}
|
values = {TEMP: values}
|
||||||
elif not isinstance(values, dict):
|
elif not isinstance(values, dict):
|
||||||
values = {None: values}
|
if pty == '4' and m in I2C_DEVICE_TYPES:
|
||||||
|
values = {I2C_DEVICE_TYPES[m]: values}
|
||||||
|
else:
|
||||||
|
values = {None: values}
|
||||||
for key in values:
|
for key in values:
|
||||||
self.lg.debug(f'add sensor {key}')
|
self.lg.debug(f'add sensor {key}')
|
||||||
ret['sensor'][port].append(dict(
|
ret['sensor'][port].append(dict(
|
||||||
|
|||||||
@@ -15,5 +15,5 @@
|
|||||||
"@andvikt"
|
"@andvikt"
|
||||||
],
|
],
|
||||||
"issue_tracker": "https://github.com/andvikt/mega_hacs/issues",
|
"issue_tracker": "https://github.com/andvikt/mega_hacs/issues",
|
||||||
"version": "v0.3.12"
|
"version": "v0.3.13"
|
||||||
}
|
}
|
||||||
@@ -15,7 +15,7 @@ get_port:
|
|||||||
description: ID меги, можно оставить пустым, тогда будут сохранены все зарегистрированные меги
|
description: ID меги, можно оставить пустым, тогда будут сохранены все зарегистрированные меги
|
||||||
example: "mega"
|
example: "mega"
|
||||||
port:
|
port:
|
||||||
description: Номер порта (если не заполнять, будут запрошены все порты сразу)
|
description: Номер порта или список портов (если не заполнять, будут запрошены все порты сразу)
|
||||||
example: 1
|
example: 1
|
||||||
|
|
||||||
run_cmd:
|
run_cmd:
|
||||||
@@ -25,9 +25,10 @@ run_cmd:
|
|||||||
mega_id:
|
mega_id:
|
||||||
description: ID меги, можно оставить пустым, тогда будут сохранены все зарегистрированные меги
|
description: ID меги, можно оставить пустым, тогда будут сохранены все зарегистрированные меги
|
||||||
example: "mega"
|
example: "mega"
|
||||||
port:
|
|
||||||
description: Номер порта (это не порт, которым мы управляем, а порт с которого шлем команду)
|
|
||||||
example: 1
|
|
||||||
cmd:
|
cmd:
|
||||||
description: Любая поддерживаемая мегой команда
|
description: Любая поддерживаемая мегой команда
|
||||||
example: "1:0"
|
example: "1:0"
|
||||||
|
port:
|
||||||
|
description: (Deprecated, больше не нужен) Номер порта (это не порт, которым мы управляем, а порт с которого шлем команду)
|
||||||
|
example: 1
|
||||||
|
|
||||||
|
|||||||
@@ -14,7 +14,8 @@
|
|||||||
"invert": "[%key:common::config_flow::data::invert%]",
|
"invert": "[%key:common::config_flow::data::invert%]",
|
||||||
"mqtt_inputs": "[%key:common::config_flow::data::mqtt_inputs%]",
|
"mqtt_inputs": "[%key:common::config_flow::data::mqtt_inputs%]",
|
||||||
"nports": "[%key:common::config_flow::data::nports%]",
|
"nports": "[%key:common::config_flow::data::nports%]",
|
||||||
"update_all": "[%key:common::config_flow::data::update_all%]"
|
"update_all": "[%key:common::config_flow::data::update_all%]",
|
||||||
|
"poll_outs": "[%key:common::config_flow::data::poll_outs%]"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@@ -38,7 +39,8 @@
|
|||||||
"invert": "[%key:common::config_flow::data::invert%]",
|
"invert": "[%key:common::config_flow::data::invert%]",
|
||||||
"mqtt_inputs": "[%key:common::config_flow::data::mqtt_inputs%]",
|
"mqtt_inputs": "[%key:common::config_flow::data::mqtt_inputs%]",
|
||||||
"nports": "[%key:common::config_flow::data::nports%]",
|
"nports": "[%key:common::config_flow::data::nports%]",
|
||||||
"update_all": "[%key:common::config_flow::data::update_all%]"
|
"update_all": "[%key:common::config_flow::data::update_all%]",
|
||||||
|
"poll_outs": "[%key:common::config_flow::data::poll_outs%]"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -22,7 +22,8 @@
|
|||||||
"port_to_scan": "Port to poll aliveness (needed only if no sensors used)",
|
"port_to_scan": "Port to poll aliveness (needed only if no sensors used)",
|
||||||
"nports": "Number of ports",
|
"nports": "Number of ports",
|
||||||
"update_all": "Update all outs when input",
|
"update_all": "Update all outs when input",
|
||||||
"mqtt_inputs": "Use MQTT"
|
"mqtt_inputs": "Use MQTT",
|
||||||
|
"poll_outs": "Poll outs"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -35,7 +36,8 @@
|
|||||||
"port_to_scan": "Port to poll aliveness (needed only if no sensors used)",
|
"port_to_scan": "Port to poll aliveness (needed only if no sensors used)",
|
||||||
"reload": "Reload objects",
|
"reload": "Reload objects",
|
||||||
"mqtt_inputs": "Use MQTT",
|
"mqtt_inputs": "Use MQTT",
|
||||||
"update_all": "Update all outs when input"
|
"update_all": "Update all outs when input",
|
||||||
|
"poll_outs": "Poll outs"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -21,7 +21,8 @@
|
|||||||
"port_to_scan": "Порт, который сканируется когда нет датчиков",
|
"port_to_scan": "Порт, который сканируется когда нет датчиков",
|
||||||
"mqtt_inputs": "Использовать MQTT",
|
"mqtt_inputs": "Использовать MQTT",
|
||||||
"nports": "Кол-во портов",
|
"nports": "Кол-во портов",
|
||||||
"update_all": "Обновить все выходы когда срабатывает вход"
|
"update_all": "Обновить все выходы когда срабатывает вход",
|
||||||
|
"poll_outs": "Обновлять выходы (регулярно)"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -36,7 +37,8 @@
|
|||||||
"invert": "Список портов (через ,) с инвертированной логикой",
|
"invert": "Список портов (через ,) с инвертированной логикой",
|
||||||
"mqtt_inputs": "Использовать MQTT",
|
"mqtt_inputs": "Использовать MQTT",
|
||||||
"nports": "Кол-во портов",
|
"nports": "Кол-во портов",
|
||||||
"update_all": "Обновить все выходы когда срабатывает вход"
|
"update_all": "Обновить все выходы когда срабатывает вход",
|
||||||
|
"poll_outs": "Обновлять выходы (регулярно)"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -21,7 +21,8 @@
|
|||||||
"port_to_scan": "Порт для сканування при відсутності датчиків",
|
"port_to_scan": "Порт для сканування при відсутності датчиків",
|
||||||
"mqtt_inputs": "Використовувати MQTT",
|
"mqtt_inputs": "Використовувати MQTT",
|
||||||
"nports": "Кількість портів",
|
"nports": "Кількість портів",
|
||||||
"update_all": "Оновити всі виходи коли спрацьовує вхід"
|
"update_all": "Оновити всі виходи коли спрацьовує вхід",
|
||||||
|
"poll_outs": "Оновити виходи"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -36,7 +37,8 @@
|
|||||||
"invert": "Список портів з інвертованою логікою (через ,)",
|
"invert": "Список портів з інвертованою логікою (через ,)",
|
||||||
"mqtt_inputs": "Використовувати MQTT",
|
"mqtt_inputs": "Використовувати MQTT",
|
||||||
"nports": "Кількість портів",
|
"nports": "Кількість портів",
|
||||||
"update_all": "Оновити всі виходи коли спрацьовує вхід"
|
"update_all": "Оновити всі виходи коли спрацьовує вхід",
|
||||||
|
"poll_outs": "Оновити виходи"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -73,6 +73,9 @@ mega:
|
|||||||
Начиная с версии `0.3.1` интеграция стала поддерживать обратную связь без mqtt, используя http-сервер. Для этого в настройках
|
Начиная с версии `0.3.1` интеграция стала поддерживать обратную связь без mqtt, используя http-сервер. Для этого в настройках
|
||||||
интеграции необходимо снять галку с `использовать mqtt`
|
интеграции необходимо снять галку с `использовать mqtt`
|
||||||
|
|
||||||
|
**Внимание!** Не используйте srv loop на контроллере, это может приводить к ложным срабатываниям входов. Вместо srv loop
|
||||||
|
интеграция будет сама обновлять все состояния портов с заданным интервалом
|
||||||
|
|
||||||
В самой меге необходимо прописать настройки:
|
В самой меге необходимо прописать настройки:
|
||||||
```yaml
|
```yaml
|
||||||
srv: "192.168.1.4:8123" # ip:port вашего HA
|
srv: "192.168.1.4:8123" # ip:port вашего HA
|
||||||
@@ -155,12 +158,15 @@ mega:
|
|||||||
```
|
```
|
||||||
Возможные варианты поля `type`:
|
Возможные варианты поля `type`:
|
||||||
- `long`: долгое нажатие
|
- `long`: долгое нажатие
|
||||||
- `release`: размыкание (с гарантией что не было долгого нажатия)
|
- `release`: размыкание (с гарантией** что не было долгого нажатия)
|
||||||
- `long_release`: размыкание после долгого нажатия
|
- `long_release`: размыкание после долгого нажатия
|
||||||
- `press`: замыкание
|
- `press`: замыкание
|
||||||
- `single`: одинарный клик (в режиме кликов)
|
- `single`: одинарный клик (в режиме кликов)
|
||||||
- `double`: двойной клик
|
- `double`: двойной клик
|
||||||
|
|
||||||
|
**гарантия есть только при использовании http-метода синхронизации, mqtt не гарантирует порядок доставки сообщений, хотя
|
||||||
|
маловероятно, что порядок будет нарушен, но все же сам протокол не дает таких гарантий.
|
||||||
|
|
||||||
Чтобы понять, какие события происходят, лучше всего воспользоваться панелью разработчика и подписаться
|
Чтобы понять, какие события происходят, лучше всего воспользоваться панелью разработчика и подписаться
|
||||||
на вкладке события на событие `mega.sensor`, понажимать кнопки.
|
на вкладке события на событие `mega.sensor`, понажимать кнопки.
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user