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,490 +1,545 @@
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 raw dataset handler
7
- """
8
-
9
- from typing import Any, Mapping
10
-
11
- from power_grid_model._core.buffer_handling import (
12
- BufferProperties,
13
- CAttributeBuffer,
14
- CBuffer,
15
- create_buffer,
16
- get_buffer_properties,
17
- get_buffer_view,
18
- )
19
- from power_grid_model._core.dataset_definitions import ComponentType, DatasetType, _str_to_component_type
20
- from power_grid_model._core.error_handling import VALIDATOR_MSG, assert_no_error
21
- from power_grid_model._core.power_grid_core import (
22
- ConstDatasetPtr,
23
- DatasetInfoPtr,
24
- MutableDatasetPtr,
25
- WritableDatasetPtr,
26
- power_grid_core as pgc,
27
- )
28
- from power_grid_model._core.power_grid_meta import ComponentMetaData, DatasetMetaData, power_grid_meta_data
29
- from power_grid_model._utils import get_dataset_type, is_columnar, is_nan_or_equivalent, process_data_filter
30
- from power_grid_model.data_types import AttributeType, ComponentData, Dataset
31
- from power_grid_model.enum import ComponentAttributeFilterOptions
32
- from power_grid_model.typing import ComponentAttributeMapping, _ComponentAttributeMappingDict
33
-
34
-
35
- class CDatasetInfo: # pylint: disable=too-few-public-methods
36
- """
37
- Raw Power Grid Model dataset info.
38
- """
39
-
40
- def __init__(self, info: DatasetInfoPtr):
41
- self._info: DatasetInfoPtr = info
42
-
43
- def name(self) -> str:
44
- """
45
- The name of the dataset type.
46
-
47
- Returns:
48
- The name of the dataset type
49
- """
50
- return pgc.dataset_info_name(self._info)
51
-
52
- def dataset_type(self):
53
- """
54
- The name of the dataset type.
55
-
56
- Returns:
57
- The name of the dataset type
58
- """
59
- return self.name()
60
-
61
- def is_batch(self) -> bool:
62
- """
63
- Whether the dataset is a batch dataset.
64
-
65
- Returns:
66
- Whether the dataset is a batch dataset
67
- """
68
- return bool(pgc.dataset_info_is_batch(self._info))
69
-
70
- def batch_size(self) -> int:
71
- """
72
- The size of the dataset.
73
-
74
- Returns:
75
- The size of the dataset
76
- """
77
- return pgc.dataset_info_batch_size(self._info)
78
-
79
- def n_components(self) -> int:
80
- """
81
- The amount of components in the dataset.
82
-
83
- Returns:
84
- The amount of components in the dataset
85
- """
86
- return pgc.dataset_info_n_components(self._info)
87
-
88
- def components(self) -> list[ComponentType]:
89
- """
90
- The components in the dataset.
91
-
92
- Returns:
93
- A list of the component names in the dataset
94
- """
95
- return [
96
- _str_to_component_type(pgc.dataset_info_component_name(self._info, idx))
97
- for idx in range(self.n_components())
98
- ]
99
-
100
- def elements_per_scenario(self) -> Mapping[ComponentType, int]:
101
- """
102
- The number of elements per scenario in the dataset.
103
-
104
- Returns:
105
- The number of elements per senario for each component in the dataset;
106
- or -1 if the scenario is not uniform (different amount per scenario)
107
- """
108
- return {
109
- component_name: pgc.dataset_info_elements_per_scenario(self._info, idx)
110
- for idx, component_name in enumerate(self.components())
111
- }
112
-
113
- def total_elements(self) -> Mapping[ComponentType, int]:
114
- """
115
- The total number of elements in the dataset.
116
-
117
- Returns:
118
- The total number of elements for each component.
119
- For each component, if the number of elements per scenario is uniform, its value shall be equal to
120
- the product of the batch size and the amount of elements per scenario for that component.
121
- """
122
- return {
123
- component_name: pgc.dataset_info_total_elements(self._info, idx)
124
- for idx, component_name in enumerate(self.components())
125
- }
126
-
127
-
128
- class CMutableDataset:
129
- """
130
- A view of a user-owned dataset.
131
-
132
- This may be used to provide a user dataset to the Power Grid Model.
133
-
134
- The dataset will create mutable buffers that the Power Grid Model can use to load data.
135
- """
136
-
137
- _dataset_type: DatasetType
138
- _schema: DatasetMetaData
139
- _is_batch: bool
140
- _batch_size: int
141
- _mutable_dataset: MutableDatasetPtr
142
- _buffer_views: list[CBuffer]
143
-
144
- def __new__(cls, data: Dataset, dataset_type: Any = None):
145
- instance = super().__new__(cls)
146
- instance._mutable_dataset = MutableDatasetPtr()
147
- instance._buffer_views = []
148
-
149
- instance._dataset_type = dataset_type if dataset_type in DatasetType else get_dataset_type(data)
150
- instance._schema = power_grid_meta_data[instance._dataset_type]
151
-
152
- if data:
153
- first_component, first_component_data = next(iter(data.items()))
154
- first_sub_info = get_buffer_properties(data=first_component_data, schema=instance._schema[first_component])
155
- instance._is_batch = first_sub_info.is_batch
156
- instance._batch_size = first_sub_info.batch_size
157
- else:
158
- instance._is_batch = False
159
- instance._batch_size = 1
160
-
161
- instance._mutable_dataset = pgc.create_dataset_mutable(
162
- instance._dataset_type.value, instance._is_batch, instance._batch_size
163
- )
164
- assert_no_error()
165
-
166
- instance._add_data(data)
167
- assert_no_error()
168
-
169
- return instance
170
-
171
- def get_dataset_ptr(self) -> MutableDatasetPtr:
172
- """
173
- Get the raw underlying mutable dataset pointer.
174
-
175
- Returns:
176
- MutableDatasetPtr: the raw underlying mutable dataset pointer.
177
- """
178
- return self._mutable_dataset
179
-
180
- def get_info(self) -> CDatasetInfo:
181
- """
182
- Get the info for this dataset.
183
-
184
- Returns:
185
- The dataset info for this dataset.
186
- """
187
- return CDatasetInfo(pgc.dataset_mutable_get_info(self._mutable_dataset))
188
-
189
- def get_buffer_views(self) -> list[CBuffer]:
190
- """
191
- Get list of buffer views
192
-
193
- Returns:
194
- list of buffer view
195
- """
196
- return self._buffer_views
197
-
198
- def _add_data(self, data: Dataset):
199
- """
200
- Add Power Grid Model data to the mutable dataset view.
201
-
202
- Args:
203
- data: the data.
204
-
205
- Raises:
206
- ValueError: if the component is unknown and allow_unknown is False.
207
- ValueError: if the data is inconsistent with the rest of the dataset.
208
- PowerGridError: if there was an internal error.
209
- """
210
- for component, component_data in data.items():
211
- self._add_component_data(component, component_data, allow_unknown=False)
212
-
213
- def _add_component_data(self, component: ComponentType, data: ComponentData, allow_unknown: bool = False):
214
- """
215
- Add Power Grid Model data for a single component to the mutable dataset view.
216
-
217
- Args:
218
- component: the name of the component
219
- data: the data of the component
220
- allow_unknown (optional): ignore any unknown components. Defaults to False.
221
-
222
- Raises:
223
- ValueError: if the component is unknown and allow_unknown is False.
224
- ValueError: if the data is inconsistent with the rest of the dataset.
225
- PowerGridError: if there was an internal error.
226
- """
227
- if component not in self._schema:
228
- if not allow_unknown:
229
- raise ValueError(f"Unknown component {component} in schema. {VALIDATOR_MSG}")
230
- return
231
-
232
- self._validate_properties(data, self._schema[component])
233
- c_buffer = get_buffer_view(data, self._schema[component], self._is_batch, self._batch_size)
234
- self._buffer_views.append(c_buffer)
235
- self._register_buffer(component, c_buffer)
236
-
237
- def _register_buffer(self, component: ComponentType, buffer: CBuffer):
238
- pgc.dataset_mutable_add_buffer(
239
- dataset=self._mutable_dataset,
240
- component=component.value,
241
- elements_per_scenario=buffer.n_elements_per_scenario,
242
- total_elements=buffer.total_elements,
243
- indptr=buffer.indptr,
244
- data=buffer.data,
245
- )
246
- assert_no_error()
247
- for attr, attr_data in buffer.attribute_data.items():
248
- self._register_attribute_buffer(component, attr, attr_data)
249
-
250
- def _register_attribute_buffer(self, component, attr, attr_data):
251
- pgc.dataset_mutable_add_attribute_buffer(
252
- dataset=self._mutable_dataset,
253
- component=component.value,
254
- attribute=attr,
255
- data=attr_data.data,
256
- )
257
- assert_no_error()
258
-
259
- def _validate_properties(self, data: ComponentData, schema: ComponentMetaData):
260
- properties = get_buffer_properties(data, schema=schema, is_batch=self._is_batch, batch_size=self._batch_size)
261
- if properties.is_batch != self._is_batch:
262
- raise ValueError(
263
- f"Dataset type (single or batch) must be consistent across all components. {VALIDATOR_MSG}"
264
- )
265
- if properties.batch_size != self._batch_size:
266
- raise ValueError(f"Dataset must have a consistent batch size across all components. {VALIDATOR_MSG}")
267
-
268
- def __del__(self):
269
- pgc.destroy_dataset_mutable(self._mutable_dataset)
270
-
271
-
272
- class CConstDataset:
273
- """
274
- A view of a user-owned dataset.
275
-
276
- This may be used to provide a user dataset to the Power Grid Model.
277
-
278
- The dataset will create const buffers that the Power Grid Model can use to load data.
279
-
280
- It is created from mutable dataset.
281
- """
282
-
283
- _const_dataset: ConstDatasetPtr
284
- _buffer_views: list[CBuffer]
285
-
286
- def __new__(cls, data: Dataset, dataset_type: DatasetType | None = None):
287
- instance = super().__new__(cls)
288
- instance._const_dataset = ConstDatasetPtr()
289
-
290
- # create from mutable dataset
291
- mutable_dataset = CMutableDataset(data=data, dataset_type=dataset_type)
292
- instance._const_dataset = pgc.create_dataset_const_from_mutable(mutable_dataset.get_dataset_ptr())
293
- assert_no_error()
294
- instance._buffer_views = mutable_dataset.get_buffer_views()
295
-
296
- return instance
297
-
298
- def get_dataset_ptr(self) -> ConstDatasetPtr:
299
- """
300
- Get the raw underlying const dataset pointer.
301
-
302
- Returns:
303
- ConstDatasetPtr: the raw underlying const dataset pointer.
304
- """
305
- return self._const_dataset
306
-
307
- def get_info(self) -> CDatasetInfo:
308
- """
309
- Get the info for this dataset.
310
-
311
- Returns:
312
- The dataset info for this dataset.
313
- """
314
- return CDatasetInfo(pgc.dataset_const_get_info(self._const_dataset))
315
-
316
- def __del__(self):
317
- pgc.destroy_dataset_const(self._const_dataset)
318
-
319
-
320
- class CWritableDataset:
321
- """
322
- A view of a Power Grid Model-owned dataset.
323
-
324
- This may be used to retrieve data from the Power Grid Model.
325
-
326
- This class provides buffers to which the Power Grid Model can write data in an external call.
327
- After writing to the buffers, the data contents can be retrieved.
328
- """
329
-
330
- def __init__(self, dataset_ptr: WritableDatasetPtr, data_filter: ComponentAttributeMapping):
331
- self._writable_dataset = dataset_ptr
332
-
333
- info = self.get_info()
334
- self._dataset_type = info.dataset_type()
335
- self._schema = power_grid_meta_data[self._dataset_type]
336
-
337
- self._data_filter = process_data_filter(
338
- dataset_type=info.dataset_type(),
339
- data_filter=data_filter,
340
- available_components=info.components(),
341
- )
342
-
343
- self._data: Dataset = {}
344
- self._buffers: Mapping[str, CBuffer] = {}
345
- self._component_buffer_properties = self._get_buffer_properties(info)
346
-
347
- self._add_buffers()
348
- assert_no_error()
349
-
350
- def get_dataset_ptr(self) -> WritableDatasetPtr:
351
- """
352
- Get the raw underlying writable dataset pointer.
353
-
354
- Returns:
355
- WritableDatasetPtr: the raw underlying writable dataset pointer.
356
- """
357
- return self._writable_dataset
358
-
359
- def get_info(self) -> CDatasetInfo:
360
- """
361
- Get the info for this dataset.
362
-
363
- Returns:
364
- The dataset info for this dataset.
365
- """
366
- return CDatasetInfo(pgc.dataset_writable_get_info(self._writable_dataset))
367
-
368
- def get_data(self) -> Dataset:
369
- """
370
- Retrieve data from the Power Grid Model dataset.
371
-
372
- The Power Grid Model may write to these buffers at a later point in time.
373
-
374
- Returns:
375
- The full dataset with filters applied.
376
- """
377
- self._post_filtering()
378
- return self._data
379
-
380
- def get_component_data(self, component: ComponentType) -> ComponentData:
381
- """
382
- Retrieve Power Grid Model data from the dataset for a specific component.
383
-
384
- Args:
385
- component: the component.
386
-
387
- Returns:
388
- The dataset for the specified component.
389
- """
390
- return self._data[component]
391
-
392
- def get_data_filter(self) -> _ComponentAttributeMappingDict:
393
- """Gets the data filter requested
394
-
395
- Returns:
396
- _ComponentAttributeMappingDict: data filter
397
- """
398
- return self._data_filter
399
-
400
- def _add_buffers(self):
401
- for component, buffer_properties in self._component_buffer_properties.items():
402
- self._add_buffer(component, buffer_properties)
403
-
404
- def _add_buffer(self, component: ComponentType, buffer_properties: BufferProperties):
405
- schema = self._schema[component]
406
-
407
- self._data[component] = create_buffer(buffer_properties, schema)
408
- self._register_buffer(component, get_buffer_view(self._data[component], schema))
409
-
410
- def _register_buffer(self, component: ComponentType, buffer: CBuffer):
411
- pgc.dataset_writable_set_buffer(
412
- dataset=self._writable_dataset,
413
- component=component,
414
- indptr=buffer.indptr,
415
- data=buffer.data,
416
- )
417
- assert_no_error()
418
- for attribute, attribute_data in buffer.attribute_data.items():
419
- self._register_attribute_buffer(component, attribute, attribute_data)
420
-
421
- def _register_attribute_buffer(
422
- self,
423
- component: ComponentType,
424
- attribute: AttributeType,
425
- buffer: CAttributeBuffer,
426
- ):
427
- pgc.dataset_writable_set_attribute_buffer(
428
- dataset=self._writable_dataset,
429
- component=component,
430
- attribute=attribute,
431
- data=buffer.data,
432
- )
433
- assert_no_error()
434
-
435
- def _get_buffer_properties(self, info: CDatasetInfo) -> Mapping[ComponentType, BufferProperties]:
436
- is_batch = info.is_batch()
437
- batch_size = info.batch_size()
438
- components = info.components()
439
- n_elements_per_scenario = info.elements_per_scenario()
440
- n_total_elements = info.total_elements()
441
-
442
- return {
443
- component: BufferProperties(
444
- is_sparse=n_elements_per_scenario[component] == -1,
445
- is_batch=is_batch,
446
- batch_size=batch_size,
447
- n_elements_per_scenario=n_elements_per_scenario[component],
448
- n_total_elements=n_total_elements[component],
449
- columns=_get_filtered_attributes(
450
- schema=self._schema[component],
451
- component_data_filter=self._data_filter[component],
452
- ),
453
- )
454
- for component in components
455
- if component in self._data_filter
456
- }
457
-
458
- def _filter_attributes(self, attributes):
459
- keys_to_remove = []
460
- for attr, array in attributes.items():
461
- if is_columnar(array):
462
- continue
463
- if is_nan_or_equivalent(array):
464
- keys_to_remove.append(attr)
465
- for key in keys_to_remove:
466
- del attributes[key]
467
-
468
- def _filter_with_mapping(self):
469
- for component_type, attributes in self._data.items():
470
- if component_type in self._data_filter:
471
- filter_option = self._data_filter[component_type]
472
- if filter_option is ComponentAttributeFilterOptions.relevant:
473
- self._filter_attributes(attributes)
474
-
475
- def _post_filtering(self):
476
- if isinstance(self._data_filter, dict):
477
- self._filter_with_mapping()
478
-
479
-
480
- def _get_filtered_attributes(
481
- schema: ComponentMetaData,
482
- component_data_filter: set[str] | list[str] | None | ComponentAttributeFilterOptions,
483
- ) -> list[str] | None:
484
- if component_data_filter is None:
485
- return None
486
-
487
- if isinstance(component_data_filter, ComponentAttributeFilterOptions):
488
- return [] if schema.dtype.names is None else list(schema.dtype.names)
489
-
490
- return list(component_data_filter)
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 raw dataset handler
7
+ """
8
+
9
+ from collections.abc import Mapping
10
+ from typing import Any, cast
11
+
12
+ from power_grid_model._core.buffer_handling import (
13
+ BufferProperties,
14
+ CAttributeBuffer,
15
+ CBuffer,
16
+ create_buffer,
17
+ get_buffer_properties,
18
+ get_buffer_view,
19
+ )
20
+ from power_grid_model._core.data_types import (
21
+ AttributeType,
22
+ ColumnarData,
23
+ ComponentData,
24
+ Dataset,
25
+ DenseBatchColumnarData,
26
+ SingleColumnarData,
27
+ SparseBatchColumnarData,
28
+ )
29
+ from power_grid_model._core.dataset_definitions import ComponentType, DatasetType, _str_to_component_type
30
+ from power_grid_model._core.enum import ComponentAttributeFilterOptions
31
+ from power_grid_model._core.error_handling import VALIDATOR_MSG, assert_no_error
32
+ from power_grid_model._core.power_grid_core import (
33
+ ConstDatasetPtr,
34
+ DatasetInfoPtr,
35
+ MutableDatasetPtr,
36
+ WritableDatasetPtr,
37
+ get_power_grid_core as get_pgc,
38
+ )
39
+ from power_grid_model._core.power_grid_meta import ComponentMetaData, DatasetMetaData, power_grid_meta_data
40
+ from power_grid_model._core.typing import ComponentAttributeMapping, ComponentAttributeMappingDict
41
+ from power_grid_model._core.utils import (
42
+ get_dataset_type,
43
+ is_columnar,
44
+ is_nan_or_equivalent,
45
+ is_sparse,
46
+ process_data_filter,
47
+ )
48
+
49
+
50
+ class CDatasetInfo:
51
+ """
52
+ Raw Power Grid Model dataset info.
53
+ """
54
+
55
+ def __init__(self, info: DatasetInfoPtr):
56
+ self._info: DatasetInfoPtr = info
57
+
58
+ def name(self) -> str:
59
+ """
60
+ The name of the dataset type.
61
+
62
+ Returns:
63
+ The name of the dataset type
64
+ """
65
+ return get_pgc().dataset_info_name(self._info)
66
+
67
+ def dataset_type(self):
68
+ """
69
+ The name of the dataset type.
70
+
71
+ Returns:
72
+ The name of the dataset type
73
+ """
74
+ return self.name()
75
+
76
+ def is_batch(self) -> bool:
77
+ """
78
+ Whether the dataset is a batch dataset.
79
+
80
+ Returns:
81
+ Whether the dataset is a batch dataset
82
+ """
83
+ return bool(get_pgc().dataset_info_is_batch(self._info))
84
+
85
+ def batch_size(self) -> int:
86
+ """
87
+ The size of the dataset.
88
+
89
+ Returns:
90
+ The size of the dataset
91
+ """
92
+ return get_pgc().dataset_info_batch_size(self._info)
93
+
94
+ def n_components(self) -> int:
95
+ """
96
+ The amount of components in the dataset.
97
+
98
+ Returns:
99
+ The amount of components in the dataset
100
+ """
101
+ return get_pgc().dataset_info_n_components(self._info)
102
+
103
+ def components(self) -> list[ComponentType]:
104
+ """
105
+ The components in the dataset.
106
+
107
+ Returns:
108
+ A list of the component names in the dataset
109
+ """
110
+ return [
111
+ _str_to_component_type(get_pgc().dataset_info_component_name(self._info, idx))
112
+ for idx in range(self.n_components())
113
+ ]
114
+
115
+ def elements_per_scenario(self) -> Mapping[ComponentType, int]:
116
+ """
117
+ The number of elements per scenario in the dataset.
118
+
119
+ Returns:
120
+ The number of elements per senario for each component in the dataset;
121
+ or -1 if the scenario is not uniform (different amount per scenario)
122
+ """
123
+ return {
124
+ component_name: get_pgc().dataset_info_elements_per_scenario(self._info, idx)
125
+ for idx, component_name in enumerate(self.components())
126
+ }
127
+
128
+ def total_elements(self) -> Mapping[ComponentType, int]:
129
+ """
130
+ The total number of elements in the dataset.
131
+
132
+ Returns:
133
+ The total number of elements for each component.
134
+ For each component, if the number of elements per scenario is uniform, its value shall be equal to
135
+ the product of the batch size and the amount of elements per scenario for that component.
136
+ """
137
+ return {
138
+ component_name: get_pgc().dataset_info_total_elements(self._info, idx)
139
+ for idx, component_name in enumerate(self.components())
140
+ }
141
+
142
+ def attribute_indications(self) -> Mapping[ComponentType, None | list[AttributeType]]:
143
+ """
144
+ The attribute indications in the dataset.
145
+
146
+ Returns:
147
+ A map of component to its attribute indications.
148
+ None means no attribute indications
149
+ """
150
+ result_dict: dict[ComponentType, None | list[AttributeType]] = {}
151
+ components = self.components()
152
+ for component_idx, component_name in enumerate(components):
153
+ has_indications = get_pgc().dataset_info_has_attribute_indications(self._info, component_idx)
154
+ if has_indications == 0:
155
+ result_dict[component_name] = None
156
+ else:
157
+ n_indications = get_pgc().dataset_info_n_attribute_indications(self._info, component_idx)
158
+ result_dict[component_name] = [
159
+ get_pgc().dataset_info_attribute_name(self._info, component_idx, attribute_idx)
160
+ for attribute_idx in range(n_indications)
161
+ ]
162
+ return result_dict
163
+
164
+
165
+ class CMutableDataset:
166
+ """
167
+ A view of a user-owned dataset.
168
+
169
+ This may be used to provide a user dataset to the Power Grid Model.
170
+
171
+ The dataset will create mutable buffers that the Power Grid Model can use to load data.
172
+ """
173
+
174
+ _dataset_type: DatasetType
175
+ _schema: DatasetMetaData
176
+ _is_batch: bool
177
+ _batch_size: int
178
+ _mutable_dataset: MutableDatasetPtr
179
+ _buffer_views: list[CBuffer]
180
+
181
+ def __new__(cls, data: Dataset, dataset_type: Any = None):
182
+ instance = super().__new__(cls)
183
+ instance._mutable_dataset = MutableDatasetPtr()
184
+ instance._buffer_views = []
185
+
186
+ instance._dataset_type = dataset_type if dataset_type in DatasetType else get_dataset_type(data)
187
+ instance._schema = power_grid_meta_data[instance._dataset_type]
188
+
189
+ if data:
190
+ first_component, first_component_data = next(iter(data.items()))
191
+ first_sub_info = get_buffer_properties(data=first_component_data, schema=instance._schema[first_component])
192
+ instance._is_batch = first_sub_info.is_batch
193
+ instance._batch_size = first_sub_info.batch_size
194
+ else:
195
+ instance._is_batch = False
196
+ instance._batch_size = 1
197
+
198
+ instance._mutable_dataset = get_pgc().create_dataset_mutable(
199
+ instance._dataset_type.value, instance._is_batch, instance._batch_size
200
+ )
201
+ assert_no_error()
202
+
203
+ instance._add_data(data)
204
+ assert_no_error()
205
+
206
+ return instance
207
+
208
+ def get_dataset_ptr(self) -> MutableDatasetPtr:
209
+ """
210
+ Get the raw underlying mutable dataset pointer.
211
+
212
+ Returns:
213
+ MutableDatasetPtr: the raw underlying mutable dataset pointer.
214
+ """
215
+ return self._mutable_dataset
216
+
217
+ def get_info(self) -> CDatasetInfo:
218
+ """
219
+ Get the info for this dataset.
220
+
221
+ Returns:
222
+ The dataset info for this dataset.
223
+ """
224
+ return CDatasetInfo(get_pgc().dataset_mutable_get_info(self._mutable_dataset))
225
+
226
+ def get_buffer_views(self) -> list[CBuffer]:
227
+ """
228
+ Get list of buffer views
229
+
230
+ Returns:
231
+ list of buffer view
232
+ """
233
+ return self._buffer_views
234
+
235
+ def _add_data(self, data: Dataset):
236
+ """
237
+ Add Power Grid Model data to the mutable dataset view.
238
+
239
+ Args:
240
+ data: the data.
241
+
242
+ Raises:
243
+ ValueError: if the component is unknown.
244
+ ValueError: if the data is inconsistent with the rest of the dataset.
245
+ PowerGridError: if there was an internal error.
246
+ """
247
+ for component, component_data in data.items():
248
+ self._add_component_data(component, component_data, allow_unknown=False)
249
+
250
+ def _add_component_data(self, component: ComponentType, data: ComponentData, allow_unknown: bool = False):
251
+ """
252
+ Add Power Grid Model data for a single component to the mutable dataset view.
253
+
254
+ Args:
255
+ component: the name of the component
256
+ data: the data of the component
257
+ allow_unknown (optional): ignore any unknown components. Defaults to False.
258
+
259
+ Raises:
260
+ ValueError: if the component is unknown and allow_unknown is False.
261
+ ValueError: if the data is inconsistent with the rest of the dataset.
262
+ PowerGridError: if there was an internal error.
263
+ """
264
+ if component not in self._schema:
265
+ if not allow_unknown:
266
+ raise ValueError(f"Unknown component {component} in schema. {VALIDATOR_MSG}")
267
+ return
268
+
269
+ self._validate_properties(data, self._schema[component])
270
+ c_buffer = get_buffer_view(data, self._schema[component], self._is_batch, self._batch_size)
271
+ self._buffer_views.append(c_buffer)
272
+ self._register_buffer(component, c_buffer)
273
+
274
+ def _register_buffer(self, component: ComponentType, buffer: CBuffer):
275
+ get_pgc().dataset_mutable_add_buffer(
276
+ dataset=self._mutable_dataset,
277
+ component=component.value,
278
+ elements_per_scenario=buffer.n_elements_per_scenario,
279
+ total_elements=buffer.total_elements,
280
+ indptr=buffer.indptr,
281
+ data=buffer.data,
282
+ )
283
+ assert_no_error()
284
+ for attr, attr_data in buffer.attribute_data.items():
285
+ self._register_attribute_buffer(component, attr, attr_data)
286
+
287
+ def _register_attribute_buffer(self, component, attr, attr_data):
288
+ get_pgc().dataset_mutable_add_attribute_buffer(
289
+ dataset=self._mutable_dataset,
290
+ component=component.value,
291
+ attribute=attr,
292
+ data=attr_data.data,
293
+ )
294
+ assert_no_error()
295
+
296
+ def _validate_properties(self, data: ComponentData, schema: ComponentMetaData):
297
+ properties = get_buffer_properties(data, schema=schema, is_batch=None, batch_size=None)
298
+ if properties.is_batch != self._is_batch:
299
+ raise ValueError(
300
+ f"Dataset type (single or batch) must be consistent across all components. {VALIDATOR_MSG}"
301
+ )
302
+ if properties.batch_size != self._batch_size:
303
+ raise ValueError(f"Dataset must have a consistent batch size across all components. {VALIDATOR_MSG}")
304
+
305
+ def __del__(self):
306
+ get_pgc().destroy_dataset_mutable(self._mutable_dataset)
307
+
308
+
309
+ class CConstDataset:
310
+ """
311
+ A view of a user-owned dataset.
312
+
313
+ This may be used to provide a user dataset to the Power Grid Model.
314
+
315
+ The dataset will create const buffers that the Power Grid Model can use to load data.
316
+
317
+ It is created from mutable dataset.
318
+ """
319
+
320
+ _const_dataset: ConstDatasetPtr
321
+ _buffer_views: list[CBuffer]
322
+
323
+ def __new__(cls, data: Dataset, dataset_type: DatasetType | None = None):
324
+ instance = super().__new__(cls)
325
+ instance._const_dataset = ConstDatasetPtr()
326
+
327
+ # create from mutable dataset
328
+ mutable_dataset = CMutableDataset(data=data, dataset_type=dataset_type)
329
+ instance._const_dataset = get_pgc().create_dataset_const_from_mutable(mutable_dataset.get_dataset_ptr())
330
+ assert_no_error()
331
+ instance._buffer_views = mutable_dataset.get_buffer_views()
332
+
333
+ return instance
334
+
335
+ def get_dataset_ptr(self) -> ConstDatasetPtr:
336
+ """
337
+ Get the raw underlying const dataset pointer.
338
+
339
+ Returns:
340
+ ConstDatasetPtr: the raw underlying const dataset pointer.
341
+ """
342
+ return self._const_dataset
343
+
344
+ def get_info(self) -> CDatasetInfo:
345
+ """
346
+ Get the info for this dataset.
347
+
348
+ Returns:
349
+ The dataset info for this dataset.
350
+ """
351
+ return CDatasetInfo(get_pgc().dataset_const_get_info(self._const_dataset))
352
+
353
+ def set_next_cartesian_product_dimension(self, next_dataset: "CConstDataset") -> None:
354
+ """
355
+ Set the next dataset in the linked list.
356
+
357
+ Args:
358
+ next_dataset: The next dataset to set.
359
+ """
360
+ get_pgc().dataset_const_set_next_cartesian_product_dimension(self._const_dataset, next_dataset._const_dataset)
361
+ assert_no_error()
362
+
363
+ def __del__(self):
364
+ get_pgc().destroy_dataset_const(self._const_dataset)
365
+
366
+
367
+ class CWritableDataset:
368
+ """
369
+ A view of a Power Grid Model-owned dataset.
370
+
371
+ This may be used to retrieve data from the Power Grid Model.
372
+
373
+ This class provides buffers to which the Power Grid Model can write data in an external call.
374
+ After writing to the buffers, the data contents can be retrieved.
375
+ """
376
+
377
+ def __init__(self, dataset_ptr: WritableDatasetPtr, data_filter: ComponentAttributeMapping):
378
+ self._writable_dataset = dataset_ptr
379
+
380
+ info = self.get_info()
381
+ self._dataset_type = info.dataset_type()
382
+ self._schema = power_grid_meta_data[self._dataset_type]
383
+
384
+ self._data_filter = process_data_filter(
385
+ dataset_type=info.dataset_type(),
386
+ data_filter=data_filter,
387
+ available_components=info.components(),
388
+ )
389
+
390
+ self._data: Dataset = {}
391
+ self._buffers: Mapping[str, CBuffer] = {}
392
+ self._component_buffer_properties = self._get_buffer_properties(info)
393
+
394
+ self._add_buffers()
395
+ assert_no_error()
396
+
397
+ def get_dataset_ptr(self) -> WritableDatasetPtr:
398
+ """
399
+ Get the raw underlying writable dataset pointer.
400
+
401
+ Returns:
402
+ WritableDatasetPtr: the raw underlying writable dataset pointer.
403
+ """
404
+ return self._writable_dataset
405
+
406
+ def get_info(self) -> CDatasetInfo:
407
+ """
408
+ Get the info for this dataset.
409
+
410
+ Returns:
411
+ The dataset info for this dataset.
412
+ """
413
+ return CDatasetInfo(get_pgc().dataset_writable_get_info(self._writable_dataset))
414
+
415
+ def get_data(self) -> Dataset:
416
+ """
417
+ Retrieve data from the Power Grid Model dataset.
418
+
419
+ The Power Grid Model may write to these buffers at a later point in time.
420
+
421
+ Returns:
422
+ The full dataset with filters applied.
423
+ """
424
+ self._post_filtering()
425
+ return self._data
426
+
427
+ def get_component_data(self, component: ComponentType) -> ComponentData:
428
+ """
429
+ Retrieve Power Grid Model data from the dataset for a specific component.
430
+
431
+ Args:
432
+ component: the component.
433
+
434
+ Returns:
435
+ The dataset for the specified component.
436
+ """
437
+ return self._data[component]
438
+
439
+ def get_data_filter(self) -> ComponentAttributeMappingDict:
440
+ """Gets the data filter requested
441
+
442
+ Returns:
443
+ ComponentAttributeMappingDict: data filter
444
+ """
445
+ return self._data_filter
446
+
447
+ def _add_buffers(self):
448
+ for component, buffer_properties in self._component_buffer_properties.items():
449
+ self._add_buffer(component, buffer_properties)
450
+
451
+ def _add_buffer(self, component: ComponentType, buffer_properties: BufferProperties):
452
+ schema = self._schema[component]
453
+
454
+ self._data[component] = create_buffer(buffer_properties, schema)
455
+ self._register_buffer(component, get_buffer_view(self._data[component], schema))
456
+
457
+ def _register_buffer(self, component: ComponentType, buffer: CBuffer):
458
+ get_pgc().dataset_writable_set_buffer(
459
+ dataset=self._writable_dataset,
460
+ component=component,
461
+ indptr=buffer.indptr,
462
+ data=buffer.data,
463
+ )
464
+ assert_no_error()
465
+ for attribute, attribute_data in buffer.attribute_data.items():
466
+ self._register_attribute_buffer(component, attribute, attribute_data)
467
+
468
+ def _register_attribute_buffer(
469
+ self,
470
+ component: ComponentType,
471
+ attribute: AttributeType,
472
+ buffer: CAttributeBuffer,
473
+ ):
474
+ get_pgc().dataset_writable_set_attribute_buffer(
475
+ dataset=self._writable_dataset,
476
+ component=component,
477
+ attribute=attribute,
478
+ data=buffer.data,
479
+ )
480
+ assert_no_error()
481
+
482
+ def _get_buffer_properties(self, info: CDatasetInfo) -> Mapping[ComponentType, BufferProperties]:
483
+ is_batch = info.is_batch()
484
+ batch_size = info.batch_size()
485
+ components = info.components()
486
+ n_elements_per_scenario = info.elements_per_scenario()
487
+ n_total_elements = info.total_elements()
488
+ attribute_indications = info.attribute_indications()
489
+
490
+ return {
491
+ component: BufferProperties(
492
+ is_sparse=n_elements_per_scenario[component] == -1,
493
+ is_batch=is_batch,
494
+ batch_size=batch_size,
495
+ n_elements_per_scenario=n_elements_per_scenario[component],
496
+ n_total_elements=n_total_elements[component],
497
+ columns=_get_filtered_attributes(
498
+ schema=self._schema[component],
499
+ component_data_filter=self._data_filter[component],
500
+ attribute_indication=attribute_indications[component],
501
+ ),
502
+ )
503
+ for component in components
504
+ if component in self._data_filter
505
+ }
506
+
507
+ def _filter_attributes(self, buffer: ColumnarData):
508
+ if is_sparse(buffer):
509
+ attributes = cast(SparseBatchColumnarData, buffer)["data"]
510
+ else:
511
+ attributes = cast(SingleColumnarData | DenseBatchColumnarData, buffer)
512
+
513
+ keys_to_remove = []
514
+ for attr, array in attributes.items():
515
+ if is_nan_or_equivalent(array):
516
+ keys_to_remove.append(attr)
517
+ for key in keys_to_remove:
518
+ del attributes[key]
519
+
520
+ def _filter_with_mapping(self):
521
+ for component_type, component_buffer in self._data.items():
522
+ if component_type in self._data_filter:
523
+ filter_option = self._data_filter[component_type]
524
+ if filter_option is ComponentAttributeFilterOptions.relevant and is_columnar(component_buffer):
525
+ self._filter_attributes(component_buffer)
526
+
527
+ def _post_filtering(self):
528
+ if isinstance(self._data_filter, dict):
529
+ self._filter_with_mapping()
530
+
531
+
532
+ def _get_filtered_attributes(
533
+ schema: ComponentMetaData,
534
+ component_data_filter: set[str] | list[str] | None | ComponentAttributeFilterOptions,
535
+ attribute_indication: None | list[AttributeType],
536
+ ) -> list[AttributeType] | None:
537
+ if component_data_filter is None:
538
+ return None
539
+
540
+ if isinstance(component_data_filter, ComponentAttributeFilterOptions):
541
+ if component_data_filter == ComponentAttributeFilterOptions.relevant and attribute_indication is not None:
542
+ return attribute_indication
543
+ return [] if schema.dtype.names is None else list(schema.dtype.names)
544
+
545
+ return list(component_data_filter)