power-grid-model 1.10.17__py3-none-win_amd64.whl → 1.12.119__py3-none-win_amd64.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.

Potentially problematic release.


This version of power-grid-model might be problematic. Click here for more details.

Files changed (67) hide show
  1. power_grid_model/__init__.py +54 -29
  2. power_grid_model/_core/__init__.py +3 -3
  3. power_grid_model/_core/buffer_handling.py +507 -478
  4. power_grid_model/_core/data_handling.py +195 -141
  5. power_grid_model/_core/data_types.py +142 -0
  6. power_grid_model/_core/dataset_definitions.py +109 -109
  7. power_grid_model/_core/enum.py +226 -0
  8. power_grid_model/_core/error_handling.py +215 -198
  9. power_grid_model/_core/errors.py +134 -0
  10. power_grid_model/_core/index_integer.py +17 -17
  11. power_grid_model/_core/options.py +71 -69
  12. power_grid_model/_core/power_grid_core.py +577 -562
  13. power_grid_model/_core/power_grid_dataset.py +545 -490
  14. power_grid_model/_core/power_grid_meta.py +262 -244
  15. power_grid_model/_core/power_grid_model.py +1025 -687
  16. power_grid_model/_core/power_grid_model_c/__init__.py +3 -0
  17. power_grid_model/_core/power_grid_model_c/bin/power_grid_model_c.dll +0 -0
  18. power_grid_model/_core/power_grid_model_c/get_pgm_dll_path.py +63 -0
  19. power_grid_model/_core/power_grid_model_c/include/power_grid_model_c/basics.h +251 -0
  20. power_grid_model/_core/power_grid_model_c/include/power_grid_model_c/buffer.h +108 -0
  21. power_grid_model/_core/power_grid_model_c/include/power_grid_model_c/dataset.h +332 -0
  22. power_grid_model/_core/power_grid_model_c/include/power_grid_model_c/dataset_definitions.h +1060 -0
  23. power_grid_model/_core/power_grid_model_c/include/power_grid_model_c/handle.h +111 -0
  24. power_grid_model/_core/power_grid_model_c/include/power_grid_model_c/meta_data.h +189 -0
  25. power_grid_model/_core/power_grid_model_c/include/power_grid_model_c/model.h +130 -0
  26. power_grid_model/_core/power_grid_model_c/include/power_grid_model_c/options.h +142 -0
  27. power_grid_model/_core/power_grid_model_c/include/power_grid_model_c/serialization.h +118 -0
  28. power_grid_model/_core/power_grid_model_c/include/power_grid_model_c.h +36 -0
  29. power_grid_model/_core/power_grid_model_c/include/power_grid_model_cpp/basics.hpp +65 -0
  30. power_grid_model/_core/power_grid_model_c/include/power_grid_model_cpp/buffer.hpp +61 -0
  31. power_grid_model/_core/power_grid_model_c/include/power_grid_model_cpp/dataset.hpp +224 -0
  32. power_grid_model/_core/power_grid_model_c/include/power_grid_model_cpp/handle.hpp +108 -0
  33. power_grid_model/_core/power_grid_model_c/include/power_grid_model_cpp/meta_data.hpp +84 -0
  34. power_grid_model/_core/power_grid_model_c/include/power_grid_model_cpp/model.hpp +63 -0
  35. power_grid_model/_core/power_grid_model_c/include/power_grid_model_cpp/options.hpp +52 -0
  36. power_grid_model/_core/power_grid_model_c/include/power_grid_model_cpp/serialization.hpp +124 -0
  37. power_grid_model/_core/power_grid_model_c/include/power_grid_model_cpp/utils.hpp +81 -0
  38. power_grid_model/_core/power_grid_model_c/include/power_grid_model_cpp.hpp +19 -0
  39. power_grid_model/_core/power_grid_model_c/lib/cmake/power_grid_model/power_grid_modelConfig.cmake +37 -0
  40. power_grid_model/_core/power_grid_model_c/lib/cmake/power_grid_model/power_grid_modelConfigVersion.cmake +65 -0
  41. power_grid_model/_core/power_grid_model_c/lib/cmake/power_grid_model/power_grid_modelTargets-release.cmake +19 -0
  42. power_grid_model/_core/power_grid_model_c/lib/cmake/power_grid_model/power_grid_modelTargets.cmake +144 -0
  43. power_grid_model/_core/power_grid_model_c/lib/power_grid_model_c.lib +0 -0
  44. power_grid_model/_core/power_grid_model_c/share/LICENSE +292 -0
  45. power_grid_model/_core/power_grid_model_c/share/README.md +15 -0
  46. power_grid_model/_core/serialization.py +319 -317
  47. power_grid_model/_core/typing.py +20 -0
  48. power_grid_model/{_utils.py → _core/utils.py} +798 -783
  49. power_grid_model/data_types.py +321 -319
  50. power_grid_model/enum.py +27 -214
  51. power_grid_model/errors.py +37 -119
  52. power_grid_model/typing.py +43 -48
  53. power_grid_model/utils.py +529 -400
  54. power_grid_model/validation/__init__.py +25 -10
  55. power_grid_model/validation/{rules.py → _rules.py} +1167 -962
  56. power_grid_model/validation/{validation.py → _validation.py} +1172 -1015
  57. power_grid_model/validation/assertions.py +93 -92
  58. power_grid_model/validation/errors.py +602 -524
  59. power_grid_model/validation/utils.py +313 -318
  60. {power_grid_model-1.10.17.dist-info → power_grid_model-1.12.119.dist-info}/METADATA +162 -165
  61. power_grid_model-1.12.119.dist-info/RECORD +65 -0
  62. {power_grid_model-1.10.17.dist-info → power_grid_model-1.12.119.dist-info}/WHEEL +1 -1
  63. power_grid_model-1.12.119.dist-info/entry_points.txt +3 -0
  64. power_grid_model/_core/_power_grid_core.dll +0 -0
  65. power_grid_model-1.10.17.dist-info/RECORD +0 -32
  66. power_grid_model-1.10.17.dist-info/top_level.txt +0 -1
  67. {power_grid_model-1.10.17.dist-info → power_grid_model-1.12.119.dist-info/licenses}/LICENSE +0 -0
