Skip to content

Commit de261cf

Browse files
lord-haffiDeltaDanielhf-kklein
authored
refactor(model): Use pydantic always and explicitly (#75)
--------- Co-authored-by: DeltaDaniel <[email protected]> Co-authored-by: konstantin <[email protected]>
1 parent 20f1f75 commit de261cf

9 files changed

+48
-72
lines changed

.github/workflows/unittests.yml

-4
Original file line numberDiff line numberDiff line change
@@ -10,7 +10,6 @@ jobs:
1010
strategy:
1111
matrix:
1212
python-version: ["3.11", "3.12", "3.13"]
13-
pydantic: [install_pydantic, skip_pydantic]
1413
cli: [install_typer, skip_typer]
1514
os: [ubuntu-latest]
1615
steps:
@@ -28,9 +27,6 @@ jobs:
2827
run: |
2928
python -m pip install --upgrade pip
3029
pip install tox
31-
- name: install pydantic if requested
32-
if: matrix.run_step == 'install_pydantic'
33-
run: pip install .[pydantic]
3430
- name: install typer if requested
3531
if: matrix.run_step == 'install_typer'
3632
run: pip install .[cli]

README.md

+2-8
Original file line numberDiff line numberDiff line change
@@ -81,14 +81,8 @@ Die vollständigen Beispiele finden sich in den [unittests](unittests):
8181
- Beispiel [AHB UTILTS](unittests/example_ahb_utilts_11d.py)
8282
- Beispiel [MIG UTILTS](https://github.com/Hochfrequenz/xml-fundamend-python/blob/main/unittests/example_migs.py)
8383

84-
### Verwendung mit Pydantic
85-
Per default verwendet fundamend die [dataclasses aus der Python-Standardlibrary](https://docs.python.org/3/library/dataclasses.html).
86-
Es lässt sich aber auch direkt mit [Pydantic](https://docs.pydantic.dev/latest/) und den [Pydantic dataclasses](https://docs.pydantic.dev/2.7/concepts/dataclasses/) verwenden.
87-
Wenn entweder pydantic schon installiert ist, oder mittels
88-
```bash
89-
pip install fundamend[pydantic]
90-
```
91-
mit installiert wird, dann sind Datenmodelle, die von `AhbReader` und `MigReader` zurückgegeben werden, automatisch pydantic Objekte.
84+
### Pydantic
85+
Die Datenmodelle, die von `AhbReader` und `MigReader` zurückgegeben werden, sind pydantic Objekte.
9286

9387
Mit Pydantic können die Ergebnisse auch leicht bspw. als JSON exportiert werden (was auch über ein CLI-Tool im nächsten Abschnitt) noch einfacher möglich ist.
9488
```python

pyproject.toml

+3-6
Original file line numberDiff line numberDiff line change
@@ -17,7 +17,9 @@ classifiers = [
1717
"Programming Language :: Python :: 3.12",
1818
"Programming Language :: Python :: 3.13",
1919
]
20-
dependencies = [] # add all the fundamend dependencies here, None so far
20+
dependencies = [
21+
"pydantic>=2"
22+
]
2123
dynamic = ["readme", "version"]
2224

2325
[project.optional-dependencies]
@@ -31,12 +33,7 @@ formatting = [
3133
linting = [
3234
"pylint==3.3.3"
3335
]
34-
pydantic = [
35-
"pydantic>=2"
36-
# if you install fundamend[pydantic], the dataclasses from pydantic will be used
37-
]
3836
cli = [
39-
"fundamend[pydantic]",
4037
"typer" # if you install fundamend[cli], the cli commands are available via typer
4138
]
4239
spellcheck = [

requirements.txt

+11-1
Original file line numberDiff line numberDiff line change
@@ -2,5 +2,15 @@
22
# This file is autogenerated by pip-compile with Python 3.12
33
# by the following command:
44
#
5-
# pip-compile pyproject.toml
5+
# pip-compile '.\pyproject.toml'
66
#
7+
annotated-types==0.7.0
8+
# via pydantic
9+
pydantic==2.10.5
10+
# via fundamend (pyproject.toml)
11+
pydantic-core==2.27.2
12+
# via pydantic
13+
typing-extensions==4.12.2
14+
# via
15+
# pydantic
16+
# pydantic-core

src/fundamend/models/_dataclass_wrapper.py

-14
This file was deleted.

src/fundamend/models/anwendungshandbuch.py

+11-21
Original file line numberDiff line numberDiff line change
@@ -6,11 +6,10 @@
66
from datetime import date
77
from typing import Union
88

9-
from ._dataclass_wrapper import dataclass
9+
from fundamend.models.base import FundamendBaseModel
1010

1111

12-
@dataclass(kw_only=True, eq=True, frozen=True)
13-
class Code:
12+
class Code(FundamendBaseModel):
1413
"""
1514
A single code element inside an AHB DataElement, indicated by the `<Code>` tag.
1615
"""
@@ -24,8 +23,7 @@ class Code:
2423
ahb_status: str #: e.g. 'X' # new for AHB
2524

2625

27-
@dataclass(kw_only=True, eq=True, frozen=True)
28-
class DataElement:
26+
class DataElement(FundamendBaseModel):
2927
"""
3028
A single data element, German 'Datenelement' inside an AHB Segment, indicated by the `<D_xxxx>` tag.
3129
This element can contain a single or multiple Code elements.
@@ -40,8 +38,7 @@ class DataElement:
4038
codes: list[Code]
4139

4240

43-
@dataclass(eq=True, kw_only=True, frozen=True)
44-
class DataElementGroup:
41+
class DataElementGroup(FundamendBaseModel):
4542
"""
4643
A group of data elements, German 'Datenelementgruppe' inside the AHB, indicated by the `<C_xxxx>` tag.
4744
This model can contain both the 'Datenelement' and the 'Gruppendatenelement'
@@ -65,8 +62,7 @@ class DataElementGroup:
6562
data_elements: list[DataElement]
6663

6764

68-
@dataclass(frozen=True, eq=True, unsafe_hash=True, kw_only=True)
69-
class Segment:
65+
class Segment(FundamendBaseModel):
7066
"""
7167
A segment inside an AHB, indicated by the `<S_xxxx>` tag.
7268
This model can contain both data elements and data element groups.
@@ -91,8 +87,7 @@ class Segment:
9187
data_elements: list[DataElement | DataElementGroup]
9288

9389

94-
@dataclass(kw_only=True, eq=True, frozen=True)
95-
class SegmentGroup:
90+
class SegmentGroup(FundamendBaseModel):
9691
"""
9792
A 'Segmentgruppe' inside an AHB, indicated by the `<G_xxxx>` tag.
9893
This model can contain both Segments and segment groups.
@@ -117,8 +112,7 @@ class SegmentGroup:
117112
elements: list[Union[Segment, "SegmentGroup"]]
118113

119114

120-
@dataclass(kw_only=True, eq=True, frozen=True)
121-
class Anwendungsfall:
115+
class Anwendungsfall(FundamendBaseModel):
122116
"""
123117
One 'Anwendungsfall', indicated by `<AWF>` tag, corresponds to one Prüfidentifikator or type of Message
124118
"""
@@ -138,16 +132,14 @@ class Anwendungsfall:
138132
elements: list[Union[Segment, SegmentGroup]]
139133

140134

141-
@dataclass(kw_only=True, eq=True, frozen=True)
142-
class Bedingung:
135+
class Bedingung(FundamendBaseModel):
143136
"""Ein ConditionKeyConditionText Mapping"""
144137

145138
nummer: str #: e.g. '1'
146139
text: str #: e.g. 'Nur MP-ID aus Sparte Strom'
147140

148141

149-
@dataclass(kw_only=True, eq=True, frozen=True)
150-
class UbBedingung:
142+
class UbBedingung(FundamendBaseModel):
151143
"""Eine UB-Bedingung"""
152144

153145
# Example:
@@ -156,8 +148,7 @@ class UbBedingung:
156148
text: str #: e.g. '([931] ∧ [932] [490]) ⊻ ([931] ∧ [933] [491])'
157149

158150

159-
@dataclass(kw_only=True, eq=True, frozen=True)
160-
class Paket:
151+
class Paket(FundamendBaseModel):
161152
"""Ein Bedingungspaket/PackageKeyConditionText Mapping"""
162153

163154
# Example:
@@ -166,8 +157,7 @@ class Paket:
166157
text: str #: e.g. '--'
167158

168159

169-
@dataclass(kw_only=True, eq=True, frozen=True)
170-
class Anwendungshandbuch:
160+
class Anwendungshandbuch(FundamendBaseModel):
171161
"""
172162
Ein Anwendungshandbuch, indicated by the `<AHB` tag, bündelt verschiedene Nachrichtentypen/Anwendungsfälle im
173163
selben Format oder mit der selben regulatorischen Grundlage und stellt gemeinsame Pakete & Bedingungen bereit.

src/fundamend/models/base.py

+13
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
"""
2+
Base class for all models in the fundamend package.
3+
"""
4+
5+
from pydantic import BaseModel, ConfigDict
6+
7+
8+
class FundamendBaseModel(BaseModel):
9+
"""
10+
Base class for all models in the fundamend package. Defines all models as frozen.
11+
"""
12+
13+
model_config = ConfigDict(frozen=True)

src/fundamend/models/messageimplementationguide.py

+7-13
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@
44
from enum import StrEnum
55
from typing import Union
66

7-
from ._dataclass_wrapper import dataclass
7+
from .base import FundamendBaseModel
88

99
# I didn't invent the data model ;)
1010
# pylint:disable=too-many-instance-attributes
@@ -23,8 +23,7 @@ class MigStatus(StrEnum):
2323
O = "O"
2424

2525

26-
@dataclass(kw_only=True, eq=True)
27-
class Code:
26+
class Code(FundamendBaseModel):
2827
"""
2928
A single code element inside a MIG data element, indicated by the `<Code>` tag.
3029
"""
@@ -36,8 +35,7 @@ class Code:
3635
value: str | None # e.g. 'UTILTS'
3736

3837

39-
@dataclass(kw_only=True, eq=True)
40-
class DataElement:
38+
class DataElement(FundamendBaseModel):
4139
"""
4240
A single data element inside a MIG Segment.
4341
This models both the 'Datenelement' and the 'Gruppendatenelement', indicated by the `<D_xxxx` tag.
@@ -59,8 +57,7 @@ class DataElement:
5957
codes: list[Code]
6058

6159

62-
@dataclass(eq=True, kw_only=True)
63-
class DataElementGroup:
60+
class DataElementGroup(FundamendBaseModel):
6461
"""
6562
A group of data elements, German 'Datenelementgruppe', indicated by the `<C_xxxx>` tag.
6663
Are able to contain a single or multiple data elements.
@@ -91,8 +88,7 @@ class DataElementGroup:
9188
data_elements: list[DataElement]
9289

9390

94-
@dataclass(frozen=True, eq=True, order=True, unsafe_hash=True, kw_only=True)
95-
class Segment:
91+
class Segment(FundamendBaseModel):
9692
"""
9793
A segment inside a MIG, indicated by the `<S_xxxx>` tag. A segment contains data elements and data element groups.
9894
"""
@@ -126,8 +122,7 @@ class Segment:
126122
data_elements: list[DataElement | DataElementGroup]
127123

128124

129-
@dataclass(kw_only=True, eq=True)
130-
class SegmentGroup:
125+
class SegmentGroup(FundamendBaseModel):
131126
"""
132127
A 'Segmentgruppe' inside a MIG, indicated by the `<G_xxx>` tag. A segment contains segments and segments groups.
133128
"""
@@ -160,8 +155,7 @@ class SegmentGroup:
160155
elements: list[Union[Segment, "SegmentGroup"]]
161156

162157

163-
@dataclass(kw_only=True, eq=True)
164-
class MessageImplementationGuide:
158+
class MessageImplementationGuide(FundamendBaseModel):
165159
"""
166160
message implementation guide (MIG)
167161
"""

unittests/test_pydantic_features.py

+1-5
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,7 @@
11
from pathlib import Path
22

33
import pytest
4-
5-
try:
6-
from pydantic import RootModel, TypeAdapter
7-
except ImportError:
8-
pytest.skip("Only available with pydantic", allow_module_level=True)
4+
from pydantic import RootModel, TypeAdapter
95

106
from fundamend import AhbReader, Anwendungshandbuch, MessageImplementationGuide, MigReader
117

0 commit comments

Comments
 (0)