From 3c36c5ab42523b89911b9b71d3e26be257ce7550 Mon Sep 17 00:00:00 2001 From: Ali Hamdan Date: Fri, 29 Nov 2024 12:06:05 +0100 Subject: [PATCH] Update transformer catalogue and parameters --- pyproject.toml | 3 + .../models/tests/test_buses.py | 11 +- .../models/tests/test_flexible_parameters.py | 31 +- .../models/tests/test_line_parameters.py | 4 +- .../tests/test_transformer_parameters.py | 741 ------------------ .../models/tests/test_transformers.py | 8 +- .../models/transformers/transformers.py | 6 +- roseau/load_flow_single/network.py | 2 +- .../scripts/generate_test_networks.py | 2 +- .../data/networks/all_element_network.json | 6 +- .../tests/test_electrical_network.py | 4 +- uv.lock | 6 +- 12 files changed, 42 insertions(+), 782 deletions(-) delete mode 100644 roseau/load_flow_single/models/tests/test_transformer_parameters.py diff --git a/pyproject.toml b/pyproject.toml index 7765514..af5ea90 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -64,6 +64,9 @@ dev = [ [tool.uv] managed = true +[tool.uv.sources] +roseau-load-flow = { git = "https://github.com/RoseauTechnologies/Roseau_Load_Flow.git", branch = "tr-catalogue" } + [build-system] requires = ["hatchling"] build-backend = "hatchling.build" diff --git a/roseau/load_flow_single/models/tests/test_buses.py b/roseau/load_flow_single/models/tests/test_buses.py index 606ea1a..608357d 100644 --- a/roseau/load_flow_single/models/tests/test_buses.py +++ b/roseau/load_flow_single/models/tests/test_buses.py @@ -40,9 +40,9 @@ def test_voltage_limits(): assert bus.max_voltage == Q_(434.6, "V") # Can be reset to None - bus.nominal_voltage = None bus.min_voltage_level = None bus.max_voltage_level = None + bus.nominal_voltage = None assert bus.min_voltage_level is None assert bus.max_voltage_level is None assert bus.min_voltage is None @@ -60,9 +60,9 @@ def test_voltage_limits(): # NaNs are converted to None for na in (np.nan, float("nan"), pd.NA): - bus.nominal_voltage = na bus.min_voltage_level = na bus.max_voltage_level = na + bus.nominal_voltage = na assert bus.nominal_voltage is None assert bus.min_voltage_level is None assert bus.max_voltage_level is None @@ -70,9 +70,9 @@ def test_voltage_limits(): assert bus.max_voltage is None # Min/Max voltage values defined without nominal voltage are useless - bus.nominal_voltage = None bus.min_voltage_level = None bus.max_voltage_level = None + bus.nominal_voltage = None with pytest.warns( UserWarning, match=r"The min voltage level of the bus 'bus' is useless without a nominal voltage. Please define a nominal " @@ -105,6 +105,7 @@ def test_voltage_limits(): bus.nominal_voltage = None # Bad values + bus.nominal_voltage = 400 bus.min_voltage_level = 0.95 with pytest.raises(RoseauLoadFlowException) as e: bus.max_voltage_level = 0.92 @@ -186,7 +187,7 @@ def test_propagate_limits(): # noqa: C901 lp_mv = LineParameters(id="lp_mv", z_line=1.0, y_shunt=0.1) lp_lv = LineParameters(id="lp_lv", z_line=1.0) - tp = TransformerParameters.from_catalogue(name="SE_Minera_A0Ak_100kVA", manufacturer="SE") + tp = TransformerParameters.from_catalogue(name="SE Minera A0Ak 100kVA 15/20kV(20) 410V Dyn11") Line(id="l1_mv", bus1=b1_mv, bus2=b2_mv, length=1.5, parameters=lp_mv) Line(id="l2_mv", bus1=b2_mv, bus2=b3_mv, length=2, parameters=lp_mv) @@ -342,7 +343,7 @@ def test_get_connected_buses(): lp_mv = LineParameters(id="lp_mv", z_line=1.0, y_shunt=0.1) lp_lv = LineParameters(id="lp_lv", z_line=1.0) - tp = TransformerParameters.from_catalogue(name="SE_Minera_A0Ak_100kVA", manufacturer="SE") + tp = TransformerParameters.from_catalogue(name="SE Minera A0Ak 100kVA 15/20kV(20) 410V Dyn11") Line(id="l1_mv", bus1=b1_mv, bus2=b2_mv, length=1.5, parameters=lp_mv) Line(id="l2_mv", bus1=b2_mv, bus2=b3_mv, length=2, parameters=lp_mv) diff --git a/roseau/load_flow_single/models/tests/test_flexible_parameters.py b/roseau/load_flow_single/models/tests/test_flexible_parameters.py index 90011b5..9afc899 100644 --- a/roseau/load_flow_single/models/tests/test_flexible_parameters.py +++ b/roseau/load_flow_single/models/tests/test_flexible_parameters.py @@ -1,5 +1,3 @@ -import warnings - import numpy as np import pytest @@ -56,14 +54,14 @@ def test_control(): assert e.value.code == RoseauLoadFlowExceptionCode.BAD_CONTROL_VALUE # Warning if values provided to useless values - with warnings.catch_warnings(record=True) as records: + with pytest.warns( + UserWarning, + match=( + r"Some voltage norm value\(s\) will not be used by the 'p_max_u_production' control. " + r"Nevertheless, values different from 0 were given: 'u_min' \(2.0 V\), 'u_down' \(1.0 V\)" + ), + ): Control(type="p_max_u_production", u_min=2, u_down=1, u_up=240, u_max=250) - assert len(records) == 1 - assert ( - records[0].message.args[0] == "Some voltage norm value(s) will not be used by the 'p_max_u_production' " - "control. Nevertheless, values different from 0 were given: 'u_min' (2.0 V), 'u_down' (1.0 V)" - ) - assert records[0].category is UserWarning # Bad control value for p_max_u_consumption with pytest.raises(RoseauLoadFlowException) as e: @@ -102,15 +100,14 @@ def test_control(): assert e.value.code == RoseauLoadFlowExceptionCode.BAD_CONTROL_VALUE # Warning if values provided to useless values - with warnings.catch_warnings(record=True) as records: + with pytest.warns( + UserWarning, + match=( + r"Some voltage norm value\(s\) will not be used by the 'p_max_u_consumption' control. " + r"Nevertheless, values different from 0 were given: 'u_max' \(-1.0 V\), 'u_up' \(2.3 V\)" + ), + ): Control(type="p_max_u_consumption", u_min=210, u_down=220, u_up=2.3, u_max=-1) - assert len(records) == 1 - assert ( - records[0].message.args[0] == "Some voltage norm value(s) will not be used by the 'p_max_u_consumption' " - "control. Nevertheless, values different from 0 were given: 'u_max' (-1.0 " - "V), 'u_up' (2.3 V)" - ) - assert records[0].category is UserWarning # Bad control value for q_u with pytest.raises(RoseauLoadFlowException) as e: diff --git a/roseau/load_flow_single/models/tests/test_line_parameters.py b/roseau/load_flow_single/models/tests/test_line_parameters.py index 2766b30..ef8d621 100644 --- a/roseau/load_flow_single/models/tests/test_line_parameters.py +++ b/roseau/load_flow_single/models/tests/test_line_parameters.py @@ -940,8 +940,8 @@ def test_equality(): other_data = { "id": lp.id + " other", - "z_line": lp.z_line.m + 1, - "y_shunt": lp.y_shunt.m + 1, + "z_line": lp.z_line.m + 1j, + "y_shunt": lp.y_shunt.m + 1j, "ampacities": lp.ampacities.m + 1, "line_type": LineType.OVERHEAD, "materials": Material.CU, diff --git a/roseau/load_flow_single/models/tests/test_transformer_parameters.py b/roseau/load_flow_single/models/tests/test_transformer_parameters.py deleted file mode 100644 index 1eeba3c..0000000 --- a/roseau/load_flow_single/models/tests/test_transformer_parameters.py +++ /dev/null @@ -1,741 +0,0 @@ -import numbers - -import numpy as np -import pandas as pd -import pytest -from pint import DimensionalityError - -from roseau.load_flow.exceptions import RoseauLoadFlowException, RoseauLoadFlowExceptionCode -from roseau.load_flow.units import Q_ -from roseau.load_flow_single.models import TransformerParameters - - -def test_transformer_parameters(): - # Example in the "transformers" document of Victor. - # Yzn11 - 50kVA - data = { - "id": "Yzn11 - 50kVA", - "psc": 1350.0, # W - "p0": 145.0, # W - "i0": 1.8 / 100, # % - "us": 400, # V - "up": 20000, # V - "sn": 50 * 1e3, # VA - "vsc": 4 / 100, # % - "type": "yzn11", - } - tp = TransformerParameters.from_dict(data) - - r_iron = 20e3**2 / 145 # Ohm - lm_omega = 20e3**2 / (np.sqrt((1.8 / 100 * 50e3) ** 2 - 145**2)) # H *rad/s - z2_norm = 4 / 100 * 400**2 / 50e3 - r2 = 1350 * 400**2 / 50e3**2 # Ohm - l2_omega = np.sqrt(z2_norm**2 - r2**2) # H*rad/s - - ym_expected = 1 / r_iron + 1 / (1j * lm_omega) - z2_expected = r2 + 1j * l2_omega - k_expected = 400 / (np.sqrt(3.0) * 20e3) - orientation_expected = 1.0 - - assert np.isclose(tp.z2.m, z2_expected) - assert np.isclose(tp.ym.m, ym_expected) - assert np.isclose(tp.k.m, k_expected) - assert np.isclose(tp.orientation, orientation_expected) - - # Dyn11 - 100kVA - data = { - "id": "Dyn11 - 100kVA", - "psc": 2150.0, # W - "p0": 210.0, # W - "i0": 3.5 / 100, # % - "us": 400, # V - "up": 20000, # V - "sn": 100 * 1e3, # VA - "vsc": 4 / 100, # % - "type": "dyn11", - } - tp = TransformerParameters.from_dict(data) - r_iron = 3 * 20e3**2 / 210 # Ohm - lm_omega = 3 * 20e3**2 / (np.sqrt((3.5 / 100 * 100e3) ** 2 - 210**2)) # H*rad/s - z2_norm = 4 / 100 * 400**2 / 100e3 - r2 = 2150 * 400**2 / 100e3**2 # Ohm - l2_omega = np.sqrt(z2_norm**2 - r2**2) # H*rad/s - - ym_expected = 1 / r_iron + 1 / (1j * lm_omega) - z2_expected = r2 + 1j * l2_omega - k_expected = (400 / np.sqrt(3)) / 20e3 - orientation_expected = 1.0 - - assert np.isclose(tp.z2.m, z2_expected) - assert np.isclose(tp.ym.m, ym_expected) - assert np.isclose(tp.k.m, k_expected) - assert np.isclose(tp.orientation, orientation_expected) - - # Dyn5 - 160kVA - data = { - "id": "Dyn5 - 160kVA", - "psc": 2350.0, # W - "p0": 460.0, # W - "i0": 5.6 / 100, # % - "us": 400, # V - "up": 20000, # V - "sn": 160 * 1e3, # VA - "vsc": 4 / 100, # % - "type": "dyn5", - } - tp = TransformerParameters.from_dict(data) - r_iron = 3 * 20e3**2 / 460 # Ohm - lm_omega = 3 * 20e3**2 / (np.sqrt((5.6 / 100 * 160e3) ** 2 - 460**2)) # H*rad/s - z2_norm = 4 / 100 * 400**2 / 160e3 - r2 = 2350 * 400**2 / 160e3**2 # Ohm - l2_omega = np.sqrt(z2_norm**2 - r2**2) # H*rad/s - - ym_expected = 1 / r_iron + 1 / (1j * lm_omega) - z2_expected = r2 + 1j * l2_omega - k_expected = 400 / np.sqrt(3) / 20e3 - orientation_expected = -1.0 - - assert np.isclose(tp.z2.m, z2_expected) - assert np.isclose(tp.ym.m, ym_expected) - assert np.isclose(tp.k.m, k_expected) - assert np.isclose(tp.orientation, orientation_expected) - - # Check that there is an error if the winding is not good - data = { - "id": "test", - "psc": 2350.0, # W - "p0": 460.0, # W - "i0": 5.6 / 100, # % - "us": 400, # V - "up": 20000, # V - "sn": 160 * 1e3, # VA - "vsc": 4 / 100, # % - "type": "dtotoyn11", - } - with pytest.raises(RoseauLoadFlowException) as e: - TransformerParameters.from_dict(data) - assert "Transformer windings cannot be extracted from the string" in e.value.msg - assert e.value.code == RoseauLoadFlowExceptionCode.BAD_TRANSFORMER_WINDINGS - - # Bad i0 - data = { - "id": "test", - "psc": 2350.0, # W - "p0": 460.0, # W - "i0": 5.6, # % - "us": 400, # V - "up": 20000, # V - "sn": 160 * 1e3, # VA - "vsc": 4 / 100, # % - "type": "dyn11", - } - with pytest.raises(RoseauLoadFlowException) as e: - TransformerParameters.from_dict(data) - assert "Invalid open-circuit test current i0=" in e.value.msg - assert e.value.code == RoseauLoadFlowExceptionCode.BAD_TRANSFORMER_PARAMETERS - - # Bad vsc - data = { - "id": "test", - "psc": 2350.0, # W - "p0": 460.0, # W - "i0": 5.6 / 100, # % - "us": 400, # V - "up": 20000, # V - "sn": 160 * 1e3, # VA - "vsc": 4, # % - "type": "dyn11", - } - with pytest.raises(RoseauLoadFlowException) as e: - TransformerParameters.from_dict(data) - assert "Invalid short-circuit test voltage vsc=" in e.value.msg - assert e.value.code == RoseauLoadFlowExceptionCode.BAD_TRANSFORMER_PARAMETERS - - # Bad l2_omega - data = { - "id": "test", - "type": "Dyn11", - "sn": 50000.0, - "up": 20000.0, - "us": 400.0, - "i0": 0.027, - "p0": 210.0, - "psc": 2150.0, - "vsc": 0.04, - } - with pytest.raises(RoseauLoadFlowException) as e: - TransformerParameters.from_dict(data) - assert "The following inequality must be respected: psc/sn <= vsc" in e.value.msg - assert e.value.code == RoseauLoadFlowExceptionCode.BAD_TRANSFORMER_PARAMETERS - - -def test_transformers_parameters_units(): - # Example in the "transformers" document of Victor. - # Yzn11 - 50kVA. Good units - data = { - "id": "Yzn11 - 50kVA", - "z2": Q_(8.64 + 9.444j, "centiohm"), # Ohm - "ym": Q_(0.3625 - 2.2206j, "uS"), # S - "us": Q_(400, "V"), # V - "up": Q_(20, "kV"), # V - "sn": Q_(50, "kVA"), # VA - "type": "yzn11", - } - tp = TransformerParameters(**data) - assert np.isclose(tp._z2, (0.0864 + 0.0944406692j)) - assert np.isclose(tp._ym, (3.625e-07 - 2.2206e-06j)) - assert np.isclose(tp._us, 400) - assert np.isclose(tp._up, 20000) - assert np.isclose(tp._sn, 50e3) - - # Bad unit for each of them - for param, fake_quantity in ( - ("z2", Q_(1350.0, "A")), - ("ym", Q_(145.0, "A")), - ("us", Q_(400, "A")), - ("up", Q_(20, "A")), - ("sn", Q_(50, "A")), - ): - copy_data = data.copy() - copy_data[param] = fake_quantity - with pytest.raises( - DimensionalityError, match=r"Cannot convert from 'ampere' \(\[current\]\) to '\w+?' \(.+?\)" - ): - TransformerParameters(**copy_data) - - -def test_transformers_parameters_units_from_tests(): - # Example in the "transformers" document of Victor. - # Yzn11 - 50kVA. Good units - data = { - "id": "Yzn11 - 50kVA", - "psc": Q_(1350.0, "W"), # W - "p0": Q_(145.0, "W"), # W - "i0": Q_(1.8, "percent"), # % - "us": Q_(400, "V"), # V - "up": Q_(20, "kV"), # V - "sn": Q_(50, "kVA"), # VA - "vsc": Q_(4, "percent"), # % - "type": "yzn11", - } - tp = TransformerParameters.from_open_and_short_circuit_tests(**data) - assert np.isclose(tp._psc, 1350.0) - assert np.isclose(tp._p0, 145.0) - assert np.isclose(tp._i0, 1.8e-2) - assert np.isclose(tp._us, 400) - assert np.isclose(tp._up, 20000) - assert np.isclose(tp._sn, 50e3) - assert np.isclose(tp._vsc, 4e-2) - - # Bad unit for each of them - for param, fake_quantity in ( - ("psc", Q_(1350.0, "A")), - ("p0", Q_(145.0, "A")), - ("i0", Q_(1.8 / 100, "A")), - ("us", Q_(400, "A")), - ("up", Q_(20, "A")), - ("sn", Q_(50, "A")), - ("vsc", Q_(4 / 100, "A")), - ): - copy_data = data.copy() - copy_data[param] = fake_quantity - with pytest.raises( - DimensionalityError, match=r"Cannot convert from 'ampere' \(\[current\]\) to '\w+?' \(.+?\)" - ): - TransformerParameters.from_open_and_short_circuit_tests(**copy_data) - - -def test_transformer_type(): - valid_windings = ("y", "yn", "z", "zn", "d") - valid_phase_displacements = (0, 5, 6, 11) - valid_types = {"dd", "yy", "yny", "yyn", "ynyn", "dz", "dzn", "dy", "dyn", "yd", "ynd", "yz", "ynz", "yzn", "ynzn"} - valid_full_types = { - "dd0", - "dd6", - "yy0", - "yy6", - "yny0", - "yny6", - "yyn0", - "yyn6", - "ynyn0", - "ynyn6", - "dz0", - "dz6", - "dzn0", - "dzn6", - "dy5", - "dy11", - "dyn5", - "dyn11", - "yd5", - "yd11", - "ynd5", - "ynd11", - "yz5", - "yz11", - "ynz5", - "ynz11", - "yzn5", - "yzn11", - "ynzn5", - "ynzn11", - } - - for winding1 in valid_windings: - for winding2 in valid_windings: - t = f"{winding1}{winding2}" - if t in valid_types: - with pytest.raises(RoseauLoadFlowException) as e: - TransformerParameters.extract_windings(t) - assert "Transformer windings cannot be extracted from the string" in e.value.msg - assert e.value.code == RoseauLoadFlowExceptionCode.BAD_TRANSFORMER_WINDINGS - for phase_displacement in valid_phase_displacements: - t = f"{winding1}{winding2}{phase_displacement}" - if t in valid_full_types: - w1, w2, p = TransformerParameters.extract_windings(t) - assert w1 == winding1.upper() - assert w2 == winding2 - assert p == phase_displacement - else: - with pytest.raises(RoseauLoadFlowException) as e: - TransformerParameters.extract_windings(t) - assert e.value.code == RoseauLoadFlowExceptionCode.BAD_TRANSFORMER_WINDINGS - else: - with pytest.raises(RoseauLoadFlowException) as e: - TransformerParameters.extract_windings(t) - assert "Transformer windings cannot be extracted from the string" in e.value.msg - assert e.value.code == RoseauLoadFlowExceptionCode.BAD_TRANSFORMER_WINDINGS - - -def test_catalogue_data(): - # The catalogue data path exists - catalogue_path = TransformerParameters.catalogue_path() - assert catalogue_path.exists() - - # Read it and copy it - catalogue_data = TransformerParameters.catalogue_data() - - # Iterate over the folder and ensure that the elements are in the catalogue data - error_message = ( - "Something changed in the transformers parameters catalogue. Please regenerate the json files for the " - "transformers catalogue by using the python file `scripts/generate_transformer_parameters_catalogue.py`. " - "Don't forget to delete files that are useless too." - ) - - # Check that the name is unique - assert catalogue_data["name"].is_unique, error_message - - catalogue_data.set_index(keys=["name"], inplace=True) - for idx in catalogue_data.index: - tp = TransformerParameters.from_catalogue(name=idx) - - # The entry of the catalogue has been found - assert tp.id in catalogue_data.index, error_message - - # Check the values are the same - assert tp.type == catalogue_data.at[tp.id, "type"] - assert np.isclose(tp.up.m, catalogue_data.at[tp.id, "up"]) - assert np.isclose(tp.us.m, catalogue_data.at[tp.id, "us"]) - assert np.isclose(tp.sn.m, catalogue_data.at[tp.id, "sn"]) - assert np.isclose(tp.p0.m, catalogue_data.at[tp.id, "p0"]) - assert np.isclose(tp.i0.m, catalogue_data.at[tp.id, "i0"]) - assert np.isclose(tp.psc.m, catalogue_data.at[tp.id, "psc"]) - assert np.isclose(tp.vsc.m, catalogue_data.at[tp.id, "vsc"]) - assert tp.manufacturer == catalogue_data.at[tp.id, "manufacturer"] - assert tp.range == catalogue_data.at[tp.id, "range"] - assert tp.efficiency == catalogue_data.at[tp.id, "efficiency"] - - # Check that the parameters are valid - assert isinstance(tp.z2.m, numbers.Complex) - assert isinstance(tp.ym.m, numbers.Complex) - assert isinstance(tp.k.m, numbers.Real) - assert tp.orientation in (-1.0, 1.0) - - -def test_from_catalogue(): - # Unknown strings - for field_name in ("name", "manufacturer", "range", "efficiency", "type"): - # String - with pytest.raises(RoseauLoadFlowException) as e: - TransformerParameters.from_catalogue(**{field_name: "unknown"}) - assert e.value.msg.startswith(f"No {field_name} matching 'unknown' has been found. Available ") - assert e.value.code == RoseauLoadFlowExceptionCode.CATALOGUE_NOT_FOUND - - # Regexp - with pytest.raises(RoseauLoadFlowException) as e: - TransformerParameters.from_catalogue(**{field_name: r"unknown[a-z]+"}) - assert e.value.msg.startswith(f"No {field_name} matching 'unknown[a-z]+' has been found. Available ") - assert e.value.code == RoseauLoadFlowExceptionCode.CATALOGUE_NOT_FOUND - - # Unknown floats - for field_name, display_name, display_unit in ( - ("sn", "nominal power", "kVA"), - ("up", "primary side voltage", "kV"), - ("us", "secondary side voltage", "kV"), - ): - # Without unit - with pytest.raises(RoseauLoadFlowException) as e: - TransformerParameters.from_catalogue(**{field_name: 3141.5}) - assert e.value.msg.startswith(f"No {display_name} matching 3.1 {display_unit} has been found. Available ") - assert e.value.code == RoseauLoadFlowExceptionCode.CATALOGUE_NOT_FOUND - - # With unit - with pytest.raises(RoseauLoadFlowException) as e: - TransformerParameters.from_catalogue(**{field_name: Q_(3141.5, display_unit.removeprefix("k"))}) - assert e.value.msg.startswith(f"No {display_name} matching 3.1 {display_unit} has been found. Available ") - assert e.value.code == RoseauLoadFlowExceptionCode.CATALOGUE_NOT_FOUND - - # Several transformers - with pytest.raises(RoseauLoadFlowException) as e: - TransformerParameters.from_catalogue(type=r"yzn.*", sn=50e3) - assert e.value.msg == ( - "Several transformers matching the query (type='yzn.*', nominal power=50.0 kVA) have been " - "found: 'SE_Minera_A0Ak_50kVA', 'SE_Minera_B0Bk_50kVA', 'SE_Minera_C0Bk_50kVA', " - "'SE_Minera_Standard_50kVA'." - ) - assert e.value.code == RoseauLoadFlowExceptionCode.CATALOGUE_SEVERAL_FOUND - - # Success - tp = TransformerParameters.from_catalogue(name="SE_Minera_AA0Ak_160kVA") - assert tp.id == "SE_Minera_AA0Ak_160kVA" - tp = TransformerParameters.from_catalogue(name="SE_Minera_AA0Ak_160kVA", id="tp-test1") - assert tp.id == "tp-test1" - - -def test_get_catalogue(): - # Get the entire catalogue - catalogue = TransformerParameters.get_catalogue() - assert isinstance(catalogue, pd.DataFrame) - assert catalogue.shape == (162, 7) - - # Filter on a single attribute - for field_name, value, expected_size in ( - ("name", "SE_Minera_A0Ak_50kVA", 1), - ("manufacturer", "SE", 148), - ("range", r"min.*", 67), - ("efficiency", r"c0.*", 29), - ("type", r"dy.*", 158), - ("sn", Q_(160, "kVA"), 12), - ("up", Q_(20, "kV"), 162), - ("us", 400, 162), - ): - filtered_catalogue = TransformerParameters.get_catalogue(**{field_name: value}) - assert filtered_catalogue.shape == (expected_size, 7), f"{field_name}={value!r}" - - # Filter on two attributes - for field_name, value, expected_size in ( - ("name", "SE_Minera_A0Ak_50kVA", 1), - ("range", "minera", 67), - ("efficiency", r"c0.*", 29), - ("type", r"^d.*11$", 144), - ("sn", Q_(160, "kVA"), 11), - ("up", Q_(20, "kV"), 148), - ("us", 400, 148), - ): - filtered_catalogue = TransformerParameters.get_catalogue(**{field_name: value}, manufacturer="se") - assert filtered_catalogue.shape == (expected_size, 7), f"{field_name}={value!r}" - - # Filter on three attributes - for field_name, value, expected_size in ( - ("name", "se_VEGETA_C0BK_3150kva", 1), - ("efficiency", r"c0[abc]k", 15), - ("type", r"dyn\d+", 41), - ("sn", Q_(160, "kVA"), 3), - ("up", Q_(20, "kV"), 41), - ("us", 400, 41), - ): - filtered_catalogue = TransformerParameters.get_catalogue( - **{field_name: value}, manufacturer="se", range=r"^vegeta$" - ) - assert filtered_catalogue.shape == (expected_size, 7), f"{field_name}={value!r}" - - # No results - empty_catalogue = TransformerParameters.get_catalogue(us=250) - assert empty_catalogue.shape == (0, 7) - - -def test_from_open_dss(): - """https://sourceforge.net/p/electricdss/discussion/beginners/thread/742e6c9665/ - - Main input data for transformer: - - - Type: two windings transformer - - Phases numbers: 3 - - Apparent rated power: 1800 kVA - - Frequency = 50 Hz - - Rated line voltage in = 33 kV - - Rated line voltage out = 0.405 kV - - Short circuit voltage: 6% - - Short circuit losses: 0.902% - - No load current: 0.3% - - No load losses: 0.136% - - Connections: DYn - - Group No: 11 - - Neutral: distributed at side out (low voltage level) - - Impedance to earth in = 1 MΩ (insulated) - - Impedance to earth out = 0 Ω - - Impedance to earth common = 5 Ω " - - OpenDss Model:: - - New Transformer.Isacco phases=3 windings=2 XHL=6 - ~ wdg=1 bus=MVbusname kV=33 kVA=1800 conn=delta - ~ Wdg=2 bus=LVBusname.1.2.3.4 kV=0.405 kVA=1800 conn=wye - ~ %Loadloss=0.902 %imag=0.3 %noload=.136 LeadLag=Euro - - // Neutral reactor - New Reactor.5-ohm phases=1 bus=LVBusname.4 R=0 X=5 - """ - sn = Q_(1800, "kVA") - tp_rlf = TransformerParameters.from_open_and_short_circuit_tests( - id="tp-test", - # Electrical parameters - type="Dyn11", - up=Q_(33, "kV"), - us=Q_(0.405, "kV"), - sn=sn, - p0=Q_(0.136, "percent") * sn, - i0=Q_(0.3, "percent"), - psc=Q_(0.902, "percent") * sn, - vsc=Q_(6, "percent"), - # Optional parameters - manufacturer="Roseau", - range="Tech+", - efficiency="Wonderful", - ) - - tp_dss = TransformerParameters.from_open_dss( - id="tp-test", - # Electrical parameters - conns=("delta", "wye"), - kvs=(33, 0.405), - kvas=1800, - leadlag="euro", - xhl=6, - loadloss=0.902, - noloadloss=0.136, - imag=0.3, - # Optional parameters - manufacturer="Roseau", - range="Tech+", - efficiency="Wonderful", - ) - - # Electrical parameters - assert tp_rlf.up == tp_dss.up - assert tp_rlf.us == tp_dss.us - assert tp_rlf.sn == tp_dss.sn - assert tp_rlf.k == tp_dss.k - assert tp_rlf.orientation == tp_dss.orientation - np.testing.assert_allclose(tp_rlf.z2.m, tp_dss.z2.m) - np.testing.assert_allclose(tp_rlf.ym.m, tp_dss.ym.m) - - # Optional parameters - assert tp_rlf.manufacturer == tp_dss.manufacturer - assert tp_rlf.range == tp_dss.range - assert tp_rlf.efficiency == tp_dss.efficiency - - -def test_from_power_factory(): - # Parameters from tests/data/dgs/MV_LV_Transformer.json - tp_pwf = TransformerParameters.from_power_factory( - id="Transformer 100 kVA Dyn11", - # Electrical parameters - tech=3, # Three Phase Transformer - sn=0.1, # MVA - uhv=20, # kV - ulv=0.4, # kV - vg_hv="D", - vg_lv="yn", - phase_shift=11, - uk=4, # Vsc (%) - pc=2.15, # Psc (kW) - curmg=2.5, # i0 (%) - pfe=0.21, # P0 (kW) - # Optional parameters - manufacturer="Roseau", - range="Tech+", - efficiency="Wonderful", - ) - tp_rlf = TransformerParameters.from_open_and_short_circuit_tests( - id="Transformer 100 kVA Dyn11", - # Electrical parameters - type="Dyn11", - up=Q_(20, "kV"), - us=Q_(0.4, "kV"), - sn=Q_(0.1, "MVA"), - p0=Q_(0.21, "kW"), - i0=Q_(2.5, "percent"), - psc=Q_(2.15, "kW"), - vsc=Q_(4, "percent"), - # Optional parameters - manufacturer="Roseau", - range="Tech+", - efficiency="Wonderful", - ) - - # Electrical parameters - assert tp_pwf.up == tp_rlf.up - assert tp_pwf.us == tp_rlf.us - assert tp_pwf.sn == tp_rlf.sn - assert tp_pwf.k == tp_rlf.k - assert tp_pwf.orientation == tp_rlf.orientation - np.testing.assert_allclose(tp_pwf.z2.m, tp_rlf.z2.m) - np.testing.assert_allclose(tp_pwf.ym.m, tp_rlf.ym.m) - - # Optional parameters - assert tp_pwf.manufacturer == tp_rlf.manufacturer - assert tp_pwf.range == tp_rlf.range - assert tp_pwf.efficiency == tp_rlf.efficiency - - # Test single phase (only the technology has been changed) - tp_pwf = TransformerParameters.from_power_factory( - id="Transformer 100 kVA Dyn11", - # Electrical parameters - tech="single-phase", # Single Phase Transformer - sn=0.1, # MVA - uhv=20, # kV - ulv=0.4, # kV - vg_hv="D", - vg_lv="yn", - phase_shift=11, - uk=4, # Vsc (%) - pc=2.15, # Psc (kW) - curmg=2.5, # i0 (%) - pfe=0.21, # P0 (kW) - # Optional parameters - manufacturer="Roseau", - range="Tech+", - efficiency="Wonderful", - ) - assert tp_pwf.type == "single" - - # Bad technology - with pytest.raises(RoseauLoadFlowException) as e: - TransformerParameters.from_power_factory( - id="Transformer 100 kVA Dyn11", - # Electrical parameters - tech="unknown value", # <-------------Error - sn=0.1, # MVA - uhv=20, # kV - ulv=0.4, # kV - vg_hv="D", - vg_lv="yn", - phase_shift=11, - uk=4, # Vsc (%) - pc=2.15, # Psc (kW) - curmg=2.5, # i0 (%) - pfe=0.21, # P0 (kW) - # Optional parameters - manufacturer="Roseau", - range="Tech+", - efficiency="Wonderful", - ) - assert ( - e.value.msg == "Expected tech='single-phase' or 'three-phase', got 'unknown value' for transformer parameters " - "'Transformer 100 kVA Dyn11'." - ) - assert e.value.code == RoseauLoadFlowExceptionCode.BAD_TRANSFORMER_TYPE - - -def test_to_dict(): - # No results to export - tp = TransformerParameters.from_catalogue(name="SE_Minera_A0Ak_100kVA", manufacturer="SE") - with pytest.raises(RoseauLoadFlowException) as e: - tp.results_to_dict() - assert e.value.msg == "The TransformerParameters has no results to export." - assert e.value.code == RoseauLoadFlowExceptionCode.JSON_NO_RESULTS - - # All the options for to_dict: no short-circuit data but Optional data - tp2 = TransformerParameters( - id=tp.id, - type=tp.type, - up=tp.up, - us=tp.us, - sn=tp.sn, - z2=tp.z2, - ym=tp.ym, - manufacturer=tp.manufacturer, - range=tp.range, - efficiency=tp.efficiency, - ) - d = tp2.to_dict() - assert d.pop("id") == tp2.id - assert d.pop("sn") == tp2.sn.m - assert d.pop("up") == tp2.up.m - assert d.pop("us") == tp2.us.m - assert d.pop("type") == tp2.type - assert d.pop("z2") == [tp2.z2.m.real, tp2.z2.m.imag] - assert d.pop("ym") == [tp2.ym.m.real, tp2.ym.m.imag] - assert d.pop("manufacturer") == tp2.manufacturer - assert d.pop("range") == tp2.range - assert d.pop("efficiency") == tp2.efficiency - assert not d - - # Test the from_dict without "p0", ... (only z2 and ym) - tp3 = TransformerParameters.from_dict(tp2.to_dict()) - assert tp3 == tp2 - - -def test_equality(): - data = { - "id": "Yzn11 - 50kVA", - "z2": Q_(8.64 + 9.444j, "centiohm"), # Ohm - "ym": Q_(0.3625 - 2.2206j, "uS"), # S - "us": Q_(400, "V"), # V - "up": Q_(20, "kV"), # V - "sn": Q_(50, "kVA"), # VA - "type": "yzn11", - "manufacturer": "Roseau", - "range": "Tech+", - "efficiency": "Extraordinary", - } - - tp = TransformerParameters(**data) - tp2 = TransformerParameters(**data) - assert tp2 == tp - - # Fails - other_data = { - "id": "Dyn11 - 49kVA", - "z2": Q_(8.63 + 9.444j, "centiohm"), # Ohm - "ym": Q_(0.48 - 2.2206j, "uS"), # S - "us": Q_(399, "V"), # V - "up": Q_(19, "kV"), # V - "sn": Q_(49, "kVA"), # VA - "type": "dyn11", - "manufacturer": "Roso", - "range": "Tech-", - "efficiency": "Less extraordinary", - } - for k, v in other_data.items(): - other_tp = TransformerParameters(**(data | {k: v})) - assert other_tp != tp, k - - # Test the case which returns NotImplemented in the equality operator - assert tp != object() - - -@pytest.mark.no_patch_engine -def test_compute_open_short_circuit_parameters(): - tp = TransformerParameters.from_catalogue(name="SE_Minera_A0Ak_100kVA", manufacturer="SE") - p0, i0 = tp._compute_open_circuit_parameters() - assert np.isclose(p0.m, tp.p0.m) - assert np.isclose(i0.m, tp.i0.m) - - psc, vsc = tp._compute_short_circuit_parameters() - assert np.isclose(psc.m, tp.psc.m, rtol=0.001) - assert np.isclose(vsc.m, tp.vsc.m) - - -def test_ideal_transformer(): - # Ideal transformer not yet supported - with pytest.raises(RoseauLoadFlowException) as e: - TransformerParameters(id="test", type="Dyn11", sn=50e3, up=20e3, us=400, z2=0.0, ym=0.0) - assert e.value.msg == ( - "Transformer type 'test' has a null series impedance z2. Ideal transformers are not supported." - ) - assert e.value.code == RoseauLoadFlowExceptionCode.BAD_TRANSFORMER_IMPEDANCE - # OK - TransformerParameters(id="test", type="Dyn11", sn=50e3, up=20e3, us=400, z2=0.0000001, ym=0.0) diff --git a/roseau/load_flow_single/models/tests/test_transformers.py b/roseau/load_flow_single/models/tests/test_transformers.py index a61fd91..b0a80ff 100644 --- a/roseau/load_flow_single/models/tests/test_transformers.py +++ b/roseau/load_flow_single/models/tests/test_transformers.py @@ -6,7 +6,7 @@ def test_max_power(): - tp = TransformerParameters.from_catalogue(name="FT_Standard_Standard_100kVA") + tp = TransformerParameters.from_catalogue(name="FT 100kVA 15/20kV(20) 400V Dyn11") assert tp.sn == Q_(100, "kVA") bus1 = Bus(id="bus1") @@ -24,7 +24,7 @@ def test_max_loading(): bus1 = Bus(id="bus1") bus2 = Bus(id="bus2") tp = TransformerParameters.from_open_and_short_circuit_tests( - id="tp", psc=1350.0, p0=145.0, i0=1.8 / 100, us=400, up=20000, sn=50 * 1e3, vsc=4 / 100, type="yzn11" + id="tp", vg="Yzn11", sn=50e3, uhv=20e3, ulv=400, p0=145, i0=0.018, psc=1350, vsc=0.04 ) transformer = Transformer(id="transformer", bus1=bus1, bus2=bus2, parameters=tp) @@ -44,7 +44,7 @@ def test_res_violated(): bus1 = Bus(id="bus1") bus2 = Bus(id="bus2") tp = TransformerParameters.from_open_and_short_circuit_tests( - id="tp", psc=1350.0, p0=145.0, i0=1.8 / 100, us=400, up=20000, sn=50 * 1e3, vsc=4 / 100, type="yzn11" + id="tp", vg="Yzn11", sn=50e3, uhv=20e3, ulv=400, p0=145, i0=0.018, psc=1350, vsc=0.04 ) transformer = Transformer(id="transformer", bus1=bus1, bus2=bus2, parameters=tp) @@ -82,7 +82,7 @@ def test_transformer_results(): bus1 = Bus(id="bus1") bus2 = Bus(id="bus2") tp = TransformerParameters.from_open_and_short_circuit_tests( - id="tp", psc=1350, p0=145, i0=0.018, us=400, up=20e3, sn=50e3, vsc=0.04, type="yzn11" + id="tp", vg="Yzn11", sn=50e3, uhv=20e3, ulv=400, p0=145, i0=0.018, psc=1350, vsc=0.04 ) transformer = Transformer(id="transformer", bus1=bus1, bus2=bus2, parameters=tp) diff --git a/roseau/load_flow_single/models/transformers/transformers.py b/roseau/load_flow_single/models/transformers/transformers.py index 125d16f..2ed981d 100644 --- a/roseau/load_flow_single/models/transformers/transformers.py +++ b/roseau/load_flow_single/models/transformers/transformers.py @@ -63,12 +63,12 @@ def __init__( self._parameters = parameters self.max_loading = max_loading - if parameters.type == "center": + if parameters.type == "center-tapped": msg = "Center-tapped transformers are not allowed." logger.error(msg) raise RoseauLoadFlowException(msg=msg, code=RoseauLoadFlowExceptionCode.BAD_TRANSFORMER_WINDINGS) - elif parameters.type == "single": - z2, ym, k_single = parameters._z2, parameters._ym, parameters._k + elif parameters.type == "single-phase": + z2, ym, k_single = parameters._z2, parameters._ym, parameters._k * parameters._orientation else: z2, ym = parameters._z2, parameters._ym k_complex_factor = { diff --git a/roseau/load_flow_single/network.py b/roseau/load_flow_single/network.py index 0cc3b99..47a92dd 100644 --- a/roseau/load_flow_single/network.py +++ b/roseau/load_flow_single/network.py @@ -1046,7 +1046,7 @@ def _propagate_potentials(self) -> None: for e in element._connected_elements: if e not in visited: if isinstance(element, Transformer): - k = element.parameters._us / element.parameters._up + k = element.parameters._ulv / element.parameters._uhv elements.append((e, potential * k, element)) # TODO dephasage visited.add(e) else: diff --git a/roseau/load_flow_single/scripts/generate_test_networks.py b/roseau/load_flow_single/scripts/generate_test_networks.py index 0f4ff84..fd5e008 100644 --- a/roseau/load_flow_single/scripts/generate_test_networks.py +++ b/roseau/load_flow_single/scripts/generate_test_networks.py @@ -83,7 +83,7 @@ def generate_all_element_network(apl=None) -> None: line0 = rlfs.Line(id="line0", bus1=bus0, bus2=bus1, parameters=lp0, length=rlf.Q_(1.5, "km")) # Transformer between bus1 and bus2 - tp0 = rlfs.TransformerParameters(id="630kVA", type="single", sn=630e3, up=20e3, us=400, z2=0.02, ym=1e-7) + tp0 = rlfs.TransformerParameters(id="630kVA", vg="Ii0", sn=630e3, uhv=20e3, ulv=400, z2=0.02, ym=1e-7) transformer0 = rlfs.Transformer(id="transformer0", bus1=bus1, bus2=bus2, parameters=tp0, tap=1.0) # Switch between the bus2 and the bus3 diff --git a/roseau/load_flow_single/tests/data/networks/all_element_network.json b/roseau/load_flow_single/tests/data/networks/all_element_network.json index a356b15..919653e 100644 --- a/roseau/load_flow_single/tests/data/networks/all_element_network.json +++ b/roseau/load_flow_single/tests/data/networks/all_element_network.json @@ -333,9 +333,9 @@ { "id": "630kVA", "sn": 630000.0, - "up": 20000.0, - "us": 400, - "type": "single", + "uhv": 20000.0, + "ulv": 400, + "vg": "Ii0", "z2": [0.02, 0.0], "ym": [1e-7, 0.0] } diff --git a/roseau/load_flow_single/tests/test_electrical_network.py b/roseau/load_flow_single/tests/test_electrical_network.py index a1003b3..428d34b 100644 --- a/roseau/load_flow_single/tests/test_electrical_network.py +++ b/roseau/load_flow_single/tests/test_electrical_network.py @@ -77,7 +77,7 @@ def test_connect_and_disconnect(): # Connection of a new connected component load_bus2 = Bus(id="load_bus2") - tp = TransformerParameters(id="630kVA", type="single", sn=630e3, up=20e3, us=400, z2=0.02, ym=1e-7) + tp = TransformerParameters(id="630kVA", vg="Ii0", sn=630e3, uhv=20e3, ulv=400, z2=0.02, ym=1e-7) Transformer(id="transfo", bus1=load_bus, bus2=load_bus2, parameters=tp) # Disconnection of a load @@ -244,7 +244,7 @@ def test_bad_networks(): # No potential reference bus3 = Bus(id="bus3") tp = TransformerParameters.from_open_and_short_circuit_tests( - id="t", type="single", up=20000, us=400, sn=160 * 1e3, p0=460, i0=2.3 / 100, psc=2350, vsc=4 / 100 + id="t", vg="Ii0", uhv=20000, ulv=400, sn=160 * 1e3, p0=460, i0=2.3 / 100, psc=2350, vsc=4 / 100 ) t = Transformer(id="transfo", bus1=bus2, bus2=bus3, parameters=tp) diff --git a/uv.lock b/uv.lock index 979e743..103dceb 100644 --- a/uv.lock +++ b/uv.lock @@ -1,5 +1,5 @@ version = 1 -requires-python = ">=3.10, <4.0.0" +requires-python = ">=3.10" resolution-markers = [ "python_full_version < '3.11'", "python_full_version == '3.11.*'", @@ -854,7 +854,7 @@ wheels = [ [[package]] name = "roseau-load-flow" version = "0.10.0" -source = { git = "ssh://git@github.com/RoseauTechnologies/Roseau_Load_Flow.git?branch=develop#ed1c3dbf0c85ae1e1d1f5a2409cf58281b47d541" } +source = { git = "https://github.com/RoseauTechnologies/Roseau_Load_Flow.git?branch=tr-catalogue#6c5aee50dbe346d273e275f0f5c8dd4968bd2d7d" } dependencies = [ { name = "certifi" }, { name = "geopandas" }, @@ -939,7 +939,7 @@ requires-dist = [ { name = "platformdirs", specifier = ">=4.0.0" }, { name = "pyproj", specifier = ">=3.3.0" }, { name = "regex", specifier = ">=2022.1.18" }, - { name = "roseau-load-flow", git = "ssh://git@github.com/RoseauTechnologies/Roseau_Load_Flow.git?branch=develop" }, + { name = "roseau-load-flow", git = "https://github.com/RoseauTechnologies/Roseau_Load_Flow.git?branch=tr-catalogue" }, { name = "roseau-load-flow-engine", specifier = ">=0.16.0a0" }, { name = "shapely", specifier = ">=2.0.0" }, { name = "typing-extensions", specifier = ">=4.6.2" },