* Initial commit
* Correct settings for config flow
* Use scan interval
* Store proper data
* Remove circular dependency
* Remove circular dependency
* Integration can be initialized
* Fix defaults
* Add setup entry
* Add setup entry
* Dont block forever
* Poll during async_setup_entry
* Remove not needed async methods
* Add debug info
* Parse binary data
* Parse binary data
* Use data to update device
* Use data to update device
* Add CCM15DeviceState
* Use DataCoordinator
* Use DataCoordinator
* Use DataCoordinator
* Use CoordinatorEntity
* Use CoordinatorEntity
* Call update API
* Call update API
* Call update API
* Call update API
* Use dataclass
* Use dataclass
* Use dataclass
* Use dataclass
* Use dataclass
* Use dataclass
* Use dataclass
* Use dataclass
* Fix bugs
* Implement swing
* Support swing mode, read only
* Add unit test
* Swing should work
* Set swing mode
* Add DeviceInfo
* Add error code
* Add error code
* Add error code
* Add error code
* Initial commit
* Refactor
* Remove comment code
* Try remove circular ref
* Try remove circular ref
* Remove circular ref
* Fix bug
* Fix tests
* Fix tests
* Increase test coverage
* Increase test coverage
* Increase test coverrage
* Add more unit tests
* Increase coverage
* Update coordinator.py
* Fix ruff
* Set unit of temperature
* Add bounds check
* Fix unit tests
* Add test coverage
* Use Py-ccm15
* Update tests
* Upgrade dependency
* Apply PR feedback
* Upgrade dependency
* Upgrade dependency
* Upgrade dependency
* Force ruff
* Delete not needed consts
* Fix mypy
* Update homeassistant/components/ccm15/coordinator.py
Co-authored-by: Robert Resch <robert@resch.dev>
* Apply PR Feedback
* Apply PR Feedback
* Apply PR Feedback
* Apply PR Feedback
* Apply PR Feedback
* Apply PR Feedback
* Fix unit tests
* Move climate instance
* Revert "Move climate instance"
This reverts commit cc5b9916b7
.
* Apply PR feedback
* Apply PR Feedback
* Remove scan internal parameter
* Update homeassistant/components/ccm15/coordinator.py
Co-authored-by: Robert Resch <robert@resch.dev>
* Remove empty keys
* Fix tests
* Use attr fields
* Try refactor
* Check for multiple hosts
* Check for duplicates
* Fix tests
* Use PRECISION_WHOLE
* Use str(ac_index)
* Move {self._ac_host}.{self._ac_index} to construtor
* Make it fancy
* Update homeassistant/components/ccm15/coordinator.py
Co-authored-by: Joost Lekkerkerker <joostlek@outlook.com>
* Move const to class variables
* Use actual config host
* Move device info to construtor
* Update homeassistant/components/ccm15/climate.py
Co-authored-by: Joost Lekkerkerker <joostlek@outlook.com>
* Set name to none, dont ask for poll
* Undo name change
* Dont use coordinator in config flow
* Dont use coordinator in config flow
* Check already configured
* Apply PR comments
* Move above
* Use device info name
* Update tests/components/ccm15/test_coordinator.py
Co-authored-by: Joost Lekkerkerker <joostlek@outlook.com>
* Update tests/components/ccm15/test_config_flow.py
Co-authored-by: Joost Lekkerkerker <joostlek@outlook.com>
* Apply feedback
* Remove logger debug calls
* Add new test to check for dupplicates
* Test error
* Use better name for test
* Update homeassistant/components/ccm15/config_flow.py
Co-authored-by: Joost Lekkerkerker <joostlek@outlook.com>
* Update homeassistant/components/ccm15/climate.py
Co-authored-by: Joost Lekkerkerker <joostlek@outlook.com>
* Update homeassistant/components/ccm15/config_flow.py
Co-authored-by: Joost Lekkerkerker <joostlek@outlook.com>
* Use prop data for all getters
* Fix tests
* Improve tests
* Improve tests, v2
* Replace log message by comment
* No need to do bounds check
* Update config_flow.py
* Update test_config_flow.py
* Update test_coordinator.py
* Update test_coordinator.py
* Create test_climate.py
* Delete tests/components/ccm15/test_coordinator.py
* Update coordinator.py
* Update __init__.py
* Create test_climate.ambr
* Update conftest.py
* Update test_climate.py
* Create test_init.py
* Update .coveragerc
* Update __init__.py
* We need to check bounds after all
* Add more test coverage
* Test is not None
* Use better naming
* fix tests
* Add available property
* Update homeassistant/components/ccm15/climate.py
Co-authored-by: Joost Lekkerkerker <joostlek@outlook.com>
* Use snapshots to simulate netwrok failure or power failure
* Remove not needed test
* Use walrus
---------
Co-authored-by: Robert Resch <robert@resch.dev>
Co-authored-by: Joost Lekkerkerker <joostlek@outlook.com>
76 lines
2.8 KiB
Python
76 lines
2.8 KiB
Python
"""Climate device for CCM15 coordinator."""
|
|
import datetime
|
|
import logging
|
|
|
|
from ccm15 import CCM15Device, CCM15DeviceState, CCM15SlaveDevice
|
|
import httpx
|
|
|
|
from homeassistant.components.climate import HVACMode
|
|
from homeassistant.core import HomeAssistant
|
|
from homeassistant.helpers.update_coordinator import DataUpdateCoordinator, UpdateFailed
|
|
|
|
from .const import (
|
|
CONST_FAN_CMD_MAP,
|
|
CONST_STATE_CMD_MAP,
|
|
DEFAULT_INTERVAL,
|
|
DEFAULT_TIMEOUT,
|
|
)
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
class CCM15Coordinator(DataUpdateCoordinator[CCM15DeviceState]):
|
|
"""Class to coordinate multiple CCM15Climate devices."""
|
|
|
|
def __init__(self, hass: HomeAssistant, host: str, port: int) -> None:
|
|
"""Initialize the coordinator."""
|
|
super().__init__(
|
|
hass,
|
|
_LOGGER,
|
|
name=host,
|
|
update_interval=datetime.timedelta(seconds=DEFAULT_INTERVAL),
|
|
)
|
|
self._ccm15 = CCM15Device(host, port, DEFAULT_TIMEOUT)
|
|
self._host = host
|
|
|
|
def get_host(self) -> str:
|
|
"""Get the host."""
|
|
return self._host
|
|
|
|
async def _async_update_data(self) -> CCM15DeviceState:
|
|
"""Fetch data from Rain Bird device."""
|
|
try:
|
|
return await self._fetch_data()
|
|
except httpx.RequestError as err: # pragma: no cover
|
|
raise UpdateFailed("Error communicating with Device") from err
|
|
|
|
async def _fetch_data(self) -> CCM15DeviceState:
|
|
"""Get the current status of all AC devices."""
|
|
return await self._ccm15.get_status_async()
|
|
|
|
async def async_set_state(self, ac_index: int, state: str, value: int) -> None:
|
|
"""Set new target states."""
|
|
if await self._ccm15.async_set_state(ac_index, state, value):
|
|
await self.async_request_refresh()
|
|
|
|
def get_ac_data(self, ac_index: int) -> CCM15SlaveDevice | None:
|
|
"""Get ac data from the ac_index."""
|
|
if ac_index < 0 or ac_index >= len(self.data.devices):
|
|
# Network latency may return an empty or incomplete array
|
|
return None
|
|
return self.data.devices[ac_index]
|
|
|
|
async def async_set_hvac_mode(self, ac_index, hvac_mode: HVACMode) -> None:
|
|
"""Set the hvac mode."""
|
|
_LOGGER.debug("Set Hvac[%s]='%s'", ac_index, str(hvac_mode))
|
|
await self.async_set_state(ac_index, "mode", CONST_STATE_CMD_MAP[hvac_mode])
|
|
|
|
async def async_set_fan_mode(self, ac_index, fan_mode: str) -> None:
|
|
"""Set the fan mode."""
|
|
_LOGGER.debug("Set Fan[%s]='%s'", ac_index, fan_mode)
|
|
await self.async_set_state(ac_index, "fan", CONST_FAN_CMD_MAP[fan_mode])
|
|
|
|
async def async_set_temperature(self, ac_index, temp) -> None:
|
|
"""Set the target temperature mode."""
|
|
_LOGGER.debug("Set Temp[%s]='%s'", ac_index, temp)
|
|
await self.async_set_state(ac_index, "temp", temp)
|