@@ -1,1015 +1,1172 @@
1
- # SPDX-FileCopyrightText: Contributors to the Power Grid Model project <powergridmodel@lfenergy.org>
2
- #
3
- # SPDX-License-Identifier: MPL-2.0
4
-
5
- """
6
- Power Grid Model Validation Functions.
7
-
8
- Although all functions are 'public', you probably only need validate_input_data() and validate_batch_data().
9
-
10
- """
11
-
12
- # pylint: disable=too-many-lines
13
-
14
- import copy
15
- from collections.abc import Sized as ABCSized
16
- from itertools import chain
17
- from typing import cast
18
-
19
- import numpy as np
20
-
21
- from power_grid_model import ComponentType, DatasetType, power_grid_meta_data
22
- from power_grid_model._utils import (
23
- compatibility_convert_row_columnar_dataset as _compatibility_convert_row_columnar_dataset,
24
- convert_batch_dataset_to_batch_list as _convert_batch_dataset_to_batch_list,
25
- )
26
- from power_grid_model.data_types import BatchDataset, Dataset, SingleDataset
27
- from power_grid_model.enum import (
28
- Branch3Side,
29
- BranchSide,
30
- CalculationType,
31
- FaultPhase,
32
- FaultType,
33
- LoadGenType,
34
- MeasuredTerminalType,
35
- WindingType,
36
- )
37
- from power_grid_model.validation.errors import (
38
- IdNotInDatasetError,
39
- InvalidIdError,
40
- MissingValueError,
41
- MultiComponentNotUniqueError,
42
- ValidationError,
43
- )
44
- from power_grid_model.validation.rules import (
45
- all_between as _all_between,
46
- all_between_or_at as _all_between_or_at,
47
- all_boolean as _all_boolean,
48
- all_cross_unique as _all_cross_unique,
49
- all_enabled_identical as _all_enabled_identical,
50
- all_finite as _all_finite,
51
- all_greater_or_equal as _all_greater_or_equal,
52
- all_greater_than_or_equal_to_zero as _all_greater_than_or_equal_to_zero,
53
- all_greater_than_zero as _all_greater_than_zero,
54
- all_less_than as _all_less_than,
55
- all_not_two_values_equal as _all_not_two_values_equal,
56
- all_not_two_values_zero as _all_not_two_values_zero,
57
- all_supported_tap_control_side as _all_supported_tap_control_side,
58
- all_unique as _all_unique,
59
- all_valid_associated_enum_values as _all_valid_associated_enum_values,
60
- all_valid_clocks as _all_valid_clocks,
61
- all_valid_enum_values as _all_valid_enum_values,
62
- all_valid_fault_phases as _all_valid_fault_phases,
63
- all_valid_ids as _all_valid_ids,
64
- ids_valid_in_update_data_set as _ids_valid_in_update_data_set,
65
- none_missing as _none_missing,
66
- valid_p_q_sigma as _valid_p_q_sigma,
67
- )
68
- from power_grid_model.validation.utils import _update_input_data
69
-
70
-
71
- def validate_input_data(
72
- input_data: SingleDataset, calculation_type: CalculationType | None = None, symmetric: bool = True
73
- ) -> list[ValidationError] | None:
74
- """
75
- Validates the entire input dataset:
76
-
77
- 1. Is the data structure correct? (checking data types and numpy array shapes)
78
- 2. Are all required values provided? (checking NaNs)
79
- 3. Are all ID's unique? (checking object identifiers across all components)
80
- 4. Are the supplied values valid? (checking limits and other logic as described in "Graph Data Model")
81
-
82
- Args:
83
- input_data: A power-grid-model input dataset
84
- calculation_type: Supply a calculation method, to allow missing values for unused fields
85
- symmetric: A boolean to state whether input data will be used for a symmetric or asymmetric calculation
86
-
87
- Returns:
88
- None if the data is valid, or a list containing all validation errors.
89
-
90
- Raises:
91
- Error: KeyError | TypeError | ValueError: if the data structure is invalid.
92
- """
93
- # Convert to row based if in columnar or mixed format format
94
- row_input_data = _compatibility_convert_row_columnar_dataset(input_data, None, DatasetType.input)
95
-
96
- # A deep copy is made of the input data, since default values will be added in the validation process
97
- input_data_copy = copy.deepcopy(row_input_data)
98
-
99
- assert_valid_data_structure(input_data_copy, DatasetType.input)
100
-
101
- errors: list[ValidationError] = []
102
- errors += validate_required_values(input_data_copy, calculation_type, symmetric)
103
- errors += validate_unique_ids_across_components(input_data_copy)
104
- errors += validate_values(input_data_copy, calculation_type)
105
- return errors if errors else None
106
-
107
-
108
- def validate_batch_data(
109
- input_data: SingleDataset,
110
- update_data: BatchDataset,
111
- calculation_type: CalculationType | None = None,
112
- symmetric: bool = True,
113
- ) -> dict[int, list[ValidationError]] | None:
114
- """
115
- The input dataset is validated:
116
-
117
- 1. Is the data structure correct? (checking data types and numpy array shapes)
118
- 2. Are all input data ID's unique? (checking object identifiers across all components)
119
-
120
- For each batch the update data is validated:
121
- 3. Is the update data structure correct? (checking data types and numpy array shapes)
122
- 4. Are all update ID's valid? (checking object identifiers across update and input data)
123
-
124
- Then (for each batch independently) the input dataset is updated with the batch's update data and validated:
125
- 5. Are all required values provided? (checking NaNs)
126
- 6. Are the supplied values valid? (checking limits and other logic as described in "Graph Data Model")
127
-
128
- Args:
129
- input_data: A power-grid-model input dataset
130
- update_data: A power-grid-model update dataset (one or more batches)
131
- calculation_type: Supply a calculation method, to allow missing values for unused fields
132
- symmetric: A boolean to state whether input data will be used for a symmetric or asymmetric calculation
133
-
134
- Returns:
135
- None if the data is valid, or a dictionary containing all validation errors,
136
- where the key is the batch number (0-indexed).
137
-
138
- Raises:
139
- Error: KeyError | TypeError | ValueError: if the data structure is invalid.
140
- """
141
- # Convert to row based if in columnar or mixed format
142
- row_input_data = _compatibility_convert_row_columnar_dataset(input_data, None, DatasetType.input)
143
-
144
- # A deep copy is made of the input data, since default values will be added in the validation process
145
- input_data_copy = copy.deepcopy(row_input_data)
146
- assert_valid_data_structure(input_data_copy, DatasetType.input)
147
-
148
- input_errors: list[ValidationError] = list(validate_unique_ids_across_components(input_data_copy))
149
-
150
- batch_data = _convert_batch_dataset_to_batch_list(update_data, DatasetType.update)
151
-
152
- errors = {}
153
- for batch, batch_update_data in enumerate(batch_data):
154
- row_update_data = _compatibility_convert_row_columnar_dataset(batch_update_data, None, DatasetType.update)
155
- assert_valid_data_structure(row_update_data, DatasetType.update)
156
- id_errors: list[IdNotInDatasetError | InvalidIdError] = validate_ids(row_update_data, input_data_copy)
157
-
158
- batch_errors = input_errors + id_errors
159
-
160
- if not id_errors:
161
- batch_errors = input_errors
162
- merged_data = _update_input_data(input_data_copy, row_update_data)
163
- batch_errors += validate_required_values(merged_data, calculation_type, symmetric)
164
- batch_errors += validate_values(merged_data, calculation_type)
165
-
166
- if batch_errors:
167
- errors[batch] = batch_errors
168
-
169
- return errors if errors else None
170
-
171
-
172
- def assert_valid_data_structure(data: Dataset, data_type: DatasetType) -> None:
173
- """
174
- Checks if all component names are valid and if the data inside the component matches the required Numpy
175
- structured array as defined in the Power Grid Model meta data.
176
-
177
- Args:
178
- data: A power-grid-model input/update dataset
179
- data_type: 'input' or 'update'
180
-
181
- Raises:
182
- Error: KeyError, TypeError
183
-
184
- """
185
- if data_type not in {DatasetType.input, DatasetType.update}:
186
- raise KeyError(f"Unexpected data type '{data_type}' (should be 'input' or 'update')")
187
-
188
- component_dtype = {component: meta.dtype for component, meta in power_grid_meta_data[data_type].items()}
189
- for component, array in data.items():
190
- # Check if component name is valid
191
- if component not in component_dtype:
192
- raise KeyError(f"Unknown component '{component}' in {data_type}_data.")
193
-
194
- # Check if component definition is as expected
195
- dtype = component_dtype[component]
196
- if isinstance(array, np.ndarray):
197
- if array.dtype != dtype:
198
- if not hasattr(array.dtype, "names") or not array.dtype.names:
199
- raise TypeError(
200
- f"Unexpected Numpy array ({array.dtype}) for '{component}' {data_type}_data "
201
- "(should be a Numpy structured array)."
202
- )
203
- raise TypeError(f"Unexpected Numpy structured array; (expected = {dtype}, actual = {array.dtype}).")
204
- else:
205
- raise TypeError(
206
- f"Unexpected data type {type(array).__name__} for '{component}' {data_type}_data "
207
- "(should be a Numpy structured array)."
208
- )
209
-
210
-
211
- def validate_unique_ids_across_components(data: SingleDataset) -> list[MultiComponentNotUniqueError]:
212
- """
213
- Checks if all ids in the input dataset are unique
214
-
215
- Args:
216
- data: A power-grid-model input dataset
217
-
218
- Returns:
219
- An empty list if all ids are unique, or a list of MultiComponentNotUniqueErrors for all components that
220
- have non-unique ids
221
- """
222
- return _all_cross_unique(data, [(component, "id") for component in data])
223
-
224
-
225
- def validate_ids(update_data: SingleDataset, input_data: SingleDataset) -> list[IdNotInDatasetError | InvalidIdError]:
226
- """
227
- Checks if all ids of the components in the update data:
228
- - exist and match those in the input data
229
- - are not present but qualifies for optional id
230
-
231
- This function should be called for every update dataset in a batch set
232
-
233
- Args:
234
- update_data: A single update dataset
235
- input_data: Input dataset
236
-
237
- Returns:
238
- An empty list if all update data ids are valid, or a list of IdNotInDatasetErrors or InvalidIdError for
239
- all update components that have invalid ids
240
-
241
- """
242
- errors = (
243
- _ids_valid_in_update_data_set(update_data, input_data, component, "update_data") for component in update_data
244
- )
245
- return list(chain(*errors))
246
-
247
-
248
- def _process_power_sigma_and_p_q_sigma(
249
- data: SingleDataset,
250
- sensor: ComponentType,
251
- required_list: dict[ComponentType | str, list[str | list[str]]],
252
- ) -> None:
253
- """
254
- Helper function to process the required list when both `p_sigma` and `q_sigma` exist
255
- and valid but `power_sigma` is missing. The field `power_sigma` is set to the norm of
256
- `p_sigma` and `q_sigma`in this case. Happens only on proxy data (not the original data).
257
- However, note that this value is eventually not used in the calculation.
258
- """
259
-
260
- def _check_sensor_in_data(_data, _sensor):
261
- return _sensor in _data and isinstance(_data[_sensor], np.ndarray)
262
-
263
- def _contains_p_q_sigma(_sensor_data):
264
- return "p_sigma" in _sensor_data.dtype.names and "q_sigma" in _sensor_data.dtype.names
265
-
266
- def _process_power_sigma_in_list(_sensor_mask, _power_sigma, _p_sigma, _q_sigma):
267
- _mask = np.logical_not(np.logical_or(np.isnan(_p_sigma), np.isnan(_q_sigma)))
268
- if _power_sigma.ndim < _mask.ndim:
269
- _mask = np.any(_mask, axis=tuple(range(_power_sigma.ndim, _mask.ndim)))
270
-
271
- for sublist, should_remove in zip(_sensor_mask, _mask):
272
- if should_remove and "power_sigma" in sublist:
273
- sublist = cast(list[str], sublist)
274
- sublist.remove("power_sigma")
275
-
276
- if _check_sensor_in_data(data, sensor):
277
- sensor_data = data[sensor]
278
- sensor_mask = required_list[sensor]
279
- if _contains_p_q_sigma(sensor_data):
280
- p_sigma = sensor_data["p_sigma"]
281
- q_sigma = sensor_data["q_sigma"]
282
- power_sigma = sensor_data["power_sigma"]
283
-
284
- _process_power_sigma_in_list(sensor_mask, power_sigma, p_sigma, q_sigma)
285
-
286
-
287
- def validate_required_values(
288
- data: SingleDataset, calculation_type: CalculationType | None = None, symmetric: bool = True
289
- ) -> list[MissingValueError]:
290
- """
291
- Checks if all required data is available.
292
-
293
- Args:
294
- data: A power-grid-model input dataset
295
- calculation_type: Supply a calculation method, to allow missing values for unused fields
296
- symmetric: A boolean to state whether input data will be used for a symmetric or asymmetric calculation
297
-
298
- Returns:
299
- An empty list if all required data is available, or a list of MissingValueErrors.
300
- """
301
- # Base
302
- required: dict[ComponentType | str, list[str | list[str]]] = {"base": ["id"]}
303
-
304
- # Nodes
305
- required["node"] = required["base"] + ["u_rated"]
306
-
307
- # Branches
308
- required["branch"] = required["base"] + ["from_node", "to_node", "from_status", "to_status"]
309
- required["link"] = required["branch"].copy()
310
- required["line"] = required["branch"] + ["r1", "x1", "c1", "tan1"]
311
- required["transformer"] = required["branch"] + [
312
- "u1",
313
- "u2",
314
- "sn",
315
- "uk",
316
- "pk",
317
- "i0",
318
- "p0",
319
- "winding_from",
320
- "winding_to",
321
- "clock",
322
- "tap_side",
323
- "tap_min",
324
- "tap_max",
325
- "tap_size",
326
- ]
327
- # Branch3
328
- required["branch3"] = required["base"] + ["node_1", "node_2", "node_3", "status_1", "status_2", "status_3"]
329
- required["three_winding_transformer"] = required["branch3"] + [
330
- "u1",
331
- "u2",
332
- "u3",
333
- "sn_1",
334
- "sn_2",
335
- "sn_3",
336
- "uk_12",
337
- "uk_13",
338
- "uk_23",
339
- "pk_12",
340
- "pk_13",
341
- "pk_23",
342
- "i0",
343
- "p0",
344
- "winding_1",
345
- "winding_2",
346
- "winding_3",
347
- "clock_12",
348
- "clock_13",
349
- "tap_side",
350
- "tap_min",
351
- "tap_max",
352
- "tap_size",
353
- ]
354
-
355
- # Regulators
356
- required["regulator"] = required["base"] + ["regulated_object", "status"]
357
- required["transformer_tap_regulator"] = required["regulator"]
358
- if calculation_type is None or calculation_type == CalculationType.power_flow:
359
- required["transformer_tap_regulator"] += ["control_side", "u_set", "u_band"]
360
-
361
- # Appliances
362
- required["appliance"] = required["base"] + ["node", "status"]
363
- required["source"] = required["appliance"].copy()
364
- if calculation_type is None or calculation_type == CalculationType.power_flow:
365
- required["source"] += ["u_ref"]
366
- required["shunt"] = required["appliance"] + ["g1", "b1"]
367
- required["generic_load_gen"] = required["appliance"] + ["type"]
368
- if calculation_type is None or calculation_type == CalculationType.power_flow:
369
- required["generic_load_gen"] += ["p_specified", "q_specified"]
370
- required["sym_load"] = required["generic_load_gen"].copy()
371
- required["asym_load"] = required["generic_load_gen"].copy()
372
- required["sym_gen"] = required["generic_load_gen"].copy()
373
- required["asym_gen"] = required["generic_load_gen"].copy()
374
-
375
- # Sensors
376
- required["sensor"] = required["base"] + ["measured_object"]
377
- required["voltage_sensor"] = required["sensor"].copy()
378
- required["power_sensor"] = required["sensor"] + ["measured_terminal_type"]
379
- if calculation_type is None or calculation_type == CalculationType.state_estimation:
380
- required["voltage_sensor"] += ["u_sigma", "u_measured"]
381
- required["power_sensor"] += ["power_sigma", "p_measured", "q_measured"]
382
- required["sym_voltage_sensor"] = required["voltage_sensor"].copy()
383
- required["asym_voltage_sensor"] = required["voltage_sensor"].copy()
384
- # Different requirements for individual sensors. Avoid shallow copy.
385
- for sensor_type in ("sym_power_sensor", "asym_power_sensor"):
386
- try:
387
- required[sensor_type] = [
388
- required["power_sensor"].copy() for _ in range(data[sensor_type].shape[0]) # type: ignore
389
- ]
390
- except KeyError:
391
- pass
392
-
393
- # Faults
394
- required["fault"] = required["base"] + ["fault_object"]
395
- asym_sc = False
396
- if calculation_type is None or calculation_type == CalculationType.short_circuit:
397
- required["fault"] += ["status", "fault_type"]
398
- if "fault" in data:
399
- for elem in data[ComponentType.fault]["fault_type"]:
400
- if elem not in (FaultType.three_phase, FaultType.nan):
401
- asym_sc = True
402
- break
403
-
404
- if not symmetric or asym_sc:
405
- required["line"] += ["r0", "x0", "c0", "tan0"]
406
- required["shunt"] += ["g0", "b0"]
407
-
408
- _process_power_sigma_and_p_q_sigma(data, ComponentType.sym_power_sensor, required)
409
- _process_power_sigma_and_p_q_sigma(data, ComponentType.asym_power_sensor, required)
410
-
411
- return _validate_required_in_data(data, required)
412
-
413
-
414
- def _validate_required_in_data(data, required):
415
- """
416
- Checks if all required data is available.
417
-
418
- Args:
419
- data: A power-grid-model input dataset
420
- required: a list of required fields (a list of str), per component when applicaple (a list of str or str lists)
421
-
422
- Returns:
423
- An empty list if all required data is available, or a list of MissingValueErrors.
424
- """
425
-
426
- def is_valid_component(data, component):
427
- return (
428
- not (isinstance(data[component], np.ndarray) and data[component].size == 0)
429
- and data[component] is not None
430
- and isinstance(data[component], ABCSized)
431
- )
432
-
433
- def is_nested_list(items):
434
- return isinstance(items, list) and all(isinstance(i, list) for i in items)
435
-
436
- def process_nested_items(component, items, data, results):
437
- for index, item in enumerate(sublist for sublist in items):
438
- if index < len(data[component]):
439
- results.append(_none_missing(data, component, item, index))
440
-
441
- results = []
442
-
443
- for component in data:
444
- if is_valid_component(data, component):
445
- items = required.get(component, [])
446
- if is_nested_list(items):
447
- process_nested_items(component, items, data, results)
448
- else:
449
- results.append(_none_missing(data, component, items, 0))
450
-
451
- return list(chain(*results))
452
-
453
-
454
- def validate_values(data: SingleDataset, calculation_type: CalculationType | None = None) -> list[ValidationError]:
455
- """
456
- For each component supplied in the data, call the appropriate validation function
457
-
458
- Args:
459
- data: A power-grid-model input dataset
460
- calculation_type: Supply a calculation method, to allow missing values for unused fields
461
-
462
- Returns:
463
- An empty list if all required data is valid, or a list of ValidationErrors.
464
-
465
- """
466
- errors: list[ValidationError] = list(
467
- _all_finite(
468
- data,
469
- {
470
- ComponentType.sym_power_sensor: ["power_sigma"],
471
- ComponentType.asym_power_sensor: ["power_sigma"],
472
- ComponentType.sym_voltage_sensor: ["u_sigma"],
473
- ComponentType.asym_voltage_sensor: ["u_sigma"],
474
- },
475
- )
476
- )
477
-
478
- component_validators = {
479
- "node": validate_node,
480
- "line": validate_line,
481
- "link": lambda d: validate_branch(d, ComponentType.link),
482
- "generic_branch": validate_generic_branch,
483
- "transformer": validate_transformer,
484
- "three_winding_transformer": validate_three_winding_transformer,
485
- "source": validate_source,
486
- "sym_load": lambda d: validate_generic_load_gen(d, ComponentType.sym_load),
487
- "sym_gen": lambda d: validate_generic_load_gen(d, ComponentType.sym_gen),
488
- "asym_load": lambda d: validate_generic_load_gen(d, ComponentType.asym_load),
489
- "asym_gen": lambda d: validate_generic_load_gen(d, ComponentType.asym_gen),
490
- "shunt": validate_shunt,
491
- }
492
-
493
- for component, validator in component_validators.items():
494
- if component in data:
495
- errors += validator(data)
496
-
497
- if calculation_type in (None, CalculationType.state_estimation):
498
- if "sym_voltage_sensor" in data:
499
- errors += validate_generic_voltage_sensor(data, ComponentType.sym_voltage_sensor)
500
- if "asym_voltage_sensor" in data:
501
- errors += validate_generic_voltage_sensor(data, ComponentType.asym_voltage_sensor)
502
- if "sym_power_sensor" in data:
503
- errors += validate_generic_power_sensor(data, ComponentType.sym_power_sensor)
504
- if "asym_power_sensor" in data:
505
- errors += validate_generic_power_sensor(data, ComponentType.asym_power_sensor)
506
-
507
- if calculation_type in (None, CalculationType.short_circuit) and "fault" in data:
508
- errors += validate_fault(data)
509
-
510
- if calculation_type in (None, CalculationType.power_flow) and "transformer_tap_regulator" in data:
511
- errors += validate_transformer_tap_regulator(data)
512
-
513
- return errors
514
-
515
-
516
- # pylint: disable=missing-function-docstring
517
-
518
-
519
- def validate_base(data: SingleDataset, component: ComponentType) -> list[ValidationError]:
520
- errors: list[ValidationError] = list(_all_unique(data, component, "id"))
521
- return errors
522
-
523
-
524
- def validate_node(data: SingleDataset) -> list[ValidationError]:
525
- errors = validate_base(data, ComponentType.node)
526
- errors += _all_greater_than_zero(data, ComponentType.node, "u_rated")
527
- return errors
528
-
529
-
530
- def validate_branch(data: SingleDataset, component: ComponentType) -> list[ValidationError]:
531
- errors = validate_base(data, component)
532
- errors += _all_valid_ids(data, component, "from_node", ComponentType.node)
533
- errors += _all_valid_ids(data, component, "to_node", ComponentType.node)
534
- errors += _all_not_two_values_equal(data, component, "to_node", "from_node")
535
- errors += _all_boolean(data, component, "from_status")
536
- errors += _all_boolean(data, component, "to_status")
537
- return errors
538
-
539
-
540
- def validate_line(data: SingleDataset) -> list[ValidationError]:
541
- errors = validate_branch(data, ComponentType.line)
542
- errors += _all_not_two_values_zero(data, ComponentType.line, "r1", "x1")
543
- errors += _all_not_two_values_zero(data, ComponentType.line, "r0", "x0")
544
- errors += _all_greater_than_zero(data, ComponentType.line, "i_n")
545
- return errors
546
-
547
-
548
- def validate_generic_branch(data: SingleDataset) -> list[ValidationError]:
549
- errors = validate_branch(data, ComponentType.generic_branch)
550
- errors += _all_greater_than_zero(data, ComponentType.generic_branch, "k")
551
- errors += _all_greater_than_or_equal_to_zero(data, ComponentType.generic_branch, "sn")
552
- return errors
553
-
554
-
555
- def validate_transformer(data: SingleDataset) -> list[ValidationError]:
556
- errors = validate_branch(data, ComponentType.transformer)
557
- errors += _all_greater_than_zero(data, ComponentType.transformer, "u1")
558
- errors += _all_greater_than_zero(data, ComponentType.transformer, "u2")
559
- errors += _all_greater_than_zero(data, ComponentType.transformer, "sn")
560
- errors += _all_greater_or_equal(data, ComponentType.transformer, "uk", "pk/sn")
561
- errors += _all_between(data, ComponentType.transformer, "uk", 0, 1)
562
- errors += _all_greater_than_or_equal_to_zero(data, ComponentType.transformer, "pk")
563
- errors += _all_greater_or_equal(data, ComponentType.transformer, "i0", "p0/sn")
564
- errors += _all_less_than(data, ComponentType.transformer, "i0", 1)
565
- errors += _all_greater_than_or_equal_to_zero(data, ComponentType.transformer, "p0")
566
- errors += _all_valid_enum_values(data, ComponentType.transformer, "winding_from", WindingType)
567
- errors += _all_valid_enum_values(data, ComponentType.transformer, "winding_to", WindingType)
568
- errors += _all_between_or_at(data, ComponentType.transformer, "clock", 0, 12)
569
- errors += _all_valid_clocks(data, ComponentType.transformer, "clock", "winding_from", "winding_to")
570
- errors += _all_valid_enum_values(data, ComponentType.transformer, "tap_side", BranchSide)
571
- errors += _all_between_or_at(
572
- data, ComponentType.transformer, "tap_pos", "tap_min", "tap_max", data[ComponentType.transformer]["tap_nom"], 0
573
- )
574
- errors += _all_between_or_at(data, ComponentType.transformer, "tap_nom", "tap_min", "tap_max", 0)
575
- errors += _all_greater_than_or_equal_to_zero(data, ComponentType.transformer, "tap_size")
576
- errors += _all_greater_or_equal(
577
- data, ComponentType.transformer, "uk_min", "pk_min/sn", data[ComponentType.transformer]["uk"]
578
- )
579
- errors += _all_between(data, ComponentType.transformer, "uk_min", 0, 1, data[ComponentType.transformer]["uk"])
580
- errors += _all_greater_or_equal(
581
- data, ComponentType.transformer, "uk_max", "pk_max/sn", data[ComponentType.transformer]["uk"]
582
- )
583
- errors += _all_between(data, ComponentType.transformer, "uk_max", 0, 1, data[ComponentType.transformer]["uk"])
584
- errors += _all_greater_than_or_equal_to_zero(
585
- data, ComponentType.transformer, "pk_min", data[ComponentType.transformer]["pk"]
586
- )
587
- errors += _all_greater_than_or_equal_to_zero(
588
- data, ComponentType.transformer, "pk_max", data[ComponentType.transformer]["pk"]
589
- )
590
- return errors
591
-
592
-
593
- def validate_branch3(data: SingleDataset, component: ComponentType) -> list[ValidationError]:
594
- errors = validate_base(data, component)
595
- errors += _all_valid_ids(data, component, "node_1", ComponentType.node)
596
- errors += _all_valid_ids(data, component, "node_2", ComponentType.node)
597
- errors += _all_valid_ids(data, component, "node_3", ComponentType.node)
598
- errors += _all_not_two_values_equal(data, component, "node_1", "node_2")
599
- errors += _all_not_two_values_equal(data, component, "node_1", "node_3")
600
- errors += _all_not_two_values_equal(data, component, "node_2", "node_3")
601
- errors += _all_boolean(data, component, "status_1")
602
- errors += _all_boolean(data, component, "status_2")
603
- errors += _all_boolean(data, component, "status_3")
604
- return errors
605
-
606
-
607
- # pylint: disable=R0915
608
- def validate_three_winding_transformer(data: SingleDataset) -> list[ValidationError]:
609
- errors = validate_branch3(data, ComponentType.three_winding_transformer)
610
- errors += _all_greater_than_zero(data, ComponentType.three_winding_transformer, "u1")
611
- errors += _all_greater_than_zero(data, ComponentType.three_winding_transformer, "u2")
612
- errors += _all_greater_than_zero(data, ComponentType.three_winding_transformer, "u3")
613
- errors += _all_greater_than_zero(data, ComponentType.three_winding_transformer, "sn_1")
614
- errors += _all_greater_than_zero(data, ComponentType.three_winding_transformer, "sn_2")
615
- errors += _all_greater_than_zero(data, ComponentType.three_winding_transformer, "sn_3")
616
- errors += _all_greater_or_equal(data, ComponentType.three_winding_transformer, "uk_12", "pk_12/sn_1")
617
- errors += _all_greater_or_equal(data, ComponentType.three_winding_transformer, "uk_12", "pk_12/sn_2")
618
- errors += _all_greater_or_equal(data, ComponentType.three_winding_transformer, "uk_13", "pk_13/sn_1")
619
- errors += _all_greater_or_equal(data, ComponentType.three_winding_transformer, "uk_13", "pk_13/sn_3")
620
- errors += _all_greater_or_equal(data, ComponentType.three_winding_transformer, "uk_23", "pk_23/sn_2")
621
- errors += _all_greater_or_equal(data, ComponentType.three_winding_transformer, "uk_23", "pk_23/sn_3")
622
- errors += _all_between(data, ComponentType.three_winding_transformer, "uk_12", 0, 1)
623
- errors += _all_between(data, ComponentType.three_winding_transformer, "uk_13", 0, 1)
624
- errors += _all_between(data, ComponentType.three_winding_transformer, "uk_23", 0, 1)
625
- errors += _all_greater_than_or_equal_to_zero(data, ComponentType.three_winding_transformer, "pk_12")
626
- errors += _all_greater_than_or_equal_to_zero(data, ComponentType.three_winding_transformer, "pk_13")
627
- errors += _all_greater_than_or_equal_to_zero(data, ComponentType.three_winding_transformer, "pk_23")
628
- errors += _all_greater_or_equal(data, ComponentType.three_winding_transformer, "i0", "p0/sn_1")
629
- errors += _all_less_than(data, ComponentType.three_winding_transformer, "i0", 1)
630
- errors += _all_greater_than_or_equal_to_zero(data, ComponentType.three_winding_transformer, "p0")
631
- errors += _all_valid_enum_values(data, ComponentType.three_winding_transformer, "winding_1", WindingType)
632
- errors += _all_valid_enum_values(data, ComponentType.three_winding_transformer, "winding_2", WindingType)
633
- errors += _all_valid_enum_values(data, ComponentType.three_winding_transformer, "winding_3", WindingType)
634
- errors += _all_between_or_at(data, ComponentType.three_winding_transformer, "clock_12", 0, 12)
635
- errors += _all_between_or_at(data, ComponentType.three_winding_transformer, "clock_13", 0, 12)
636
- errors += _all_valid_clocks(data, ComponentType.three_winding_transformer, "clock_12", "winding_1", "winding_2")
637
- errors += _all_valid_clocks(data, ComponentType.three_winding_transformer, "clock_13", "winding_1", "winding_3")
638
- errors += _all_valid_enum_values(data, ComponentType.three_winding_transformer, "tap_side", Branch3Side)
639
- errors += _all_between_or_at(
640
- data,
641
- ComponentType.three_winding_transformer,
642
- "tap_pos",
643
- "tap_min",
644
- "tap_max",
645
- data[ComponentType.three_winding_transformer]["tap_nom"],
646
- 0,
647
- )
648
- errors += _all_between_or_at(data, ComponentType.three_winding_transformer, "tap_nom", "tap_min", "tap_max", 0)
649
- errors += _all_greater_than_or_equal_to_zero(data, ComponentType.three_winding_transformer, "tap_size")
650
- errors += _all_greater_or_equal(
651
- data,
652
- ComponentType.three_winding_transformer,
653
- "uk_12_min",
654
- "pk_12_min/sn_1",
655
- data[ComponentType.three_winding_transformer]["uk_12"],
656
- )
657
- errors += _all_greater_or_equal(
658
- data,
659
- ComponentType.three_winding_transformer,
660
- "uk_12_min",
661
- "pk_12_min/sn_2",
662
- data[ComponentType.three_winding_transformer]["uk_12"],
663
- )
664
- errors += _all_greater_or_equal(
665
- data,
666
- ComponentType.three_winding_transformer,
667
- "uk_13_min",
668
- "pk_13_min/sn_1",
669
- data[ComponentType.three_winding_transformer]["uk_13"],
670
- )
671
- errors += _all_greater_or_equal(
672
- data,
673
- ComponentType.three_winding_transformer,
674
- "uk_13_min",
675
- "pk_13_min/sn_3",
676
- data[ComponentType.three_winding_transformer]["uk_13"],
677
- )
678
- errors += _all_greater_or_equal(
679
- data,
680
- ComponentType.three_winding_transformer,
681
- "uk_23_min",
682
- "pk_23_min/sn_2",
683
- data[ComponentType.three_winding_transformer]["uk_23"],
684
- )
685
- errors += _all_greater_or_equal(
686
- data,
687
- ComponentType.three_winding_transformer,
688
- "uk_23_min",
689
- "pk_23_min/sn_3",
690
- data[ComponentType.three_winding_transformer]["uk_23"],
691
- )
692
- errors += _all_between(
693
- data,
694
- ComponentType.three_winding_transformer,
695
- "uk_12_min",
696
- 0,
697
- 1,
698
- data[ComponentType.three_winding_transformer]["uk_12"],
699
- )
700
- errors += _all_between(
701
- data,
702
- ComponentType.three_winding_transformer,
703
- "uk_13_min",
704
- 0,
705
- 1,
706
- data[ComponentType.three_winding_transformer]["uk_13"],
707
- )
708
- errors += _all_between(
709
- data,
710
- ComponentType.three_winding_transformer,
711
- "uk_23_min",
712
- 0,
713
- 1,
714
- data[ComponentType.three_winding_transformer]["uk_23"],
715
- )
716
- errors += _all_greater_or_equal(
717
- data,
718
- ComponentType.three_winding_transformer,
719
- "uk_12_max",
720
- "pk_12_max/sn_1",
721
- data[ComponentType.three_winding_transformer]["uk_12"],
722
- )
723
- errors += _all_greater_or_equal(
724
- data,
725
- ComponentType.three_winding_transformer,
726
- "uk_12_max",
727
- "pk_12_max/sn_2",
728
- data[ComponentType.three_winding_transformer]["uk_12"],
729
- )
730
- errors += _all_greater_or_equal(
731
- data,
732
- ComponentType.three_winding_transformer,
733
- "uk_13_max",
734
- "pk_13_max/sn_1",
735
- data[ComponentType.three_winding_transformer]["uk_13"],
736
- )
737
- errors += _all_greater_or_equal(
738
- data,
739
- ComponentType.three_winding_transformer,
740
- "uk_13_max",
741
- "pk_13_max/sn_3",
742
- data[ComponentType.three_winding_transformer]["uk_13"],
743
- )
744
- errors += _all_greater_or_equal(
745
- data,
746
- ComponentType.three_winding_transformer,
747
- "uk_23_max",
748
- "pk_23_max/sn_2",
749
- data[ComponentType.three_winding_transformer]["uk_23"],
750
- )
751
- errors += _all_greater_or_equal(
752
- data,
753
- ComponentType.three_winding_transformer,
754
- "uk_23_max",
755
- "pk_23_max/sn_3",
756
- data[ComponentType.three_winding_transformer]["uk_23"],
757
- )
758
- errors += _all_between(
759
- data,
760
- ComponentType.three_winding_transformer,
761
- "uk_12_max",
762
- 0,
763
- 1,
764
- data[ComponentType.three_winding_transformer]["uk_12"],
765
- )
766
- errors += _all_between(
767
- data,
768
- ComponentType.three_winding_transformer,
769
- "uk_13_max",
770
- 0,
771
- 1,
772
- data[ComponentType.three_winding_transformer]["uk_13"],
773
- )
774
- errors += _all_between(
775
- data,
776
- ComponentType.three_winding_transformer,
777
- "uk_23_max",
778
- 0,
779
- 1,
780
- data[ComponentType.three_winding_transformer]["uk_23"],
781
- )
782
- errors += _all_greater_than_or_equal_to_zero(
783
- data,
784
- ComponentType.three_winding_transformer,
785
- "pk_12_min",
786
- data[ComponentType.three_winding_transformer]["pk_12"],
787
- )
788
- errors += _all_greater_than_or_equal_to_zero(
789
- data,
790
- ComponentType.three_winding_transformer,
791
- "pk_13_min",
792
- data[ComponentType.three_winding_transformer]["pk_13"],
793
- )
794
- errors += _all_greater_than_or_equal_to_zero(
795
- data,
796
- ComponentType.three_winding_transformer,
797
- "pk_23_min",
798
- data[ComponentType.three_winding_transformer]["pk_23"],
799
- )
800
- errors += _all_greater_than_or_equal_to_zero(
801
- data,
802
- ComponentType.three_winding_transformer,
803
- "pk_12_max",
804
- data[ComponentType.three_winding_transformer]["pk_12"],
805
- )
806
- errors += _all_greater_than_or_equal_to_zero(
807
- data,
808
- ComponentType.three_winding_transformer,
809
- "pk_13_max",
810
- data[ComponentType.three_winding_transformer]["pk_13"],
811
- )
812
- errors += _all_greater_than_or_equal_to_zero(
813
- data,
814
- ComponentType.three_winding_transformer,
815
- "pk_23_max",
816
- data[ComponentType.three_winding_transformer]["pk_23"],
817
- )
818
- return errors
819
-
820
-
821
- def validate_appliance(data: SingleDataset, component: ComponentType) -> list[ValidationError]:
822
- errors = validate_base(data, component)
823
- errors += _all_boolean(data, component, "status")
824
- errors += _all_valid_ids(data, component, "node", ComponentType.node)
825
- return errors
826
-
827
-
828
- def validate_source(data: SingleDataset) -> list[ValidationError]:
829
- errors = validate_appliance(data, ComponentType.source)
830
- errors += _all_greater_than_zero(data, ComponentType.source, "u_ref")
831
- errors += _all_greater_than_zero(data, ComponentType.source, "sk")
832
- errors += _all_greater_than_or_equal_to_zero(data, ComponentType.source, "rx_ratio")
833
- errors += _all_greater_than_zero(data, ComponentType.source, "z01_ratio")
834
- return errors
835
-
836
-
837
- def validate_generic_load_gen(data: SingleDataset, component: ComponentType) -> list[ValidationError]:
838
- errors = validate_appliance(data, component)
839
- errors += _all_valid_enum_values(data, component, "type", LoadGenType)
840
- return errors
841
-
842
-
843
- def validate_shunt(data: SingleDataset) -> list[ValidationError]:
844
- errors = validate_appliance(data, ComponentType.shunt)
845
- return errors
846
-
847
-
848
- def validate_generic_voltage_sensor(data: SingleDataset, component: ComponentType) -> list[ValidationError]:
849
- errors = validate_base(data, component)
850
- errors += _all_greater_than_zero(data, component, "u_sigma")
851
- errors += _all_greater_than_zero(data, component, "u_measured")
852
- errors += _all_valid_ids(data, component, "measured_object", ComponentType.node)
853
- return errors
854
-
855
-
856
- def validate_generic_power_sensor(data: SingleDataset, component: ComponentType) -> list[ValidationError]:
857
- errors = validate_base(data, component)
858
- errors += _all_greater_than_zero(data, component, "power_sigma")
859
- errors += _all_valid_enum_values(data, component, "measured_terminal_type", MeasuredTerminalType)
860
- errors += _all_valid_ids(
861
- data,
862
- component,
863
- field="measured_object",
864
- ref_components=[
865
- ComponentType.node,
866
- ComponentType.line,
867
- ComponentType.generic_branch,
868
- ComponentType.transformer,
869
- ComponentType.three_winding_transformer,
870
- ComponentType.source,
871
- ComponentType.shunt,
872
- ComponentType.sym_load,
873
- ComponentType.asym_load,
874
- ComponentType.sym_gen,
875
- ComponentType.asym_gen,
876
- ],
877
- )
878
- errors += _all_valid_ids(
879
- data,
880
- component,
881
- field="measured_object",
882
- ref_components=[ComponentType.line, ComponentType.generic_branch, ComponentType.transformer],
883
- measured_terminal_type=MeasuredTerminalType.branch_from,
884
- )
885
- errors += _all_valid_ids(
886
- data,
887
- component,
888
- field="measured_object",
889
- ref_components=[ComponentType.line, ComponentType.generic_branch, ComponentType.transformer],
890
- measured_terminal_type=MeasuredTerminalType.branch_to,
891
- )
892
- errors += _all_valid_ids(
893
- data,
894
- component,
895
- field="measured_object",
896
- ref_components=ComponentType.source,
897
- measured_terminal_type=MeasuredTerminalType.source,
898
- )
899
- errors += _all_valid_ids(
900
- data,
901
- component,
902
- field="measured_object",
903
- ref_components=ComponentType.shunt,
904
- measured_terminal_type=MeasuredTerminalType.shunt,
905
- )
906
- errors += _all_valid_ids(
907
- data,
908
- component,
909
- field="measured_object",
910
- ref_components=[ComponentType.sym_load, ComponentType.asym_load],
911
- measured_terminal_type=MeasuredTerminalType.load,
912
- )
913
- errors += _all_valid_ids(
914
- data,
915
- component,
916
- field="measured_object",
917
- ref_components=[ComponentType.sym_gen, ComponentType.asym_gen],
918
- measured_terminal_type=MeasuredTerminalType.generator,
919
- )
920
- errors += _all_valid_ids(
921
- data,
922
- component,
923
- field="measured_object",
924
- ref_components=ComponentType.three_winding_transformer,
925
- measured_terminal_type=MeasuredTerminalType.branch3_1,
926
- )
927
- errors += _all_valid_ids(
928
- data,
929
- component,
930
- field="measured_object",
931
- ref_components=ComponentType.three_winding_transformer,
932
- measured_terminal_type=MeasuredTerminalType.branch3_2,
933
- )
934
- errors += _all_valid_ids(
935
- data,
936
- component,
937
- field="measured_object",
938
- ref_components=ComponentType.three_winding_transformer,
939
- measured_terminal_type=MeasuredTerminalType.branch3_3,
940
- )
941
- errors += _all_valid_ids(
942
- data,
943
- component,
944
- field="measured_object",
945
- ref_components=ComponentType.node,
946
- measured_terminal_type=MeasuredTerminalType.node,
947
- )
948
- if component in ("sym_power_sensor", "asym_power_sensor"):
949
- errors += _valid_p_q_sigma(data, component)
950
-
951
- return errors
952
-
953
-
954
- def validate_fault(data: SingleDataset) -> list[ValidationError]:
955
- errors = validate_base(data, ComponentType.fault)
956
- errors += _all_boolean(data, ComponentType.fault, "status")
957
- errors += _all_valid_enum_values(data, ComponentType.fault, "fault_type", FaultType)
958
- errors += _all_valid_enum_values(data, ComponentType.fault, "fault_phase", FaultPhase)
959
- errors += _all_valid_fault_phases(data, ComponentType.fault, "fault_type", "fault_phase")
960
- errors += _all_valid_ids(data, ComponentType.fault, field="fault_object", ref_components=ComponentType.node)
961
- errors += _all_greater_than_or_equal_to_zero(data, ComponentType.fault, "r_f")
962
- errors += _all_enabled_identical(data, ComponentType.fault, "fault_type", "status")
963
- errors += _all_enabled_identical(data, ComponentType.fault, "fault_phase", "status")
964
- return errors
965
-
966
-
967
- def validate_regulator(data: SingleDataset, component: ComponentType) -> list[ValidationError]:
968
- errors = validate_base(data, component)
969
- errors += _all_valid_ids(
970
- data,
971
- component,
972
- field="regulated_object",
973
- ref_components=[ComponentType.transformer, ComponentType.three_winding_transformer],
974
- )
975
- return errors
976
-
977
-
978
- def validate_transformer_tap_regulator(data: SingleDataset) -> list[ValidationError]:
979
- errors = validate_regulator(data, ComponentType.transformer_tap_regulator)
980
- errors += _all_boolean(data, ComponentType.transformer_tap_regulator, "status")
981
- errors += _all_valid_enum_values(
982
- data, ComponentType.transformer_tap_regulator, "control_side", [BranchSide, Branch3Side]
983
- )
984
- errors += _all_valid_associated_enum_values(
985
- data,
986
- ComponentType.transformer_tap_regulator,
987
- "control_side",
988
- "regulated_object",
989
- [ComponentType.transformer],
990
- [BranchSide],
991
- )
992
- errors += _all_valid_associated_enum_values(
993
- data,
994
- ComponentType.transformer_tap_regulator,
995
- "control_side",
996
- "regulated_object",
997
- [ComponentType.three_winding_transformer],
998
- [Branch3Side],
999
- )
1000
- errors += _all_greater_than_or_equal_to_zero(data, ComponentType.transformer_tap_regulator, "u_set")
1001
- errors += _all_greater_than_zero(data, ComponentType.transformer_tap_regulator, "u_band")
1002
- errors += _all_greater_than_or_equal_to_zero(
1003
- data, ComponentType.transformer_tap_regulator, "line_drop_compensation_r", 0.0
1004
- )
1005
- errors += _all_greater_than_or_equal_to_zero(
1006
- data, ComponentType.transformer_tap_regulator, "line_drop_compensation_x", 0.0
1007
- )
1008
- errors += _all_supported_tap_control_side(
1009
- data,
1010
- ComponentType.transformer_tap_regulator,
1011
- "control_side",
1012
- "regulated_object",
1013
- [(ComponentType.transformer, "tap_side"), (ComponentType.three_winding_transformer, "tap_side")],
1014
- )
1015
- return errors
1
+ # SPDX-FileCopyrightText: Contributors to the Power Grid Model project <powergridmodel@lfenergy.org>
2
+ #
3
+ # SPDX-License-Identifier: MPL-2.0
4
+
5
+ """
6
+ Power Grid Model Validation Functions.
7
+
8
+ Although all functions are 'public', you probably only need validate_input_data() and validate_batch_data().
9
+
10
+ """
11
+
12
+ import copy
13
+ from collections.abc import Sized as ABCSized
14
+ from itertools import chain
15
+ from typing import Literal
16
+
17
+ import numpy as np
18
+
19
+ from power_grid_model._core.dataset_definitions import ComponentType, DatasetType, _map_to_component_types
20
+ from power_grid_model._core.power_grid_meta import power_grid_meta_data
21
+ from power_grid_model._core.utils import (
22
+ compatibility_convert_row_columnar_dataset as _compatibility_convert_row_columnar_dataset,
23
+ convert_batch_dataset_to_batch_list as _convert_batch_dataset_to_batch_list,
24
+ )
25
+ from power_grid_model.data_types import BatchDataset, Dataset, SingleDataset
26
+ from power_grid_model.enum import (
27
+ AngleMeasurementType,
28
+ Branch3Side,
29
+ BranchSide,
30
+ CalculationType,
31
+ FaultPhase,
32
+ FaultType,
33
+ LoadGenType,
34
+ MeasuredTerminalType,
35
+ WindingType,
36
+ )
37
+ from power_grid_model.validation._rules import (
38
+ all_between as _all_between,
39
+ all_between_or_at as _all_between_or_at,
40
+ all_boolean as _all_boolean,
41
+ all_cross_unique as _all_cross_unique,
42
+ all_enabled_identical as _all_enabled_identical,
43
+ all_finite as _all_finite,
44
+ all_greater_or_equal as _all_greater_or_equal,
45
+ all_greater_than_or_equal_to_zero as _all_greater_than_or_equal_to_zero,
46
+ all_greater_than_zero as _all_greater_than_zero,
47
+ all_in_valid_values as _all_in_valid_values,
48
+ all_less_than as _all_less_than,
49
+ all_not_two_values_equal as _all_not_two_values_equal,
50
+ all_not_two_values_zero as _all_not_two_values_zero,
51
+ all_same_current_angle_measurement_type_on_terminal as _all_same_current_angle_measurement_type_on_terminal,
52
+ all_same_sensor_type_on_same_terminal as _all_same_sensor_type_on_same_terminal,
53
+ all_unique as _all_unique,
54
+ all_valid_associated_enum_values as _all_valid_associated_enum_values,
55
+ all_valid_clocks as _all_valid_clocks,
56
+ all_valid_enum_values as _all_valid_enum_values,
57
+ all_valid_fault_phases as _all_valid_fault_phases,
58
+ all_valid_ids as _all_valid_ids,
59
+ any_voltage_angle_measurement_if_global_current_measurement as _any_voltage_angle_measurement_if_global_current_measurement, # noqa: E501
60
+ ids_valid_in_update_data_set as _ids_valid_in_update_data_set,
61
+ no_strict_subset_missing as _no_strict_subset_missing,
62
+ none_missing as _none_missing,
63
+ not_all_missing as _not_all_missing,
64
+ valid_p_q_sigma as _valid_p_q_sigma,
65
+ )
66
+ from power_grid_model.validation.errors import (
67
+ IdNotInDatasetError,
68
+ InvalidIdError,
69
+ MissingValueError,
70
+ MultiComponentNotUniqueError,
71
+ ValidationError,
72
+ )
73
+ from power_grid_model.validation.utils import _update_input_data
74
+
75
+
76
+ def validate_input_data(
77
+ input_data: SingleDataset, calculation_type: CalculationType | None = None, symmetric: bool = True
78
+ ) -> list[ValidationError] | None:
79
+ """
80
+ Validates the entire input dataset:
81
+
82
+ 1. Is the data structure correct? (checking data types and numpy array shapes)
83
+ 2. Are all required values provided? (checking NaNs)
84
+ 3. Are all ID's unique? (checking object identifiers across all components)
85
+ 4. Are the supplied values valid? (checking limits and other logic as described in "Graph Data Model")
86
+
87
+ Args:
88
+ input_data: A power-grid-model input dataset
89
+ calculation_type: Supply a calculation method, to allow missing values for unused fields
90
+ symmetric: A boolean to state whether input data will be used for a symmetric or asymmetric calculation
91
+
92
+ Returns:
93
+ None if the data is valid, or a list containing all validation errors.
94
+
95
+ Raises:
96
+ Error: KeyError | TypeError | ValueError: if the data structure is invalid.
97
+ """
98
+ input_data = _map_to_component_types(input_data)
99
+
100
+ # Convert to row based if in columnar or mixed format format
101
+ row_input_data = _compatibility_convert_row_columnar_dataset(input_data, None, DatasetType.input)
102
+
103
+ # A deep copy is made of the input data, since default values will be added in the validation process
104
+ input_data_copy = copy.deepcopy(row_input_data)
105
+
106
+ assert_valid_data_structure(input_data_copy, DatasetType.input)
107
+
108
+ errors: list[ValidationError] = []
109
+ errors += validate_required_values(input_data_copy, calculation_type, symmetric)
110
+ errors += validate_unique_ids_across_components(input_data_copy)
111
+ errors += validate_values(input_data_copy, calculation_type)
112
+ return errors if errors else None
113
+
114
+
115
+ def validate_batch_data(
116
+ input_data: SingleDataset,
117
+ update_data: BatchDataset,
118
+ calculation_type: CalculationType | None = None,
119
+ symmetric: bool = True,
120
+ ) -> dict[int, list[ValidationError]] | None:
121
+ """
122
+ The input dataset is validated:
123
+
124
+ 1. Is the data structure correct? (checking data types and numpy array shapes)
125
+ 2. Are all input data ID's unique? (checking object identifiers across all components)
126
+
127
+ For each batch the update data is validated:
128
+ 3. Is the update data structure correct? (checking data types and numpy array shapes)
129
+ 4. Are all update ID's valid? (checking object identifiers across update and input data)
130
+
131
+ Then (for each batch independently) the input dataset is updated with the batch's update data and validated:
132
+ 5. Are all required values provided? (checking NaNs)
133
+ 6. Are the supplied values valid? (checking limits and other logic as described in "Graph Data Model")
134
+
135
+ Args:
136
+ input_data: A power-grid-model input dataset
137
+ update_data: A power-grid-model update dataset (one or more batches)
138
+ calculation_type: Supply a calculation method, to allow missing values for unused fields
139
+ symmetric: A boolean to state whether input data will be used for a symmetric or asymmetric calculation
140
+
141
+ Returns:
142
+ None if the data is valid, or a dictionary containing all validation errors,
143
+ where the key is the batch number (0-indexed).
144
+
145
+ Raises:
146
+ Error: KeyError | TypeError | ValueError: if the data structure is invalid.
147
+ """
148
+ input_data = _map_to_component_types(input_data)
149
+ update_data = _map_to_component_types(update_data)
150
+
151
+ # Convert to row based if in columnar or mixed format
152
+ row_input_data = _compatibility_convert_row_columnar_dataset(input_data, None, DatasetType.input)
153
+
154
+ # A deep copy is made of the input data, since default values will be added in the validation process
155
+ input_data_copy = copy.deepcopy(row_input_data)
156
+ assert_valid_data_structure(input_data_copy, DatasetType.input)
157
+
158
+ input_errors: list[ValidationError] = list(validate_unique_ids_across_components(input_data_copy))
159
+
160
+ batch_data = _convert_batch_dataset_to_batch_list(update_data, DatasetType.update)
161
+
162
+ errors = {}
163
+ for batch, batch_update_data in enumerate(batch_data):
164
+ row_update_data = _compatibility_convert_row_columnar_dataset(batch_update_data, None, DatasetType.update)
165
+ assert_valid_data_structure(row_update_data, DatasetType.update)
166
+ id_errors: list[IdNotInDatasetError | InvalidIdError] = validate_ids(row_update_data, input_data_copy)
167
+
168
+ batch_errors = input_errors + id_errors
169
+
170
+ if not id_errors:
171
+ merged_data = _update_input_data(input_data_copy, row_update_data)
172
+ batch_errors += validate_required_values(merged_data, calculation_type, symmetric)
173
+ batch_errors += validate_values(merged_data, calculation_type)
174
+
175
+ if batch_errors:
176
+ errors[batch] = batch_errors
177
+
178
+ return errors if errors else None
179
+
180
+
181
+ def assert_valid_data_structure(data: Dataset, data_type: DatasetType) -> None:
182
+ """
183
+ Checks if all component names are valid and if the data inside the component matches the required Numpy
184
+ structured array as defined in the Power Grid Model meta data.
185
+
186
+ Args:
187
+ data: A power-grid-model input/update dataset
188
+ data_type: 'input' or 'update'
189
+
190
+ Raises:
191
+ Error: KeyError, TypeError
192
+
193
+ """
194
+ if data_type not in {DatasetType.input, DatasetType.update}:
195
+ raise KeyError(f"Unexpected data type '{data_type}' (should be 'input' or 'update')")
196
+
197
+ component_dtype = {component: meta.dtype for component, meta in power_grid_meta_data[data_type].items()}
198
+ for component, array in data.items():
199
+ # Check if component name is valid
200
+ if component not in component_dtype:
201
+ raise KeyError(f"Unknown component '{component}' in {data_type} data.")
202
+
203
+ # Check if component definition is as expected
204
+ dtype = component_dtype[component]
205
+ if isinstance(array, np.ndarray):
206
+ if array.dtype != dtype:
207
+ if not hasattr(array.dtype, "names") or not array.dtype.names:
208
+ raise TypeError(
209
+ f"Unexpected Numpy array ({array.dtype}) for '{component}' {data_type} data "
210
+ "(should be a Numpy structured array)."
211
+ )
212
+ raise TypeError(
213
+ f"Unexpected Numpy structured array; (expected = {dtype}, actual = {array.dtype}). "
214
+ f"For component '{component}'."
215
+ )
216
+ else:
217
+ raise TypeError(
218
+ f"Unexpected data type {type(array).__name__} for '{component}' {data_type} data "
219
+ "(should be a Numpy structured array)."
220
+ )
221
+
222
+
223
+ def validate_unique_ids_across_components(data: SingleDataset) -> list[MultiComponentNotUniqueError]:
224
+ """
225
+ Checks if all ids in the input dataset are unique
226
+
227
+ Args:
228
+ data: A power-grid-model input dataset
229
+
230
+ Returns:
231
+ An empty list if all ids are unique, or a list of MultiComponentNotUniqueErrors for all components that
232
+ have non-unique ids
233
+ """
234
+ return _all_cross_unique(data, [(component, "id") for component in data])
235
+
236
+
237
+ def validate_ids(update_data: SingleDataset, input_data: SingleDataset) -> list[IdNotInDatasetError | InvalidIdError]:
238
+ """
239
+ Checks if all ids of the components in the update data:
240
+ - exist and match those in the input data
241
+ - are not present but qualifies for optional id
242
+
243
+ This function should be called for every update dataset in a batch set
244
+
245
+ Args:
246
+ update_data: A single update dataset
247
+ input_data: Input dataset
248
+
249
+ Returns:
250
+ An empty list if all update data ids are valid, or a list of IdNotInDatasetErrors or InvalidIdError for
251
+ all update components that have invalid ids
252
+
253
+ """
254
+ errors = (
255
+ _ids_valid_in_update_data_set(update_data, input_data, component, DatasetType.update)
256
+ for component in update_data
257
+ )
258
+ return list(chain(*errors))
259
+
260
+
261
+ def validate_required_values( # noqa: PLR0915
262
+ data: SingleDataset, calculation_type: CalculationType | None = None, symmetric: bool = True
263
+ ) -> list[MissingValueError]:
264
+ """
265
+ Checks if all required data is available.
266
+
267
+ Args:
268
+ data: A power-grid-model input dataset
269
+ calculation_type: Supply a calculation method, to allow missing values for unused fields
270
+ symmetric: A boolean to state whether input data will be used for a symmetric or asymmetric calculation
271
+
272
+ Returns:
273
+ An empty list if all required data is available, or a list of MissingValueErrors.
274
+ """
275
+ # Base
276
+ required: dict[ComponentType | str, list[str]] = {"base": ["id"]}
277
+
278
+ # Nodes
279
+ required[ComponentType.node] = required["base"] + ["u_rated"]
280
+
281
+ # Branches
282
+ required["branch"] = required["base"] + ["from_node", "to_node", "from_status", "to_status"]
283
+ required[ComponentType.link] = required["branch"].copy()
284
+ required[ComponentType.line] = required["branch"] + ["r1", "x1", "c1", "tan1"]
285
+ required[ComponentType.asym_line] = required["branch"] + [
286
+ "r_aa",
287
+ "r_ba",
288
+ "r_bb",
289
+ "r_ca",
290
+ "r_cb",
291
+ "r_cc",
292
+ "x_aa",
293
+ "x_ba",
294
+ "x_bb",
295
+ "x_ca",
296
+ "x_cb",
297
+ "x_cc",
298
+ ]
299
+ required[ComponentType.transformer] = required["branch"] + [
300
+ "u1",
301
+ "u2",
302
+ "sn",
303
+ "uk",
304
+ "pk",
305
+ "i0",
306
+ "p0",
307
+ "winding_from",
308
+ "winding_to",
309
+ "clock",
310
+ "tap_side",
311
+ "tap_min",
312
+ "tap_max",
313
+ "tap_size",
314
+ ]
315
+ # Branch3
316
+ required["branch3"] = required["base"] + ["node_1", "node_2", "node_3", "status_1", "status_2", "status_3"]
317
+ required[ComponentType.three_winding_transformer] = required["branch3"] + [
318
+ "u1",
319
+ "u2",
320
+ "u3",
321
+ "sn_1",
322
+ "sn_2",
323
+ "sn_3",
324
+ "uk_12",
325
+ "uk_13",
326
+ "uk_23",
327
+ "pk_12",
328
+ "pk_13",
329
+ "pk_23",
330
+ "i0",
331
+ "p0",
332
+ "winding_1",
333
+ "winding_2",
334
+ "winding_3",
335
+ "clock_12",
336
+ "clock_13",
337
+ "tap_side",
338
+ "tap_min",
339
+ "tap_max",
340
+ "tap_size",
341
+ ]
342
+
343
+ # Regulators
344
+ required["regulator"] = required["base"] + ["regulated_object", "status"]
345
+ required[ComponentType.transformer_tap_regulator] = required["regulator"]
346
+ if calculation_type is None or calculation_type == CalculationType.power_flow:
347
+ required[ComponentType.transformer_tap_regulator] += ["control_side", "u_set", "u_band"]
348
+
349
+ # Appliances
350
+ required["appliance"] = required["base"] + ["node", "status"]
351
+ required[ComponentType.source] = required["appliance"].copy()
352
+ if calculation_type is None or calculation_type == CalculationType.power_flow:
353
+ required[ComponentType.source] += ["u_ref"]
354
+ required[ComponentType.shunt] = required["appliance"] + ["g1", "b1"]
355
+ required["generic_load_gen"] = required["appliance"] + ["type"]
356
+ if calculation_type is None or calculation_type == CalculationType.power_flow:
357
+ required["generic_load_gen"] += ["p_specified", "q_specified"]
358
+ required[ComponentType.sym_load] = required["generic_load_gen"].copy()
359
+ required[ComponentType.asym_load] = required["generic_load_gen"].copy()
360
+ required[ComponentType.sym_gen] = required["generic_load_gen"].copy()
361
+ required[ComponentType.asym_gen] = required["generic_load_gen"].copy()
362
+
363
+ # Sensors
364
+ required["sensor"] = required["base"] + ["measured_object"]
365
+ required["voltage_sensor"] = required["sensor"].copy()
366
+ required["power_sensor"] = required["sensor"] + ["measured_terminal_type"]
367
+ required["current_sensor"] = required["sensor"] + ["measured_terminal_type", "angle_measurement_type"]
368
+ if calculation_type is None or calculation_type == CalculationType.state_estimation:
369
+ required["voltage_sensor"] += ["u_sigma", "u_measured"]
370
+ required["power_sensor"] += ["p_measured", "q_measured"] # power_sigma, p_sigma and q_sigma are checked later
371
+ required["current_sensor"] += ["i_sigma", "i_angle_sigma", "i_measured", "i_angle_measured"]
372
+ required[ComponentType.sym_voltage_sensor] = required["voltage_sensor"].copy()
373
+ required[ComponentType.asym_voltage_sensor] = required["voltage_sensor"].copy()
374
+ required[ComponentType.sym_current_sensor] = required["current_sensor"].copy()
375
+ required[ComponentType.asym_current_sensor] = required["current_sensor"].copy()
376
+
377
+ # Different requirements for individual sensors. Avoid shallow copy.
378
+ for sensor_type in (ComponentType.sym_power_sensor, ComponentType.asym_power_sensor):
379
+ required[sensor_type] = required["power_sensor"].copy()
380
+
381
+ # Faults
382
+ required[ComponentType.fault] = required["base"] + ["fault_object"]
383
+ asym_sc = False
384
+ if calculation_type is None or calculation_type == CalculationType.short_circuit:
385
+ required[ComponentType.fault] += ["status", "fault_type"]
386
+ if ComponentType.fault in data:
387
+ for elem in data[ComponentType.fault]["fault_type"]:
388
+ if elem not in (FaultType.three_phase, FaultType.nan):
389
+ asym_sc = True
390
+ break
391
+
392
+ if not symmetric or asym_sc:
393
+ required[ComponentType.line] += ["r0", "x0", "c0", "tan0"]
394
+ required[ComponentType.shunt] += ["g0", "b0"]
395
+
396
+ errors = _validate_required_in_data(data, required)
397
+
398
+ if calculation_type is None or calculation_type == CalculationType.state_estimation:
399
+ errors += _validate_required_power_sigma_or_p_q_sigma(data, ComponentType.sym_power_sensor)
400
+ errors += _validate_required_power_sigma_or_p_q_sigma(data, ComponentType.asym_power_sensor)
401
+
402
+ return errors
403
+
404
+
405
+ def _validate_required_in_data(data: SingleDataset, required: dict[ComponentType | str, list[str]]):
406
+ """
407
+ Checks if all required data is available.
408
+
409
+ Args:
410
+ data: A power-grid-model input dataset
411
+ required: a list of required fields (a list of str), per component when applicaple (a list of str or str lists)
412
+
413
+ Returns:
414
+ An empty list if all required data is available, or a list of MissingValueErrors.
415
+ """
416
+
417
+ def is_valid_component(data, component):
418
+ return (
419
+ not (isinstance(data[component], np.ndarray) and data[component].size == 0)
420
+ and data[component] is not None
421
+ and isinstance(data[component], ABCSized)
422
+ )
423
+
424
+ results: list[MissingValueError] = []
425
+
426
+ for component in data:
427
+ if is_valid_component(data, component):
428
+ items = required.get(component, [])
429
+ results += _none_missing(data, component, items)
430
+
431
+ return results
432
+
433
+
434
+ def _validate_required_power_sigma_or_p_q_sigma(
435
+ data: SingleDataset,
436
+ power_sensor: Literal[ComponentType.sym_power_sensor, ComponentType.asym_power_sensor],
437
+ ) -> list[MissingValueError]:
438
+ """
439
+ Check that either `p_sigma` and `q_sigma` are all provided, or that `power_sigma` is provided.
440
+
441
+ Args:
442
+ data: SingleDataset, pgm data
443
+ sensor: the power sensor type, either ComponentType.sym_power_sensor or ComponentType.asym_power_sensor
444
+ """
445
+ result: list[MissingValueError] = []
446
+
447
+ if power_sensor in data:
448
+ sensor_data = data[power_sensor]
449
+ p_sigma = sensor_data["p_sigma"]
450
+ q_sigma = sensor_data["q_sigma"]
451
+
452
+ asym_axes = tuple(range(sensor_data.ndim, p_sigma.ndim))
453
+ all_pq_sigma_missing_mask = np.all(np.isnan(p_sigma), axis=asym_axes) & np.all(
454
+ np.isnan(q_sigma), axis=asym_axes
455
+ )
456
+
457
+ result += _validate_required_in_data(
458
+ {power_sensor: sensor_data[all_pq_sigma_missing_mask]}, required={power_sensor: ["power_sigma"]}
459
+ )
460
+ result += _validate_required_in_data(
461
+ {power_sensor: sensor_data[~all_pq_sigma_missing_mask]}, required={power_sensor: ["p_sigma", "q_sigma"]}
462
+ )
463
+
464
+ return result
465
+
466
+
467
+ def validate_values(data: SingleDataset, calculation_type: CalculationType | None = None) -> list[ValidationError]:
468
+ """
469
+ For each component supplied in the data, call the appropriate validation function
470
+
471
+ Args:
472
+ data: A power-grid-model input dataset
473
+ calculation_type: Supply a calculation method, to allow missing values for unused fields
474
+
475
+ Returns:
476
+ An empty list if all required data is valid, or a list of ValidationErrors.
477
+
478
+ """
479
+ errors: list[ValidationError] = list(
480
+ _all_finite(
481
+ data=data,
482
+ exceptions={
483
+ ComponentType.sym_power_sensor: ["power_sigma", "p_sigma", "q_sigma"],
484
+ ComponentType.asym_power_sensor: ["power_sigma", "p_sigma", "q_sigma"],
485
+ ComponentType.sym_voltage_sensor: ["u_sigma"],
486
+ ComponentType.asym_voltage_sensor: ["u_sigma"],
487
+ ComponentType.sym_current_sensor: ["i_sigma", "i_angle_sigma"],
488
+ ComponentType.asym_current_sensor: ["i_sigma", "i_angle_sigma"],
489
+ },
490
+ )
491
+ )
492
+
493
+ component_validators = {
494
+ ComponentType.node: validate_node,
495
+ ComponentType.line: validate_line,
496
+ ComponentType.asym_line: validate_asym_line,
497
+ ComponentType.link: lambda d: validate_branch(d, ComponentType.link),
498
+ ComponentType.generic_branch: validate_generic_branch,
499
+ ComponentType.transformer: validate_transformer,
500
+ ComponentType.three_winding_transformer: validate_three_winding_transformer,
501
+ ComponentType.source: validate_source,
502
+ ComponentType.sym_load: lambda d: validate_generic_load_gen(d, ComponentType.sym_load),
503
+ ComponentType.sym_gen: lambda d: validate_generic_load_gen(d, ComponentType.sym_gen),
504
+ ComponentType.asym_load: lambda d: validate_generic_load_gen(d, ComponentType.asym_load),
505
+ ComponentType.asym_gen: lambda d: validate_generic_load_gen(d, ComponentType.asym_gen),
506
+ ComponentType.shunt: validate_shunt,
507
+ }
508
+
509
+ for component, validator in component_validators.items():
510
+ if component in data:
511
+ errors += validator(data)
512
+
513
+ if calculation_type in (None, CalculationType.state_estimation):
514
+ if ComponentType.sym_voltage_sensor in data:
515
+ errors += validate_generic_voltage_sensor(data, ComponentType.sym_voltage_sensor)
516
+ if ComponentType.asym_voltage_sensor in data:
517
+ errors += validate_generic_voltage_sensor(data, ComponentType.asym_voltage_sensor)
518
+ if ComponentType.sym_power_sensor in data:
519
+ errors += validate_generic_power_sensor(data, ComponentType.sym_power_sensor)
520
+ if ComponentType.asym_power_sensor in data:
521
+ errors += validate_generic_power_sensor(data, ComponentType.asym_power_sensor)
522
+ if ComponentType.sym_current_sensor in data:
523
+ errors += validate_generic_current_sensor(data, ComponentType.sym_current_sensor)
524
+ if ComponentType.asym_current_sensor in data:
525
+ errors += validate_generic_current_sensor(data, ComponentType.asym_current_sensor)
526
+
527
+ errors += validate_no_mixed_sensors_on_same_terminal(data)
528
+
529
+ if calculation_type in (None, CalculationType.short_circuit) and ComponentType.fault in data:
530
+ errors += validate_fault(data)
531
+
532
+ if calculation_type in (None, CalculationType.power_flow) and ComponentType.transformer_tap_regulator in data:
533
+ errors += validate_transformer_tap_regulator(data)
534
+
535
+ return errors
536
+
537
+
538
+ def validate_base(data: SingleDataset, component: ComponentType) -> list[ValidationError]:
539
+ errors: list[ValidationError] = list(_all_unique(data, component, "id"))
540
+ return errors
541
+
542
+
543
+ def validate_node(data: SingleDataset) -> list[ValidationError]:
544
+ errors = validate_base(data, ComponentType.node)
545
+ errors += _all_greater_than_zero(data, ComponentType.node, "u_rated")
546
+ return errors
547
+
548
+
549
+ def validate_branch(data: SingleDataset, component: ComponentType) -> list[ValidationError]:
550
+ errors = validate_base(data, component)
551
+ errors += _all_valid_ids(data, component, "from_node", ComponentType.node)
552
+ errors += _all_valid_ids(data, component, "to_node", ComponentType.node)
553
+ errors += _all_not_two_values_equal(data, component, "to_node", "from_node")
554
+ errors += _all_boolean(data, component, "from_status")
555
+ errors += _all_boolean(data, component, "to_status")
556
+ return errors
557
+
558
+
559
+ def validate_line(data: SingleDataset) -> list[ValidationError]:
560
+ errors = validate_branch(data, ComponentType.line)
561
+ errors += _all_not_two_values_zero(data, ComponentType.line, "r1", "x1")
562
+ errors += _all_not_two_values_zero(data, ComponentType.line, "r0", "x0")
563
+ errors += _all_greater_than_zero(data, ComponentType.line, "i_n")
564
+ return errors
565
+
566
+
567
+ def validate_asym_line(data: SingleDataset) -> list[ValidationError]:
568
+ errors = validate_branch(data, ComponentType.asym_line)
569
+ errors += _all_greater_than_zero(data, ComponentType.asym_line, "i_n")
570
+ required_fields = ["r_aa", "r_ba", "r_bb", "r_ca", "r_cb", "r_cc", "x_aa", "x_ba", "x_bb", "x_ca", "x_cb", "x_cc"]
571
+ optional_r_matrix_fields = ["r_na", "r_nb", "r_nc", "r_nn"]
572
+ optional_x_matrix_fields = ["x_na", "x_nb", "x_nc", "x_nn"]
573
+ required_c_matrix_fields = ["c_aa", "c_ba", "c_bb", "c_ca", "c_cb", "c_cc"]
574
+ c_fields = ["c0", "c1"]
575
+ for field in (
576
+ required_fields + optional_r_matrix_fields + optional_x_matrix_fields + required_c_matrix_fields + c_fields
577
+ ):
578
+ errors += _all_greater_than_zero(data, ComponentType.asym_line, field)
579
+
580
+ errors += _no_strict_subset_missing(
581
+ data, optional_r_matrix_fields + optional_x_matrix_fields, ComponentType.asym_line
582
+ )
583
+ errors += _no_strict_subset_missing(data, required_c_matrix_fields, ComponentType.asym_line)
584
+ errors += _no_strict_subset_missing(data, c_fields, ComponentType.asym_line)
585
+ errors += _not_all_missing(data, required_c_matrix_fields + c_fields, ComponentType.asym_line)
586
+
587
+ return errors
588
+
589
+
590
+ def validate_generic_branch(data: SingleDataset) -> list[ValidationError]:
591
+ errors = validate_branch(data, ComponentType.generic_branch)
592
+ errors += _all_greater_than_zero(data, ComponentType.generic_branch, "k")
593
+ errors += _all_greater_than_or_equal_to_zero(data, ComponentType.generic_branch, "sn")
594
+ return errors
595
+
596
+
597
+ def validate_transformer(data: SingleDataset) -> list[ValidationError]:
598
+ errors = validate_branch(data, ComponentType.transformer)
599
+ errors += _all_greater_than_zero(data, ComponentType.transformer, "u1")
600
+ errors += _all_greater_than_zero(data, ComponentType.transformer, "u2")
601
+ errors += _all_greater_than_zero(data, ComponentType.transformer, "sn")
602
+ errors += _all_greater_or_equal(data, ComponentType.transformer, "uk", "pk/sn")
603
+ errors += _all_between(data, ComponentType.transformer, "uk", 0, 1)
604
+ errors += _all_greater_than_or_equal_to_zero(data, ComponentType.transformer, "pk")
605
+ errors += _all_greater_or_equal(data, ComponentType.transformer, "i0", "p0/sn")
606
+ errors += _all_less_than(data, ComponentType.transformer, "i0", 1)
607
+ errors += _all_greater_than_or_equal_to_zero(data, ComponentType.transformer, "p0")
608
+ errors += _all_valid_enum_values(data, ComponentType.transformer, "winding_from", WindingType)
609
+ errors += _all_valid_enum_values(data, ComponentType.transformer, "winding_to", WindingType)
610
+ errors += _all_between_or_at(data, ComponentType.transformer, "clock", -12, 12)
611
+ errors += _all_valid_clocks(data, ComponentType.transformer, "clock", "winding_from", "winding_to")
612
+ errors += _all_valid_enum_values(data, ComponentType.transformer, "tap_side", BranchSide)
613
+ errors += _all_between_or_at(
614
+ data, ComponentType.transformer, "tap_pos", "tap_min", "tap_max", data[ComponentType.transformer]["tap_nom"], 0
615
+ )
616
+ errors += _all_between_or_at(data, ComponentType.transformer, "tap_nom", "tap_min", "tap_max", 0)
617
+ errors += _all_greater_than_or_equal_to_zero(data, ComponentType.transformer, "tap_size")
618
+ errors += _all_greater_or_equal(
619
+ data, ComponentType.transformer, "uk_min", "pk_min/sn", data[ComponentType.transformer]["uk"]
620
+ )
621
+ errors += _all_between(data, ComponentType.transformer, "uk_min", 0, 1, data[ComponentType.transformer]["uk"])
622
+ errors += _all_greater_or_equal(
623
+ data, ComponentType.transformer, "uk_max", "pk_max/sn", data[ComponentType.transformer]["uk"]
624
+ )
625
+ errors += _all_between(data, ComponentType.transformer, "uk_max", 0, 1, data[ComponentType.transformer]["uk"])
626
+ errors += _all_greater_than_or_equal_to_zero(
627
+ data, ComponentType.transformer, "pk_min", data[ComponentType.transformer]["pk"]
628
+ )
629
+ errors += _all_greater_than_or_equal_to_zero(
630
+ data, ComponentType.transformer, "pk_max", data[ComponentType.transformer]["pk"]
631
+ )
632
+ return errors
633
+
634
+
635
+ def validate_branch3(data: SingleDataset, component: ComponentType) -> list[ValidationError]:
636
+ errors = validate_base(data, component)
637
+ errors += _all_valid_ids(data, component, "node_1", ComponentType.node)
638
+ errors += _all_valid_ids(data, component, "node_2", ComponentType.node)
639
+ errors += _all_valid_ids(data, component, "node_3", ComponentType.node)
640
+ errors += _all_not_two_values_equal(data, component, "node_1", "node_2")
641
+ errors += _all_not_two_values_equal(data, component, "node_1", "node_3")
642
+ errors += _all_not_two_values_equal(data, component, "node_2", "node_3")
643
+ errors += _all_boolean(data, component, "status_1")
644
+ errors += _all_boolean(data, component, "status_2")
645
+ errors += _all_boolean(data, component, "status_3")
646
+ return errors
647
+
648
+
649
+ def validate_three_winding_transformer(data: SingleDataset) -> list[ValidationError]: # noqa: PLR0915
650
+ errors = validate_branch3(data, ComponentType.three_winding_transformer)
651
+ errors += _all_greater_than_zero(data, ComponentType.three_winding_transformer, "u1")
652
+ errors += _all_greater_than_zero(data, ComponentType.three_winding_transformer, "u2")
653
+ errors += _all_greater_than_zero(data, ComponentType.three_winding_transformer, "u3")
654
+ errors += _all_greater_than_zero(data, ComponentType.three_winding_transformer, "sn_1")
655
+ errors += _all_greater_than_zero(data, ComponentType.three_winding_transformer, "sn_2")
656
+ errors += _all_greater_than_zero(data, ComponentType.three_winding_transformer, "sn_3")
657
+ errors += _all_greater_or_equal(data, ComponentType.three_winding_transformer, "uk_12", "pk_12/sn_1")
658
+ errors += _all_greater_or_equal(data, ComponentType.three_winding_transformer, "uk_12", "pk_12/sn_2")
659
+ errors += _all_greater_or_equal(data, ComponentType.three_winding_transformer, "uk_13", "pk_13/sn_1")
660
+ errors += _all_greater_or_equal(data, ComponentType.three_winding_transformer, "uk_13", "pk_13/sn_3")
661
+ errors += _all_greater_or_equal(data, ComponentType.three_winding_transformer, "uk_23", "pk_23/sn_2")
662
+ errors += _all_greater_or_equal(data, ComponentType.three_winding_transformer, "uk_23", "pk_23/sn_3")
663
+ errors += _all_between(data, ComponentType.three_winding_transformer, "uk_12", 0, 1)
664
+ errors += _all_between(data, ComponentType.three_winding_transformer, "uk_13", 0, 1)
665
+ errors += _all_between(data, ComponentType.three_winding_transformer, "uk_23", 0, 1)
666
+ errors += _all_greater_than_or_equal_to_zero(data, ComponentType.three_winding_transformer, "pk_12")
667
+ errors += _all_greater_than_or_equal_to_zero(data, ComponentType.three_winding_transformer, "pk_13")
668
+ errors += _all_greater_than_or_equal_to_zero(data, ComponentType.three_winding_transformer, "pk_23")
669
+ errors += _all_greater_or_equal(data, ComponentType.three_winding_transformer, "i0", "p0/sn_1")
670
+ errors += _all_less_than(data, ComponentType.three_winding_transformer, "i0", 1)
671
+ errors += _all_greater_than_or_equal_to_zero(data, ComponentType.three_winding_transformer, "p0")
672
+ errors += _all_valid_enum_values(data, ComponentType.three_winding_transformer, "winding_1", WindingType)
673
+ errors += _all_valid_enum_values(data, ComponentType.three_winding_transformer, "winding_2", WindingType)
674
+ errors += _all_valid_enum_values(data, ComponentType.three_winding_transformer, "winding_3", WindingType)
675
+ errors += _all_between_or_at(data, ComponentType.three_winding_transformer, "clock_12", -12, 12)
676
+ errors += _all_between_or_at(data, ComponentType.three_winding_transformer, "clock_13", -12, 12)
677
+ errors += _all_valid_clocks(data, ComponentType.three_winding_transformer, "clock_12", "winding_1", "winding_2")
678
+ errors += _all_valid_clocks(data, ComponentType.three_winding_transformer, "clock_13", "winding_1", "winding_3")
679
+ errors += _all_valid_enum_values(data, ComponentType.three_winding_transformer, "tap_side", Branch3Side)
680
+ errors += _all_between_or_at(
681
+ data,
682
+ ComponentType.three_winding_transformer,
683
+ "tap_pos",
684
+ "tap_min",
685
+ "tap_max",
686
+ data[ComponentType.three_winding_transformer]["tap_nom"],
687
+ 0,
688
+ )
689
+ errors += _all_between_or_at(data, ComponentType.three_winding_transformer, "tap_nom", "tap_min", "tap_max", 0)
690
+ errors += _all_greater_than_or_equal_to_zero(data, ComponentType.three_winding_transformer, "tap_size")
691
+ errors += _all_greater_or_equal(
692
+ data,
693
+ ComponentType.three_winding_transformer,
694
+ "uk_12_min",
695
+ "pk_12_min/sn_1",
696
+ data[ComponentType.three_winding_transformer]["uk_12"],
697
+ )
698
+ errors += _all_greater_or_equal(
699
+ data,
700
+ ComponentType.three_winding_transformer,
701
+ "uk_12_min",
702
+ "pk_12_min/sn_2",
703
+ data[ComponentType.three_winding_transformer]["uk_12"],
704
+ )
705
+ errors += _all_greater_or_equal(
706
+ data,
707
+ ComponentType.three_winding_transformer,
708
+ "uk_13_min",
709
+ "pk_13_min/sn_1",
710
+ data[ComponentType.three_winding_transformer]["uk_13"],
711
+ )
712
+ errors += _all_greater_or_equal(
713
+ data,
714
+ ComponentType.three_winding_transformer,
715
+ "uk_13_min",
716
+ "pk_13_min/sn_3",
717
+ data[ComponentType.three_winding_transformer]["uk_13"],
718
+ )
719
+ errors += _all_greater_or_equal(
720
+ data,
721
+ ComponentType.three_winding_transformer,
722
+ "uk_23_min",
723
+ "pk_23_min/sn_2",
724
+ data[ComponentType.three_winding_transformer]["uk_23"],
725
+ )
726
+ errors += _all_greater_or_equal(
727
+ data,
728
+ ComponentType.three_winding_transformer,
729
+ "uk_23_min",
730
+ "pk_23_min/sn_3",
731
+ data[ComponentType.three_winding_transformer]["uk_23"],
732
+ )
733
+ errors += _all_between(
734
+ data,
735
+ ComponentType.three_winding_transformer,
736
+ "uk_12_min",
737
+ 0,
738
+ 1,
739
+ data[ComponentType.three_winding_transformer]["uk_12"],
740
+ )
741
+ errors += _all_between(
742
+ data,
743
+ ComponentType.three_winding_transformer,
744
+ "uk_13_min",
745
+ 0,
746
+ 1,
747
+ data[ComponentType.three_winding_transformer]["uk_13"],
748
+ )
749
+ errors += _all_between(
750
+ data,
751
+ ComponentType.three_winding_transformer,
752
+ "uk_23_min",
753
+ 0,
754
+ 1,
755
+ data[ComponentType.three_winding_transformer]["uk_23"],
756
+ )
757
+ errors += _all_greater_or_equal(
758
+ data,
759
+ ComponentType.three_winding_transformer,
760
+ "uk_12_max",
761
+ "pk_12_max/sn_1",
762
+ data[ComponentType.three_winding_transformer]["uk_12"],
763
+ )
764
+ errors += _all_greater_or_equal(
765
+ data,
766
+ ComponentType.three_winding_transformer,
767
+ "uk_12_max",
768
+ "pk_12_max/sn_2",
769
+ data[ComponentType.three_winding_transformer]["uk_12"],
770
+ )
771
+ errors += _all_greater_or_equal(
772
+ data,
773
+ ComponentType.three_winding_transformer,
774
+ "uk_13_max",
775
+ "pk_13_max/sn_1",
776
+ data[ComponentType.three_winding_transformer]["uk_13"],
777
+ )
778
+ errors += _all_greater_or_equal(
779
+ data,
780
+ ComponentType.three_winding_transformer,
781
+ "uk_13_max",
782
+ "pk_13_max/sn_3",
783
+ data[ComponentType.three_winding_transformer]["uk_13"],
784
+ )
785
+ errors += _all_greater_or_equal(
786
+ data,
787
+ ComponentType.three_winding_transformer,
788
+ "uk_23_max",
789
+ "pk_23_max/sn_2",
790
+ data[ComponentType.three_winding_transformer]["uk_23"],
791
+ )
792
+ errors += _all_greater_or_equal(
793
+ data,
794
+ ComponentType.three_winding_transformer,
795
+ "uk_23_max",
796
+ "pk_23_max/sn_3",
797
+ data[ComponentType.three_winding_transformer]["uk_23"],
798
+ )
799
+ errors += _all_between(
800
+ data,
801
+ ComponentType.three_winding_transformer,
802
+ "uk_12_max",
803
+ 0,
804
+ 1,
805
+ data[ComponentType.three_winding_transformer]["uk_12"],
806
+ )
807
+ errors += _all_between(
808
+ data,
809
+ ComponentType.three_winding_transformer,
810
+ "uk_13_max",
811
+ 0,
812
+ 1,
813
+ data[ComponentType.three_winding_transformer]["uk_13"],
814
+ )
815
+ errors += _all_between(
816
+ data,
817
+ ComponentType.three_winding_transformer,
818
+ "uk_23_max",
819
+ 0,
820
+ 1,
821
+ data[ComponentType.three_winding_transformer]["uk_23"],
822
+ )
823
+ errors += _all_greater_than_or_equal_to_zero(
824
+ data,
825
+ ComponentType.three_winding_transformer,
826
+ "pk_12_min",
827
+ data[ComponentType.three_winding_transformer]["pk_12"],
828
+ )
829
+ errors += _all_greater_than_or_equal_to_zero(
830
+ data,
831
+ ComponentType.three_winding_transformer,
832
+ "pk_13_min",
833
+ data[ComponentType.three_winding_transformer]["pk_13"],
834
+ )
835
+ errors += _all_greater_than_or_equal_to_zero(
836
+ data,
837
+ ComponentType.three_winding_transformer,
838
+ "pk_23_min",
839
+ data[ComponentType.three_winding_transformer]["pk_23"],
840
+ )
841
+ errors += _all_greater_than_or_equal_to_zero(
842
+ data,
843
+ ComponentType.three_winding_transformer,
844
+ "pk_12_max",
845
+ data[ComponentType.three_winding_transformer]["pk_12"],
846
+ )
847
+ errors += _all_greater_than_or_equal_to_zero(
848
+ data,
849
+ ComponentType.three_winding_transformer,
850
+ "pk_13_max",
851
+ data[ComponentType.three_winding_transformer]["pk_13"],
852
+ )
853
+ errors += _all_greater_than_or_equal_to_zero(
854
+ data,
855
+ ComponentType.three_winding_transformer,
856
+ "pk_23_max",
857
+ data[ComponentType.three_winding_transformer]["pk_23"],
858
+ )
859
+ return errors
860
+
861
+
862
+ def validate_appliance(data: SingleDataset, component: ComponentType) -> list[ValidationError]:
863
+ errors = validate_base(data, component)
864
+ errors += _all_boolean(data, component, "status")
865
+ errors += _all_valid_ids(data, component, "node", ComponentType.node)
866
+ return errors
867
+
868
+
869
+ def validate_source(data: SingleDataset) -> list[ValidationError]:
870
+ errors = validate_appliance(data, ComponentType.source)
871
+ errors += _all_greater_than_zero(data, ComponentType.source, "u_ref")
872
+ errors += _all_greater_than_zero(data, ComponentType.source, "sk")
873
+ errors += _all_greater_than_or_equal_to_zero(data, ComponentType.source, "rx_ratio")
874
+ errors += _all_greater_than_zero(data, ComponentType.source, "z01_ratio")
875
+ return errors
876
+
877
+
878
+ def validate_generic_load_gen(data: SingleDataset, component: ComponentType) -> list[ValidationError]:
879
+ errors = validate_appliance(data, component)
880
+ errors += _all_valid_enum_values(data, component, "type", LoadGenType)
881
+ return errors
882
+
883
+
884
+ def validate_shunt(data: SingleDataset) -> list[ValidationError]:
885
+ return validate_appliance(data, ComponentType.shunt)
886
+
887
+
888
+ def validate_generic_voltage_sensor(data: SingleDataset, component: ComponentType) -> list[ValidationError]:
889
+ errors = validate_base(data, component)
890
+ errors += _all_greater_than_zero(data, component, "u_sigma")
891
+ errors += _all_greater_than_zero(data, component, "u_measured")
892
+ errors += _all_valid_ids(data, component, "measured_object", ComponentType.node)
893
+ return errors
894
+
895
+
896
+ def validate_generic_power_sensor(data: SingleDataset, component: ComponentType) -> list[ValidationError]:
897
+ errors = validate_base(data, component)
898
+ errors += _all_greater_than_zero(data, component, "power_sigma")
899
+ errors += _all_valid_enum_values(data, component, "measured_terminal_type", MeasuredTerminalType)
900
+ errors += _all_valid_ids(
901
+ data,
902
+ component,
903
+ field="measured_object",
904
+ ref_components=[
905
+ ComponentType.node,
906
+ ComponentType.line,
907
+ ComponentType.asym_line,
908
+ ComponentType.generic_branch,
909
+ ComponentType.transformer,
910
+ ComponentType.three_winding_transformer,
911
+ ComponentType.source,
912
+ ComponentType.shunt,
913
+ ComponentType.sym_load,
914
+ ComponentType.asym_load,
915
+ ComponentType.sym_gen,
916
+ ComponentType.asym_gen,
917
+ ],
918
+ )
919
+ errors += _all_valid_ids(
920
+ data,
921
+ component,
922
+ field="measured_object",
923
+ ref_components=[
924
+ ComponentType.line,
925
+ ComponentType.asym_line,
926
+ ComponentType.generic_branch,
927
+ ComponentType.transformer,
928
+ ],
929
+ measured_terminal_type=MeasuredTerminalType.branch_from,
930
+ )
931
+ errors += _all_valid_ids(
932
+ data,
933
+ component,
934
+ field="measured_object",
935
+ ref_components=[
936
+ ComponentType.line,
937
+ ComponentType.asym_line,
938
+ ComponentType.generic_branch,
939
+ ComponentType.transformer,
940
+ ],
941
+ measured_terminal_type=MeasuredTerminalType.branch_to,
942
+ )
943
+ errors += _all_valid_ids(
944
+ data,
945
+ component,
946
+ field="measured_object",
947
+ ref_components=ComponentType.source,
948
+ measured_terminal_type=MeasuredTerminalType.source,
949
+ )
950
+ errors += _all_valid_ids(
951
+ data,
952
+ component,
953
+ field="measured_object",
954
+ ref_components=ComponentType.shunt,
955
+ measured_terminal_type=MeasuredTerminalType.shunt,
956
+ )
957
+ errors += _all_valid_ids(
958
+ data,
959
+ component,
960
+ field="measured_object",
961
+ ref_components=[ComponentType.sym_load, ComponentType.asym_load],
962
+ measured_terminal_type=MeasuredTerminalType.load,
963
+ )
964
+ errors += _all_valid_ids(
965
+ data,
966
+ component,
967
+ field="measured_object",
968
+ ref_components=[ComponentType.sym_gen, ComponentType.asym_gen],
969
+ measured_terminal_type=MeasuredTerminalType.generator,
970
+ )
971
+ errors += _all_valid_ids(
972
+ data,
973
+ component,
974
+ field="measured_object",
975
+ ref_components=ComponentType.three_winding_transformer,
976
+ measured_terminal_type=MeasuredTerminalType.branch3_1,
977
+ )
978
+ errors += _all_valid_ids(
979
+ data,
980
+ component,
981
+ field="measured_object",
982
+ ref_components=ComponentType.three_winding_transformer,
983
+ measured_terminal_type=MeasuredTerminalType.branch3_2,
984
+ )
985
+ errors += _all_valid_ids(
986
+ data,
987
+ component,
988
+ field="measured_object",
989
+ ref_components=ComponentType.three_winding_transformer,
990
+ measured_terminal_type=MeasuredTerminalType.branch3_3,
991
+ )
992
+ errors += _all_valid_ids(
993
+ data,
994
+ component,
995
+ field="measured_object",
996
+ ref_components=ComponentType.node,
997
+ measured_terminal_type=MeasuredTerminalType.node,
998
+ )
999
+ if component in (ComponentType.sym_power_sensor, ComponentType.asym_power_sensor):
1000
+ errors += _valid_p_q_sigma(data, component)
1001
+
1002
+ return errors
1003
+
1004
+
1005
+ def validate_generic_current_sensor(data: SingleDataset, component: ComponentType) -> list[ValidationError]:
1006
+ errors = validate_base(data, component)
1007
+ errors += _all_greater_than_zero(data, component, "i_sigma")
1008
+ errors += _all_greater_than_zero(data, component, "i_angle_sigma")
1009
+ errors += _all_valid_enum_values(data, component, "measured_terminal_type", MeasuredTerminalType)
1010
+ errors += _all_valid_enum_values(data, component, "angle_measurement_type", AngleMeasurementType)
1011
+ errors += _all_in_valid_values(
1012
+ data,
1013
+ component,
1014
+ "measured_terminal_type",
1015
+ [
1016
+ MeasuredTerminalType.branch_from,
1017
+ MeasuredTerminalType.branch_to,
1018
+ MeasuredTerminalType.branch3_1,
1019
+ MeasuredTerminalType.branch3_2,
1020
+ MeasuredTerminalType.branch3_3,
1021
+ ],
1022
+ )
1023
+ errors += _all_valid_ids(
1024
+ data,
1025
+ component,
1026
+ field="measured_object",
1027
+ ref_components=[
1028
+ ComponentType.line,
1029
+ ComponentType.asym_line,
1030
+ ComponentType.generic_branch,
1031
+ ComponentType.transformer,
1032
+ ComponentType.three_winding_transformer,
1033
+ ],
1034
+ )
1035
+ errors += _all_valid_ids(
1036
+ data,
1037
+ component,
1038
+ field="measured_object",
1039
+ ref_components=[
1040
+ ComponentType.line,
1041
+ ComponentType.asym_line,
1042
+ ComponentType.generic_branch,
1043
+ ComponentType.transformer,
1044
+ ],
1045
+ measured_terminal_type=MeasuredTerminalType.branch_from,
1046
+ )
1047
+ errors += _all_valid_ids(
1048
+ data,
1049
+ component,
1050
+ field="measured_object",
1051
+ ref_components=[
1052
+ ComponentType.line,
1053
+ ComponentType.asym_line,
1054
+ ComponentType.generic_branch,
1055
+ ComponentType.transformer,
1056
+ ],
1057
+ measured_terminal_type=MeasuredTerminalType.branch_to,
1058
+ )
1059
+ errors += _all_valid_ids(
1060
+ data,
1061
+ component,
1062
+ field="measured_object",
1063
+ ref_components=ComponentType.three_winding_transformer,
1064
+ measured_terminal_type=MeasuredTerminalType.branch3_1,
1065
+ )
1066
+ errors += _all_valid_ids(
1067
+ data,
1068
+ component,
1069
+ field="measured_object",
1070
+ ref_components=ComponentType.three_winding_transformer,
1071
+ measured_terminal_type=MeasuredTerminalType.branch3_2,
1072
+ )
1073
+ errors += _all_valid_ids(
1074
+ data,
1075
+ component,
1076
+ field="measured_object",
1077
+ ref_components=ComponentType.three_winding_transformer,
1078
+ measured_terminal_type=MeasuredTerminalType.branch3_3,
1079
+ )
1080
+ errors += _all_same_current_angle_measurement_type_on_terminal(
1081
+ data,
1082
+ component,
1083
+ measured_object_field="measured_object",
1084
+ measured_terminal_type_field="measured_terminal_type",
1085
+ angle_measurement_type_field="angle_measurement_type",
1086
+ )
1087
+ errors += _any_voltage_angle_measurement_if_global_current_measurement(
1088
+ data,
1089
+ component,
1090
+ angle_measurement_type_filter=("angle_measurement_type", AngleMeasurementType.global_angle),
1091
+ voltage_sensor_u_angle_measured={
1092
+ ComponentType.sym_voltage_sensor: "u_angle_measured",
1093
+ ComponentType.asym_voltage_sensor: "u_angle_measured",
1094
+ },
1095
+ )
1096
+
1097
+ return errors
1098
+
1099
+
1100
+ def validate_fault(data: SingleDataset) -> list[ValidationError]:
1101
+ errors = validate_base(data, ComponentType.fault)
1102
+ errors += _all_boolean(data, ComponentType.fault, "status")
1103
+ errors += _all_valid_enum_values(data, ComponentType.fault, "fault_type", FaultType)
1104
+ errors += _all_valid_enum_values(data, ComponentType.fault, "fault_phase", FaultPhase)
1105
+ errors += _all_valid_fault_phases(data, ComponentType.fault, "fault_type", "fault_phase")
1106
+ errors += _all_valid_ids(data, ComponentType.fault, field="fault_object", ref_components=ComponentType.node)
1107
+ errors += _all_greater_than_or_equal_to_zero(data, ComponentType.fault, "r_f")
1108
+ errors += _all_enabled_identical(data, ComponentType.fault, "fault_type", "status")
1109
+ errors += _all_enabled_identical(data, ComponentType.fault, "fault_phase", "status")
1110
+ return errors
1111
+
1112
+
1113
+ def validate_regulator(data: SingleDataset, component: ComponentType) -> list[ValidationError]:
1114
+ errors = validate_base(data, component)
1115
+ errors += _all_valid_ids(
1116
+ data,
1117
+ component,
1118
+ field="regulated_object",
1119
+ ref_components=[ComponentType.transformer, ComponentType.three_winding_transformer],
1120
+ )
1121
+ return errors
1122
+
1123
+
1124
+ def validate_transformer_tap_regulator(data: SingleDataset) -> list[ValidationError]:
1125
+ errors = validate_regulator(data, ComponentType.transformer_tap_regulator)
1126
+ errors += _all_boolean(data, ComponentType.transformer_tap_regulator, "status")
1127
+ errors += _all_unique(data, ComponentType.transformer_tap_regulator, "regulated_object")
1128
+ errors += _all_valid_enum_values(
1129
+ data, ComponentType.transformer_tap_regulator, "control_side", [BranchSide, Branch3Side]
1130
+ )
1131
+ errors += _all_valid_associated_enum_values(
1132
+ data,
1133
+ ComponentType.transformer_tap_regulator,
1134
+ "control_side",
1135
+ "regulated_object",
1136
+ [ComponentType.transformer],
1137
+ [BranchSide],
1138
+ )
1139
+ errors += _all_valid_associated_enum_values(
1140
+ data,
1141
+ ComponentType.transformer_tap_regulator,
1142
+ "control_side",
1143
+ "regulated_object",
1144
+ [ComponentType.three_winding_transformer],
1145
+ [Branch3Side],
1146
+ )
1147
+ errors += _all_greater_than_or_equal_to_zero(data, ComponentType.transformer_tap_regulator, "u_set")
1148
+ errors += _all_greater_than_zero(data, ComponentType.transformer_tap_regulator, "u_band")
1149
+ errors += _all_greater_than_or_equal_to_zero(
1150
+ data, ComponentType.transformer_tap_regulator, "line_drop_compensation_r", 0.0
1151
+ )
1152
+ errors += _all_greater_than_or_equal_to_zero(
1153
+ data, ComponentType.transformer_tap_regulator, "line_drop_compensation_x", 0.0
1154
+ )
1155
+ return errors
1156
+
1157
+
1158
+ def validate_no_mixed_sensors_on_same_terminal(data: SingleDataset) -> list[ValidationError]:
1159
+ errors: list[ValidationError] = []
1160
+
1161
+ for power_sensor in [ComponentType.sym_power_sensor, ComponentType.asym_power_sensor]:
1162
+ for current_sensor in [ComponentType.sym_current_sensor, ComponentType.asym_current_sensor]:
1163
+ if power_sensor in data and current_sensor in data:
1164
+ errors += _all_same_sensor_type_on_same_terminal(
1165
+ data,
1166
+ power_sensor_type=power_sensor,
1167
+ current_sensor_type=current_sensor,
1168
+ measured_object_field="measured_object",
1169
+ measured_terminal_type_field="measured_terminal_type",
1170
+ )
1171
+
1172
+ return errors