From 0887a1950f4c541332c6f1e366894654b4363003 Mon Sep 17 00:00:00 2001 From: davidusb-geek Date: Wed, 3 Jan 2024 02:00:33 +0100 Subject: [PATCH 1/7] Fix - Fixed conflict on constraints --- CHANGELOG.md | 4 ++ Dockerfile | 20 +++--- requirements.txt | 8 +-- scripts/optim_results_analysis.py | 113 ++++++++++++++++++++++++++++++ setup.py | 8 +-- src/emhass/optimization.py | 41 ++++++----- 6 files changed, 161 insertions(+), 33 deletions(-) create mode 100644 scripts/optim_results_analysis.py diff --git a/CHANGELOG.md b/CHANGELOG.md index 4741bdf2..4ea36156 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,5 +1,9 @@ # Changelog +## [0.6.2] - Unreleased +### Fix +- Updated optimization constraints to solve conflict for `set_def_constant` and `treat_def_as_semi_cont` cases + ## [0.6.1] - 2023-12-18 ### Fix - Patching EMHASS for Python 3.11. New explicit dependecy h5py==3.10.0 diff --git a/Dockerfile b/Dockerfile index b0fb0f2d..e1909775 100644 --- a/Dockerfile +++ b/Dockerfile @@ -1,5 +1,6 @@ FROM python:3.11-slim-buster -#FROM ghcr.io/home-assistant/amd64-base-debian:bookworm # Uncomment to test add-on +# FROM ghcr.io/home-assistant/amd64-base-debian:bookworm # Uncomment to test add-on +# FROM ghcr.io/home-assistant/armhf-base-debian:bookworm # switch working directory WORKDIR /app @@ -13,13 +14,12 @@ COPY README.md README.md # Setup RUN apt-get update \ && apt-get install -y --no-install-recommends \ - # libc-bin \ # Uncomment to test add-on - # libffi-dev \ # Uncomment to test add-on - # python3 \ # Uncomment to test add-on - # python3-pip \ # Uncomment to test add-on - # python3-dev \ # Uncomment to test add-on - # git \ # Uncomment to test add-on - # build-essential \ # Uncomment to test add-on + # libffi-dev \ + # python3 \ + # python3-pip \ + # python3-dev \ + # git \ + # build-essential \ gcc \ coinor-cbc \ coinor-libcbc-dev \ @@ -27,8 +27,12 @@ RUN apt-get update \ libhdf5-serial-dev \ netcdf-bin \ libnetcdf-dev \ + # pkg-config \ + # gfortran \ + # libatlas-base-dev \ && ln -s /usr/include/hdf5/serial /usr/include/hdf5/include \ && export HDF5_DIR=/usr/include/hdf5 \ + # && pip3 install --extra-index-url=https://www.piwheels.org/simple --no-cache-dir --break-system-packages -U setuptools wheel \ && pip3 install --no-cache-dir --break-system-packages -r requirements_webserver.txt \ && apt-get purge -y --auto-remove \ gcc \ diff --git a/requirements.txt b/requirements.txt index a11be33b..bdd51541 100644 --- a/requirements.txt +++ b/requirements.txt @@ -1,7 +1,7 @@ wheel -numpy==1.26.0 -pandas==2.0.3 -scipy==1.11.3 +numpy<=1.26.0 +pandas<=2.0.3 +scipy<=1.11.3 pvlib>=0.10.2 protobuf>=3.0.0 pytz>=2021.1 @@ -10,5 +10,5 @@ beautifulsoup4>=4.9.3 h5py==3.10.0 pulp>=2.4 pyyaml>=5.4.1 -tables==3.9.1 +tables<=3.9.1 skforecast==0.11.0 \ No newline at end of file diff --git a/scripts/optim_results_analysis.py b/scripts/optim_results_analysis.py new file mode 100644 index 00000000..a08f18fd --- /dev/null +++ b/scripts/optim_results_analysis.py @@ -0,0 +1,113 @@ +# -*- coding: utf-8 -*- +import pickle +import numpy as np +import pandas as pd +import pathlib +import plotly.express as px +import plotly.subplots as sp +import plotly.io as pio +pio.renderers.default = 'browser' +pd.options.plotting.backend = "plotly" + +from emhass.retrieve_hass import retrieve_hass +from emhass.optimization import optimization +from emhass.forecast import forecast +from emhass.utils import get_root, get_yaml_parse, get_days_list, get_logger + +# the root folder +root = str(get_root(__file__, num_parent=2)) +# create logger +logger, ch = get_logger(__name__, root, save_to_file=False) + +def get_forecast_optim_objects(retrieve_hass_conf, optim_conf, plant_conf, + params, get_data_from_file): + fcst = forecast(retrieve_hass_conf, optim_conf, plant_conf, + params, root, logger, get_data_from_file=get_data_from_file) + df_weather = fcst.get_weather_forecast(method='csv') + P_PV_forecast = fcst.get_power_from_weather(df_weather) + P_load_forecast = fcst.get_load_forecast(method=optim_conf['load_forecast_method']) + df_input_data_dayahead = pd.concat([P_PV_forecast, P_load_forecast], axis=1) + df_input_data_dayahead.columns = ['P_PV_forecast', 'P_load_forecast'] + opt = optimization(retrieve_hass_conf, optim_conf, plant_conf, + fcst.var_load_cost, fcst.var_prod_price, + 'profit', root, logger) + return fcst, P_PV_forecast, P_load_forecast, df_input_data_dayahead, opt + +if __name__ == '__main__': + show_figures = False + save_figures = False + get_data_from_file = True + params = None + retrieve_hass_conf, optim_conf, plant_conf = get_yaml_parse(pathlib.Path(root+'/config_emhass.yaml'), use_secrets=False) + retrieve_hass_conf, optim_conf, plant_conf = \ + retrieve_hass_conf, optim_conf, plant_conf + rh = retrieve_hass(retrieve_hass_conf['hass_url'], retrieve_hass_conf['long_lived_token'], + retrieve_hass_conf['freq'], retrieve_hass_conf['time_zone'], + params, root, logger) + if get_data_from_file: + with open(pathlib.Path(root+'/data/test_df_final.pkl'), 'rb') as inp: + rh.df_final, days_list, var_list = pickle.load(inp) + else: + days_list = get_days_list(retrieve_hass_conf['days_to_retrieve']) + var_list = [retrieve_hass_conf['var_load'], retrieve_hass_conf['var_PV']] + rh.get_data(days_list, var_list, + minimal_response=False, significant_changes_only=False) + rh.prepare_data(retrieve_hass_conf['var_load'], load_negative = retrieve_hass_conf['load_negative'], + set_zero_min = retrieve_hass_conf['set_zero_min'], + var_replace_zero = retrieve_hass_conf['var_replace_zero'], + var_interp = retrieve_hass_conf['var_interp']) + df_input_data = rh.df_final.copy() + + fcst, P_PV_forecast, P_load_forecast, df_input_data_dayahead, opt = \ + get_forecast_optim_objects(retrieve_hass_conf, optim_conf, plant_conf, + params, get_data_from_file) + df_input_data = fcst.get_load_cost_forecast(df_input_data) + df_input_data = fcst.get_prod_price_forecast(df_input_data) + + template = 'presentation' + + # Let's plot the input data + fig_inputs1 = df_input_data[['sensor.power_photovoltaics', + 'sensor.power_load_no_var_loads_positive']].plot() + fig_inputs1.layout.template = template + fig_inputs1.update_yaxes(title_text = "Powers (W)") + fig_inputs1.update_xaxes(title_text = "Time") + if show_figures: + fig_inputs1.show() + if save_figures: + fig_inputs1.write_image(root + "/docs/images/inputs_power.svg", + width=1080, height=0.8*1080) + + fig_inputs_dah = df_input_data_dayahead.plot() + fig_inputs_dah.layout.template = template + fig_inputs_dah.update_yaxes(title_text = "Powers (W)") + fig_inputs_dah.update_xaxes(title_text = "Time") + if show_figures: + fig_inputs_dah.show() + if save_figures: + fig_inputs_dah.write_image(root + "/docs/images/inputs_dayahead.svg", + width=1080, height=0.8*1080) + + # And then perform a dayahead optimization + df_input_data_dayahead = fcst.get_load_cost_forecast(df_input_data_dayahead) + df_input_data_dayahead = fcst.get_prod_price_forecast(df_input_data_dayahead) + optim_conf['treat_def_as_semi_cont'] = [True, True] + optim_conf['set_def_constant'] = [True, True] + opt_res_dah = opt.perform_dayahead_forecast_optim(df_input_data_dayahead, P_PV_forecast, P_load_forecast) + opt_res_dah['P_PV'] = df_input_data_dayahead[['P_PV_forecast']] + fig_res_dah = opt_res_dah[['P_deferrable0', 'P_deferrable1', 'P_grid', 'P_PV', + 'P_def_start_0', 'P_def_start_1', 'P_def_bin2_0', 'P_def_bin2_1']].plot() + fig_res_dah.layout.template = template + fig_res_dah.update_yaxes(title_text = "Powers (W)") + fig_res_dah.update_xaxes(title_text = "Time") + # if show_figures: + fig_res_dah.show() + if save_figures: + fig_res_dah.write_image(root + "/docs/images/optim_results_PV_defLoads_dayaheadOptim.svg", + width=1080, height=0.8*1080) + + print("System with: PV, two deferrable loads, dayahead optimization, profit >> total cost function sum: "+\ + str(opt_res_dah['cost_profit'].sum())) + + print(opt_res_dah) + opt_res_dah.to_html('opt_res_dah.html') \ No newline at end of file diff --git a/setup.py b/setup.py index 3cad4acb..89ac60c7 100644 --- a/setup.py +++ b/setup.py @@ -40,9 +40,9 @@ python_requires='>=3.9, <3.12', install_requires=[ 'wheel', - 'numpy==1.26', - 'scipy==1.11.3', - 'pandas==2.0.3', + 'numpy<=1.26', + 'scipy<=1.11.3', + 'pandas<=2.0.3', 'pvlib>=0.10.1', 'protobuf>=3.0.0', 'pytz>=2021.1', @@ -50,7 +50,7 @@ 'beautifulsoup4>=4.9.3', 'pulp>=2.4', 'pyyaml>=5.4.1', - 'tables==3.9.1', + 'tables<=3.9.1', 'skforecast==0.11.0', ], # Optional entry_points={ # Optional diff --git a/src/emhass/optimization.py b/src/emhass/optimization.py index 219d68dc..98751718 100644 --- a/src/emhass/optimization.py +++ b/src/emhass/optimization.py @@ -287,30 +287,31 @@ def perform_optimization(self, data_opt: pd.DataFrame, P_PV: np.array, P_load: n for i in set_I}) # Treat the number of starts for a deferrable load if self.optim_conf['set_def_constant'][k]: - constraints.update({"constraint_pdef{}_start1".format(k) : - plp.LpConstraint( - e=P_def_start[k][0], - sense=plp.LpConstraintEQ, - rhs=0) - }) - constraints.update({"constraint_pdef{}_start2_{}".format(k, i) : - plp.LpConstraint( - e=P_def_start[k][i] - P_def_bin2[k][i] + P_def_bin2[k][i-1], - sense=plp.LpConstraintEQ, - rhs=0) - for i in set_I[1:]}) - constraints.update({"constraint_pdef{}_start4_{}".format(k, i) : + + constraints.update({"constraint_pdef{}_start1_{}".format(k, i) : plp.LpConstraint( e=P_deferrable[k][i] - P_def_bin2[k][i]*M, sense=plp.LpConstraintLE, rhs=0) for i in set_I}) - constraints.update({"constraint_pdef{}_start5_{}".format(k, i) : + constraints.update({"constraint_pdef{}_start2_{}".format(k, i) : plp.LpConstraint( - e=-P_deferrable[k][i] + M*(P_def_bin2[k][i]-1) + 1, - sense=plp.LpConstraintLE, + e=P_def_start[k][i] - P_def_bin2[k][i] + P_def_bin2[k][i-1], + sense=plp.LpConstraintGE, rhs=0) - for i in set_I}) + for i in set_I[1:]}) + constraints.update({"constraint_pdef{}_start3".format(k) : + plp.LpConstraint( + e = plp.lpSum(P_def_start[k][i] for i in set_I), + sense = plp.LpConstraintEQ, + rhs = 1) + }) + constraints.update({"constraint_pdef{}_start4".format(k) : + plp.LpConstraint( + e = plp.lpSum(P_def_bin2[k][i] for i in set_I), + sense = plp.LpConstraintEQ, + rhs = self.optim_conf['def_total_hours'][k]/self.timeStep) + }) # The battery constraints if self.optim_conf['set_use_battery']: @@ -465,6 +466,12 @@ def perform_optimization(self, data_opt: pd.DataFrame, P_PV: np.array, P_load: n # Add the optimization status opt_tp["optim_status"] = self.optim_status + # Debug variables + opt_tp["P_def_start_0"] = [P_def_start[0][i].varValue for i in set_I] + opt_tp["P_def_start_1"] = [P_def_start[1][i].varValue for i in set_I] + opt_tp["P_def_bin2_0"] = [P_def_bin2[0][i].varValue for i in set_I] + opt_tp["P_def_bin2_1"] = [P_def_bin2[1][i].varValue for i in set_I] + return opt_tp def perform_perfect_forecast_optim(self, df_input_data: pd.DataFrame, days_list: pd.date_range) -> pd.DataFrame: From 2faa8dc169b15cb37914aabb5382602eaf6c0d8c Mon Sep 17 00:00:00 2001 From: davidusb-geek Date: Wed, 3 Jan 2024 02:15:41 +0100 Subject: [PATCH 2/7] Fix - Added a debug option for optimization methods --- scripts/optim_results_analysis.py | 8 +++++++- src/emhass/optimization.py | 12 +++++++----- 2 files changed, 14 insertions(+), 6 deletions(-) diff --git a/scripts/optim_results_analysis.py b/scripts/optim_results_analysis.py index a08f18fd..d77fa62b 100644 --- a/scripts/optim_results_analysis.py +++ b/scripts/optim_results_analysis.py @@ -93,7 +93,13 @@ def get_forecast_optim_objects(retrieve_hass_conf, optim_conf, plant_conf, df_input_data_dayahead = fcst.get_prod_price_forecast(df_input_data_dayahead) optim_conf['treat_def_as_semi_cont'] = [True, True] optim_conf['set_def_constant'] = [True, True] - opt_res_dah = opt.perform_dayahead_forecast_optim(df_input_data_dayahead, P_PV_forecast, P_load_forecast) + unit_load_cost = df_input_data[opt.var_load_cost].values + unit_prod_price = df_input_data[opt.var_prod_price].values + opt_res_dah = opt.perform_optimization(df_input_data_dayahead, P_PV_forecast.values.ravel(), + P_load_forecast.values.ravel(), + unit_load_cost, unit_prod_price, + debug = True) + # opt_res_dah = opt.perform_dayahead_forecast_optim(df_input_data_dayahead, P_PV_forecast, P_load_forecast) opt_res_dah['P_PV'] = df_input_data_dayahead[['P_PV_forecast']] fig_res_dah = opt_res_dah[['P_deferrable0', 'P_deferrable1', 'P_grid', 'P_PV', 'P_def_start_0', 'P_def_start_1', 'P_def_bin2_0', 'P_def_bin2_1']].plot() diff --git a/src/emhass/optimization.py b/src/emhass/optimization.py index 98751718..96de0550 100644 --- a/src/emhass/optimization.py +++ b/src/emhass/optimization.py @@ -88,7 +88,8 @@ def __init__(self, retrieve_hass_conf: dict, optim_conf: dict, plant_conf: dict, def perform_optimization(self, data_opt: pd.DataFrame, P_PV: np.array, P_load: np.array, unit_load_cost: np.array, unit_prod_price: np.array, soc_init: Optional[float] = None, soc_final: Optional[float] = None, - def_total_hours: Optional[list] = None) -> pd.DataFrame: + def_total_hours: Optional[list] = None, + debug: Optional[bool] = False) -> pd.DataFrame: r""" Perform the actual optimization using linear programming (LP). @@ -467,10 +468,11 @@ def perform_optimization(self, data_opt: pd.DataFrame, P_PV: np.array, P_load: n opt_tp["optim_status"] = self.optim_status # Debug variables - opt_tp["P_def_start_0"] = [P_def_start[0][i].varValue for i in set_I] - opt_tp["P_def_start_1"] = [P_def_start[1][i].varValue for i in set_I] - opt_tp["P_def_bin2_0"] = [P_def_bin2[0][i].varValue for i in set_I] - opt_tp["P_def_bin2_1"] = [P_def_bin2[1][i].varValue for i in set_I] + if debug: + opt_tp["P_def_start_0"] = [P_def_start[0][i].varValue for i in set_I] + opt_tp["P_def_start_1"] = [P_def_start[1][i].varValue for i in set_I] + opt_tp["P_def_bin2_0"] = [P_def_bin2[0][i].varValue for i in set_I] + opt_tp["P_def_bin2_1"] = [P_def_bin2[1][i].varValue for i in set_I] return opt_tp From a2306142883ff68483e02ab3bd0f714264861a44 Mon Sep 17 00:00:00 2001 From: davidusb-geek Date: Wed, 3 Jan 2024 18:49:31 +0100 Subject: [PATCH 3/7] Fix - Updated optimization tests to improve coverage --- tests/test_optimization.py | 41 ++++++++++++++++++++++++++++++++++++++ 1 file changed, 41 insertions(+) diff --git a/tests/test_optimization.py b/tests/test_optimization.py index 52df24a9..fc78bc10 100644 --- a/tests/test_optimization.py +++ b/tests/test_optimization.py @@ -101,6 +101,47 @@ def test_perform_dayahead_forecast_optim(self): self.assertTrue(table.columns[1]=='Cost Totals') # Check status self.assertTrue('optim_status' in self.opt_res_dayahead.columns) + # Test treat_def_as_semi_cont and set_def_constant constraints + self.optim_conf.update({'treat_def_as_semi_cont': [True, True]}) + self.optim_conf.update({'set_def_constant': [True, True]}) + self.opt = optimization(self.retrieve_hass_conf, self.optim_conf, self.plant_conf, + self.fcst.var_load_cost, self.fcst.var_prod_price, + self.costfun, root, logger) + self.opt_res_dayahead = self.opt.perform_dayahead_forecast_optim( + self.df_input_data_dayahead, self.P_PV_forecast, self.P_load_forecast) + self.assertTrue(self.opt.optim_status == 'Optimal') + self.optim_conf.update({'treat_def_as_semi_cont': [False, True]}) + self.optim_conf.update({'set_def_constant': [True, True]}) + self.opt = optimization(self.retrieve_hass_conf, self.optim_conf, self.plant_conf, + self.fcst.var_load_cost, self.fcst.var_prod_price, + self.costfun, root, logger) + self.opt_res_dayahead = self.opt.perform_dayahead_forecast_optim( + self.df_input_data_dayahead, self.P_PV_forecast, self.P_load_forecast) + self.assertTrue(self.opt.optim_status == 'Optimal') + self.optim_conf.update({'treat_def_as_semi_cont': [False, True]}) + self.optim_conf.update({'set_def_constant': [False, True]}) + self.opt = optimization(self.retrieve_hass_conf, self.optim_conf, self.plant_conf, + self.fcst.var_load_cost, self.fcst.var_prod_price, + self.costfun, root, logger) + self.opt_res_dayahead = self.opt.perform_dayahead_forecast_optim( + self.df_input_data_dayahead, self.P_PV_forecast, self.P_load_forecast) + self.assertTrue(self.opt.optim_status == 'Optimal') + self.optim_conf.update({'treat_def_as_semi_cont': [False, False]}) + self.optim_conf.update({'set_def_constant': [False, True]}) + self.opt = optimization(self.retrieve_hass_conf, self.optim_conf, self.plant_conf, + self.fcst.var_load_cost, self.fcst.var_prod_price, + self.costfun, root, logger) + self.opt_res_dayahead = self.opt.perform_dayahead_forecast_optim( + self.df_input_data_dayahead, self.P_PV_forecast, self.P_load_forecast) + self.assertTrue(self.opt.optim_status == 'Optimal') + self.optim_conf.update({'treat_def_as_semi_cont': [False, False]}) + self.optim_conf.update({'set_def_constant': [False, False]}) + self.opt = optimization(self.retrieve_hass_conf, self.optim_conf, self.plant_conf, + self.fcst.var_load_cost, self.fcst.var_prod_price, + self.costfun, root, logger) + self.opt_res_dayahead = self.opt.perform_dayahead_forecast_optim( + self.df_input_data_dayahead, self.P_PV_forecast, self.P_load_forecast) + self.assertTrue(self.opt.optim_status == 'Optimal') def test_perform_dayahead_forecast_optim_costfun_selfconso(self): costfun = 'self-consumption' From 15f318a4ce5e8bf347d20f8702e297f44ddec6ea Mon Sep 17 00:00:00 2001 From: davidusb-geek Date: Wed, 3 Jan 2024 21:35:58 +0100 Subject: [PATCH 4/7] Fix - Improved coverage --- src/emhass/optimization.py | 2 +- tests/test_optimization.py | 32 +++++++++++++++++++++++++++----- 2 files changed, 28 insertions(+), 6 deletions(-) diff --git a/src/emhass/optimization.py b/src/emhass/optimization.py index 96de0550..e4713f5c 100644 --- a/src/emhass/optimization.py +++ b/src/emhass/optimization.py @@ -77,7 +77,7 @@ def __init__(self, retrieve_hass_conf: dict, optim_conf: dict, plant_conf: dict, if 'lp_solver' in optim_conf.keys(): self.lp_solver = optim_conf['lp_solver'] else: - self.lp_solver = 'PULP_CBC_CMD' + self.lp_solver = 'default' if 'lp_solver_path' in optim_conf.keys(): self.lp_solver_path = optim_conf['lp_solver_path'] else: diff --git a/tests/test_optimization.py b/tests/test_optimization.py index fc78bc10..a41d2c00 100644 --- a/tests/test_optimization.py +++ b/tests/test_optimization.py @@ -108,7 +108,7 @@ def test_perform_dayahead_forecast_optim(self): self.fcst.var_load_cost, self.fcst.var_prod_price, self.costfun, root, logger) self.opt_res_dayahead = self.opt.perform_dayahead_forecast_optim( - self.df_input_data_dayahead, self.P_PV_forecast, self.P_load_forecast) + self.df_input_data_dayahead, self.P_PV_forecast, self.P_load_forecast) self.assertTrue(self.opt.optim_status == 'Optimal') self.optim_conf.update({'treat_def_as_semi_cont': [False, True]}) self.optim_conf.update({'set_def_constant': [True, True]}) @@ -116,7 +116,7 @@ def test_perform_dayahead_forecast_optim(self): self.fcst.var_load_cost, self.fcst.var_prod_price, self.costfun, root, logger) self.opt_res_dayahead = self.opt.perform_dayahead_forecast_optim( - self.df_input_data_dayahead, self.P_PV_forecast, self.P_load_forecast) + self.df_input_data_dayahead, self.P_PV_forecast, self.P_load_forecast) self.assertTrue(self.opt.optim_status == 'Optimal') self.optim_conf.update({'treat_def_as_semi_cont': [False, True]}) self.optim_conf.update({'set_def_constant': [False, True]}) @@ -124,7 +124,7 @@ def test_perform_dayahead_forecast_optim(self): self.fcst.var_load_cost, self.fcst.var_prod_price, self.costfun, root, logger) self.opt_res_dayahead = self.opt.perform_dayahead_forecast_optim( - self.df_input_data_dayahead, self.P_PV_forecast, self.P_load_forecast) + self.df_input_data_dayahead, self.P_PV_forecast, self.P_load_forecast) self.assertTrue(self.opt.optim_status == 'Optimal') self.optim_conf.update({'treat_def_as_semi_cont': [False, False]}) self.optim_conf.update({'set_def_constant': [False, True]}) @@ -132,7 +132,7 @@ def test_perform_dayahead_forecast_optim(self): self.fcst.var_load_cost, self.fcst.var_prod_price, self.costfun, root, logger) self.opt_res_dayahead = self.opt.perform_dayahead_forecast_optim( - self.df_input_data_dayahead, self.P_PV_forecast, self.P_load_forecast) + self.df_input_data_dayahead, self.P_PV_forecast, self.P_load_forecast) self.assertTrue(self.opt.optim_status == 'Optimal') self.optim_conf.update({'treat_def_as_semi_cont': [False, False]}) self.optim_conf.update({'set_def_constant': [False, False]}) @@ -140,7 +140,29 @@ def test_perform_dayahead_forecast_optim(self): self.fcst.var_load_cost, self.fcst.var_prod_price, self.costfun, root, logger) self.opt_res_dayahead = self.opt.perform_dayahead_forecast_optim( - self.df_input_data_dayahead, self.P_PV_forecast, self.P_load_forecast) + self.df_input_data_dayahead, self.P_PV_forecast, self.P_load_forecast) + self.assertTrue(self.opt.optim_status == 'Optimal') + # Test with different default solver, debug mode and batt SOC conditions + del self.optim_conf['lp_solver'] + del self.optim_conf['lp_solver_path'] + self.optim_conf['set_use_battery'] = True + soc_init = None + soc_final = 0.3 + self.optim_conf['set_total_pv_sell'] = True + self.opt = optimization(self.retrieve_hass_conf, self.optim_conf, self.plant_conf, + self.fcst.var_load_cost, self.fcst.var_prod_price, + self.costfun, root, logger) + + unit_load_cost = self.df_input_data_dayahead[self.opt.var_load_cost].values + unit_prod_price = self.df_input_data_dayahead[self.opt.var_prod_price].values + self.opt_res_dayahead = self.opt.perform_optimization( + self.df_input_data_dayahead, self.P_PV_forecast.values.ravel(), + self.P_load_forecast.values.ravel(), unit_load_cost, unit_prod_price, + soc_init = soc_init, soc_final = soc_final, debug = True) + self.assertIsInstance(self.opt_res_dayahead, type(pd.DataFrame())) + self.assertIsInstance(self.opt_res_dayahead.index, pd.core.indexes.datetimes.DatetimeIndex) + self.assertIsInstance(self.opt_res_dayahead.index.dtype, pd.core.dtypes.dtypes.DatetimeTZDtype) + self.assertTrue('cost_fun_'+self.costfun in self.opt_res_dayahead.columns) self.assertTrue(self.opt.optim_status == 'Optimal') def test_perform_dayahead_forecast_optim_costfun_selfconso(self): From da94662b72ea77a38072026d6a19c7b9851bb07d Mon Sep 17 00:00:00 2001 From: davidusb-geek Date: Thu, 4 Jan 2024 01:49:19 +0100 Subject: [PATCH 5/7] Feat - Added option to pass additional weight for battery usage --- config_emhass.yaml | 2 ++ docs/config.md | 2 ++ options.json | 2 ++ src/emhass/optimization.py | 20 ++++++++++---------- src/emhass/utils.py | 4 ++++ src/emhass/web_server.py | 2 ++ 6 files changed, 22 insertions(+), 10 deletions(-) diff --git a/config_emhass.yaml b/config_emhass.yaml index 18d56b2c..cf8b8425 100644 --- a/config_emhass.yaml +++ b/config_emhass.yaml @@ -52,6 +52,8 @@ optim_conf: - set_battery_dynamic: False # add a constraint to limit the dynamic of the battery power in power per time unit - battery_dynamic_max: 0.9 # maximum dynamic positive power variation in percentage of battery maximum power - battery_dynamic_min: -0.9 # minimum dynamic negative power variation in percentage of battery maximum power + - weight_battery_discharge: 1.0 # weight applied in cost function to battery usage for discharge + - weight_battery_charge: 1.0 # weight applied in cost function to battery usage for charge plant_conf: - P_grid_max: 9000 # The maximum power that can be supplied by the utility grid in Watts diff --git a/docs/config.md b/docs/config.md index a389e034..02017b60 100644 --- a/docs/config.md +++ b/docs/config.md @@ -79,6 +79,8 @@ The following parameters and definitions are only needed if load_cost_forecast_m - set_battery_dynamic: Set a power dynamic limiting condition to the battery power. This is an additional constraint on the battery dynamic in power per unit of time, which allows you to set a percentage of the battery nominal full power as the maximum power allowed for (dis)charge. - battery_dynamic_max: The maximum positive (for discharge) battery power dynamic. This is the allowed power variation (in percentage) of battery maximum power per unit of time. - battery_dynamic_min: The maximum negative (for charge) battery power dynamic. This is the allowed power variation (in percentage) of battery maximum power per unit of time. +- weight_battery_discharge: An additional weight applied in cost function to battery usage for discharge. +- weight_battery_charge: An additional weight applied in cost function to battery usage for charge. ## System configuration parameters diff --git a/options.json b/options.json index 0ca6c167..7814aef8 100644 --- a/options.json +++ b/options.json @@ -13,6 +13,8 @@ "set_battery_dynamic": false, "battery_dynamic_max": 0.9, "battery_dynamic_min": -0.9, + "weight_battery_discharge": 1.0, + "weight_battery_charge": 1.0, "sensor_power_photovoltaics": "sensor.power_photovoltaics", "sensor_power_load_no_var_loads": "sensor.power_load_no_var_loads", "number_of_deferrable_loads": 3, diff --git a/src/emhass/optimization.py b/src/emhass/optimization.py index e4713f5c..924fab54 100644 --- a/src/emhass/optimization.py +++ b/src/emhass/optimization.py @@ -197,31 +197,31 @@ def perform_optimization(self, data_opt: pd.DataFrame, P_PV: np.array, P_load: n if self.costfun == 'profit': if self.optim_conf['set_total_pv_sell']: objective = plp.lpSum(-0.001*self.timeStep*(unit_load_cost[i]*(P_load[i] + P_def_sum[i]) + \ - unit_prod_price[i]*P_grid_neg[i]) - for i in set_I) + unit_prod_price[i]*P_grid_neg[i]) for i in set_I) else: objective = plp.lpSum(-0.001*self.timeStep*(unit_load_cost[i]*P_grid_pos[i] + \ - unit_prod_price[i]*P_grid_neg[i]) - for i in set_I) + unit_prod_price[i]*P_grid_neg[i]) for i in set_I) elif self.costfun == 'cost': if self.optim_conf['set_total_pv_sell']: - objective = plp.lpSum(-0.001*self.timeStep*unit_load_cost[i]*(P_load[i] + P_def_sum[i]) - for i in set_I) + objective = plp.lpSum(-0.001*self.timeStep*unit_load_cost[i]*(P_load[i] + P_def_sum[i]) for i in set_I) else: - objective = plp.lpSum(-0.001*self.timeStep*unit_load_cost[i]*P_grid_pos[i] - for i in set_I) + objective = plp.lpSum(-0.001*self.timeStep*unit_load_cost[i]*P_grid_pos[i] for i in set_I) elif self.costfun == 'self-consumption': if type_self_conso == 'bigm': bigm = 1e3 objective = plp.lpSum(-0.001*self.timeStep*(bigm*unit_load_cost[i]*P_grid_pos[i] + \ - unit_prod_price[i]*P_grid_neg[i]) - for i in set_I) + unit_prod_price[i]*P_grid_neg[i]) for i in set_I) elif type_self_conso == 'maxmin': objective = plp.lpSum(0.001*self.timeStep*unit_load_cost[i]*SC[i] for i in set_I) else: self.logger.error("Not a valida option for type_self_conso parameter") else: self.logger.error("The cost function specified type is not valid") + # Add more terms to the objective function in the case of battery use + if self.optim_conf['set_use_battery']: + objective = objective + plp.lpSum(-0.001*self.timeStep*( + self.optim_conf['weight_battery_discharge']*P_sto_pos[i] + \ + self.optim_conf['weight_battery_charge']*P_sto_neg[i]) for i in set_I) opt_model.setObjective(objective) ## Setting constraints diff --git a/src/emhass/utils.py b/src/emhass/utils.py index cd2b5a43..8cd5f6f6 100644 --- a/src/emhass/utils.py +++ b/src/emhass/utils.py @@ -324,6 +324,10 @@ def treat_runtimeparams(runtimeparams: str, params: str, retrieve_hass_conf: dic if 'solar_forecast_kwp' in runtimeparams.keys(): retrieve_hass_conf['solar_forecast_kwp'] = runtimeparams['solar_forecast_kwp'] optim_conf['weather_forecast_method'] = 'solar.forecast' + if 'weight_battery_discharge' in runtimeparams.keys(): + optim_conf['weight_battery_discharge'] = runtimeparams['weight_battery_discharge'] + if 'weight_battery_charge' in runtimeparams.keys(): + optim_conf['weight_battery_charge'] = runtimeparams['weight_battery_charge'] # Treat plant configuration parameters passed at runtime if 'SOCtarget' in runtimeparams.keys(): plant_conf['SOCtarget'] = runtimeparams['SOCtarget'] diff --git a/src/emhass/web_server.py b/src/emhass/web_server.py index 8701c6b7..91ea956e 100644 --- a/src/emhass/web_server.py +++ b/src/emhass/web_server.py @@ -130,6 +130,8 @@ def build_params(params, options, addon): params['optim_conf'][20]['set_battery_dynamic'] = options['set_battery_dynamic'] params['optim_conf'][21]['battery_dynamic_max'] = options['battery_dynamic_max'] params['optim_conf'][22]['battery_dynamic_min'] = options['battery_dynamic_min'] + params['optim_conf'][23]['weight_battery_discharge'] = options['weight_battery_discharge'] + params['optim_conf'][24]['weight_battery_charge'] = options['weight_battery_charge'] # Updating variables in plant_conf params['plant_conf'][0]['P_grid_max'] = options['maximum_power_from_grid'] params['plant_conf'][1]['module_model'] = [i['pv_module_model'] for i in options['list_pv_module_model']] From 69b7d1084b6fff0147f37859b34276f63fbaa681 Mon Sep 17 00:00:00 2001 From: davidusb-geek Date: Thu, 4 Jan 2024 02:05:08 +0100 Subject: [PATCH 6/7] Fix - Improved coverage for utils --- CHANGELOG.md | 3 +++ docs/conf.py | 2 +- setup.py | 2 +- tests/test_utils.py | 25 +++++++++++++++++++++++++ 4 files changed, 30 insertions(+), 2 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 4ea36156..2117396d 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,6 +1,9 @@ # Changelog ## [0.6.2] - Unreleased +### Improvement +- Added option to pass additional weight for battery usage +- Improved coverage ### Fix - Updated optimization constraints to solve conflict for `set_def_constant` and `treat_def_as_semi_cont` cases diff --git a/docs/conf.py b/docs/conf.py index 64c86692..f37f20e6 100644 --- a/docs/conf.py +++ b/docs/conf.py @@ -22,7 +22,7 @@ author = 'David HERNANDEZ' # The full version, including alpha/beta/rc tags -release = '0.6.1' +release = '0.6.2' # -- General configuration --------------------------------------------------- diff --git a/setup.py b/setup.py index 89ac60c7..3ef8916c 100644 --- a/setup.py +++ b/setup.py @@ -19,7 +19,7 @@ setup( name='emhass', # Required - version='0.6.1', # Required + version='0.6.2', # Required description='An Energy Management System for Home Assistant', # Optional long_description=long_description, # Optional long_description_content_type='text/markdown', # Optional (see note above) diff --git a/tests/test_utils.py b/tests/test_utils.py index 76c4d40b..f57f51c1 100644 --- a/tests/test_utils.py +++ b/tests/test_utils.py @@ -114,11 +114,24 @@ def test_treat_runtimeparams(self): runtimeparams.update({'def_total_hours':[5, 8, 10]}) runtimeparams.update({'treat_def_as_semi_cont':[True, True, True]}) runtimeparams.update({'set_def_constant':[False, False, False]}) + runtimeparams.update({'weight_battery_discharge':2.0}) + runtimeparams.update({'weight_battery_charge':2.0}) runtimeparams.update({'solcast_api_key':'yoursecretsolcastapikey'}) runtimeparams.update({'solcast_rooftop_id':'yourrooftopid'}) runtimeparams.update({'solar_forecast_kwp':5.0}) runtimeparams.update({'SOCtarget':0.4}) runtimeparams.update({'publish_prefix':'emhass_'}) + runtimeparams.update({'custom_pv_forecast_id':'my_custom_pv_forecast_id'}) + runtimeparams.update({'custom_load_forecast_id':'my_custom_load_forecast_id'}) + runtimeparams.update({'custom_batt_forecast_id':'my_custom_batt_forecast_id'}) + runtimeparams.update({'custom_batt_soc_forecast_id':'my_custom_batt_soc_forecast_id'}) + runtimeparams.update({'custom_grid_forecast_id':'my_custom_grid_forecast_id'}) + runtimeparams.update({'custom_cost_fun_id':'my_custom_cost_fun_id'}) + runtimeparams.update({'custom_optim_status_id':'my_custom_optim_status_id'}) + runtimeparams.update({'custom_unit_load_cost_id':'my_custom_unit_load_cost_id'}) + runtimeparams.update({'custom_unit_prod_price_id':'my_custom_unit_prod_price_id'}) + runtimeparams.update({'custom_deferrable_forecast_id':'my_custom_deferrable_forecast_id'}) + runtimeparams_json = json.dumps(runtimeparams) retrieve_hass_conf, optim_conf, plant_conf = utils.get_yaml_parse( pathlib.Path(root+'/config_emhass.yaml'), use_secrets=True, params=self.params_json) @@ -137,11 +150,23 @@ def test_treat_runtimeparams(self): self.assertTrue(optim_conf['def_total_hours'] == [5, 8, 10]) self.assertTrue(optim_conf['treat_def_as_semi_cont'] == [True, True, True]) self.assertTrue(optim_conf['set_def_constant'] == [False, False, False]) + self.assertTrue(optim_conf['weight_battery_discharge'] == 2.0) + self.assertTrue(optim_conf['weight_battery_charge'] == 2.0) self.assertTrue(retrieve_hass_conf['solcast_api_key'] == 'yoursecretsolcastapikey') self.assertTrue(retrieve_hass_conf['solcast_rooftop_id'] == 'yourrooftopid') self.assertTrue(retrieve_hass_conf['solar_forecast_kwp'] == 5.0) self.assertTrue(plant_conf['SOCtarget'] == 0.4) self.assertTrue(params['passed_data']['publish_prefix'] == 'emhass_') + self.assertTrue(params['passed_data']['custom_pv_forecast_id'] == 'my_custom_pv_forecast_id') + self.assertTrue(params['passed_data']['custom_load_forecast_id'] == 'my_custom_load_forecast_id') + self.assertTrue(params['passed_data']['custom_batt_forecast_id'] == 'my_custom_battforecast_id') + self.assertTrue(params['passed_data']['custom_batt_soc_forecast_id'] == 'my_custom_batt_soc_forecast_id') + self.assertTrue(params['passed_data']['custom_grid_forecast_id'] == 'my_custom_grid_forecast_id') + self.assertTrue(params['passed_data']['custom_cost_fun_id'] == 'my_custom_cost_fun_id') + self.assertTrue(params['passed_data']['custom_optim_status_id'] == 'my_custom_optim_status_id') + self.assertTrue(params['passed_data']['custom_unit_load_cost_id'] == 'my_custom_unit_load_cost_id') + self.assertTrue(params['passed_data']['custom_unit_prod_price_id'] == 'my_custom_unit_prod_price_id') + self.assertTrue(params['passed_data']['custom_deferrable_forecast_id'] == 'my_custom_deferrable_forecast_id') def test_treat_runtimeparams_failed(self): params = TestCommandLineUtils.get_test_params() From 7eb7fc0f1e0f7a91755a97f09b3eac42799644d8 Mon Sep 17 00:00:00 2001 From: davidusb-geek Date: Thu, 4 Jan 2024 02:11:31 +0100 Subject: [PATCH 7/7] Fix - Fixed test utils type error --- CHANGELOG.md | 2 +- tests/test_utils.py | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 2117396d..7ac6df8d 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,6 +1,6 @@ # Changelog -## [0.6.2] - Unreleased +## [0.6.2] - 2024-01-04 ### Improvement - Added option to pass additional weight for battery usage - Improved coverage diff --git a/tests/test_utils.py b/tests/test_utils.py index f57f51c1..430a6897 100644 --- a/tests/test_utils.py +++ b/tests/test_utils.py @@ -159,7 +159,7 @@ def test_treat_runtimeparams(self): self.assertTrue(params['passed_data']['publish_prefix'] == 'emhass_') self.assertTrue(params['passed_data']['custom_pv_forecast_id'] == 'my_custom_pv_forecast_id') self.assertTrue(params['passed_data']['custom_load_forecast_id'] == 'my_custom_load_forecast_id') - self.assertTrue(params['passed_data']['custom_batt_forecast_id'] == 'my_custom_battforecast_id') + self.assertTrue(params['passed_data']['custom_batt_forecast_id'] == 'my_custom_batt_forecast_id') self.assertTrue(params['passed_data']['custom_batt_soc_forecast_id'] == 'my_custom_batt_soc_forecast_id') self.assertTrue(params['passed_data']['custom_grid_forecast_id'] == 'my_custom_grid_forecast_id') self.assertTrue(params['passed_data']['custom_cost_fun_id'] == 'my_custom_cost_fun_id')