diff --git a/.gitignore b/.gitignore
index 110b0a6..b6f5604 100644
--- a/.gitignore
+++ b/.gitignore
@@ -9,3 +9,5 @@ wheels/
# Virtual environments
.venv
.env
+tmp
+pytest.log
diff --git a/pyproject.toml b/pyproject.toml
index c7b43f8..df4d6e3 100644
--- a/pyproject.toml
+++ b/pyproject.toml
@@ -19,6 +19,16 @@ simulate = [
requires = ["uv_build>=0.9.15,<0.10.0"]
build-backend = "uv_build"
+[tool.poe.tasks.precommit]
+sequence = [
+ {cmd = "uv sync --all-groups --all-extras"},
+ {cmd = "uvx ruff format"},
+ {cmd = "uvx ruff check"},
+ {cmd = "uvx ty check"},
+ {cmd = "pytest --basetemp=tmp"},
+]
+
+
[tool.pytest.ini_options]
testpaths = ["tests"]
log_cli_level = "ERROR"
@@ -27,5 +37,6 @@ log_file = "./pytest.log"
[dependency-groups]
dev = [
+ "poethepoet>=0.40.0",
"pytest>=9.0.2",
]
diff --git a/src/pyxems/csx.py b/src/pyxems/csx.py
new file mode 100644
index 0000000..bc4900f
--- /dev/null
+++ b/src/pyxems/csx.py
@@ -0,0 +1,189 @@
+from dataclasses import dataclass, field
+from typing import Literal, get_args, Protocol
+
+Axes = Literal["X", "Y", "Z"]
+
+
+@dataclass(frozen=True)
+class Line:
+ axe: Axes
+ position: list[float] = field(default_factory=list)
+
+ def to_xml(self) -> str:
+ xml = f'<{self.axe.upper()}Lines Qty="{len(self.position)}">'
+ xml += ",".join([f"{value}" for value in self.position])
+ xml += f"{self.axe.upper()}Lines>"
+ return xml
+
+
+@dataclass()
+class Physical:
+ value: tuple[float, float, float]
+
+ def __init__(self, value: float | int | tuple[float, float, float]):
+ if isinstance(value, float) or isinstance(value, int):
+ self.value = (value, 1.0, 1.0)
+ else:
+ self.value = value
+
+ def __str__(self, short=True) -> str:
+ if short:
+ return f"{self.value[0]:g}"
+ else:
+ return ",".join([f"{v:e}" for v in self.value])
+
+
+@dataclass()
+class Material:
+ name: str
+ epsilon_r: Physical
+ mu_r: Physical
+ kappa: Physical = field(default_factory=lambda: Physical((0, 0, 0)))
+ sigma: Physical = field(default_factory=lambda: Physical((0, 0, 0)))
+ density: float = 0.0
+
+ def to_xml(self, short=True) -> str:
+ if short:
+ return f'<{self.name} Epsilon="{self.epsilon_r}" Mue="{self.mu_r}" Kappa="{self.kappa}" Sigma="{self.sigma}" />'
+ else:
+ return f'<{self.name} Epsilon="{self.epsilon_r.__str__(short=False)}" Mue="{self.mu_r.__str__(short=False)}" Kappa="{self.kappa.__str__(short=False)}" Sigma="{self.sigma.__str__(short=False)}" Density="{self.density:e}" />'
+
+
+@dataclass(frozen=True)
+class Color:
+ r: int
+ g: int
+ b: int
+ a: int = 255
+
+ def to_xml(self) -> str:
+ return f'R="{self.r}" G="{self.g}" B="{self.b}" a="{self.a}"'
+
+
+class Primitive(Protocol):
+ def to_xml(self) -> str: ...
+
+
+point = tuple[float, float, float]
+
+
+@dataclass(frozen=True)
+class Box(Primitive):
+ start: point
+ stop: point
+ priority: int = 0
+
+ def to_xml(self) -> str:
+ xml = f'\n'
+ xml += f' \n'
+ xml += f' \n'
+ xml += ""
+ return xml
+
+
+@dataclass(frozen=True)
+class Property:
+ name: str
+ id: int
+ kind: Literal["Metal", "Material"] = "Material"
+ fillcolor: Color = field(default_factory=lambda: Color(255, 255, 255, 255))
+ edgecolor: Color = field(default_factory=lambda: Color(0, 0, 0, 255))
+ material: Material = field(
+ default_factory=lambda: Material("Property", Physical(1.0), Physical(1.0))
+ )
+ weight: Material = field(
+ default_factory=lambda: Material(
+ "Weight",
+ Physical(1.0),
+ Physical(1.0),
+ Physical((1.0, 1.0, 1.0)),
+ Physical((1.0, 1.0, 1.0)),
+ 1.0,
+ )
+ )
+ _primitive: list[Primitive] = field(default_factory=list)
+
+ def to_xml(self) -> str:
+ iso = ' Isotropy="1"' if self.kind == "Material" else ""
+ xml = f'<{self.kind} ID="{self.id}" Name="{self.name}"{iso}>\n'
+ xml += f" \n"
+ xml += f" \n"
+ xml += " \n"
+ for primitive in self._primitive:
+ for line in primitive.to_xml().splitlines():
+ xml += f" {line}\n"
+ xml += " \n"
+ if self.kind == "Material":
+ xml += f" {self.material.to_xml(False)}\n"
+ xml += f" {self.weight.to_xml(False)}\n"
+ xml += f"{self.kind}>\n"
+ return xml
+
+
+@dataclass(frozen=True)
+class ContinousStructure:
+ coordinates_system: int = 0
+ lines: dict[Axes, Line] = field(default_factory=dict)
+ background_material: Material = field(
+ default_factory=lambda: Material(
+ "BackgroundMaterial", Physical(1.0), Physical(1.0)
+ )
+ )
+ properties: list[Property] = field(default_factory=list)
+
+ def __post_init__(self):
+ for axe in get_args(Axes):
+ self.lines[axe] = Line(axe.upper())
+
+ def add_line(self, axe: Axes, position: float):
+ self.lines[axe].position.append(position)
+
+ def add_property(
+ self,
+ kind: Literal["Metal", "Material"],
+ name: str,
+ fillcolor: Color = Color(255, 255, 255, 255),
+ edgecolor: Color = Color(0, 0, 0, 255),
+ eps: float = 1.0,
+ mu: float = 1.0,
+ kappa: float = 0.0,
+ sigma: float = 0.0,
+ ):
+ id = len(self.properties)
+ prop = Property(
+ name,
+ id,
+ kind,
+ fillcolor,
+ edgecolor,
+ material=Material(
+ "Property",
+ Physical(eps),
+ Physical(mu),
+ Physical((kappa, 0, 0)),
+ Physical((sigma, 0, 0)),
+ ),
+ )
+ self.properties.append(prop)
+
+ def add_box(
+ self, start: point, stop: point, priority: int = 0, property_id: int = 0
+ ):
+ box = Box(start, stop, priority)
+ self.properties[property_id]._primitive.append(box)
+
+ def to_xml(self) -> str:
+ xml = f'\n'
+ xml += ' \n'
+ for line in self.lines.values():
+ xml += f" {line.to_xml()}\n"
+ xml += " \n"
+ xml += f" {self.background_material.to_xml()}\n"
+ xml += " \n"
+ xml += " \n"
+ for property in self.properties:
+ for line in property.to_xml().splitlines():
+ xml += f" {line}\n"
+ xml += " \n"
+ xml += "\n"
+ return xml
diff --git a/src/pyxems/fdtd.py b/src/pyxems/fdtd.py
new file mode 100644
index 0000000..e235e17
--- /dev/null
+++ b/src/pyxems/fdtd.py
@@ -0,0 +1,35 @@
+from dataclasses import dataclass, field
+from typing import Literal
+
+Boundary = Literal["MUR", "PEC", "PMC", "PML"]
+
+
+@dataclass(frozen=True)
+class BoundaryCond:
+ xmin: Boundary = "MUR"
+ xmax: Boundary = "MUR"
+ ymin: Boundary = "MUR"
+ ymax: Boundary = "MUR"
+ zmin: Boundary = "MUR"
+ zmax: Boundary = "MUR"
+
+ def to_xml(self) -> str:
+ elem = ""
+ return elem
+
+
+@dataclass(frozen=True)
+class FDTDConfig:
+ max_time_step: int = 1_000_000
+ boundary_cond: BoundaryCond = field(default_factory=BoundaryCond)
+ exitation: int = 0
+
+ def to_xml(self) -> str:
+ xml = f'\n'
+ xml += f" {self.boundary_cond.to_xml()}\n"
+ xml += f" \n"
+ xml += "\n"
+ return xml
diff --git a/src/pyxems/main.py b/src/pyxems/main.py
index 56eacec..2795d5c 100644
--- a/src/pyxems/main.py
+++ b/src/pyxems/main.py
@@ -1,93 +1,8 @@
from pathlib import Path
-from typing import Literal, get_args
from dataclasses import dataclass, field
-Boundary = Literal["MUR", "PEC", "PMC", "PML"]
-
-
-@dataclass(frozen=True)
-class Material:
- name: str
- epsilon_r: float
- mu_r: float
- kappa: float = 0.0
- sigma: float = 0.0
-
- def to_xml(self) -> str:
- return f'<{self.name} Epsilon="{self.epsilon_r:g}" Mue="{self.mu_r:g}" Kappa="{self.kappa:g}" Sigma="{self.sigma:g}" />'
-
-
-@dataclass(frozen=True)
-class BoundaryCond:
- xmin: Boundary = "MUR"
- xmax: Boundary = "MUR"
- ymin: Boundary = "MUR"
- ymax: Boundary = "MUR"
- zmin: Boundary = "MUR"
- zmax: Boundary = "MUR"
-
- def to_xml(self) -> str:
- elem = ""
- return elem
-
-
-@dataclass(frozen=True)
-class FDTDConfig:
- max_time_step: int = 1_000_000
- boundary_cond: BoundaryCond = field(default_factory=BoundaryCond)
- exitation: int = 0
-
- def to_xml(self) -> str:
- xml = f'\n'
- xml += f" {self.boundary_cond.to_xml()}\n"
- xml += f" \n"
- xml += "\n"
- return xml
-
-
-Axes = Literal["X", "Y", "Z"]
-
-
-@dataclass(frozen=True)
-class Line:
- axe: Axes
- position: list[float] = field(default_factory=list)
-
- def to_xml(self) -> str:
- xml = f'<{self.axe.upper()}Lines Qty="{len(self.position)}">'
- xml += ",".join([f"{value}" for value in self.position])
- xml += f"{self.axe.upper()}Lines>"
- return xml
-
-
-@dataclass(frozen=True)
-class ContinousStructure:
- coordinates_system: int = 0
- lines: dict[Axes, Line] = field(default_factory=dict)
- background_material: Material = field(
- default_factory=lambda: Material("BackgroundMaterial", 1.0, 1.0)
- )
- # TODO: add parameters settings for the structure
-
- def __post_init__(self):
- for axe in get_args(Axes):
- self.lines[axe] = Line(axe.upper())
-
- def add_line(self, axe: Axes, position: float):
- self.lines[axe].position.append(position)
-
- def to_xml(self) -> str:
- xml = f'\n'
- xml += ' \n'
- for line in self.lines.values():
- xml += f" {line.to_xml()}\n"
- xml += " \n"
- xml += f" {self.background_material.to_xml()}\n"
- xml += "\n"
- return xml
+from pyxems.fdtd import FDTDConfig
+from pyxems.csx import ContinousStructure
@dataclass(frozen=True)
diff --git a/src/pyxems/run.py b/src/pyxems/run.py
index 702a750..ec63a17 100644
--- a/src/pyxems/run.py
+++ b/src/pyxems/run.py
@@ -17,14 +17,29 @@
app = cyclopts.App(__name__)
+def find_openems_executable() -> Optional[Path]:
+ if which("openEMS") is not None:
+ return Path(which("openEMS")) # type: ignore
+ load_dotenv()
+ if "OPENEMS_PATH" in os.environ:
+ openems_path = Path(os.environ["OPENEMS_PATH"]) / "openEMS"
+ if openems_path.is_file():
+ return openems_path
+ return None
+
+
+def check_config() -> bool:
+ openems_executable = find_openems_executable()
+ return openems_executable is not None
+
+
@app.command()
def simulate(config_path: Path, run_dir: Optional[Path]) -> CompletedProcess:
- if which("openEMS") is None:
- load_dotenv()
- if "OPENEMS_PATH" in os.environ:
- openems_path = Path(os.environ["OPENEMS_PATH"]) / "openEMS"
- else:
- openems_path = "openEMS"
+ openems_path = find_openems_executable()
+ if openems_path is None:
+ raise FileNotFoundError(
+ "OpenEMS executable not found. Please ensure it is installed and in your PATH, or set the OPENEMS_PATH environment variable."
+ )
config_path = config_path.resolve()
if run_dir is None:
run_dir = config_path.parent
diff --git a/tests/test_main.py b/tests/test_main.py
index 71b32a7..1a9ed1e 100644
--- a/tests/test_main.py
+++ b/tests/test_main.py
@@ -1,5 +1,6 @@
from pathlib import Path
from pyxems.main import PyXEMSConfig, write_openEMS_xml
+from pyxems.csx import Color
import logging
@@ -157,5 +158,40 @@ def test_generate_simp_patch(tmp_path: Path):
100,
):
oems_config.csx.add_line("Z", k)
+ oems_config.csx.add_property(
+ "Metal", "patch", Color(41, 35, 190), Color(41, 35, 190, 255)
+ )
+ oems_config.csx.add_box(
+ start=(-16, -20, 1.524),
+ stop=(16, 20, 1.524),
+ priority=10,
+ property_id=0,
+ )
+ oems_config.csx.add_property(
+ "Material",
+ "substrate",
+ Color(132, 225, 108, 123),
+ Color(132, 225, 108, 123),
+ eps=3.38,
+ kappa=4.606928e-4,
+ )
+ oems_config.csx.add_box(
+ start=(-30, -30, 0),
+ stop=(30, 30, 1.524),
+ priority=0,
+ property_id=1,
+ )
+ oems_config.csx.add_property(
+ "Metal",
+ "gnd",
+ Color(214, 174, 82),
+ Color(214, 174, 82),
+ )
+ oems_config.csx.add_box(
+ start=(-30, -30, 0),
+ stop=(30, 30, 0),
+ priority=10,
+ property_id=2,
+ )
write_openEMS_xml(tmp_path / "openEMS_config.xml", oems_config)
assert (tmp_path / "openEMS_config.xml").read_text() == ref.read_text()
diff --git a/tests/test_run.py b/tests/test_run.py
index 812c34b..1d901ce 100644
--- a/tests/test_run.py
+++ b/tests/test_run.py
@@ -1,7 +1,9 @@
from pathlib import Path
-from pyxems.run import simulate
+from pyxems.run import simulate, check_config
+import pytest
+@pytest.mark.skipif(not check_config(), reason="OpenEMS exe not found.")
def test_simulate(tmp_path: Path):
config_path = Path(__file__).parent / "data" / "simp_patch.xml"
result = simulate(config_path, tmp_path)
diff --git a/uv.lock b/uv.lock
index 25548e4..f398fa9 100644
--- a/uv.lock
+++ b/uv.lock
@@ -92,6 +92,15 @@ wheels = [
{ url = "https://files.pythonhosted.org/packages/20/12/38679034af332785aac8774540895e234f4d07f7545804097de4b666afd8/packaging-25.0-py3-none-any.whl", hash = "sha256:29572ef2b1f17581046b3a2227d5c611fb25ec70ca1ba8554b24b0e69331a484", size = 66469, upload-time = "2025-04-19T11:48:57.875Z" },
]
+[[package]]
+name = "pastel"
+version = "0.2.1"
+source = { registry = "https://pypi.org/simple" }
+sdist = { url = "https://files.pythonhosted.org/packages/76/f1/4594f5e0fcddb6953e5b8fe00da8c317b8b41b547e2b3ae2da7512943c62/pastel-0.2.1.tar.gz", hash = "sha256:e6581ac04e973cac858828c6202c1e1e81fee1dc7de7683f3e1ffe0bfd8a573d", size = 7555, upload-time = "2020-09-16T19:21:12.43Z" }
+wheels = [
+ { url = "https://files.pythonhosted.org/packages/aa/18/a8444036c6dd65ba3624c63b734d3ba95ba63ace513078e1580590075d21/pastel-0.2.1-py2.py3-none-any.whl", hash = "sha256:4349225fcdf6c2bb34d483e523475de5bb04a5c10ef711263452cb37d7dd4364", size = 5955, upload-time = "2020-09-16T19:21:11.409Z" },
+]
+
[[package]]
name = "pluggy"
version = "1.6.0"
@@ -101,6 +110,19 @@ wheels = [
{ url = "https://files.pythonhosted.org/packages/54/20/4d324d65cc6d9205fabedc306948156824eb9f0ee1633355a8f7ec5c66bf/pluggy-1.6.0-py3-none-any.whl", hash = "sha256:e920276dd6813095e9377c0bc5566d94c932c33b27a3e3945d8389c374dd4746", size = 20538, upload-time = "2025-05-15T12:30:06.134Z" },
]
+[[package]]
+name = "poethepoet"
+version = "0.40.0"
+source = { registry = "https://pypi.org/simple" }
+dependencies = [
+ { name = "pastel" },
+ { name = "pyyaml" },
+]
+sdist = { url = "https://files.pythonhosted.org/packages/38/9d/054c8435b03324ed9abd5d5ab8c45065b1f42c23952cd23f13a5921d8465/poethepoet-0.40.0.tar.gz", hash = "sha256:91835f00d03d6c4f0e146f80fa510e298ad865e7edd27fe4cb9c94fdc090791b", size = 81114, upload-time = "2026-01-05T19:09:13.116Z" }
+wheels = [
+ { url = "https://files.pythonhosted.org/packages/fb/bc/73327d12b176abea7a3c6c7d760e1a953992f7b59d72c0354e39d7a353b5/poethepoet-0.40.0-py3-none-any.whl", hash = "sha256:afd276ae31d5c53573c0c14898118d4848ccee3709b6b0be6a1c6cbe522bbc8a", size = 106672, upload-time = "2026-01-05T19:09:11.536Z" },
+]
+
[[package]]
name = "pygments"
version = "2.19.2"
@@ -148,6 +170,7 @@ simulate = [
[package.dev-dependencies]
dev = [
+ { name = "poethepoet" },
{ name = "pytest" },
]
@@ -159,7 +182,56 @@ requires-dist = [
provides-extras = ["simulate"]
[package.metadata.requires-dev]
-dev = [{ name = "pytest", specifier = ">=9.0.2" }]
+dev = [
+ { name = "poethepoet", specifier = ">=0.40.0" },
+ { name = "pytest", specifier = ">=9.0.2" },
+]
+
+[[package]]
+name = "pyyaml"
+version = "6.0.3"
+source = { registry = "https://pypi.org/simple" }
+sdist = { url = "https://files.pythonhosted.org/packages/05/8e/961c0007c59b8dd7729d542c61a4d537767a59645b82a0b521206e1e25c2/pyyaml-6.0.3.tar.gz", hash = "sha256:d76623373421df22fb4cf8817020cbb7ef15c725b9d5e45f17e189bfc384190f", size = 130960, upload-time = "2025-09-25T21:33:16.546Z" }
+wheels = [
+ { url = "https://files.pythonhosted.org/packages/d1/33/422b98d2195232ca1826284a76852ad5a86fe23e31b009c9886b2d0fb8b2/pyyaml-6.0.3-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:7f047e29dcae44602496db43be01ad42fc6f1cc0d8cd6c83d342306c32270196", size = 182063, upload-time = "2025-09-25T21:32:11.445Z" },
+ { url = "https://files.pythonhosted.org/packages/89/a0/6cf41a19a1f2f3feab0e9c0b74134aa2ce6849093d5517a0c550fe37a648/pyyaml-6.0.3-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:fc09d0aa354569bc501d4e787133afc08552722d3ab34836a80547331bb5d4a0", size = 173973, upload-time = "2025-09-25T21:32:12.492Z" },
+ { url = "https://files.pythonhosted.org/packages/ed/23/7a778b6bd0b9a8039df8b1b1d80e2e2ad78aa04171592c8a5c43a56a6af4/pyyaml-6.0.3-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:9149cad251584d5fb4981be1ecde53a1ca46c891a79788c0df828d2f166bda28", size = 775116, upload-time = "2025-09-25T21:32:13.652Z" },
+ { url = "https://files.pythonhosted.org/packages/65/30/d7353c338e12baef4ecc1b09e877c1970bd3382789c159b4f89d6a70dc09/pyyaml-6.0.3-cp312-cp312-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:5fdec68f91a0c6739b380c83b951e2c72ac0197ace422360e6d5a959d8d97b2c", size = 844011, upload-time = "2025-09-25T21:32:15.21Z" },
+ { url = "https://files.pythonhosted.org/packages/8b/9d/b3589d3877982d4f2329302ef98a8026e7f4443c765c46cfecc8858c6b4b/pyyaml-6.0.3-cp312-cp312-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:ba1cc08a7ccde2d2ec775841541641e4548226580ab850948cbfda66a1befcdc", size = 807870, upload-time = "2025-09-25T21:32:16.431Z" },
+ { url = "https://files.pythonhosted.org/packages/05/c0/b3be26a015601b822b97d9149ff8cb5ead58c66f981e04fedf4e762f4bd4/pyyaml-6.0.3-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:8dc52c23056b9ddd46818a57b78404882310fb473d63f17b07d5c40421e47f8e", size = 761089, upload-time = "2025-09-25T21:32:17.56Z" },
+ { url = "https://files.pythonhosted.org/packages/be/8e/98435a21d1d4b46590d5459a22d88128103f8da4c2d4cb8f14f2a96504e1/pyyaml-6.0.3-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:41715c910c881bc081f1e8872880d3c650acf13dfa8214bad49ed4cede7c34ea", size = 790181, upload-time = "2025-09-25T21:32:18.834Z" },
+ { url = "https://files.pythonhosted.org/packages/74/93/7baea19427dcfbe1e5a372d81473250b379f04b1bd3c4c5ff825e2327202/pyyaml-6.0.3-cp312-cp312-win32.whl", hash = "sha256:96b533f0e99f6579b3d4d4995707cf36df9100d67e0c8303a0c55b27b5f99bc5", size = 137658, upload-time = "2025-09-25T21:32:20.209Z" },
+ { url = "https://files.pythonhosted.org/packages/86/bf/899e81e4cce32febab4fb42bb97dcdf66bc135272882d1987881a4b519e9/pyyaml-6.0.3-cp312-cp312-win_amd64.whl", hash = "sha256:5fcd34e47f6e0b794d17de1b4ff496c00986e1c83f7ab2fb8fcfe9616ff7477b", size = 154003, upload-time = "2025-09-25T21:32:21.167Z" },
+ { url = "https://files.pythonhosted.org/packages/1a/08/67bd04656199bbb51dbed1439b7f27601dfb576fb864099c7ef0c3e55531/pyyaml-6.0.3-cp312-cp312-win_arm64.whl", hash = "sha256:64386e5e707d03a7e172c0701abfb7e10f0fb753ee1d773128192742712a98fd", size = 140344, upload-time = "2025-09-25T21:32:22.617Z" },
+ { url = "https://files.pythonhosted.org/packages/d1/11/0fd08f8192109f7169db964b5707a2f1e8b745d4e239b784a5a1dd80d1db/pyyaml-6.0.3-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:8da9669d359f02c0b91ccc01cac4a67f16afec0dac22c2ad09f46bee0697eba8", size = 181669, upload-time = "2025-09-25T21:32:23.673Z" },
+ { url = "https://files.pythonhosted.org/packages/b1/16/95309993f1d3748cd644e02e38b75d50cbc0d9561d21f390a76242ce073f/pyyaml-6.0.3-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:2283a07e2c21a2aa78d9c4442724ec1eb15f5e42a723b99cb3d822d48f5f7ad1", size = 173252, upload-time = "2025-09-25T21:32:25.149Z" },
+ { url = "https://files.pythonhosted.org/packages/50/31/b20f376d3f810b9b2371e72ef5adb33879b25edb7a6d072cb7ca0c486398/pyyaml-6.0.3-cp313-cp313-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:ee2922902c45ae8ccada2c5b501ab86c36525b883eff4255313a253a3160861c", size = 767081, upload-time = "2025-09-25T21:32:26.575Z" },
+ { url = "https://files.pythonhosted.org/packages/49/1e/a55ca81e949270d5d4432fbbd19dfea5321eda7c41a849d443dc92fd1ff7/pyyaml-6.0.3-cp313-cp313-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:a33284e20b78bd4a18c8c2282d549d10bc8408a2a7ff57653c0cf0b9be0afce5", size = 841159, upload-time = "2025-09-25T21:32:27.727Z" },
+ { url = "https://files.pythonhosted.org/packages/74/27/e5b8f34d02d9995b80abcef563ea1f8b56d20134d8f4e5e81733b1feceb2/pyyaml-6.0.3-cp313-cp313-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:0f29edc409a6392443abf94b9cf89ce99889a1dd5376d94316ae5145dfedd5d6", size = 801626, upload-time = "2025-09-25T21:32:28.878Z" },
+ { url = "https://files.pythonhosted.org/packages/f9/11/ba845c23988798f40e52ba45f34849aa8a1f2d4af4b798588010792ebad6/pyyaml-6.0.3-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:f7057c9a337546edc7973c0d3ba84ddcdf0daa14533c2065749c9075001090e6", size = 753613, upload-time = "2025-09-25T21:32:30.178Z" },
+ { url = "https://files.pythonhosted.org/packages/3d/e0/7966e1a7bfc0a45bf0a7fb6b98ea03fc9b8d84fa7f2229e9659680b69ee3/pyyaml-6.0.3-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:eda16858a3cab07b80edaf74336ece1f986ba330fdb8ee0d6c0d68fe82bc96be", size = 794115, upload-time = "2025-09-25T21:32:31.353Z" },
+ { url = "https://files.pythonhosted.org/packages/de/94/980b50a6531b3019e45ddeada0626d45fa85cbe22300844a7983285bed3b/pyyaml-6.0.3-cp313-cp313-win32.whl", hash = "sha256:d0eae10f8159e8fdad514efdc92d74fd8d682c933a6dd088030f3834bc8e6b26", size = 137427, upload-time = "2025-09-25T21:32:32.58Z" },
+ { url = "https://files.pythonhosted.org/packages/97/c9/39d5b874e8b28845e4ec2202b5da735d0199dbe5b8fb85f91398814a9a46/pyyaml-6.0.3-cp313-cp313-win_amd64.whl", hash = "sha256:79005a0d97d5ddabfeeea4cf676af11e647e41d81c9a7722a193022accdb6b7c", size = 154090, upload-time = "2025-09-25T21:32:33.659Z" },
+ { url = "https://files.pythonhosted.org/packages/73/e8/2bdf3ca2090f68bb3d75b44da7bbc71843b19c9f2b9cb9b0f4ab7a5a4329/pyyaml-6.0.3-cp313-cp313-win_arm64.whl", hash = "sha256:5498cd1645aa724a7c71c8f378eb29ebe23da2fc0d7a08071d89469bf1d2defb", size = 140246, upload-time = "2025-09-25T21:32:34.663Z" },
+ { url = "https://files.pythonhosted.org/packages/9d/8c/f4bd7f6465179953d3ac9bc44ac1a8a3e6122cf8ada906b4f96c60172d43/pyyaml-6.0.3-cp314-cp314-macosx_10_13_x86_64.whl", hash = "sha256:8d1fab6bb153a416f9aeb4b8763bc0f22a5586065f86f7664fc23339fc1c1fac", size = 181814, upload-time = "2025-09-25T21:32:35.712Z" },
+ { url = "https://files.pythonhosted.org/packages/bd/9c/4d95bb87eb2063d20db7b60faa3840c1b18025517ae857371c4dd55a6b3a/pyyaml-6.0.3-cp314-cp314-macosx_11_0_arm64.whl", hash = "sha256:34d5fcd24b8445fadc33f9cf348c1047101756fd760b4dacb5c3e99755703310", size = 173809, upload-time = "2025-09-25T21:32:36.789Z" },
+ { url = "https://files.pythonhosted.org/packages/92/b5/47e807c2623074914e29dabd16cbbdd4bf5e9b2db9f8090fa64411fc5382/pyyaml-6.0.3-cp314-cp314-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:501a031947e3a9025ed4405a168e6ef5ae3126c59f90ce0cd6f2bfc477be31b7", size = 766454, upload-time = "2025-09-25T21:32:37.966Z" },
+ { url = "https://files.pythonhosted.org/packages/02/9e/e5e9b168be58564121efb3de6859c452fccde0ab093d8438905899a3a483/pyyaml-6.0.3-cp314-cp314-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:b3bc83488de33889877a0f2543ade9f70c67d66d9ebb4ac959502e12de895788", size = 836355, upload-time = "2025-09-25T21:32:39.178Z" },
+ { url = "https://files.pythonhosted.org/packages/88/f9/16491d7ed2a919954993e48aa941b200f38040928474c9e85ea9e64222c3/pyyaml-6.0.3-cp314-cp314-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:c458b6d084f9b935061bc36216e8a69a7e293a2f1e68bf956dcd9e6cbcd143f5", size = 794175, upload-time = "2025-09-25T21:32:40.865Z" },
+ { url = "https://files.pythonhosted.org/packages/dd/3f/5989debef34dc6397317802b527dbbafb2b4760878a53d4166579111411e/pyyaml-6.0.3-cp314-cp314-musllinux_1_2_aarch64.whl", hash = "sha256:7c6610def4f163542a622a73fb39f534f8c101d690126992300bf3207eab9764", size = 755228, upload-time = "2025-09-25T21:32:42.084Z" },
+ { url = "https://files.pythonhosted.org/packages/d7/ce/af88a49043cd2e265be63d083fc75b27b6ed062f5f9fd6cdc223ad62f03e/pyyaml-6.0.3-cp314-cp314-musllinux_1_2_x86_64.whl", hash = "sha256:5190d403f121660ce8d1d2c1bb2ef1bd05b5f68533fc5c2ea899bd15f4399b35", size = 789194, upload-time = "2025-09-25T21:32:43.362Z" },
+ { url = "https://files.pythonhosted.org/packages/23/20/bb6982b26a40bb43951265ba29d4c246ef0ff59c9fdcdf0ed04e0687de4d/pyyaml-6.0.3-cp314-cp314-win_amd64.whl", hash = "sha256:4a2e8cebe2ff6ab7d1050ecd59c25d4c8bd7e6f400f5f82b96557ac0abafd0ac", size = 156429, upload-time = "2025-09-25T21:32:57.844Z" },
+ { url = "https://files.pythonhosted.org/packages/f4/f4/a4541072bb9422c8a883ab55255f918fa378ecf083f5b85e87fc2b4eda1b/pyyaml-6.0.3-cp314-cp314-win_arm64.whl", hash = "sha256:93dda82c9c22deb0a405ea4dc5f2d0cda384168e466364dec6255b293923b2f3", size = 143912, upload-time = "2025-09-25T21:32:59.247Z" },
+ { url = "https://files.pythonhosted.org/packages/7c/f9/07dd09ae774e4616edf6cda684ee78f97777bdd15847253637a6f052a62f/pyyaml-6.0.3-cp314-cp314t-macosx_10_13_x86_64.whl", hash = "sha256:02893d100e99e03eda1c8fd5c441d8c60103fd175728e23e431db1b589cf5ab3", size = 189108, upload-time = "2025-09-25T21:32:44.377Z" },
+ { url = "https://files.pythonhosted.org/packages/4e/78/8d08c9fb7ce09ad8c38ad533c1191cf27f7ae1effe5bb9400a46d9437fcf/pyyaml-6.0.3-cp314-cp314t-macosx_11_0_arm64.whl", hash = "sha256:c1ff362665ae507275af2853520967820d9124984e0f7466736aea23d8611fba", size = 183641, upload-time = "2025-09-25T21:32:45.407Z" },
+ { url = "https://files.pythonhosted.org/packages/7b/5b/3babb19104a46945cf816d047db2788bcaf8c94527a805610b0289a01c6b/pyyaml-6.0.3-cp314-cp314t-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:6adc77889b628398debc7b65c073bcb99c4a0237b248cacaf3fe8a557563ef6c", size = 831901, upload-time = "2025-09-25T21:32:48.83Z" },
+ { url = "https://files.pythonhosted.org/packages/8b/cc/dff0684d8dc44da4d22a13f35f073d558c268780ce3c6ba1b87055bb0b87/pyyaml-6.0.3-cp314-cp314t-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:a80cb027f6b349846a3bf6d73b5e95e782175e52f22108cfa17876aaeff93702", size = 861132, upload-time = "2025-09-25T21:32:50.149Z" },
+ { url = "https://files.pythonhosted.org/packages/b1/5e/f77dc6b9036943e285ba76b49e118d9ea929885becb0a29ba8a7c75e29fe/pyyaml-6.0.3-cp314-cp314t-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:00c4bdeba853cc34e7dd471f16b4114f4162dc03e6b7afcc2128711f0eca823c", size = 839261, upload-time = "2025-09-25T21:32:51.808Z" },
+ { url = "https://files.pythonhosted.org/packages/ce/88/a9db1376aa2a228197c58b37302f284b5617f56a5d959fd1763fb1675ce6/pyyaml-6.0.3-cp314-cp314t-musllinux_1_2_aarch64.whl", hash = "sha256:66e1674c3ef6f541c35191caae2d429b967b99e02040f5ba928632d9a7f0f065", size = 805272, upload-time = "2025-09-25T21:32:52.941Z" },
+ { url = "https://files.pythonhosted.org/packages/da/92/1446574745d74df0c92e6aa4a7b0b3130706a4142b2d1a5869f2eaa423c6/pyyaml-6.0.3-cp314-cp314t-musllinux_1_2_x86_64.whl", hash = "sha256:16249ee61e95f858e83976573de0f5b2893b3677ba71c9dd36b9cf8be9ac6d65", size = 829923, upload-time = "2025-09-25T21:32:54.537Z" },
+ { url = "https://files.pythonhosted.org/packages/f0/7a/1c7270340330e575b92f397352af856a8c06f230aa3e76f86b39d01b416a/pyyaml-6.0.3-cp314-cp314t-win_amd64.whl", hash = "sha256:4ad1906908f2f5ae4e5a8ddfce73c320c2a1429ec52eafd27138b7f1cbe341c9", size = 174062, upload-time = "2025-09-25T21:32:55.767Z" },
+ { url = "https://files.pythonhosted.org/packages/f1/12/de94a39c2ef588c7e6455cfbe7343d3b2dc9d6b6b2f40c4c6565744c873d/pyyaml-6.0.3-cp314-cp314t-win_arm64.whl", hash = "sha256:ebc55a14a21cb14062aa4162f906cd962b28e2e9ea38f9b4391244cd8de4ae0b", size = 149341, upload-time = "2025-09-25T21:32:56.828Z" },
+]
[[package]]
name = "rich"