Source code for edisgo.flex_opt.reinforce_measures

import logging
import math

import networkx as nx
import numpy as np
import pandas as pd

from networkx.algorithms.shortest_paths.weighted import (
    _dijkstra as dijkstra_shortest_path_length,
)

from edisgo.network.grids import LVGrid, MVGrid

logger = logging.getLogger(__name__)


[docs]def reinforce_mv_lv_station_overloading(edisgo_obj, critical_stations): """ Reinforce MV/LV substations due to overloading issues. In a first step a parallel transformer of the same kind is installed. If this is not sufficient as many standard transformers as needed are installed. Parameters ---------- edisgo_obj : :class:`~.EDisGo` critical_stations : :pandas:`pandas.DataFrame<DataFrame>` Dataframe containing over-loaded MV/LV stations, their missing apparent power at maximal over-loading and the corresponding time step. Index of the dataframe are the representatives of the grids with over-loaded stations. Columns are 's_missing' containing the missing apparent power at maximal over-loading in MVA as float and 'time_index' containing the corresponding time step the over-loading occured in as :pandas:`pandas.Timestamp<Timestamp>`. Returns ------- dict Dictionary with added and removed transformers in the form:: {'added': {'Grid_1': ['transformer_reinforced_1', ..., 'transformer_reinforced_x'], 'Grid_10': ['transformer_reinforced_10'] }, 'removed': {'Grid_1': ['transformer_1']} } """ transformers_changes = _station_overloading( edisgo_obj, critical_stations, voltage_level="lv" ) if transformers_changes["added"]: logger.debug( "==> {} LV station(s) has/have been reinforced due to " "overloading issues.".format(str(len(transformers_changes["added"]))) ) return transformers_changes
[docs]def reinforce_hv_mv_station_overloading(edisgo_obj, critical_stations): """ Reinforce HV/MV station due to overloading issues. In a first step a parallel transformer of the same kind is installed. If this is not sufficient as many standard transformers as needed are installed. Parameters ---------- edisgo_obj : :class:`~.EDisGo` critical_stations : pandas:`pandas.DataFrame<DataFrame>` Dataframe containing over-loaded HV/MV stations, their missing apparent power at maximal over-loading and the corresponding time step. Index of the dataframe are the representatives of the grids with over-loaded stations. Columns are 's_missing' containing the missing apparent power at maximal over-loading in MVA as float and 'time_index' containing the corresponding time step the over-loading occured in as :pandas:`pandas.Timestamp<Timestamp>`. Returns ------- dict Dictionary with added and removed transformers in the form:: {'added': {'Grid_1': ['transformer_reinforced_1', ..., 'transformer_reinforced_x'], 'Grid_10': ['transformer_reinforced_10'] }, 'removed': {'Grid_1': ['transformer_1']} } """ transformers_changes = _station_overloading( edisgo_obj, critical_stations, voltage_level="mv" ) if transformers_changes["added"]: logger.debug("==> MV station has been reinforced due to overloading issues.") return transformers_changes
def _station_overloading(edisgo_obj, critical_stations, voltage_level): """ Reinforce stations due to overloading issues. In a first step a parallel transformer of the same kind is installed. If this is not sufficient as many standard transformers as needed are installed. Parameters ---------- edisgo_obj : :class:`~.EDisGo` critical_stations : :pandas:`pandas.DataFrame<DataFrame>` Dataframe containing over-loaded MV/LV stations, their missing apparent power at maximal over-loading and the corresponding time step. Index of the dataframe are the representatives of the grids with over-loaded stations. Columns are 's_missing' containing the missing apparent power at maximal over-loading in MVA as float and 'time_index' containing the corresponding time step the over-loading occured in as :pandas:`pandas.Timestamp<Timestamp>`. voltage_level : str Voltage level, over-loading is handled for. Possible options are "mv" or "lv". Returns ------- dict Dictionary with added and removed transformers in the form:: {'added': {'Grid_1': ['transformer_reinforced_1', ..., 'transformer_reinforced_x'], 'Grid_10': ['transformer_reinforced_10'] }, 'removed': {'Grid_1': ['transformer_1']} } """ if voltage_level == "lv": try: standard_transformer = edisgo_obj.topology.equipment_data[ "lv_transformers" ].loc[ edisgo_obj.config["grid_expansion_standard_equipment"][ "mv_lv_transformer" ] ] except KeyError: raise KeyError("Standard MV/LV transformer is not in equipment list.") elif voltage_level == "mv": try: standard_transformer = edisgo_obj.topology.equipment_data[ "mv_transformers" ].loc[ edisgo_obj.config["grid_expansion_standard_equipment"][ "hv_mv_transformer" ] ] except KeyError: raise KeyError("Standard HV/MV transformer is not in equipment list.") else: raise ValueError( "{} is not a valid option for input variable 'voltage_level' in " "function _station_overloading. Try 'mv' or " "'lv'.".format(voltage_level) ) transformers_changes = {"added": {}, "removed": {}} for grid_name in critical_stations.index: if "MV" in grid_name: grid = edisgo_obj.topology.mv_grid else: grid = edisgo_obj.topology.get_lv_grid(grid_name) # list of maximum power of each transformer in the station s_max_per_trafo = grid.transformers_df.s_nom # missing capacity s_trafo_missing = critical_stations.at[grid_name, "s_missing"] # check if second transformer of the same kind is sufficient # if true install second transformer, otherwise install as many # standard transformers as needed if max(s_max_per_trafo) >= s_trafo_missing: # if station has more than one transformer install a new # transformer of the same kind as the transformer that best # meets the missing power demand new_transformers = grid.transformers_df.loc[ [ grid.transformers_df[s_max_per_trafo >= s_trafo_missing][ "s_nom" ].idxmin() ] ] name = new_transformers.index[0].split("_") name.insert(-1, "reinforced") name[-1] = len(grid.transformers_df) + 1 new_transformers.index = ["_".join([str(_) for _ in name])] # add new transformer to list of added transformers transformers_changes["added"][grid_name] = [new_transformers.index[0]] else: # get any transformer to get attributes for new transformer from duplicated_transformer = grid.transformers_df.iloc[[0]] name = duplicated_transformer.index[0].split("_") name.insert(-1, "reinforced") duplicated_transformer.s_nom = standard_transformer.S_nom duplicated_transformer.type_info = standard_transformer.name if voltage_level == "lv": duplicated_transformer.r_pu = standard_transformer.r_pu duplicated_transformer.x_pu = standard_transformer.x_pu # set up as many new transformers as needed number_transformers = math.ceil( (s_trafo_missing + s_max_per_trafo.sum()) / standard_transformer.S_nom ) index = [] for i in range(number_transformers): name[-1] = i + 1 index.append("_".join([str(_) for _ in name])) if number_transformers > 1: new_transformers = duplicated_transformer.iloc[ np.arange(len(duplicated_transformer)).repeat(number_transformers) ] else: new_transformers = duplicated_transformer.copy() new_transformers.index = index # add new transformer to list of added transformers transformers_changes["added"][grid_name] = new_transformers.index.values # add previous transformers to list of removed transformers transformers_changes["removed"][ grid_name ] = grid.transformers_df.index.values # remove previous transformers from topology if voltage_level == "lv": edisgo_obj.topology.transformers_df.drop( grid.transformers_df.index.values, inplace=True ) else: edisgo_obj.topology.transformers_hvmv_df.drop( grid.transformers_df.index.values, inplace=True ) # add new transformers to topology if voltage_level == "lv": edisgo_obj.topology.transformers_df = pd.concat( [ edisgo_obj.topology.transformers_df, new_transformers, ] ) else: edisgo_obj.topology.transformers_hvmv_df = pd.concat( [ edisgo_obj.topology.transformers_hvmv_df, new_transformers, ] ) return transformers_changes
[docs]def reinforce_mv_lv_station_voltage_issues(edisgo_obj, critical_stations): """ Reinforce MV/LV substations due to voltage issues. A parallel standard transformer is installed. Parameters ---------- edisgo_obj : :class:`~.EDisGo` critical_stations : :obj:`dict` Dictionary with representative of :class:`~.network.grids.LVGrid` as key and a :pandas:`pandas.DataFrame<DataFrame>` with station's voltage deviation from allowed lower or upper voltage limit as value. Index of the dataframe is the station with voltage issues. Columns are 'v_diff_max' containing the maximum voltage deviation as float and 'time_index' containing the corresponding time step the voltage issue occured in as :pandas:`pandas.Timestamp<Timestamp>`. Returns ------- :obj:`dict` Dictionary with added transformers in the form:: {'added': {'Grid_1': ['transformer_reinforced_1', ..., 'transformer_reinforced_x'], 'Grid_10': ['transformer_reinforced_10'] } } """ # get parameters for standard transformer try: standard_transformer = edisgo_obj.topology.equipment_data[ "lv_transformers" ].loc[ edisgo_obj.config["grid_expansion_standard_equipment"]["mv_lv_transformer"] ] except KeyError: raise KeyError("Standard MV/LV transformer is not in equipment list.") transformers_changes = {"added": {}} for grid_name in critical_stations.keys(): if "MV" in grid_name: grid = edisgo_obj.topology.mv_grid else: grid = edisgo_obj.topology.get_lv_grid(grid_name) # get any transformer to get attributes for new transformer from duplicated_transformer = grid.transformers_df.iloc[[0]] # change transformer parameters name = duplicated_transformer.index[0].split("_") name.insert(-1, "reinforced") name[-1] = len(grid.transformers_df) + 1 duplicated_transformer.index = ["_".join([str(_) for _ in name])] duplicated_transformer.s_nom = standard_transformer.S_nom duplicated_transformer.r_pu = standard_transformer.r_pu duplicated_transformer.x_pu = standard_transformer.x_pu duplicated_transformer.type_info = standard_transformer.name # add new transformer to topology edisgo_obj.topology.transformers_df = pd.concat( [ edisgo_obj.topology.transformers_df, duplicated_transformer, ] ) transformers_changes["added"][grid_name] = duplicated_transformer.index.tolist() if transformers_changes["added"]: logger.debug( "==> {} LV station(s) has/have been reinforced due to voltage " "issues.".format(len(transformers_changes["added"])) ) return transformers_changes
[docs]def reinforce_lines_voltage_issues(edisgo_obj, grid, crit_nodes): """ Reinforce lines in MV and LV topology due to voltage issues. Parameters ---------- edisgo_obj : :class:`~.EDisGo` grid : :class:`~.network.grids.MVGrid` or :class:`~.network.grids.LVGrid` crit_nodes : :pandas:`pandas.DataFrame<DataFrame>` Dataframe with all nodes with voltage issues in the grid and their maximal deviations from allowed lower or upper voltage limits sorted descending from highest to lowest voltage deviation (it is not distinguished between over- or undervoltage). Columns of the dataframe are 'v_diff_max' containing the maximum absolute voltage deviation as float and 'time_index' containing the corresponding time step the voltage issue occured in as :pandas:`pandas.Timestamp<Timestamp>`. Index of the dataframe are the names of all buses with voltage issues. Returns ------- dict Dictionary with name of lines as keys and the corresponding number of lines added as values. Notes ----- Reinforce measures: 1. Disconnect line at 2/3 of the length between station and critical node farthest away from the station and install new standard line 2. Install parallel standard line In LV grids only lines outside buildings are reinforced; loads and generators in buildings cannot be directly connected to the MV/LV station. In MV grids lines can only be disconnected at LV stations because they have switch disconnectors needed to operate the lines as half rings (loads in MV would be suitable as well because they have a switch bay (Schaltfeld) but loads in dingo are only connected to MV busbar). If there is no suitable LV station the generator is directly connected to the MV busbar. There is no need for a switch disconnector in that case because generators don't need to be n-1 safe. """ # load standard line data if isinstance(grid, LVGrid): standard_line = edisgo_obj.config["grid_expansion_standard_equipment"][ "lv_line" ] elif isinstance(grid, MVGrid): standard_line = edisgo_obj.config["grid_expansion_standard_equipment"][ f"mv_line_{int(grid.nominal_voltage)}kv" ] else: raise ValueError("Inserted grid is invalid.") # find path to each node in order to find node with voltage issues farthest # away from station in each feeder station_node = grid.transformers_df.bus1.iloc[0] graph = grid.graph paths = {} nodes_feeder = {} for node in crit_nodes.index: path = nx.shortest_path(graph, station_node, node) paths[node] = path # raise exception if voltage issue occurs at station's secondary side # because voltage issues should have been solved during extension of # distribution substations due to overvoltage issues. if len(path) == 1: logging.error( "Voltage issues at busbar in LV network {} should have " "been solved in previous steps.".format(grid) ) nodes_feeder.setdefault(path[1], []).append(node) lines_changes = {} for repr_node in nodes_feeder.keys(): # find node farthest away get_weight = lambda u, v, data: data["length"] # noqa: E731 path_length = 0 for n in nodes_feeder[repr_node]: path_length_dict_tmp = dijkstra_shortest_path_length( graph, station_node, get_weight, target=n ) if path_length_dict_tmp[n] > path_length: node = n path_length = path_length_dict_tmp[n] path_length_dict = path_length_dict_tmp path = paths[node] # find first node in path that exceeds 2/3 of the line length # from station to critical node farthest away from the station node_2_3 = next( j for j in path if path_length_dict[j] >= path_length_dict[node] * 2 / 3 ) # if LVGrid: check if node_2_3 is outside of a house # and if not find next BranchTee outside the house if isinstance(grid, LVGrid): while ( ~np.isnan(grid.buses_df.loc[node_2_3].in_building) and grid.buses_df.loc[node_2_3].in_building ): node_2_3 = path[path.index(node_2_3) - 1] # break if node is station if node_2_3 is path[0]: logger.error("Could not reinforce voltage issue.") break # if MVGrid: check if node_2_3 is LV station and if not find # next LV station else: while node_2_3 not in edisgo_obj.topology.transformers_df.bus0.values: try: # try to find LVStation behind node_2_3 node_2_3 = path[path.index(node_2_3) + 1] except IndexError: # if no LVStation between node_2_3 and node with # voltage problem, connect node directly to # MVStation node_2_3 = node break # if node_2_3 is a representative (meaning it is already # directly connected to the station), line cannot be # disconnected and must therefore be reinforced if node_2_3 in nodes_feeder.keys(): crit_line_name = graph.get_edge_data(station_node, node_2_3)["branch_name"] crit_line = grid.lines_df.loc[crit_line_name] # if critical line is already a standard line install one # more parallel line if crit_line.type_info == standard_line: edisgo_obj.topology.update_number_of_parallel_lines( pd.Series( index=[crit_line_name], data=[ edisgo_obj.topology._lines_df.at[ crit_line_name, "num_parallel" ] + 1 ], ) ) lines_changes[crit_line_name] = 1 # if critical line is not yet a standard line replace old # line by a standard line else: # number of parallel standard lines could be calculated # following [2] p.103; for now number of parallel # standard lines is iterated edisgo_obj.topology.change_line_type([crit_line_name], standard_line) lines_changes[crit_line_name] = 1 # if node_2_3 is not a representative, disconnect line else: # get line between node_2_3 and predecessor node (that is # closer to the station) pred_node = path[path.index(node_2_3) - 1] crit_line_name = graph.get_edge_data(node_2_3, pred_node)["branch_name"] if grid.lines_df.at[crit_line_name, "bus0"] == pred_node: edisgo_obj.topology._lines_df.at[crit_line_name, "bus0"] = station_node elif grid.lines_df.at[crit_line_name, "bus1"] == pred_node: edisgo_obj.topology._lines_df.at[crit_line_name, "bus1"] = station_node else: raise ValueError("Bus not in line buses. Please check.") # change line length and type edisgo_obj.topology._lines_df.at[ crit_line_name, "length" ] = path_length_dict[node_2_3] edisgo_obj.topology.change_line_type([crit_line_name], standard_line) lines_changes[crit_line_name] = 1 # ToDo: Include switch disconnector if not lines_changes: logger.debug( "==> {} line(s) was/were reinforced due to voltage " "issues.".format(len(lines_changes)) ) return lines_changes
[docs]def reinforce_lines_overloading(edisgo_obj, crit_lines): """ Reinforce lines in MV and LV topology due to overloading. Parameters ---------- edisgo_obj : :class:`~.EDisGo` crit_lines : :pandas:`pandas.DataFrame<DataFrame>` Dataframe containing over-loaded lines, their maximum relative over-loading (maximum calculated current over allowed current) and the corresponding time step. Index of the dataframe are the names of the over-loaded lines. Columns are 'max_rel_overload' containing the maximum relative over-loading as float, 'time_index' containing the corresponding time step the over-loading occured in as :pandas:`pandas.Timestamp<Timestamp>`, and 'voltage_level' specifying the voltage level the line is in (either 'mv' or 'lv'). Returns ------- dict Dictionary with name of lines as keys and the corresponding number of lines added as values. Notes ----- Reinforce measures: 1. Install parallel line of the same type as the existing line (Only if line is a cable, not an overhead line. Otherwise a standard equipment cable is installed right away.) 2. Remove old line and install as many parallel standard lines as needed. """ lines_changes = {} # reinforce mv lines lines_changes.update( _reinforce_lines_overloading_per_grid_level(edisgo_obj, "mv", crit_lines) ) # reinforce lv lines lines_changes.update( _reinforce_lines_overloading_per_grid_level(edisgo_obj, "lv", crit_lines) ) if not crit_lines.empty: logger.debug( "==> {} line(s) was/were reinforced due to over-loading " "issues.".format(crit_lines.shape[0]) ) return lines_changes
def _reinforce_lines_overloading_per_grid_level(edisgo_obj, voltage_level, crit_lines): """ Reinforce lines in MV or LV topology due to overloading. Parameters ---------- edisgo_obj : :class:`~.EDisGo` voltage_level : str Voltage level, over-loading is handled for. Possible options are "mv" or "lv". crit_lines : :pandas:`pandas.DataFrame<DataFrame>` Dataframe containing over-loaded lines, their maximum relative over-loading (maximum calculated current over allowed current) and the corresponding time step. Index of the dataframe are the names of the over-loaded lines. Columns are 'max_rel_overload' containing the maximum relative over-loading as float, 'time_index' containing the corresponding time step the over-loading occured in as :pandas:`pandas.Timestamp<Timestamp>`, and 'voltage_level' specifying the voltage level the line is in (either 'mv' or 'lv'). Returns ------- dict Dictionary with name of lines as keys and the corresponding number of lines added as values. """ def _add_parallel_standard_lines(lines): """ Adds as many parallel standard lines as needed so solve overloading. Adds number of added lines to `lines_changes` dictionary. Parameters ---------- lines : list(str) List of line names to add parallel standard lines for. """ # calculate necessary number of parallel lines number_parallel_lines = np.ceil( crit_lines.max_rel_overload[lines] * edisgo_obj.topology.lines_df.loc[lines, "num_parallel"] ) # add number of added lines to lines_changes number_parallel_lines_pre = edisgo_obj.topology.lines_df.loc[ lines, "num_parallel" ] lines_changes.update( (number_parallel_lines - number_parallel_lines_pre).to_dict() ) # update number of parallel lines and line accordingly attributes edisgo_obj.topology.update_number_of_parallel_lines(number_parallel_lines) def _add_one_parallel_line_of_same_type(lines): """ Adds one parallel line of same type. Adds number of added lines to `lines_changes` dictionary. Parameters ---------- lines : list(str) List of line names to add parallel line of same type for. """ # add number of added lines to lines_changes lines_changes.update(pd.Series(index=lines, data=[1] * len(lines)).to_dict()) # update number of lines and accordingly line attributes edisgo_obj.topology.update_number_of_parallel_lines( pd.Series(index=lines, data=[2] * len(lines)) ) def _replace_by_parallel_standard_lines(lines): """ Replaces existing line with as many parallel standard lines as needed. Adds number of added lines to `lines_changes` dictionary. Parameters ---------- lines : list(str) List of line names to replace by parallel standard lines. """ # save old nominal power to calculate number of parallel standard lines s_nom_old = edisgo_obj.topology.lines_df.loc[lines, "s_nom"] # change line type to standard line edisgo_obj.topology.change_line_type(lines, standard_line_type) # calculate and update number of parallel lines number_parallel_lines = np.ceil( s_nom_old * crit_lines.loc[lines, "max_rel_overload"] / edisgo_obj.topology.lines_df.loc[lines, "s_nom"] ) edisgo_obj.topology.update_number_of_parallel_lines(number_parallel_lines) lines_changes.update(number_parallel_lines.to_dict()) lines_changes = {} # chose lines of right grid level relevant_lines = edisgo_obj.topology.lines_df.loc[ crit_lines[crit_lines.voltage_level == voltage_level].index ] if not relevant_lines.empty: nominal_voltage = edisgo_obj.topology.buses_df.loc[ edisgo_obj.topology.lines_df.loc[relevant_lines.index[0], "bus0"], "v_nom" ] if nominal_voltage == 0.4: standard_line_type = edisgo_obj.config["grid_expansion_standard_equipment"][ "lv_line" ] else: standard_line_type = edisgo_obj.config["grid_expansion_standard_equipment"][ f"mv_line_{int(nominal_voltage)}kv" ] # handling of standard lines lines_standard = relevant_lines.loc[ relevant_lines.type_info == standard_line_type ] if not lines_standard.empty: _add_parallel_standard_lines(lines_standard.index) # get lines that have not been updated yet (i.e. that are not standard # lines) relevant_lines = relevant_lines.loc[ ~relevant_lines.index.isin(lines_standard.index) ] # handling of cables where adding one cable is sufficient lines_single = ( relevant_lines.loc[relevant_lines.num_parallel == 1] .loc[relevant_lines.kind == "cable"] .loc[crit_lines.max_rel_overload < 2] ) if not lines_single.empty: _add_one_parallel_line_of_same_type(lines_single.index) # handle rest of lines (replace by as many parallel standard lines as # needed) relevant_lines = relevant_lines.loc[ ~relevant_lines.index.isin(lines_single.index) ] if not relevant_lines.empty: _replace_by_parallel_standard_lines(relevant_lines.index) return lines_changes