5 Commits

Author SHA1 Message Date
sususweet
273d4e41bf fix: reform water heater entity. 2025-09-30 00:23:56 +08:00
sususweet
e9a6c65787 fix: release lua cjson critical limit. 2025-09-30 00:07:03 +08:00
sususweet
beaf837e3b feat: version 0.1.1 2025-09-29 23:55:43 +08:00
sususweet
a7efce2d26 feat: add indoor status of T0xAC. 2025-09-29 23:44:07 +08:00
sususweet
9ff67da49b feat: fallback to cloud control when lua script encounter error. 2025-09-29 23:07:12 +08:00
11 changed files with 65 additions and 237 deletions

View File

@@ -131,16 +131,16 @@ async def async_setup(hass: HomeAssistant, config: ConfigType):
hass.data.setdefault(DOMAIN, {})
cjson = os.getcwd() + "/cjson.lua"
bit = os.getcwd() + "/bit.lua"
if not os.path.exists(cjson):
from .const import CJSON_LUA
cjson_lua = base64.b64decode(CJSON_LUA.encode("utf-8")).decode("utf-8")
with open(cjson, "wt") as fp:
fp.write(cjson_lua)
if not os.path.exists(bit):
from .const import BIT_LUA
bit_lua = base64.b64decode(BIT_LUA.encode("utf-8")).decode("utf-8")
with open(bit, "wt") as fp:
fp.write(bit_lua)
# if not os.path.exists(cjson):
from .const import CJSON_LUA
cjson_lua = base64.b64decode(CJSON_LUA.encode("utf-8")).decode("utf-8")
with open(cjson, "wt") as fp:
fp.write(cjson_lua)
# if not os.path.exists(bit):
from .const import BIT_LUA
bit_lua = base64.b64decode(BIT_LUA.encode("utf-8")).decode("utf-8")
with open(bit, "wt") as fp:
fp.write(bit_lua)
return True

File diff suppressed because one or more lines are too long

View File

@@ -140,8 +140,13 @@ class MiedaDevice(threading.Thread):
for attr in self._centralized:
new_status[attr] = self._attributes.get(attr)
new_status[attribute] = value
if set_cmd := self._lua_runtime.build_control(new_status):
await self._build_send(set_cmd)
try:
if set_cmd := self._lua_runtime.build_control(new_status):
await self._build_send(set_cmd)
except Exception as e:
cloud = self._cloud
if cloud and hasattr(cloud, "send_device_control"):
await cloud.send_device_control(self._device_id, control=new_status, status=self._attributes)
async def set_attributes(self, attributes):
new_status = {}
@@ -153,8 +158,13 @@ class MiedaDevice(threading.Thread):
has_new = True
new_status[attribute] = value
if has_new:
if set_cmd := self._lua_runtime.build_control(new_status):
await self._build_send(set_cmd)
try:
if set_cmd := self._lua_runtime.build_control(new_status):
await self._build_send(set_cmd)
except Exception as e:
cloud = self._cloud
if cloud and hasattr(cloud, "send_device_control"):
await cloud.send_device_control(self._device_id, control=new_status, status=self._attributes)
def set_ip_address(self, ip_address):
MideaLogger.debug(f"Update IP address to {ip_address}")

View File

@@ -6,7 +6,7 @@ from .logger import MideaLogger
class LuaRuntime:
def __init__(self, file):
self._runtimes = lupa.LuaRuntime()
self._runtimes = lupa.lua51.LuaRuntime()
string = f'dofile("{file}")'
self._runtimes.execute(string)
self._lock = threading.Lock()

View File

