Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Custom connection/disconnection of terminals #2754

Merged
merged 35 commits into from
Nov 9, 2023
Merged
Show file tree
Hide file tree
Changes from 14 commits
Commits
Show all changes
35 commits
Select commit Hold shift + click to select a range
ca699e0
new methods disconnect with Predicate<Switch> isSwitchOpenable parameter
rolnico Oct 23, 2023
3f37b2e
add condition
rolnico Oct 23, 2023
89bda0f
add static predicates
rolnico Oct 24, 2023
9cdfc52
changed default predicate in order not to get a breaking change
rolnico Oct 24, 2023
c0de505
add predicate to exclude load break switches
rolnico Oct 24, 2023
97ff059
modify stopping condition in path detection
rolnico Oct 25, 2023
5adc462
new class SwitchPredicates
rolnico Oct 27, 2023
4f62dd2
allow path filtering before sizing
rolnico Oct 27, 2023
84d377f
default predicate correction
rolnico Oct 30, 2023
fa87677
stop path on open switches during disconnect
rolnico Oct 31, 2023
7d16299
sort paths by size according to the given predicate then by actual size
rolnico Oct 31, 2023
d01adbb
update comment
rolnico Oct 31, 2023
b4ceb33
stop added when terminal is disconnected
rolnico Oct 31, 2023
d529823
add check on connect + comments
rolnico Oct 31, 2023
29d41f3
changed SwitchPredicates type to Predicate<Switch> + moved to iidm.ne…
rolnico Oct 31, 2023
7ca9573
create list of switch to open then open if all paths
rolnico Nov 2, 2023
62e639c
add isTypeSwitchToOperate parameter to connect method + coverage tests
rolnico Nov 2, 2023
4127413
add connect method with parameter in interface
rolnico Nov 2, 2023
cb65c8d
add test for coverage + typo
rolnico Nov 2, 2023
c56e25e
Merge branch 'main' into nro/custom_disconnection_of_connectables
rolnico Nov 2, 2023
e69dde7
add disconnection test cases + author mentions
rolnico Nov 3, 2023
c686e6a
author mentions
rolnico Nov 3, 2023
b1be0ff
add setup method in AbstractSwitchPredicatesTest + swap ComparisonCha…
rolnico Nov 6, 2023
dd4cd15
comment correction
rolnico Nov 6, 2023
3b3276a
comment correction
rolnico Nov 6, 2023
e6f401e
Added comment for checkNonClosableSwitches + renamed this method + si…
rolnico Nov 6, 2023
af91e07
typo in comment
rolnico Nov 7, 2023
560a480
Merge branch 'main' into nro/custom_disconnection_of_connectables
olperr1 Nov 7, 2023
63c1cf2
authors reinstated + blank lines deleted + changed parameter from fil…
rolnico Nov 8, 2023
9a05515
Merge remote-tracking branch 'origin/nro/custom_disconnection_of_conn…
rolnico Nov 8, 2023
9b96fb6
unused import
rolnico Nov 8, 2023
8e9d4af
unused import
rolnico Nov 8, 2023
e02a0b8
changed Function to Predicate to solve code smell
rolnico Nov 9, 2023
9bc9270
typos and improvements thanks to reviewer
rolnico Nov 9, 2023
bd651d3
Merge branch 'main' into nro/custom_disconnection_of_connectables
flo-dup Nov 9, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -159,6 +159,14 @@ public static interface BusView {
*/
boolean connect();

/**
* Try to connect the terminal.
* <p>Depends on the working variant.
olperr1 marked this conversation as resolved.
Show resolved Hide resolved
* @return true if terminal has been connected, false otherwise
* @see VariantManager
*/
boolean connect(Predicate<Switch> isTypeSwitchToOperate);

/**
* Disconnect the terminal.
* <p>Depends on the working variant.
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,31 @@
/*
* Copyright (c) 2023. , RTE (http://www.rte-france.com)
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
* SPDX-License-Identifier: MPL-2.0
*/

flo-dup marked this conversation as resolved.
Show resolved Hide resolved
package com.powsybl.iidm.network.util;

import com.powsybl.iidm.network.Switch;
import com.powsybl.iidm.network.SwitchKind;

import java.util.Objects;
import java.util.function.Predicate;

/**
* @author Nicolas Rol {@literal <nicolas.rol at rte-france.com>}
*/
public final class SwitchPredicates {

private SwitchPredicates() { }

public static final Predicate<Switch> IS_NONFICTIONAL_CLOSED_BREAKER = switchObject -> switchObject != null && switchObject.getKind() == SwitchKind.BREAKER && !switchObject.isOpen() && !switchObject.isFictitious();
public static final Predicate<Switch> IS_NONFICTIONAL_BREAKER = switchObject -> switchObject != null && switchObject.getKind() == SwitchKind.BREAKER && !switchObject.isFictitious();
public static final Predicate<Switch> IS_CLOSED_BREAKER = switchObject -> switchObject != null && switchObject.getKind() == SwitchKind.BREAKER && !switchObject.isOpen();
public static final Predicate<Switch> IS_BREAKER_OR_DISCONNECTOR = switchObject -> switchObject != null && (switchObject.getKind() == SwitchKind.BREAKER || switchObject.getKind() == SwitchKind.DISCONNECTOR);
public static final Predicate<Switch> IS_OPEN_DISCONNECTOR = switchObject -> switchObject != null && switchObject.getKind() == SwitchKind.DISCONNECTOR && switchObject.isOpen();
public static final Predicate<Switch> IS_OPEN = switchObject -> switchObject != null && switchObject.isOpen();
public static final Predicate<Switch> IS_NON_NULL = Objects::nonNull;
}
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@
import com.powsybl.commons.PowsyblException;
import com.powsybl.iidm.network.*;
import com.powsybl.iidm.network.impl.util.Ref;
import com.powsybl.iidm.network.util.SwitchPredicates;
import gnu.trove.list.array.TDoubleArrayList;

import java.util.ArrayList;
Expand Down Expand Up @@ -157,18 +158,28 @@ public double getI() {

@Override
public boolean connect() {
return connect(SwitchPredicates.IS_NONFICTIONAL_BREAKER);
}


flo-dup marked this conversation as resolved.
Show resolved Hide resolved
/**
* Try to connect the terminal.
* <p>Depends on the working variant.
olperr1 marked this conversation as resolved.
Show resolved Hide resolved
* @param isTypeSwitchToOperate Predicate telling if a switch is operable according to conditions. Examples of predicates are available in the class {@link SwitchPredicates}
flo-dup marked this conversation as resolved.
Show resolved Hide resolved
* @return true if terminal has been connected, false otherwise
* @see VariantManager
*/
@Override
public boolean connect(Predicate<Switch> isTypeSwitchToOperate) {
if (removed) {
throw new PowsyblException(UNMODIFIABLE_REMOVED_EQUIPMENT + connectable.id);
}
return voltageLevel.connect(this);
return voltageLevel.connect(this, isTypeSwitchToOperate);
}

@Override
public boolean disconnect() {
if (removed) {
throw new PowsyblException(UNMODIFIABLE_REMOVED_EQUIPMENT + connectable.id);
}
return voltageLevel.disconnect(this);
return disconnect(SwitchPredicates.IS_CLOSED_BREAKER);
}


flo-dup marked this conversation as resolved.
Show resolved Hide resolved
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -927,6 +927,11 @@ public boolean connect(TerminalExt terminal) {
return true;
}

@Override
public boolean connect(TerminalExt terminal, Predicate<? super SwitchImpl> isTypeSwitchToOperate) {
return connect(terminal);
}

@Override
public boolean disconnect(TerminalExt terminal) {
if (!(terminal instanceof BusTerminal)) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@
import com.powsybl.iidm.network.impl.util.Ref;
import com.powsybl.iidm.network.util.Identifiables;
import com.powsybl.iidm.network.util.ShortIdDictionary;
import com.powsybl.iidm.network.util.SwitchPredicates;
import com.powsybl.math.graph.*;
import gnu.trove.TCollections;
import gnu.trove.list.array.TDoubleArrayList;
Expand Down Expand Up @@ -1165,16 +1166,31 @@ private static boolean isBusbarSection(Terminal t) {
return t != null && t.getConnectable().getType() == IdentifiableType.BUSBAR_SECTION;
}

private static boolean isOpenedDisconnector(Switch s) {
return s != null && s.getKind() == SwitchKind.DISCONNECTOR && s.isOpen();
private boolean checkNonClosableSwitches(SwitchImpl sw, Predicate<? super SwitchImpl> isTypeSwitchToOperate) {
olperr1 marked this conversation as resolved.
Show resolved Hide resolved
return SwitchPredicates.IS_OPEN.test(sw) && isTypeSwitchToOperate.negate().test(sw);
}

private static boolean isOpenedDisconnectorOpenLoadBreakSwitchOrOpenedFictitiousBreaker(Switch s) {
return s != null && s.isOpen() && (s.getKind() != SwitchKind.BREAKER || s.isFictitious());
}

/**
* Connect the terminal to a busbar section in its voltage level.
* The switches that can be operated are the non-fictional breakers.
* @param terminal Terminal to connect
* @return true if the terminal has been connected, false if it hasn't or if it was already connected
*/
@Override
public boolean connect(TerminalExt terminal) {
// Only keep the closed non-fictional breakers in the nominal case
return connect(terminal, SwitchPredicates.IS_NONFICTIONAL_BREAKER);
}

/**
* Connect the terminal to a busbar section in its voltage level.
* @param terminal Terminal to connect
* @param isTypeSwitchToOperate Predicate used to identify the switches that can be operated on. <b>Warning:</b> do not include a test to see if the switch is opened, it is already done in this method
flo-dup marked this conversation as resolved.
Show resolved Hide resolved
* @return true if the terminal has been connected, false if it hasn't or if it was already connected
*/
@Override
public boolean connect(TerminalExt terminal, Predicate<? super SwitchImpl> isTypeSwitchToOperate) {
if (!(terminal instanceof NodeTerminal)) {
throw new IllegalStateException(WRONG_TERMINAL_TYPE_EXCEPTION_MESSAGE + terminal.getClass().getName());
}
Expand All @@ -1184,21 +1200,28 @@ public boolean connect(TerminalExt terminal) {
}

int node = ((NodeTerminal) terminal).getNode();
// find all paths starting from the current terminal to a busbar section that does not contain an open disconnector
// paths are already sorted
List<TIntArrayList> paths = graph.findAllPaths(node, NodeBreakerVoltageLevel::isBusbarSection, NodeBreakerVoltageLevel::isOpenedDisconnectorOpenLoadBreakSwitchOrOpenedFictitiousBreaker, SwitchPredicates.IS_OPEN);
// find all paths starting from the current terminal to a busbar section that does not contain an open switch
// that is not of the type of switch the user wants to operate
// Paths are already sorted by the number of open switches and by the size of the paths
List<TIntArrayList> paths = graph.findAllPaths(node, NodeBreakerVoltageLevel::isBusbarSection, sw -> checkNonClosableSwitches(sw, isTypeSwitchToOperate), SwitchPredicates.IS_OPEN);
boolean connected = false;
if (!paths.isEmpty()) {
// the shorted path is the best, close all opened breakers of the path
// the shortest path is the best
TIntArrayList shortestPath = paths.get(0);

// close all open switches on the path
for (int i = 0; i < shortestPath.size(); i++) {
int e = shortestPath.get(i);
SwitchImpl sw = graph.getEdgeObject(e);
if (sw != null && sw.getKind() == SwitchKind.BREAKER && sw.isOpen()) {
if (SwitchPredicates.IS_OPEN.test(sw)) {
// Since the paths were constructed using the method checkNonClosableSwitches, only operable switches can be open
sw.setOpen(false);
connected = true;
}
}
// Check that the terminal is connected (it should always be true, given how the paths are found)
if (terminal.isConnected()) {
connected = true;
}
olperr1 marked this conversation as resolved.
Show resolved Hide resolved
}
return connected;
}
Expand All @@ -1222,32 +1245,52 @@ public boolean disconnect(TerminalExt terminal, Predicate<? super SwitchImpl> is
int node = ((NodeTerminal) terminal).getNode();
// find all paths starting from the current terminal to a busbar section that does not contain an open disconnector
// (because otherwise there is nothing we can do to connected the terminal using only breakers)
olperr1 marked this conversation as resolved.
Show resolved Hide resolved
List<TIntArrayList> paths = graph.findAllPaths(node, NodeBreakerVoltageLevel::isBusbarSection, NodeBreakerVoltageLevel::isOpenedDisconnector);
List<TIntArrayList> paths = graph.findAllPaths(node, NodeBreakerVoltageLevel::isBusbarSection, SwitchPredicates.IS_OPEN::test);
if (paths.isEmpty()) {
return false;
}

// Each path is visited and for each, the first openable switch found is opened
// Set of switch that are opened
Set<SwitchImpl> openedSwitches = new HashSet<>(paths.size());
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
// Set of switch that are opened
Set<SwitchImpl> openedSwitches = new HashSet<>(paths.size());
// Set of switches that have to be opened
Set<SwitchImpl> switchesToOpen = new HashSet<>(paths.size());


// Boolean becomes false if a path cannot be opened
boolean pathsOpen = true;

// Each path is visited and for each, the first openable switch found is added in the set of switches to open
for (TIntArrayList path : paths) {
boolean pathOpen = false;
for (int i = 0; i < path.size(); i++) {
int e = path.get(i);
SwitchImpl sw = graph.getEdgeObject(e);
if (isSwitchOpenable.test(sw)) {
if (!sw.isOpen()) {
sw.setOpen(true);
}
// just one open breaker is enough to disconnect the terminal, so we can stop
pathOpen = true;
break;
}
}
if (!pathOpen) {
// Getting here meant no openable switch was found and therefor the terminal cannot be disconnected
return false;
// Open the first openable switch
olperr1 marked this conversation as resolved.
Show resolved Hide resolved
pathsOpen = openPath(path, isSwitchOpenable, openedSwitches) && pathsOpen;
olperr1 marked this conversation as resolved.
Show resolved Hide resolved
}

// The switches are opened if and only if every path can be opened
if (pathsOpen) {
openedSwitches.forEach(sw -> sw.setOpen(true));
return true;
}
return false;
}

/**
* Add the first openable switch in the given path to the set of switch to open
* @param path the path to open
* @param isSwitchOpenable predicate used to know if the switches can be opened
* @param openedSwitches set of switch to be opened
flo-dup marked this conversation as resolved.
Show resolved Hide resolved
* @return true if the path has been opened, else false
*/
boolean openPath(TIntArrayList path, Predicate<? super SwitchImpl> isSwitchOpenable, Set<SwitchImpl> openedSwitches) {
for (int i = 0; i < path.size(); i++) {
int e = path.get(i);
SwitchImpl sw = graph.getEdgeObject(e);
if (openedSwitches.contains(sw)) {
// A switch in the path has already been opened
return true;
flo-dup marked this conversation as resolved.
Show resolved Hide resolved
} else if (isSwitchOpenable.test(sw)) {
openedSwitches.add(sw);
// just one open breaker is enough to disconnect the terminal, so we can stop
return true;
}
}
return true;
return false;
}

boolean isConnected(TerminalExt terminal) {
Expand Down

This file was deleted.

Original file line number Diff line number Diff line change
Expand Up @@ -56,6 +56,8 @@ interface BusViewExt extends BusView {

boolean connect(TerminalExt terminal);

boolean connect(TerminalExt terminal, Predicate<? super SwitchImpl> isTypeSwitchToOperate);

boolean disconnect(TerminalExt terminal);

boolean disconnect(TerminalExt terminal, Predicate<? super SwitchImpl> isSwitchOpenable);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -149,6 +149,21 @@ void testNodeBreakerConnectConnectedLoad() {
assertTrue(l.getTerminal().isConnected());
}

@Test
void testNodeBreakerConnectViaVoltageLevelConnectedLoad() {
Network network = createNetwork();
Load l = network.getLoad("LD");
assertTrue(l.getTerminal().isConnected());
assertTrue(network.getSwitch("B2").isOpen());

if (l.getTerminal() instanceof TerminalExt terminal) {
NodeBreakerVoltageLevel voltageLevel = (NodeBreakerVoltageLevel) network.getVoltageLevel("VL");
voltageLevel.connect(terminal);
}
assertTrue(network.getSwitch("B2").isOpen());
assertTrue(l.getTerminal().isConnected());
}

@Test
void testNodeBreakerDisconnectDisconnectedLoad() {
Network network = createNetwork();
Expand All @@ -166,7 +181,10 @@ void testNodeBreakerDisconnectionDiamond() {
Network network = createDiamondNetwork();
Load l = network.getLoad("L");
assertTrue(l.getTerminal().isConnected());
l.getTerminal().disconnect();
if (l.getTerminal() instanceof TerminalExt terminal) {
NodeBreakerVoltageLevel voltageLevel = (NodeBreakerVoltageLevel) network.getVoltageLevel("VL");
voltageLevel.disconnect(terminal);
}
assertFalse(l.getTerminal().isConnected());
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
/*
* Copyright (c) 2023. , RTE (http://www.rte-france.com)
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
* SPDX-License-Identifier: MPL-2.0
*/

flo-dup marked this conversation as resolved.
Show resolved Hide resolved
package com.powsybl.iidm.network.impl.tck;

import com.powsybl.iidm.network.tck.AbstractNodeBreakerDisconnectionDiamondPathBugTest;

/**
* @author Nicolas Rol {@literal <nicolas.rol at rte-france.com>}
*/
public class NodeBreakerDisconnectionDiamondPathBugTest extends AbstractNodeBreakerDisconnectionDiamondPathBugTest { }
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
/*
* Copyright (c) 2023. , RTE (http://www.rte-france.com)
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
* SPDX-License-Identifier: MPL-2.0
*/

flo-dup marked this conversation as resolved.
Show resolved Hide resolved
package com.powsybl.iidm.network.impl.tck.util;

import com.powsybl.iidm.network.tck.util.AbstractSwitchPredicatesTest;

class SwitchPredicatesTest extends AbstractSwitchPredicatesTest { }
Loading
Loading