Skip to content

Commit 9976322

Browse files
committed
Calculate the maximum charging duration at the time of the minimum as in this case, it is more likely that the upcoming energy rates are available
1 parent 2abd329 commit 9976322

File tree

2 files changed

+31
-37
lines changed

2 files changed

+31
-37
lines changed

source/inverter_charge_controller.py

+16-8
Original file line numberDiff line numberDiff line change
@@ -142,6 +142,12 @@ def _do_iteration(self, current_energy_rate: EnergyRate) -> EnergyRate:
142142
next_price_minimum = self._get_next_price_minimum()
143143
self.log.info(f"The next price minimum is at {next_price_minimum}")
144144

145+
self.tibber_api_handler.set_maximum_charging_duration_of_current_energy_rate(current_energy_rate)
146+
self.log.info(
147+
f"The maximum charging duration of the current energy rate is "
148+
f"{current_energy_rate.maximum_charging_duration}"
149+
)
150+
145151
if next_price_minimum.rate > current_energy_rate.rate:
146152
# Information is unused at the moment
147153
self.log.info("The price of the upcoming minimum is higher than the current energy rate")
@@ -249,16 +255,18 @@ def _do_iteration(self, current_energy_rate: EnergyRate) -> EnergyRate:
249255

250256
def _charge_inverter(self, target_state_of_charge: int, maximum_charging_duration: timedelta) -> None:
251257
"""
252-
Charges the inverter until a given state of charge is reached.
253-
Checks every few minutes the current state of charge and compares to the target value.
254-
Charges the inverter for a maximum of one hour. If consecutive_energy_rate_is_cheap is True it charges for a
255-
maximum of two hours.
256-
--> Stops when the target state of charge or the maximum charge time is reached (whichever comes first)
258+
Charges the inverter battery to the target state of charge within a specified maximum charging duration.
259+
Monitors the charging progress at regular intervals and stops the charging process if specific conditions are
260+
met:
261+
- Charging limit reached
262+
- Maximum charging duration reached (at this point the energy prices would be to high to charge)
263+
- The state of the inverter was changed manually
264+
- Too many errors occurred while trying to communicate with the inverter
257265
258266
Args:
259-
target_state_of_charge: The desired state of charge percentage to reach before stopping the charging process.
260-
maximum_charging_duration: The maximum duration for which charging is feasible under given energy rate
261-
constraints.
267+
target_state_of_charge: The desired battery state of charge percentage to reach during the
268+
charging process.
269+
maximum_charging_duration: The maximum duration allowed for the charging process to complete.
262270
"""
263271
charging_progress_check_interval = timedelta(minutes=5)
264272

source/tibber_api_handler.py

+15-29
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,8 @@ def __init__(self):
2323

2424
self.database_handler = DatabaseHandler("energy_prices")
2525

26+
self.upcoming_energy_rates_cache = []
27+
2628
def get_next_price_minimum(self, first_iteration: bool = False) -> EnergyRate:
2729
"""
2830
This method performs a series of operations to determine the most cost-effective time to charge by analyzing
@@ -57,6 +59,7 @@ def get_next_price_minimum(self, first_iteration: bool = False) -> EnergyRate:
5759
all_energy_rates = self._extract_energy_rates_from_api_response(api_result)
5860
self.write_energy_rates_to_database(all_energy_rates)
5961
upcoming_energy_rates = self._remove_energy_rates_from_the_past(all_energy_rates)
62+
self.upcoming_energy_rates_cache = upcoming_energy_rates
6063
if first_iteration and not self._check_if_next_three_prices_are_greater_than_current_one(
6164
upcoming_energy_rates
6265
):
@@ -80,44 +83,29 @@ def get_next_price_minimum(self, first_iteration: bool = False) -> EnergyRate:
8083
):
8184
minimum_of_energy_rates.has_to_be_rechecked = True
8285

83-
minimum_of_energy_rates.maximum_charging_duration = self._calculate_maximum_charging_duration(
84-
minimum_of_energy_rates, upcoming_energy_rates
85-
)
86-
8786
return minimum_of_energy_rates
8887

89-
@staticmethod
90-
def _calculate_maximum_charging_duration(
91-
minimum_of_energy_rates: EnergyRate, upcoming_energy_rates: list[EnergyRate]
92-
) -> timedelta:
88+
def set_maximum_charging_duration_of_current_energy_rate(self, current_energy_rate: EnergyRate) -> None:
9389
"""
94-
Calculates the maximum duration for which charging is feasible under given energy rate constraints.
90+
It takes the current energy rate from the InverterChargeController and compares it against the upcoming energy
91+
rates. It then calculates and sets the maximum possible charging duration based on the comparison of the current
92+
price and the consecutive ones.
9593
96-
This method computes the maximum charging duration based on the minimum energy rate within the list of upcoming
97-
energy rates and MAXIMUM_THRESHOLD. This ensures that charging only continues while subsequent energy rates
98-
remain within the allowed threshold of the minimum energy rate.
94+
We have to provide the current energy rate as the upcoming energy rates (which are saved as an instance
95+
variable) only include the **upcoming** energy rates and not the current one.
9996
10097
Args:
101-
minimum_of_energy_rates: The energy rate which is considered as the minimum acceptable
102-
rate for initiating or continuing charging.
103-
upcoming_energy_rates: A list of energy rates representing projected energy pricing
104-
in subsequent time periods.
105-
106-
Returns:
107-
A timedelta object representing the maximum charging duration under the given energy rate constraints.
98+
current_energy_rate (EnergyRate): The current energy rate for comparison against upcoming energy rates.
10899
"""
109-
index = upcoming_energy_rates.index(minimum_of_energy_rates)
110100
charging_duration = timedelta(hours=1)
111-
if index + 1 >= len(upcoming_energy_rates):
112-
return charging_duration
113101

114-
for energy_rate in upcoming_energy_rates[index + 1 :]:
115-
if energy_rate.rate <= minimum_of_energy_rates.rate + TibberAPIHandler.MAXIMUM_THRESHOLD:
102+
for energy_rate in self.upcoming_energy_rates_cache:
103+
if energy_rate.rate <= current_energy_rate.rate + TibberAPIHandler.MAXIMUM_THRESHOLD:
116104
charging_duration += timedelta(hours=1)
117-
continue
118-
break
105+
else:
106+
break
119107

120-
return charging_duration
108+
current_energy_rate.maximum_charging_duration = charging_duration
121109

122110
@staticmethod
123111
def _check_if_next_three_prices_are_greater_than_current_one(all_upcoming_energy_rates: list[EnergyRate]) -> bool:
@@ -168,8 +156,6 @@ def _fetch_upcoming_prices_from_api(self) -> dict:
168156
}
169157
"""
170158
)
171-
# Note: Sometimes we only get the prices for today from the tibber api and the prices for tomorrow stay empty
172-
# I guess they are not determined yet...?
173159
self.log.debug("Crawling the Tibber API for the electricity prices")
174160
response = self.client.execute(query)
175161
self.log.trace(f"Retrieved data: {response}")

0 commit comments

Comments
 (0)