From 6dc197d01b24aad451fb50c6f63ad21d46b5176c Mon Sep 17 00:00:00 2001 From: fboundy Date: Thu, 4 Apr 2024 20:16:18 +0100 Subject: [PATCH] Correct consumption for EV --- apps/pv_opt/config/config.yaml | 3 +- apps/pv_opt/pv_opt.py | 136 ++++++++++++++++----------------- 2 files changed, 68 insertions(+), 71 deletions(-) diff --git a/apps/pv_opt/config/config.yaml b/apps/pv_opt/config/config.yaml index 2998afa..41e9245 100644 --- a/apps/pv_opt/config/config.yaml +++ b/apps/pv_opt/config/config.yaml @@ -226,7 +226,7 @@ pv_opt: # update_cycle_seconds: 300 # Tariff comparison - id_daily_solar: sensor.{device_name}_power_generation_today + # id_daily_solar: sensor.{device_name}_power_generation_today id_solar_power: sensor.{device_name}_pv_total_power alt_tariffs: - name: Agile_Fix @@ -236,6 +236,7 @@ pv_opt: - name: Eco7_Fix octopus_import_tariff_code: E-2R-VAR-22-11-01-G octopus_export_tariff_code: E-1R-OUTGOING-FIX-12M-19-05-13-G + - name: Flux octopus_import_tariff_code: E-1R-FLUX-IMPORT-23-02-14-G octopus_export_tariff_code: E-1R-FLUX-EXPORT-23-02-14-G \ No newline at end of file diff --git a/apps/pv_opt/pv_opt.py b/apps/pv_opt/pv_opt.py index e3c5025..ba6c60a 100644 --- a/apps/pv_opt/pv_opt.py +++ b/apps/pv_opt/pv_opt.py @@ -572,11 +572,14 @@ def _check_for_zappi(self): self.log("No Zappi sensors found") self.log("") - def _get_zappi(self, **kwargs): + def _get_zappi(self, start, end, log=False): + df = pd.DataFrame() for entity_id in self.zappi_entities: - df = self.hass2df(entity_id=entity_id, **kwargs) - self.rlog(f">>> Zappi entity {entity_id}") - self.log(f">>> {df.to_string()}") + df += self._get_hass_power_from_daily_kwh(entity_id, start=start, end=end) + if log: + self.rlog(f">>> Zappi entity {entity_id}") + self.log(f">>>\n{df.to_string()}") + return df def rlog(self, str, **kwargs): if self.redact: @@ -1555,7 +1558,7 @@ def optimise(self): self.time_now = pd.Timestamp.utcnow() self.static = self.static[self.time_now.floor("30min") :].fillna(0) - # self.load_consumption() + self.soc_now = self.get_config("id_battery_soc") x = self.hass2df(self.config["id_battery_soc"], days=1, log=self.debug) if self.debug: @@ -2266,19 +2269,20 @@ def load_consumption(self, start, end): f"Getting expected consumption data for {start.strftime(DATE_TIME_FORMAT_LONG)} to {end.strftime(DATE_TIME_FORMAT_LONG)}:" ) - time_now = pd.Timestamp.now(tz="UTC") - if (start < time_now) and (end < time_now): - self.log(" - Start and end are both in past so actuals will be used with no weighting") - - index = pd.date_range(start, end, inclusive="left", freq="30min") - consumption = pd.DataFrame(index=index, data={"consumption": 0}) - df = None + if self.get_config("use_consumption_history"): + time_now = pd.Timestamp.now(tz="UTC") + if (start < time_now) and (end < time_now): + self.log(" - Start and end are both in past so actuals will be used with no weighting") + days = (time_now - start).days + 1 + else: + days = int(self.get_config("consumption_history_days")) - entity_ids = [] - entity_id = None + index = pd.date_range(start, end, inclusive="left", freq="30min") + consumption = pd.DataFrame(index=index, data={"consumption": 0}) + df = None - if self.get_config("use_consumption_history"): - days = int(self.get_config("consumption_history_days")) + entity_ids = [] + entity_id = None if "id_consumption" in self.config: entity_ids = self.config["id_consumption"] @@ -2294,71 +2298,59 @@ def load_consumption(self, start, end): ): entity_id = self.config["id_consumption_today"] - if (start < time_now) and (end < time_now): - for entity_id in entity_ids: - power = self.hass2df(entity_id=entity_id, days=days).loc[ - start.floor("30min") : end.ceil("30min") + pd.Timedelta("30min") - ] - - if power is not None: - power = self.riemann_avg(power) - consumption["consumption"] += power + for entity_id in entity_ids: + power = self.hass2df(entity_id=entity_id, days=days) - if consumption["consumption"].sum() == 0: - consumption["consumption"] = self._get_hass_power_from_daily_kwh( - entity_id, - start=start, - end=end, - log=self.debug, - ) + power = self.riemann_avg(power) + if df is None: + df = power + else: + df += power - if consumption["consumption"].sum() == 0: - self._status("ERROR: No consumption history.") - return + if df is None: + self.log("Getting consumpion") + df = self._get_hass_power_from_daily_kwh( + entity_id, + days=days, + log=self.debug, + ) - else: - for entity_id in entity_ids: - power = self.hass2df(entity_id=entity_id, days=days) + if df is None: + self._status("ERROR: No consumption history.") + return - power = self.riemann_avg(power) - if df is None: - df = power - else: - df += power + actual_days = int( + round( + (df.index[-1] - df.index[0]).total_seconds() / 3600 / 24, + 0, + ) + ) - if df is None: - self.log("Getting consumpo") - df = self._get_hass_power_from_daily_kwh( - entity_id, - days=days, - log=self.debug, - ) + self.log( + f" - Got {actual_days} days history from {entity_id} from {df.index[0].strftime(DATE_TIME_FORMAT_SHORT)} to {df.index[-1].strftime(DATE_TIME_FORMAT_SHORT)}" + ) + if int(actual_days) == days: + str_days = "OK" + else: + self._status(f"WARNING: Consumption < {days} days.") + str_days = "Potential error. <<<" - if df is None: - self._status("ERROR: No consumption history.") - return + self.log(f" - {days} days was expected. {str_days}") - actual_days = int( - round( - (df.index[-1] - df.index[0]).total_seconds() / 3600 / 24, - 0, - ) + if (len(self.zappi_entities) > 0) and (self.get_config("ev_charger") == "Zappi"): + ev_power = self._get_zappi(start=df.index[0], end=df.index[-1]) + self.log("") + self.log(f" Deducting EV consumption of {ev_power.sum()/2000}") + self.log( + f">>> EV consumption from {ev_power.index[0].strftime(DATE_TIME_FORMAT_SHORT)} to {ev_power.index[-1].strftime(DATE_TIME_FORMAT_LONG)}" ) - self.log( - f" - Got {actual_days} days history from {entity_id} from {df.index[0].strftime(DATE_TIME_FORMAT_SHORT)} to {df.index[-1].strftime(DATE_TIME_FORMAT_SHORT)}" + f">>> House consumption from {df.index[0].strftime(DATE_TIME_FORMAT_SHORT)} to {df.index[-1].strftime(DATE_TIME_FORMAT_LONG)}" ) - if int(actual_days) == days: - str_days = "OK" - else: - self._status(f"WARNING: Consumption < {days} days.") - str_days = "Potential error. <<<" - - self.log(f" - {days} days was expected. {str_days}") - - if len(self.zappi_entities) > 0: - zappi = self._get_zappi(days=days, log=True) + if (start < time_now) and (end < time_now): + consumption["consumption"] = df.loc[start:end] + else: df = df * (1 + self.get_config("consumption_margin") / 100) dfx = pd.Series(index=df.index, data=df.to_list()) # Group by time and take the mean @@ -2391,6 +2383,10 @@ def load_consumption(self, start, end): self.log(f" - Ignoring 'Day of Week Weighting' because only {days} days of history is available") consumption["consumption"] = consumption_mean + if len(entity_ids) > 0: + self.log(f" - Estimated consumption from {entity_ids} loaded OK ") + + else: self.log(f" - Estimated consumption from {entity_id} loaded OK ") else: