From 5ae1e18bd46954f394793dad80aaedb66e5fa67f Mon Sep 17 00:00:00 2001 From: GeoDerp <18461782+GeoDerp@users.noreply.github.com> Date: Sun, 28 Jan 2024 10:19:31 +0000 Subject: [PATCH 1/3] webserver if url or token is empty --- src/emhass/web_server.py | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/src/emhass/web_server.py b/src/emhass/web_server.py index ae8e76ca..7322fb49 100644 --- a/src/emhass/web_server.py +++ b/src/emhass/web_server.py @@ -322,13 +322,13 @@ def action_call(action_name): # Some data from options logging_level = options['logging_level'] url_from_options = options.get('hass_url', 'empty') - if url_from_options == 'empty': + if url_from_options == 'empty' or url_from_options == '': url = hass_url+"/config" else: hass_url = url_from_options url = hass_url+"/api/config" token_from_options = options.get('long_lived_token', 'empty') - if token_from_options == 'empty': + if token_from_options == 'empty' or token_from_options == '': long_lived_token = key else: long_lived_token = token_from_options @@ -336,7 +336,7 @@ def action_call(action_name): "Authorization": "Bearer " + long_lived_token, "content-type": "application/json" } - response = get(url, headers=headers) + response = get(url, headers=headers) config_hass = response.json() params_secrets = { 'hass_url': hass_url, From ee252310e8c7c8834710c1bcbe06d4f02b97591c Mon Sep 17 00:00:00 2001 From: GeoDerp <18461782+GeoDerp@users.noreply.github.com> Date: Sun, 28 Jan 2024 11:43:12 +0000 Subject: [PATCH 2/3] make options optional --- src/emhass/web_server.py | 201 ++++++++++++++++++++++++--------------- 1 file changed, 125 insertions(+), 76 deletions(-) diff --git a/src/emhass/web_server.py b/src/emhass/web_server.py index 7322fb49..7e3e6b24 100644 --- a/src/emhass/web_server.py +++ b/src/emhass/web_server.py @@ -97,75 +97,117 @@ def get_injection_dict_forecast_model_tune(df_pred_optim, mlf): injection_dict['figure_0'] = image_path_0 return injection_dict -def build_params(params, options, addon): +def build_params(params, params_secrets, options, addon): if addon == 1: # Updating variables in retrieve_hass_conf - params['retrieve_hass_conf']['freq'] = options['optimization_time_step'] - params['retrieve_hass_conf']['days_to_retrieve'] = options['historic_days_to_retrieve'] - params['retrieve_hass_conf']['var_PV'] = options['sensor_power_photovoltaics'] - params['retrieve_hass_conf']['var_load'] = options['sensor_power_load_no_var_loads'] - params['retrieve_hass_conf']['load_negative'] = [options['load_negative']] - params['retrieve_hass_conf']['set_zero_min'] = [options['set_zero_min']] - params['retrieve_hass_conf']['var_replace_zero'] = [options['sensor_power_photovoltaics']] - params['retrieve_hass_conf']['var_interp'] = [options['sensor_power_photovoltaics'], options['sensor_power_load_no_var_loads']] - params['retrieve_hass_conf']['method_ts_round'] = options['method_ts_round'] - params['retrieve_hass_conf']['solcast_api_key'] = options['optional_solcast_api_key'] - params['retrieve_hass_conf']['solcast_rooftop_id'] = options['optional_solcast_rooftop_id'] - params['retrieve_hass_conf']['solar_forecast_kwp'] = options['optional_solar_forecast_kwp'] - params['retrieve_hass_conf']['time_zone'] = options['time_zone'] - params['retrieve_hass_conf']['lat'] = options['Latitude'] - params['retrieve_hass_conf']['lon'] = options['Longitude'] - params['retrieve_hass_conf']['alt'] = options['Altitude'] + params['retrieve_hass_conf']['freq'] = options.get('optimization_time_step',params['retrieve_hass_conf']['freq']) + params['retrieve_hass_conf']['days_to_retrieve'] = options.get('historic_days_to_retrieve',params['retrieve_hass_conf']['days_to_retrieve']) + params['retrieve_hass_conf']['var_PV'] = options.get('sensor_power_photovoltaics',params['retrieve_hass_conf']['var_PV']) + params['retrieve_hass_conf']['var_load'] = options.get('sensor_power_load_no_var_loads',params['retrieve_hass_conf']['var_load']) + params['retrieve_hass_conf']['load_negative'] = options.get('load_negative',params['retrieve_hass_conf']['load_negative']) + params['retrieve_hass_conf']['set_zero_min'] = options.get('set_zero_min',params['retrieve_hass_conf']['set_zero_min']) + params['retrieve_hass_conf']['var_replace_zero'] = options.get('sensor_power_photovoltaics',params['retrieve_hass_conf']['var_replace_zero']) + params['retrieve_hass_conf']['var_interp'] = options.get('sensor_power_photovoltaics',params['retrieve_hass_conf']['var_PV']), options.get('sensor_power_load_no_var_loads',params['retrieve_hass_conf']['var_load']) + params['retrieve_hass_conf']['method_ts_round'] = options.get('method_ts_round',params['retrieve_hass_conf']['method_ts_round']) + # Update params Secrets if specified + params['params_secrets'] = params_secrets + params['params_secrets']['time_zone'] = options.get('time_zone',params_secrets['time_zone']) + params['params_secrets']['lat'] = options.get('Latitude',params_secrets['lat']) + params['params_secrets']['lon'] = options.get('Longitude',params_secrets['lon']) + params['params_secrets']['alt'] = options.get('Altitude',params_secrets['alt']) # Updating variables in optim_conf - params['optim_conf']['set_use_battery'] = options['set_use_battery'] - params['optim_conf']['num_def_loads'] = options['number_of_deferrable_loads'] - params['optim_conf']['P_deferrable_nom'] = [i['nominal_power_of_deferrable_loads'] for i in options['list_nominal_power_of_deferrable_loads']] - params['optim_conf']['def_total_hours'] = [i['operating_hours_of_each_deferrable_load'] for i in options['list_operating_hours_of_each_deferrable_load']] - params['optim_conf']['treat_def_as_semi_cont'] = [i['treat_deferrable_load_as_semi_cont'] for i in options['list_treat_deferrable_load_as_semi_cont']] - params['optim_conf']['weather_forecast_method'] = options['weather_forecast_method'] - params['optim_conf']['load_forecast_method'] = options['load_forecast_method'] - params['optim_conf']['delta_forecast'] = options['delta_forecast_daily'] - params['optim_conf']['load_cost_forecast_method'] = options['load_cost_forecast_method'] - params['optim_conf']['set_def_constant'] = [i['set_deferrable_load_single_constant'] for i in options['list_set_deferrable_load_single_constant']] - start_hours_list = [i['peak_hours_periods_start_hours'] for i in options['list_peak_hours_periods_start_hours']] - end_hours_list = [i['peak_hours_periods_end_hours'] for i in options['list_peak_hours_periods_end_hours']] - num_peak_hours = len(start_hours_list) - list_hp_periods_list = [{'period_hp_'+str(i+1):[{'start':start_hours_list[i]},{'end':end_hours_list[i]}]} for i in range(num_peak_hours)] - params['optim_conf']['list_hp_periods'] = list_hp_periods_list - params['optim_conf']['load_cost_hp'] = options['load_peak_hours_cost'] - params['optim_conf']['load_cost_hc'] = options['load_offpeak_hours_cost'] - params['optim_conf']['prod_price_forecast_method'] = options['production_price_forecast_method'] - params['optim_conf']['prod_sell_price'] = options['photovoltaic_production_sell_price'] - params['optim_conf']['set_total_pv_sell'] = options['set_total_pv_sell'] - params['optim_conf']['lp_solver'] = options['lp_solver'] - params['optim_conf']['lp_solver_path'] = options['lp_solver_path'] - params['optim_conf']['set_nocharge_from_grid'] = options['set_nocharge_from_grid'] - params['optim_conf']['set_nodischarge_to_grid'] = options['set_nodischarge_to_grid'] - params['optim_conf']['set_battery_dynamic'] = options['set_battery_dynamic'] - params['optim_conf']['battery_dynamic_max'] = options['battery_dynamic_max'] - params['optim_conf']['battery_dynamic_min'] = options['battery_dynamic_min'] - params['optim_conf']['weight_battery_discharge'] = options['weight_battery_discharge'] - params['optim_conf']['weight_battery_charge'] = options['weight_battery_charge'] - params['optim_conf']['def_start_timestep'] = [i['start_timesteps_of_each_deferrable_load'] for i in options['list_start_timesteps_of_each_deferrable_load']] - params['optim_conf']['def_end_timestep'] = [i['end_timesteps_of_each_deferrable_load'] for i in options['list_end_timesteps_of_each_deferrable_load']] - # Updating variables in plant_conf - params['plant_conf']['P_grid_max'] = options['maximum_power_from_grid'] - params['plant_conf']['module_model'] = [i['pv_module_model'] for i in options['list_pv_module_model']] - params['plant_conf']['inverter_model'] = [i['pv_inverter_model'] for i in options['list_pv_inverter_model']] - params['plant_conf']['surface_tilt'] = [i['surface_tilt'] for i in options['list_surface_tilt']] - params['plant_conf']['surface_azimuth'] = [i['surface_azimuth'] for i in options['list_surface_azimuth']] - params['plant_conf']['modules_per_string'] = [i['modules_per_string'] for i in options['list_modules_per_string']] - params['plant_conf']['strings_per_inverter'] = [i['strings_per_inverter'] for i in options['list_strings_per_inverter']] - params['plant_conf']['Pd_max'] = options['battery_discharge_power_max'] - params['plant_conf']['Pc_max'] = options['battery_charge_power_max'] - params['plant_conf']['eta_disch'] = options['battery_discharge_efficiency'] - params['plant_conf']['eta_ch'] = options['battery_charge_efficiency'] - params['plant_conf']['Enom'] = options['battery_nominal_energy_capacity'] - params['plant_conf']['SOCmin'] = options['battery_minimum_state_of_charge'] - params['plant_conf']['SOCmax'] = options['battery_maximum_state_of_charge'] - params['plant_conf']['SOCtarget'] = options['battery_target_state_of_charge'] + params['optim_conf']['set_use_battery'] = options.get('set_use_battery',params['optim_conf']['set_use_battery']) + params['optim_conf']['num_def_loads'] = options.get('number_of_deferrable_loads',params['optim_conf']['num_def_loads']) + try: + params['optim_conf']['P_deferrable_nom'] = [i['nominal_power_of_deferrable_loads'] for i in options.get('list_nominal_power_of_deferrable_loads')] + except: + app.logger.debug("no list_nominal_power_of_deferrable_loads, defaulting to config file") + try: + params['optim_conf']['def_total_hours'] = [i['operating_hours_of_each_deferrable_load'] for i in options.get('list_operating_hours_of_each_deferrable_load')] + except: + app.logger.debug("no list_operating_hours_of_each_deferrable_load, defaulting to config file") + try: + params['optim_conf']['treat_def_as_semi_cont'] = [i['treat_deferrable_load_as_semi_cont'] for i in options.get('list_treat_deferrable_load_as_semi_cont')] + except: + app.logger.debug("no list_treat_deferrable_load_as_semi_cont, defaulting to config file") + params['optim_conf']['weather_forecast_method'] = options.get('weather_forecast_method',params['optim_conf']['weather_forecast_method']) + if params['optim_conf']['weather_forecast_method'] == "solcast": + params['retrieve_hass_conf']['solcast_api_key'] = options.get('optional_solcast_api_key',params['retrieve_hass_conf']['solcast_api_key']) + params['retrieve_hass_conf']['solcast_rooftop_id'] = options.get('optional_solcast_rooftop_id',params['retrieve_hass_conf']['solcast_rooftop_id']) + elif params['optim_conf']['weather_forecast_method'] == "solar.forecast": + params['retrieve_hass_conf']['solar_forecast_kwp'] = options.get('optional_solar_forecast_kwp',params['retrieve_hass_conf']['solar_forecast_kwp']) + params['optim_conf']['load_forecast_method'] = options.get('load_forecast_method',params['optim_conf']['load_forecast_method']) + params['optim_conf']['delta_forecast'] = options.get('delta_forecast_daily',params['optim_conf']['delta_forecast']) + params['optim_conf']['load_cost_forecast_method'] = options.get('load_cost_forecast_method',params['optim_conf']['load_cost_forecast_method']) + try: + params['optim_conf']['set_def_constant'] = [i['set_deferrable_load_single_constant'] for i in options.get('list_set_deferrable_load_single_constant')] + except: + app.logger.debug("no list_set_deferrable_load_single_constant, defaulting to config file") + try: + start_hours_list = [i['peak_hours_periods_start_hours'] for i in options['list_peak_hours_periods_start_hours']] + end_hours_list = [i['peak_hours_periods_end_hours'] for i in options['list_peak_hours_periods_end_hours']] + num_peak_hours = len(start_hours_list) + list_hp_periods_list = [{'period_hp_'+str(i+1):[{'start':start_hours_list[i]},{'end':end_hours_list[i]}]} for i in range(num_peak_hours)] + params['optim_conf']['list_hp_periods'] = list_hp_periods_list + except: + app.logger.debug("no list_peak_hours_periods_start_hours, defaulting to config file") + params['optim_conf']['load_cost_hp'] = options.get('load_peak_hours_cost',params['optim_conf']['load_cost_hp']) + params['optim_conf']['load_cost_hc'] = options.get('load_offpeak_hours_cost', params['optim_conf']['load_cost_hc']) + params['optim_conf']['prod_price_forecast_method'] = options.get('production_price_forecast_method', params['optim_conf']['prod_price_forecast_method']) + params['optim_conf']['prod_sell_price'] = options.get('photovoltaic_production_sell_price',params['optim_conf']['prod_sell_price']) + params['optim_conf']['set_total_pv_sell'] = options.get('set_total_pv_sell',params['optim_conf']['set_total_pv_sell']) + params['optim_conf']['lp_solver'] = options.get('lp_solver',params['optim_conf']['lp_solver']) + params['optim_conf']['lp_solver_path'] = options.get('lp_solver_path',params['optim_conf']['lp_solver_path']) + params['optim_conf']['set_nocharge_from_grid'] = options.get('set_nocharge_from_grid',params['optim_conf']['set_nocharge_from_grid']) + params['optim_conf']['set_nodischarge_to_grid'] = options.get('set_nodischarge_to_grid',params['optim_conf']['set_nodischarge_to_grid']) + params['optim_conf']['set_battery_dynamic'] = options.get('set_battery_dynamic',params['optim_conf']['set_battery_dynamic']) + params['optim_conf']['battery_dynamic_max'] = options.get('battery_dynamic_max',params['optim_conf']['battery_dynamic_max']) + params['optim_conf']['battery_dynamic_min'] = options.get('battery_dynamic_min',params['optim_conf']['battery_dynamic_min']) + params['optim_conf']['weight_battery_discharge'] = options.get('weight_battery_discharge',params['optim_conf']['weight_battery_discharge']) + params['optim_conf']['weight_battery_charge'] = options.get('weight_battery_charge',params['optim_conf']['weight_battery_charge']) + try: + params['optim_conf']['def_start_timestep'] = [i['start_timesteps_of_each_deferrable_load'] for i in options.get('list_start_timesteps_of_each_deferrable_load')] + except: + app.logger.debug("no list_start_timesteps_of_each_deferrable_load, defaulting to config file") + try: + params['optim_conf']['def_end_timestep'] = [i['end_timesteps_of_each_deferrable_load'] for i in options.get('list_end_timesteps_of_each_deferrable_load')] + except: + app.logger.debug("no list_end_timesteps_of_each_deferrable_load, defaulting to config file") + # Updating variables in plant_con + params['plant_conf']['P_grid_max'] = options.get('maximum_power_from_grid',params['plant_conf']['P_grid_max']) + try: + params['plant_conf']['module_model'] = [i['pv_module_model'] for i in options.get('list_pv_module_model')] + except: + app.logger.debug("no list_pv_module_model, defaulting to config file") + try: + params['plant_conf']['inverter_model'] = [i['pv_inverter_model'] for i in options.get('list_pv_inverter_model')] + except: + app.logger.debug("no list_pv_inverter_model, defaulting to config file") + try: + params['plant_conf']['surface_tilt'] = [i['surface_tilt'] for i in options.get('list_surface_tilt')] + except: + app.logger.debug("no list_surface_tilt, defaulting to config file") + try: + params['plant_conf']['surface_azimuth'] = [i['surface_azimuth'] for i in options.get('list_surface_azimuth')] + except: + app.logger.debug("no list_surface_azimuth, defaulting to config file") + try: + params['plant_conf']['modules_per_string'] = [i['modules_per_string'] for i in options.get('list_modules_per_string')] + except: + app.logger.debug("no list_modules_per_string, defaulting to config file") + try: + params['plant_conf']['strings_per_inverter'] = [i['strings_per_inverter'] for i in options.get('list_strings_per_inverter')] + except: + app.logger.debug("no list_strings_per_inverter, defaulting to config file") + params['plant_conf']['Pd_max'] = options.get('battery_discharge_power_max',params['plant_conf']['Pd_max']) + params['plant_conf']['Pc_max'] = options.get('battery_charge_power_max',params['plant_conf']['Pc_max']) + params['plant_conf']['eta_disch'] = options.get('battery_discharge_efficiency',params['plant_conf']['eta_disch']) + params['plant_conf']['eta_ch'] = options.get('battery_charge_efficiency',params['plant_conf']['eta_ch']) + params['plant_conf']['Enom'] = options.get('battery_nominal_energy_capacity',params['plant_conf']['Enom']) + params['plant_conf']['SOCmin'] = options.get('battery_minimum_state_of_charge',params['plant_conf']['SOCmin']) + params['plant_conf']['SOCmax'] = options.get('battery_maximum_state_of_charge',params['plant_conf']['SOCmax']) + params['plant_conf']['SOCtarget'] = options.get('battery_target_state_of_charge',params['plant_conf']['SOCtarget']) # The params dict - params['params_secrets'] = params_secrets params['passed_data'] = {'pv_power_forecast':None,'load_power_forecast':None,'load_cost_forecast':None,'prod_price_forecast':None, 'prediction_horizon':None,'soc_init':None,'soc_final':None,'def_total_hours':None,'def_start_timestep':None,'def_end_timestep':None,'alpha':None,'beta':None} return params @@ -320,7 +362,7 @@ def action_call(action_name): # The cost function costfun = options.get('costfun', 'profit') # Some data from options - logging_level = options['logging_level'] + logging_level = options.get('logging_level','INFO') url_from_options = options.get('hass_url', 'empty') if url_from_options == 'empty' or url_from_options == '': url = hass_url+"/config" @@ -337,15 +379,22 @@ def action_call(action_name): "content-type": "application/json" } response = get(url, headers=headers) - config_hass = response.json() - params_secrets = { - 'hass_url': hass_url, - 'long_lived_token': long_lived_token, - 'time_zone': config_hass['time_zone'], - 'lat': config_hass['latitude'], - 'lon': config_hass['longitude'], - 'alt': config_hass['elevation'] - } + try: + config_hass = response.json() + + params_secrets = { + 'hass_url': hass_url, + 'long_lived_token': long_lived_token, + 'time_zone': config_hass['time_zone'], + 'lat': config_hass['latitude'], + 'lon': config_hass['longitude'], + 'alt': config_hass['elevation'] + } + except: #if addon testing (use secrets) + with open(os.getenv('SECRETS_PATH', default='/app/secrets_emhass.yaml'), 'r') as file: + params_secrets = yaml.load(file, Loader=yaml.FullLoader) + params_secrets['hass_url'] = hass_url + params_secrets['long_lived_token'] = long_lived_token else: costfun = os.getenv('LOCAL_COSTFUN', default='profit') logging_level = os.getenv('LOGGING_LEVEL', default='INFO') @@ -354,7 +403,7 @@ def action_call(action_name): hass_url = params_secrets['hass_url'] # Build params - params = build_params(params, options, args.addon) + params = build_params(params, params_secrets, options, args.addon) with open(str(data_path / 'params.pkl'), "wb") as fid: pickle.dump((config_path, params), fid) From 8e447d9e29141b3721328ee8a92df0ff109b044d Mon Sep 17 00:00:00 2001 From: GeoDerp <18461782+GeoDerp@users.noreply.github.com> Date: Sun, 28 Jan 2024 12:38:27 +0000 Subject: [PATCH 3/3] webserver added else clause in build --- src/emhass/web_server.py | 2 ++ 1 file changed, 2 insertions(+) diff --git a/src/emhass/web_server.py b/src/emhass/web_server.py index 7e3e6b24..596e8c37 100644 --- a/src/emhass/web_server.py +++ b/src/emhass/web_server.py @@ -207,6 +207,8 @@ def build_params(params, params_secrets, options, addon): params['plant_conf']['SOCmin'] = options.get('battery_minimum_state_of_charge',params['plant_conf']['SOCmin']) params['plant_conf']['SOCmax'] = options.get('battery_maximum_state_of_charge',params['plant_conf']['SOCmax']) params['plant_conf']['SOCtarget'] = options.get('battery_target_state_of_charge',params['plant_conf']['SOCtarget']) + else: + params['params_secrets'] = params_secrets # The params dict params['passed_data'] = {'pv_power_forecast':None,'load_power_forecast':None,'load_cost_forecast':None,'prod_price_forecast':None, 'prediction_horizon':None,'soc_init':None,'soc_final':None,'def_total_hours':None,'def_start_timestep':None,'def_end_timestep':None,'alpha':None,'beta':None}