Skip to content

Commit

Permalink
Merge branch 'develop' into fix/fdxb_algorithm_id
Browse files Browse the repository at this point in the history
  • Loading branch information
rbolgaryn authored Jan 2, 2024
2 parents 66b3a30 + 401d573 commit e9b0577
Show file tree
Hide file tree
Showing 18 changed files with 378 additions and 72 deletions.
4 changes: 4 additions & 0 deletions CHANGELOG.rst
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@ Change Log

[upcoming release] - 2023-..-..
-------------------------------
- [ADDED] function to search std_types from the basic standard type library
- [ADDED] Documentation for running powerflow using power-grid-model
- [ADDED] exporting to :code:`GeoJSON` with all properties from :code:`bus`, :code:`res_bus` and :code:`line`, :code:`res_line`
- [ADDED] function to run powerflow using the power-grid-model library
Expand Down Expand Up @@ -35,7 +36,10 @@ Change Log
- [CHANGED] parameters of function merge_internal_net_and_equivalent_external_net()
- [FIXED] :code:`convert_format.py`: update the attributes of the characteristic objects to match the new characteristic
- [FIXED] fixed the wrong id numbers for pypower powerflow algorithms fdxb and fdbx
- [FIXED] additional arguments from mpc saved to net._options: create "_options" if it does not exist
- [CHANGED] cim2pp: extracted getting default classes, added generic setting datatypes from CGMES XMI schema
- [ADDED] function :code:`getOTDF` to obtain Outage Transfer Distribution Factors, that can be used to analyse outages using the DC approximation of the power system
- [ADDED] function :code:`outage_results_OTDF` to obtain the matrix of results for all outage scenarios, with rows as outage scenarios and columns as branch power flows in that scenario


[2.13.1] - 2023-05-12
Expand Down
2 changes: 1 addition & 1 deletion doc/plotting/matplotlib/simple_plot.rst
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@
Simple Plotting
=============================

The function simple_plot() can be used for simple plotting. For advanced possibilities see the tutorials
The function simple_plot() can be used for simple plotting. For advanced possibilities see the `tutorial <http://nbviewer.jupyter.org/github/e2nIEE/pandapower/blob/develop/tutorials/plotting_basic.ipynb>`_.

.. _simple_plot:

Expand Down
4 changes: 2 additions & 2 deletions pandapower/auxiliary.py
Original file line number Diff line number Diff line change
Expand Up @@ -318,9 +318,9 @@ def __repr__(self): # pragma: no cover

def plural_s(number):
if number > 1:
return ""
else:
return "s"
else:
return ""

def _preserve_dtypes(df, dtypes):
for item, dtype in list(dtypes.items()):
Expand Down
5 changes: 4 additions & 1 deletion pandapower/control/controller/trafo/ContinuousTapControl.py
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@
# and Energy System Technology (IEE), Kassel. All rights reserved.

import numpy as np

from pandapower.auxiliary import read_from_net, write_to_net
from pandapower.control.controller.trafo_control import TrafoController

Expand Down Expand Up @@ -97,6 +98,8 @@ def is_converged(self, net):
return True

vm_pu = read_from_net(net, "res_bus", self.controlled_bus, "vm_pu", self._read_write_flag)
# this is possible in case the trafo is set out of service by the connectivity check
is_nan = np.isnan(vm_pu)
self.tap_pos = read_from_net(net, self.trafotable, self.controlled_tid, "tap_pos", self._read_write_flag)
difference = 1 - self.vm_set_pu / vm_pu

Expand All @@ -110,4 +113,4 @@ def is_converged(self, net):
else:
converged = np.abs(difference) < self.tol

return np.all(converged)
return np.all(np.logical_or(converged, is_nan))
7 changes: 5 additions & 2 deletions pandapower/control/controller/trafo/DiscreteTapControl.py
Original file line number Diff line number Diff line change
Expand Up @@ -3,9 +3,11 @@
# Copyright (c) 2016-2023 by University of Kassel and Fraunhofer Institute for Energy Economics
# and Energy System Technology (IEE), Kassel. All rights reserved.
import numpy as np