@@ -6,7 +6,7 @@ from homeassistant.components.switch import SwitchDeviceClass
DEVICE_MAPPING = {
"default": {
"rationale": ["off", "on"],
"queries": [{}],
"queries": [{}, {"query_type":"run_status"}],
"centralized": [
"power", "temperature", "small_temperature", "mode", "eco",
"comfort_power_save", "strong_wind",
@@ -77,11 +77,11 @@ DEVICE_MAPPING = {
"unit_of_measurement": UnitOfTemperature.CELSIUS,
"state_class": SensorStateClass.MEASUREMENT
},
"outdoor_temperature": {
"device_class": SensorDeviceClass.TEMPERATURE,
"unit_of_measurement": UnitOfTemperature.CELSIUS,
"indoor_humidity": {
"device_class": SensorDeviceClass.HUMIDITY,
"unit_of_measurement": "%",
"state_class": SensorStateClass.MEASUREMENT
},
}
}
}
},
@@ -155,8 +155,12 @@ DEVICE_MAPPING = {
}
},
Platform.SENSOR: {
"indoor_temperature": {
"device_class": SensorDeviceClass.TEMPERATURE,
"unit_of_measurement": UnitOfTemperature.CELSIUS,
"state_class": SensorStateClass.MEASUREMENT
},
"outdoor_temperature": {
"name": "室外机温度",
"device_class": SensorDeviceClass.TEMPERATURE,
"unit_of_measurement": UnitOfTemperature.CELSIUS,
"state_class": SensorStateClass.MEASUREMENT

View File

@@ -127,19 +127,19 @@ DEVICE_MAPPING = {
"state_class": SensorStateClass.MEASUREMENT
},
"people_number": {
"device_class": SensorDeviceClass.NONE,
"device_class": SensorDeviceClass.DATA_RATE,
"state_class": SensorStateClass.MEASUREMENT
},
"steam_quantity": {
"device_class": SensorDeviceClass.NONE,
"device_class": SensorDeviceClass.ENUM,
"state_class": SensorStateClass.MEASUREMENT
},
"totalstep": {
"device_class": SensorDeviceClass.NONE,
"device_class": SensorDeviceClass.ENUM,
"state_class": SensorStateClass.MEASUREMENT
},
"stepnum": {
"device_class": SensorDeviceClass.NONE,
"device_class": SensorDeviceClass.ENUM,
"state_class": SensorStateClass.MEASUREMENT
},
"hour_set": {
@@ -158,23 +158,23 @@ DEVICE_MAPPING = {
"state_class": SensorStateClass.MEASUREMENT
},
"ota": {
"device_class": SensorDeviceClass.NONE,
"device_class": SensorDeviceClass.ENUM,
"state_class": SensorStateClass.MEASUREMENT
},
"error_code": {
"device_class": SensorDeviceClass.NONE,
"device_class": SensorDeviceClass.ENUM,
"state_class": SensorStateClass.MEASUREMENT
},
"version": {
"device_class": SensorDeviceClass.NONE,
"device_class": SensorDeviceClass.ENUM,
"state_class": SensorStateClass.MEASUREMENT
},
"cbs_version": {
"device_class": SensorDeviceClass.NONE,
"device_class": SensorDeviceClass.ENUM,
"state_class": SensorStateClass.MEASUREMENT
},
"cloudmenuid": {
"device_class": SensorDeviceClass.NONE,
"device_class": SensorDeviceClass.ENUM,
"state_class": SensorStateClass.MEASUREMENT
}
}

View File

@@ -50,48 +50,15 @@ DEVICE_MAPPING = {
}
},
Platform.SWITCH: {
"music": {
"device_class": SwitchDeviceClass.SWITCH,
},
"ti_protect": {
"device_class": SwitchDeviceClass.SWITCH,
},
"fast_wash": {
"device_class": SwitchDeviceClass.SWITCH,
},
"ali_manager": {
"device_class": SwitchDeviceClass.SWITCH,
},
"heat": {
"device_class": SwitchDeviceClass.SWITCH,
},
"ele_exception": {
"device_class": SwitchDeviceClass.SWITCH,
},
"communication_error": {
"device_class": SwitchDeviceClass.SWITCH,
},
"eplus": {
"device_class": SwitchDeviceClass.SWITCH,
},
"summer": {
"device_class": SwitchDeviceClass.SWITCH,
},
"winter": {
"device_class": SwitchDeviceClass.SWITCH,
},
"efficient": {
"device_class": SwitchDeviceClass.SWITCH,
},
"night": {
"device_class": SwitchDeviceClass.SWITCH,
},
"bath_person": {
"device_class": SwitchDeviceClass.SWITCH,
},
"cloud": {
"device_class": SwitchDeviceClass.SWITCH,
},
"bath": {
"device_class": SwitchDeviceClass.SWITCH,
},
@@ -101,87 +68,12 @@ DEVICE_MAPPING = {
"whole_heat": {
"device_class": SwitchDeviceClass.SWITCH,
},
"sterilization": {
"device_class": SwitchDeviceClass.SWITCH,
},
"frequency_hot": {
"device_class": SwitchDeviceClass.SWITCH,
},
"scene": {
"device_class": SwitchDeviceClass.SWITCH,
},
"big_water": {
"device_class": SwitchDeviceClass.SWITCH,
},
"wash": {
"device_class": SwitchDeviceClass.SWITCH,
},
"negative_ions": {
"device_class": SwitchDeviceClass.SWITCH,
},
"screen_off": {
"device_class": SwitchDeviceClass.SWITCH,
},
"t_hot": {
"device_class": SwitchDeviceClass.SWITCH,
},
"baby_wash": {
"device_class": SwitchDeviceClass.SWITCH,
},
"dad_wash": {
"device_class": SwitchDeviceClass.SWITCH,
},
"mom_wash": {
"device_class": SwitchDeviceClass.SWITCH,
},
"wash_with_temp": {
"device_class": SwitchDeviceClass.SWITCH,
},
"single_wash": {
"device_class": SwitchDeviceClass.SWITCH,
},
"people_wash": {
"device_class": SwitchDeviceClass.SWITCH,
},
"one_egg": {
"device_class": SwitchDeviceClass.SWITCH,
},
"two_egg": {
"device_class": SwitchDeviceClass.SWITCH,
},
"always_fell": {
"device_class": SwitchDeviceClass.SWITCH,
},
"smart_sterilize": {
"device_class": SwitchDeviceClass.SWITCH,
},
"sound_dad": {
"device_class": SwitchDeviceClass.SWITCH,
},
"door_status": {
"device_class": SwitchDeviceClass.SWITCH,
},
"limit_error": {
"device_class": SwitchDeviceClass.SWITCH,
},
"sensor_error": {
"device_class": SwitchDeviceClass.SWITCH,
},
"auto_off": {
"device_class": SwitchDeviceClass.SWITCH,
},
"clean": {
"device_class": SwitchDeviceClass.SWITCH,
},
"cloud_appoint": {
"device_class": SwitchDeviceClass.SWITCH,
},
"protect": {
"device_class": SwitchDeviceClass.SWITCH,
},
"midea_manager": {
"device_class": SwitchDeviceClass.SWITCH,
},
"sleep": {
"device_class": SwitchDeviceClass.SWITCH,
},
@@ -191,9 +83,6 @@ DEVICE_MAPPING = {
"shower": {
"device_class": SwitchDeviceClass.SWITCH,
},
"scroll_hot": {
"device_class": SwitchDeviceClass.SWITCH,
},
"fast_hot_power": {
"device_class": SwitchDeviceClass.SWITCH,
},
@@ -206,54 +95,12 @@ DEVICE_MAPPING = {
"water_flow": {
"device_class": SwitchDeviceClass.SWITCH,
},
"appoint_wash": {
"device_class": SwitchDeviceClass.SWITCH,
},
"now_wash": {
"device_class": SwitchDeviceClass.SWITCH,
},
"get_time": {
"device_class": SwitchDeviceClass.SWITCH,
},
"get_temp": {
"device_class": SwitchDeviceClass.SWITCH,
},
"warm_power": {
"device_class": SwitchDeviceClass.SWITCH,
},
"sterilize_high_temp": {
"device_class": SwitchDeviceClass.SWITCH,
},
"bottom_heat": {
"device_class": SwitchDeviceClass.SWITCH,
},
"top_heat": {
"device_class": SwitchDeviceClass.SWITCH,
},
"show_h": {
"device_class": SwitchDeviceClass.SWITCH,
},
"uv_sterilize": {
"device_class": SwitchDeviceClass.SWITCH,
},
"need_discharge": {
"device_class": SwitchDeviceClass.SWITCH,
},
"elec_warning": {
"device_class": SwitchDeviceClass.SWITCH,
},
"water_cyclic": {
"device_class": SwitchDeviceClass.SWITCH,
},
"tech_water": {
"device_class": SwitchDeviceClass.SWITCH,
},
"protect_show": {
"device_class": SwitchDeviceClass.SWITCH,
},
"appoint_power": {
"device_class": SwitchDeviceClass.SWITCH,
}
},
Platform.SELECT: {
"mode": {
@@ -365,22 +212,11 @@ DEVICE_MAPPING = {
"unit_of_measurement": UnitOfTime.MINUTES,
"state_class": SensorStateClass.MEASUREMENT
},
"version": {
"device_class": SensorDeviceClass.ENUM
},
"tds_value": {
"device_class": SensorDeviceClass.WATER,
"unit_of_measurement": "ppm",
"state_class": SensorStateClass.MEASUREMENT
},
"scene_id": {
"device_class": SensorDeviceClass.ENUM
},
"volume": {
"device_class": SensorDeviceClass.SOUND_PRESSURE,
"unit_of_measurement": "%",
"state_class": SensorStateClass.MEASUREMENT
},
"heat_water_level": {
"device_class": SensorDeviceClass.WATER,
"unit_of_measurement": "%",

View File

@@ -107,7 +107,7 @@ DEVICE_MAPPING = {
},
Platform.SENSOR: {
"real_gear": {
"device_class": SensorDeviceClass.NONE,
"device_class": SensorDeviceClass.ENUM,
"state_class": SensorStateClass.MEASUREMENT
},
"dust_life_time": {
@@ -130,7 +130,7 @@ DEVICE_MAPPING = {
"state_class": SensorStateClass.MEASUREMENT
},
"error_code": {
"device_class": SensorDeviceClass.NONE,
"device_class": SensorDeviceClass.ENUM,
"state_class": SensorStateClass.MEASUREMENT
},
"temperature_feedback": {
@@ -139,7 +139,7 @@ DEVICE_MAPPING = {
"state_class": SensorStateClass.MEASUREMENT
},
"water_feedback": {
"device_class": SensorDeviceClass.NONE,
"device_class": SensorDeviceClass.ENUM,
"state_class": SensorStateClass.MEASUREMENT
},
"timer_off_hour": {
@@ -163,7 +163,7 @@ DEVICE_MAPPING = {
"state_class": SensorStateClass.MEASUREMENT
},
"version": {
"device_class": SensorDeviceClass.NONE,
"device_class": SensorDeviceClass.ENUM,
"state_class": SensorStateClass.MEASUREMENT
},
"pm25": {
@@ -172,26 +172,26 @@ DEVICE_MAPPING = {
"state_class": SensorStateClass.MEASUREMENT
},
"ud_swing_angle": {
"device_class": SensorDeviceClass.NONE,
"device_class": SensorDeviceClass.ENUM,
"state_class": SensorStateClass.MEASUREMENT
},
"lr_diy_down_percent": {
"device_class": SensorDeviceClass.NONE,
"device_class": SensorDeviceClass.ENUM,
"unit_of_measurement": PERCENTAGE,
"state_class": SensorStateClass.MEASUREMENT
},
"lr_diy_up_percent": {
"device_class": SensorDeviceClass.NONE,
"device_class": SensorDeviceClass.ENUM,
"unit_of_measurement": PERCENTAGE,
"state_class": SensorStateClass.MEASUREMENT
},
"ud_diy_down_percent": {
"device_class": SensorDeviceClass.NONE,
"device_class": SensorDeviceClass.ENUM,
"unit_of_measurement": PERCENTAGE,
"state_class": SensorStateClass.MEASUREMENT
},
"ud_diy_up_percent": {
"device_class": SensorDeviceClass.NONE,
"device_class": SensorDeviceClass.ENUM,
"unit_of_measurement": PERCENTAGE,
"state_class": SensorStateClass.MEASUREMENT
}

View File

@@ -7,5 +7,5 @@
"iot_class": "cloud_push",
"issue_tracker": "https://github.com/sususweet/midea-meiju-codec/issues",
"requirements": ["lupa>=2.0"],
"version": "v0.1.0"
"version": "v0.1.1"
}

View File

@@ -5,7 +5,6 @@ from homeassistant.core import HomeAssistant
from homeassistant.helpers.entity_platform import AddEntitiesCallback
from .const import DOMAIN
from .core.logger import MideaLogger
from .midea_entity import MideaEntity
from . import load_device_config

View File

@@ -1,14 +1,23 @@
from homeassistant.components.water_heater import WaterHeaterEntity, WaterHeaterEntityFeature
from homeassistant.config_entries import ConfigEntry
from homeassistant.const import (
Platform,
ATTR_TEMPERATURE
)
from homeassistant.core import HomeAssistant
from homeassistant.helpers.entity_platform import AddEntitiesCallback
from .const import DOMAIN
from .midea_entity import MideaEntity
from . import load_device_config
async def async_setup_entry(hass, config_entry, async_add_entities):
async def async_setup_entry(
hass: HomeAssistant,
config_entry: ConfigEntry,
async_add_entities: AddEntitiesCallback,
) -> None:
"""Set up water heater entities for Midea devices."""
account_bucket = hass.data.get(DOMAIN, {}).get("accounts", {}).get(config_entry.entry_id)
if not account_bucket:
async_add_entities([])
@@ -51,30 +60,6 @@ class MideaWaterHeaterEntityEntity(MideaEntity, WaterHeaterEntity):
self._manufacturer = manufacturer
self._rationale = rationale
self._config = config
# Legacy compatibility: register update and restore display attributes
if self._device:
self._device.register_update(self.update_state)
if (rationale_local := self._config.get("rationale")) is not None:
self._rationale = rationale_local
if self._rationale is None:
self._rationale = ["off", "on"]
self._attr_native_unit_of_measurement = self._config.get("unit_of_measurement")
self._attr_device_class = self._config.get("device_class")
self._attr_state_class = self._config.get("state_class")
self._attr_icon = self._config.get("icon")
from .const import DOMAIN as _DOMAIN
self._attr_unique_id = f"{_DOMAIN}.{self._device.device_id}_{self._entity_key}"
self._attr_device_info = {
"manufacturer": "Midea" if self._manufacturer is None else self._manufacturer,
"model": f"{self._device.model}",
"identifiers": {( _DOMAIN, self._device.device_id)},
"name": self._device.device_name
}
name = self._config.get("name")
if name is None:
name = self._entity_key.replace("_", " ").title()
self._attr_name = f"{self._device.device_name} {name}"
self.entity_id = self._attr_unique_id
self._key_power = self._config.get("power")
self._key_operation_list = self._config.get("operation_list")
self._key_min_temp = self._config.get("min_temp")
@@ -167,9 +152,3 @@ class MideaWaterHeaterEntityEntity(MideaEntity, WaterHeaterEntity):
if new_status:
await self.async_set_attributes(new_status)
def update_state(self, status):
try:
self.schedule_update_ha_state()
except Exception:
pass