-
-
Notifications
You must be signed in to change notification settings - Fork 37.6k
Add Viessmann ViCare Climate platform #26151
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
Merged
Merged
Changes from all commits
Commits
Show all changes
8 commits
Select commit
Hold shift + click to select a range
6ef8805
Add Viessmann ViCare Climate platform
oischinger 9ce9b38
Add water_heater and fix review comments
oischinger e4d79df
Move PyVicare API creation to component
oischinger 1527b02
More review fixes
oischinger ce54cd4
Return false if api creation fails
oischinger 8175143
Fix logging format
oischinger e6bb2cc
Update PyVicare 0.1.1 to fix json issues
oischinger 50cace4
Formatting fixes
oischinger File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| Original file line number | Diff line number | Diff line change |
|---|---|---|
| @@ -0,0 +1,58 @@ | ||
| """The ViCare integration.""" | ||
| import logging | ||
|
|
||
| import voluptuous as vol | ||
| from PyViCare.PyViCareDevice import Device | ||
|
|
||
| import homeassistant.helpers.config_validation as cv | ||
| from homeassistant.const import CONF_USERNAME, CONF_PASSWORD, CONF_NAME | ||
| from homeassistant.helpers import discovery | ||
|
|
||
| _LOGGER = logging.getLogger(__name__) | ||
|
|
||
| VICARE_PLATFORMS = ["climate", "water_heater"] | ||
|
|
||
| DOMAIN = "vicare" | ||
| VICARE_API = "api" | ||
| VICARE_NAME = "name" | ||
|
|
||
| CONF_CIRCUIT = "circuit" | ||
|
|
||
| CONFIG_SCHEMA = vol.Schema( | ||
| { | ||
| DOMAIN: vol.Schema( | ||
| { | ||
| vol.Required(CONF_USERNAME): cv.string, | ||
| vol.Required(CONF_PASSWORD): cv.string, | ||
| vol.Optional(CONF_CIRCUIT): int, | ||
| vol.Optional(CONF_NAME, default="ViCare"): cv.string, | ||
| } | ||
| ) | ||
| }, | ||
| extra=vol.ALLOW_EXTRA, | ||
| ) | ||
|
|
||
|
|
||
| def setup(hass, config): | ||
| """Create the ViCare component.""" | ||
| conf = config[DOMAIN] | ||
| params = {"token_file": "/tmp/vicare_token.save"} | ||
| if conf.get(CONF_CIRCUIT) is not None: | ||
| params["circuit"] = conf[CONF_CIRCUIT] | ||
|
|
||
| try: | ||
| vicare_api = Device(conf[CONF_USERNAME], conf[CONF_PASSWORD], **params) | ||
| except AttributeError: | ||
| _LOGGER.error( | ||
| "Failed to create PyViCare API client. Please check your credentials." | ||
| ) | ||
| return False | ||
|
|
||
| hass.data[DOMAIN] = {} | ||
| hass.data[DOMAIN][VICARE_API] = vicare_api | ||
| hass.data[DOMAIN][VICARE_NAME] = conf[CONF_NAME] | ||
|
|
||
| for platform in VICARE_PLATFORMS: | ||
| discovery.load_platform(hass, platform, DOMAIN, {}, config) | ||
|
|
||
| return True | ||
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| Original file line number | Diff line number | Diff line change |
|---|---|---|
| @@ -0,0 +1,207 @@ | ||
| """Viessmann ViCare climate device.""" | ||
| import logging | ||
|
|
||
| from homeassistant.components.climate import ClimateDevice | ||
| from homeassistant.components.climate.const import ( | ||
| SUPPORT_PRESET_MODE, | ||
| SUPPORT_TARGET_TEMPERATURE, | ||
| PRESET_ECO, | ||
| PRESET_COMFORT, | ||
| HVAC_MODE_OFF, | ||
| HVAC_MODE_HEAT, | ||
| HVAC_MODE_AUTO, | ||
| ) | ||
| from homeassistant.const import TEMP_CELSIUS, ATTR_TEMPERATURE, PRECISION_WHOLE | ||
|
|
||
| from . import DOMAIN as VICARE_DOMAIN | ||
| from . import VICARE_API | ||
| from . import VICARE_NAME | ||
|
|
||
| _LOGGER = logging.getLogger(__name__) | ||
|
|
||
| VICARE_MODE_DHW = "dhw" | ||
| VICARE_MODE_DHWANDHEATING = "dhwAndHeating" | ||
| VICARE_MODE_FORCEDREDUCED = "forcedReduced" | ||
| VICARE_MODE_FORCEDNORMAL = "forcedNormal" | ||
| VICARE_MODE_OFF = "standby" | ||
|
|
||
| VICARE_PROGRAM_ACTIVE = "active" | ||
| VICARE_PROGRAM_COMFORT = "comfort" | ||
| VICARE_PROGRAM_ECO = "eco" | ||
| VICARE_PROGRAM_EXTERNAL = "external" | ||
| VICARE_PROGRAM_HOLIDAY = "holiday" | ||
| VICARE_PROGRAM_NORMAL = "normal" | ||
| VICARE_PROGRAM_REDUCED = "reduced" | ||
| VICARE_PROGRAM_STANDBY = "standby" | ||
|
|
||
| VICARE_HOLD_MODE_AWAY = "away" | ||
| VICARE_HOLD_MODE_HOME = "home" | ||
| VICARE_HOLD_MODE_OFF = "off" | ||
|
|
||
| VICARE_TEMP_HEATING_MIN = 3 | ||
| VICARE_TEMP_HEATING_MAX = 37 | ||
|
|
||
| SUPPORT_FLAGS_HEATING = SUPPORT_TARGET_TEMPERATURE | SUPPORT_PRESET_MODE | ||
|
|
||
| VICARE_TO_HA_HVAC_HEATING = { | ||
| VICARE_MODE_DHW: HVAC_MODE_OFF, | ||
| VICARE_MODE_DHWANDHEATING: HVAC_MODE_AUTO, | ||
| VICARE_MODE_FORCEDREDUCED: HVAC_MODE_OFF, | ||
| VICARE_MODE_FORCEDNORMAL: HVAC_MODE_HEAT, | ||
| VICARE_MODE_OFF: HVAC_MODE_OFF, | ||
| } | ||
|
|
||
| HA_TO_VICARE_HVAC_HEATING = { | ||
| HVAC_MODE_HEAT: VICARE_MODE_FORCEDNORMAL, | ||
| HVAC_MODE_OFF: VICARE_MODE_FORCEDREDUCED, | ||
| HVAC_MODE_AUTO: VICARE_MODE_DHWANDHEATING, | ||
| } | ||
|
|
||
| VICARE_TO_HA_PRESET_HEATING = { | ||
| VICARE_PROGRAM_COMFORT: PRESET_COMFORT, | ||
| VICARE_PROGRAM_ECO: PRESET_ECO, | ||
| } | ||
|
|
||
| HA_TO_VICARE_PRESET_HEATING = { | ||
| PRESET_COMFORT: VICARE_PROGRAM_COMFORT, | ||
| PRESET_ECO: VICARE_PROGRAM_ECO, | ||
| } | ||
|
|
||
| PYVICARE_ERROR = "error" | ||
|
|
||
|
|
||
| def setup_platform(hass, config, add_entities, discovery_info=None): | ||
| """Create the ViCare climate devices.""" | ||
| if discovery_info is None: | ||
| return | ||
| vicare_api = hass.data[VICARE_DOMAIN][VICARE_API] | ||
| add_entities( | ||
| [ViCareClimate(f"{hass.data[VICARE_DOMAIN][VICARE_NAME]} Heating", vicare_api)] | ||
| ) | ||
|
|
||
|
|
||
| class ViCareClimate(ClimateDevice): | ||
| """Representation of the ViCare heating climate device.""" | ||
|
|
||
| def __init__(self, name, api): | ||
| """Initialize the climate device.""" | ||
| self._name = name | ||
| self._state = None | ||
| self._api = api | ||
| self._target_temperature = None | ||
| self._current_mode = None | ||
| self._current_temperature = None | ||
| self._current_program = None | ||
|
|
||
| def update(self): | ||
| """Let HA know there has been an update from the ViCare API.""" | ||
| _room_temperature = self._api.getRoomTemperature() | ||
| if _room_temperature is not None and _room_temperature != "error": | ||
| self._current_temperature = _room_temperature | ||
| else: | ||
| self._current_temperature = self._api.getSupplyTemperature() | ||
| self._current_program = self._api.getActiveProgram() | ||
|
|
||
| # The getCurrentDesiredTemperature call can yield 'error' (str) when the system is in standby | ||
| desired_temperature = self._api.getCurrentDesiredTemperature() | ||
| if desired_temperature == PYVICARE_ERROR: | ||
| desired_temperature = None | ||
|
|
||
| self._target_temperature = desired_temperature | ||
|
|
||
| self._current_mode = self._api.getActiveMode() | ||
|
|
||
| @property | ||
| def supported_features(self): | ||
| """Return the list of supported features.""" | ||
| return SUPPORT_FLAGS_HEATING | ||
|
|
||
| @property | ||
| def name(self): | ||
| """Return the name of the climate device.""" | ||
| return self._name | ||
|
|
||
| @property | ||
| def temperature_unit(self): | ||
| """Return the unit of measurement.""" | ||
| return TEMP_CELSIUS | ||
|
|
||
| @property | ||
| def current_temperature(self): | ||
| """Return the current temperature.""" | ||
| return self._current_temperature | ||
|
|
||
| @property | ||
| def target_temperature(self): | ||
| """Return the temperature we try to reach.""" | ||
| return self._target_temperature | ||
|
|
||
| @property | ||
| def hvac_mode(self): | ||
| """Return current hvac mode.""" | ||
| return VICARE_TO_HA_HVAC_HEATING.get(self._current_mode) | ||
|
|
||
| def set_hvac_mode(self, hvac_mode): | ||
| """Set a new hvac mode on the ViCare API.""" | ||
| vicare_mode = HA_TO_VICARE_HVAC_HEATING.get(hvac_mode) | ||
| if vicare_mode is None: | ||
| _LOGGER.error( | ||
| "Cannot set invalid vicare mode: %s / %s", hvac_mode, vicare_mode | ||
| ) | ||
| return | ||
|
|
||
| _LOGGER.debug("Setting hvac mode to %s / %s", hvac_mode, vicare_mode) | ||
| self._api.setMode(vicare_mode) | ||
|
|
||
| @property | ||
| def hvac_modes(self): | ||
| """Return the list of available hvac modes.""" | ||
| return list(HA_TO_VICARE_HVAC_HEATING) | ||
|
|
||
| @property | ||
| def min_temp(self): | ||
| """Return the minimum temperature.""" | ||
| return VICARE_TEMP_HEATING_MIN | ||
|
|
||
| @property | ||
| def max_temp(self): | ||
| """Return the maximum temperature.""" | ||
| return VICARE_TEMP_HEATING_MAX | ||
|
|
||
| @property | ||
| def precision(self): | ||
| """Return the precision of the system.""" | ||
| return PRECISION_WHOLE | ||
|
|
||
| def set_temperature(self, **kwargs): | ||
| """Set new target temperatures.""" | ||
| temp = kwargs.get(ATTR_TEMPERATURE) | ||
| if temp is not None: | ||
| self._api.setProgramTemperature( | ||
| self._current_program, self._target_temperature | ||
| ) | ||
|
|
||
| @property | ||
| def preset_mode(self): | ||
| """Return the current preset mode, e.g., home, away, temp.""" | ||
| return VICARE_TO_HA_PRESET_HEATING.get(self._current_program) | ||
|
|
||
| @property | ||
| def preset_modes(self): | ||
| """Return the available preset mode.""" | ||
| return list(VICARE_TO_HA_PRESET_HEATING) | ||
|
|
||
| def set_preset_mode(self, preset_mode): | ||
| """Set new preset mode and deactivate any existing programs.""" | ||
| vicare_program = HA_TO_VICARE_PRESET_HEATING.get(preset_mode) | ||
| if vicare_program is None: | ||
| _LOGGER.error( | ||
| "Cannot set invalid vicare program: %s / %s", | ||
| preset_mode, | ||
| vicare_program, | ||
| ) | ||
| return | ||
|
|
||
| _LOGGER.debug("Setting preset to %s / %s", preset_mode, vicare_program) | ||
| self._api.deactivateProgram(self._current_program) | ||
| self._api.activateProgram(vicare_program) |
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| Original file line number | Diff line number | Diff line change |
|---|---|---|
| @@ -0,0 +1,9 @@ | ||
| { | ||
| "domain": "vicare", | ||
| "name": "Viessmann ViCare", | ||
| "documentation": "https://www.home-assistant.io/components/vicare", | ||
| "dependencies": [], | ||
| "codeowners": ["@oischinger"], | ||
| "requirements": ["PyViCare==0.1.1"] | ||
| } | ||
|
|
Oops, something went wrong.
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
Uh oh!
There was an error while loading. Please reload this page.