from pandapower.auxiliary import read_from_net, write_to_net
from pandapower.control.controller.trafo_control import TrafoController


class DiscreteTapControl(TrafoController):
"""
Trafo Controller with local tap changer voltage control.
Expand Down Expand Up @@ -115,6 +117,8 @@ def is_converged(self, net):
return True

vm_pu = read_from_net(net, "res_bus", self.controlled_bus, "vm_pu", self._read_write_flag)
# this is possible in case the trafo is set out of service by the connectivity check
is_nan = np.isnan(vm_pu)
self.tap_pos = read_from_net(net, self.trafotable, self.controlled_tid, "tap_pos", self._read_write_flag)

reached_limit = np.where(self.tap_side_coeff * self.tap_sign == 1,
Expand All @@ -125,5 +129,4 @@ def is_converged(self, net):

converged = np.logical_or(reached_limit, np.logical_and(self.vm_lower_pu < vm_pu, vm_pu < self.vm_upper_pu))

return np.all(converged)

return np.all(np.logical_or(converged, is_nan))
2 changes: 2 additions & 0 deletions pandapower/converter/matpower/from_mpc.py
Original file line number Diff line number Diff line change
Expand Up @@ -66,6 +66,8 @@ def from_mpc(mpc_file, f_hz=50, casename_mpc_file='mpc', validate_conversion=Fal
ppc = _m2ppc(mpc_file, casename_mpc_file)
net = from_ppc(ppc, f_hz=f_hz, validate_conversion=validate_conversion, **kwargs)
if "mpc_additional_data" in ppc:
if "_options" not in net:
net["_options"] = dict()
net._options.update(ppc["mpc_additional_data"])
logger.info('added fields %s in net._options' % list(ppc["mpc_additional_data"].keys()))

Expand Down
26 changes: 18 additions & 8 deletions pandapower/plotting/plotly/traces.py
Original file line number Diff line number Diff line change
Expand Up @@ -1094,15 +1094,25 @@ def draw_traces(traces, on_map=False, map_style='basic', showlegend=True, figsiz
for trace in traces:
xs += trace.get('x') or trace['lon']
ys += trace.get('y') or trace['lat']
x_dropna = pd.Series(xs).dropna()
y_dropna = pd.Series(ys).dropna()
xrange = x_dropna.max() - x_dropna.min()
yrange = y_dropna.max() - y_dropna.min()
ratio = xrange / yrange
if ratio < 1:
aspectratio = (ratio, 1.)
xs_arr = np.array(xs)
ys_arr = np.array(ys)
xrange = np.nanmax(xs_arr) - np.nanmin(xs_arr)
yrange = np.nanmax(ys_arr) - np.nanmin(ys_arr)

# the ratio only makes sense, if xrange and yrange != 0
if xrange == 0 and yrange == 0:
aspectratio = (1, 1)
elif xrange == 0:
aspectratio = (0.35, 1)
elif yrange == 0:
aspectratio = (1, 0.35)

else:
aspectratio = (1., 1 / ratio)
ratio = xrange / yrange
if ratio < 1:
aspectratio = (ratio, 1.)
else:
aspectratio = (1., 1 / ratio)

aspectratio = np.array(aspectratio) / max(aspectratio)
fig['layout']['width'], fig['layout']['height'] = ([ar * figsize * 700 for ar in aspectratio])
Expand Down
115 changes: 115 additions & 0 deletions pandapower/pypower/makeLODF.py
Original file line number Diff line number Diff line change
Expand Up @@ -66,3 +66,118 @@ def makeLODF(branch, PTDF):
update_LODF_diag(LODF)

return LODF


def makeOTDF(PTDF, LODF, outage_branches):
"""
Compute the Outage Transfer Distribution Factors (OTDF) matrix.
This function creates the OTDF matrix that relates bus power injections
to branch flows for specified outage scenarios. It's essential that
outage branches do not lead to isolated nodes or disconnected islands
in the grid.
The grid cannot have isolated nodes or disconnected islands. Use the
pandapower.topology module to identify branches that, if outaged, would
lead to isolated nodes (determine_stubs) or islands (find_graph_characteristics).
The resulting matrix has a width equal to the number of nodes and a length
equal to the number of outage branches multiplied by the total number of branches.
The dot product of OTDF and the bus power vector in generation reference frame
(positive for generation, negative for consumption - the opposite of res_bus.p_mw)
yields an array with outage branch power flows for every outage scenario,
facilitating the analysis of all outage scenarios under a DC
power flow approximation.
Parameters
----------
PTDF : numpy.ndarray
The Power Transfer Distribution Factor matrix, defining the sensitivity
of branch flows to bus power injections.
LODF : numpy.ndarray
The Line Outage Distribution Factor matrix, describing how branch flows
are affected by outages of other branches.
outage_branches : list or numpy.ndarray
Indices of branches for which outage scenarios are to be considered.
Returns
-------
OTDF : numpy.ndarray
The Outage Transfer Distribution Factor matrix. Rows correspond to
outage scenarios, and columns correspond to branch flows.
Examples
--------
>>> H = makePTDF(baseMVA, bus, branch)
>>> LODF = makeLODF(branch, H)
>>> outage_branches = [0, 2] # Example branch indices for outage scenarios
>>> OTDF = makeOTDF(H, LODF, outage_branches)
>>> # To obtain a 2D array with the outage results:
>>> outage_results = (OTDF @ Pbus).reshape(len(outage_branches), -1)
Notes
-----
- The function assumes a DC power flow model.
- Ensure that the specified outage branches do not lead to grid
disconnection or isolated nodes.
"""
OTDF = np.vstack([PTDF + LODF[:, [i]] @ PTDF[[i], :] for i in outage_branches])
return OTDF


def outage_results_OTDF(OTDF, Pbus, outage_branches):
"""
Calculate the branch power flows for each outage scenario based on the given
Outage Transfer Distribution Factors (OTDF), bus power injections (Pbus), and
specified outage branches.
This function computes how branch flows are affected under N-1 contingency
scenarios (i.e., for each branch outage specified). It uses the OTDF matrix and
the bus power vector (Pbus) to determine the branch flows in each outage scenario.
Pbus should represent the net power at each bus in the generation reference case.
Parameters
----------
OTDF : numpy.ndarray
The Outage Transfer Distribution Factor matrix, which relates bus power
injections to branch flows under specific outage scenarios. Its shape
should be (num_outage_scenarios * num_branches, num_buses).
Pbus : numpy.ndarray
A vector representing the net power injections at each bus. Positive values
for generation, negative for consumption. Its length should be equal to
the total number of buses.
outage_branches : numpy.ndarray
An array of indices representing the branches that are outaged in each
scenario. Its length should be equal to the number of outage scenarios.
Returns
-------
numpy.ndarray
A 2D array where each row corresponds to an outage scenario and each column
represents the resulting power flow in a branch. The number of rows is equal
to the number of outage scenarios, and the number of columns is equal to the
number of branches.
Examples
--------
>>> OTDF = np.array([...]) # example OTDF matrix
>>> Pbus = np.array([...]) # example bus power vector
>>> outage_branches = np.array([...]) # example outage branches
>>> branch_flows = outage_results_OTDF(OTDF,Pbus,outage_branches)
Notes
-----
The function assumes a linear relationship between bus power injections and
branch flows, which is typical in DC power flow models.
"""
# get branch flows as an array first:
nminus1_otdf = (OTDF @ Pbus.reshape(-1, 1))
# reshape to a 2D array with rows relating to outage scenarios and columns to
# the resulting branch power flows
nminus1_otdf = nminus1_otdf.reshape(outage_branches.shape[0], -1)
return nminus1_otdf




Loading

0 comments on commit e9b0577

Please sign in to comment.