From 9cb4d50f221f027ffdec5b359c745c0cf9cd3fd3 Mon Sep 17 00:00:00 2001 From: yyc12345 Date: Mon, 26 Jan 2026 22:52:56 +0800 Subject: [PATCH] refactor: refactor enum migration but not finished --- .../EnumsAnalyzer/CommonHelper.java | 6 +- .../EnumsAnalyzer/EnumsHelper.java | 6 +- .../EnumsAnalyzer/MainRunner.java | 100 ++++----- .../EnumsMigration/EnumsRender/json_loader.py | 190 ++++++++++++++++++ .../EnumsMigration/EnumsRender/main.py | 102 +++++++++- .../EnumsMigration/EnumsRender/pyproject.toml | 4 +- .../EnumsRender/template_render.py | 105 ++++++++++ .../templates/CKERROR.docstring.cpp.jinja | 6 + .../templates/CKERROR.docstring.hpp.jinja | 4 + .../templates/CK_CLASSID.docstring.cpp.jinja | 6 + .../templates/CK_CLASSID.docstring.hpp.jinja | 5 + .../EnumsRender/templates/generic.cs.jinja | 13 ++ .../templates/generic.docstring.cpp.jinja | 7 + .../templates/generic.docstring.cs.jinja | 7 + .../templates/generic.docstring.hpp.jinja | 8 + .../templates/generic.docstring.py.jinja | 0 .../templates/generic.docstring.rs.jinja | 0 .../EnumsRender/templates/generic.hpp.jinja | 12 ++ .../EnumsRender/templates/generic.py.jinja | 14 ++ .../EnumsRender/templates/generic.rs.jinja | 0 .../EnumsMigration/EnumsRender/utils.py | 28 +++ .../EnumsMigration/EnumsRender/uv.lock | 100 +++++++++ 22 files changed, 669 insertions(+), 54 deletions(-) create mode 100644 Assets/CodeGen/EnumsMigration/EnumsRender/json_loader.py create mode 100644 Assets/CodeGen/EnumsMigration/EnumsRender/template_render.py create mode 100644 Assets/CodeGen/EnumsMigration/EnumsRender/templates/CKERROR.docstring.cpp.jinja create mode 100644 Assets/CodeGen/EnumsMigration/EnumsRender/templates/CKERROR.docstring.hpp.jinja create mode 100644 Assets/CodeGen/EnumsMigration/EnumsRender/templates/CK_CLASSID.docstring.cpp.jinja create mode 100644 Assets/CodeGen/EnumsMigration/EnumsRender/templates/CK_CLASSID.docstring.hpp.jinja create mode 100644 Assets/CodeGen/EnumsMigration/EnumsRender/templates/generic.cs.jinja create mode 100644 Assets/CodeGen/EnumsMigration/EnumsRender/templates/generic.docstring.cpp.jinja create mode 100644 Assets/CodeGen/EnumsMigration/EnumsRender/templates/generic.docstring.cs.jinja create mode 100644 Assets/CodeGen/EnumsMigration/EnumsRender/templates/generic.docstring.hpp.jinja create mode 100644 Assets/CodeGen/EnumsMigration/EnumsRender/templates/generic.docstring.py.jinja create mode 100644 Assets/CodeGen/EnumsMigration/EnumsRender/templates/generic.docstring.rs.jinja create mode 100644 Assets/CodeGen/EnumsMigration/EnumsRender/templates/generic.hpp.jinja create mode 100644 Assets/CodeGen/EnumsMigration/EnumsRender/templates/generic.py.jinja create mode 100644 Assets/CodeGen/EnumsMigration/EnumsRender/templates/generic.rs.jinja create mode 100644 Assets/CodeGen/EnumsMigration/EnumsRender/utils.py create mode 100644 Assets/CodeGen/EnumsMigration/EnumsRender/uv.lock diff --git a/Assets/CodeGen/EnumsMigration/EnumsAnalyzer/CommonHelper.java b/Assets/CodeGen/EnumsMigration/EnumsAnalyzer/CommonHelper.java index a4cc7ab..995823e 100644 --- a/Assets/CodeGen/EnumsMigration/EnumsAnalyzer/CommonHelper.java +++ b/Assets/CodeGen/EnumsMigration/EnumsAnalyzer/CommonHelper.java @@ -137,7 +137,11 @@ public class CommonHelper { private static Path getRootDirectoryPath() throws Exception { String rootDir = System.getenv("ENUMS_MIGRATION_ROOT"); - return Paths.get(rootDir); + if (rootDir == null) { + throw new RuntimeException("Can not find essential environment variable ENUMS_MIGRATION_ROOT"); + } else { + return Paths.get(rootDir); + } } public static class InputFilePair { diff --git a/Assets/CodeGen/EnumsMigration/EnumsAnalyzer/EnumsHelper.java b/Assets/CodeGen/EnumsMigration/EnumsAnalyzer/EnumsHelper.java index 6804a4d..6c653fc 100644 --- a/Assets/CodeGen/EnumsMigration/EnumsAnalyzer/EnumsHelper.java +++ b/Assets/CodeGen/EnumsMigration/EnumsAnalyzer/EnumsHelper.java @@ -29,7 +29,11 @@ public class EnumsHelper { mHierarchy = new Vector(); } - /** The list to store this CK_CLASSID inheritance relationship. */ + /** + * The list to store this CK_CLASSID inheritance relationship. + * The first item is the oldest parent in inheritance. + * The last item is self. + */ public Vector mHierarchy; } diff --git a/Assets/CodeGen/EnumsMigration/EnumsAnalyzer/MainRunner.java b/Assets/CodeGen/EnumsMigration/EnumsAnalyzer/MainRunner.java index 8e1b078..66fb579 100644 --- a/Assets/CodeGen/EnumsMigration/EnumsAnalyzer/MainRunner.java +++ b/Assets/CodeGen/EnumsMigration/EnumsAnalyzer/MainRunner.java @@ -106,30 +106,30 @@ public class MainRunner { // =========== CKERROR =========== EnumsHelper.BEnum ckerror = organiseDefines("CKERROR.txt", "CKERROR"); JsonWriter.writeEnum(CommonHelper.getOutputFilePath("CKERROR.json"), ckerror); -// CppWriter.writeEnum("dest/CKERROR.hpp", ckerror); -// PythonWriter.writeEnum("dest/CKERROR.py", ckerror); -// CSharpWriter.writeEnum("dest/CKERROR.cs", ckerror); -// CppWriter.writeCkErrorAccVal("dest/CKERROR.AccVal.hpp", ckerror); -// PythonWriter.writeAccVal("dest/CKERROR.AccVal.py", ckerror); -// CSharpWriter.writeAccVal("dest/CKERROR.AccVal.cs", ckerror); +// CppWriter.writeEnum("CKERROR.hpp", ckerror); +// PythonWriter.writeEnum("CKERROR.py", ckerror); +// CSharpWriter.writeEnum("CKERROR.cs", ckerror); +// CppWriter.writeCkErrorAccVal("CKERROR.AccVal.hpp", ckerror); +// PythonWriter.writeAccVal("CKERROR.AccVal.py", ckerror); +// CSharpWriter.writeAccVal("CKERROR.AccVal.cs", ckerror); // =========== CK_CLASSID =========== EnumsHelper.BEnum classid = organiseClassid("CK_CLASSID.txt"); JsonWriter.writeEnum(CommonHelper.getOutputFilePath("CK_CLASSID.json"), classid); -// CppWriter.writeEnum("dest/CK_CLASSID.hpp", classid); -// PythonWriter.writeEnum("dest/CK_CLASSID.py", classid); -// CSharpWriter.writeEnum("dest/CK_CLASSID.cs", classid); -// CppWriter.writeCkClassidAccVal("dest/CK_CLASSID.AccVal.hpp", classid); -// PythonWriter.writeAccVal("dest/CK_CLASSID.AccVal.py", classid); +// CppWriter.writeEnum("CK_CLASSID.hpp", classid); +// PythonWriter.writeEnum("CK_CLASSID.py", classid); +// CSharpWriter.writeEnum("CK_CLASSID.cs", classid); +// CppWriter.writeCkClassidAccVal("CK_CLASSID.AccVal.hpp", classid); +// PythonWriter.writeAccVal("CK_CLASSID.AccVal.py", classid); // =========== Define2 =========== // Define2 do not need annotation output. // Because they are CKStateChunk used value which are not exposed to outside. EnumsHelper.BEnumCollection def2 = getEnumsCollection("Defines2.txt"); JsonWriter.writeEnums(CommonHelper.getOutputFilePath("Defines2.json"), def2); -// CppWriter.writeEnums("dest/Defines2.hpp", def2); -// PythonWriter.writeEnums("dest/Defines2.py", def2); -// CSharpWriter.writeEnums("dest/Defines2.cs", def2); +// CppWriter.writeEnums("Defines2.hpp", def2); +// PythonWriter.writeEnums("Defines2.py", def2); +// CSharpWriter.writeEnums("Defines2.cs", def2); // =========== Combined enums =========== EnumsHelper.BEnumCollection ck2Enums = getEnumsCollection("CKEnums.txt"), @@ -137,58 +137,58 @@ public class MainRunner { JsonWriter.writeEnums(CommonHelper.getOutputFilePath("CKEnums.json"), ck2Enums); JsonWriter.writeEnums(CommonHelper.getOutputFilePath("VxEnums.json"), vxEnums); -// CppWriter.writeEnums("dest/CKEnums.hpp", ck2Enums); -// PythonWriter.writeEnums("dest/CKEnums.py", ck2Enums); -// CSharpWriter.writeEnums("dest/CKEnums.cs", ck2Enums); -// CppWriter.writeAccVals("dest/CKEnums.AccVal.hpp", ck2Enums, CommonHelper.CKParts.CK2); -// PythonWriter.writeAccVals("dest/CKEnums.AccVal.py", ck2Enums); -// CSharpWriter.writeAccVals("dest/CKEnums.AccVal.cs", ck2Enums); +// CppWriter.writeEnums("CKEnums.hpp", ck2Enums); +// PythonWriter.writeEnums("CKEnums.py", ck2Enums); +// CSharpWriter.writeEnums("CKEnums.cs", ck2Enums); +// CppWriter.writeAccVals("CKEnums.AccVal.hpp", ck2Enums, CommonHelper.CKParts.CK2); +// PythonWriter.writeAccVals("CKEnums.AccVal.py", ck2Enums); +// CSharpWriter.writeAccVals("CKEnums.AccVal.cs", ck2Enums); -// CppWriter.writeEnums("dest/VxEnums.hpp", vxEnums); -// PythonWriter.writeEnums("dest/VxEnums.py", vxEnums); -// CSharpWriter.writeEnums("dest/VxEnums.cs", vxEnums); -// CppWriter.writeAccVals("dest/VxEnums.AccVal.hpp", vxEnums, CommonHelper.CKParts.VxMath); -// PythonWriter.writeAccVals("dest/VxEnums.AccVal.py", vxEnums); -// CSharpWriter.writeAccVals("dest/VxEnums.AccVal.cs", vxEnums); +// CppWriter.writeEnums("VxEnums.hpp", vxEnums); +// PythonWriter.writeEnums("VxEnums.py", vxEnums); +// CSharpWriter.writeEnums("VxEnums.cs", vxEnums); +// CppWriter.writeAccVals("VxEnums.AccVal.hpp", vxEnums, CommonHelper.CKParts.VxMath); +// PythonWriter.writeAccVals("VxEnums.AccVal.py", vxEnums); +// CSharpWriter.writeAccVals("VxEnums.AccVal.cs", vxEnums); // =========== Single enums =========== EnumsHelper.BEnum single; single = organiseDefines("CK_STATECHUNK_CHUNKVERSION.txt", "CK_STATECHUNK_CHUNKVERSION"); JsonWriter.writeEnum(CommonHelper.getOutputFilePath("CK_STATECHUNK_CHUNKVERSION.json"), single); -// CppWriter.writeEnum("dest/CK_STATECHUNK_CHUNKVERSION.hpp", single); -// PythonWriter.writeEnum("dest/CK_STATECHUNK_CHUNKVERSION.py", single); -// CSharpWriter.writeEnum("dest/CK_STATECHUNK_CHUNKVERSION.cs", single); -// CppWriter.writeAccVal("dest/CK_STATECHUNK_CHUNKVERSION.AccVal.hpp", single, CommonHelper.CKParts.CK2); -// PythonWriter.writeAccVal("dest/CK_STATECHUNK_CHUNKVERSION.AccVal.py", single); -// CSharpWriter.writeAccVal("dest/CK_STATECHUNK_CHUNKVERSION.AccVal.cs", single); +// CppWriter.writeEnum("CK_STATECHUNK_CHUNKVERSION.hpp", single); +// PythonWriter.writeEnum("CK_STATECHUNK_CHUNKVERSION.py", single); +// CSharpWriter.writeEnum("CK_STATECHUNK_CHUNKVERSION.cs", single); +// CppWriter.writeAccVal("CK_STATECHUNK_CHUNKVERSION.AccVal.hpp", single, CommonHelper.CKParts.CK2); +// PythonWriter.writeAccVal("CK_STATECHUNK_CHUNKVERSION.AccVal.py", single); +// CSharpWriter.writeAccVal("CK_STATECHUNK_CHUNKVERSION.AccVal.cs", single); single = organiseDefines("CK_STATECHUNK_DATAVERSION.txt", "CK_STATECHUNK_DATAVERSION"); JsonWriter.writeEnum(CommonHelper.getOutputFilePath("CK_STATECHUNK_DATAVERSION.json"), single); -// CppWriter.writeEnum("dest/CK_STATECHUNK_DATAVERSION.hpp", single); -// PythonWriter.writeEnum("dest/CK_STATECHUNK_DATAVERSION.py", single); -// CSharpWriter.writeEnum("dest/CK_STATECHUNK_DATAVERSION.cs", single); -// CppWriter.writeAccVal("dest/CK_STATECHUNK_DATAVERSION.AccVal.hpp", single, CommonHelper.CKParts.CK2); -// PythonWriter.writeAccVal("dest/CK_STATECHUNK_DATAVERSION.AccVal.py", single); -// CSharpWriter.writeAccVal("dest/CK_STATECHUNK_DATAVERSION.AccVal.cs", single); +// CppWriter.writeEnum("CK_STATECHUNK_DATAVERSION.hpp", single); +// PythonWriter.writeEnum("CK_STATECHUNK_DATAVERSION.py", single); +// CSharpWriter.writeEnum("CK_STATECHUNK_DATAVERSION.cs", single); +// CppWriter.writeAccVal("CK_STATECHUNK_DATAVERSION.AccVal.hpp", single, CommonHelper.CKParts.CK2); +// PythonWriter.writeAccVal("CK_STATECHUNK_DATAVERSION.AccVal.py", single); +// CSharpWriter.writeAccVal("CK_STATECHUNK_DATAVERSION.AccVal.cs", single); single = organiseDefines("CK_BITMAPDATA_FLAGS.txt", "CK_BITMAPDATA_FLAGS"); JsonWriter.writeEnum(CommonHelper.getOutputFilePath("CK_BITMAPDATA_FLAGS.json"), single); -// CppWriter.writeEnum("dest/CK_BITMAPDATA_FLAGS.hpp", single); -// PythonWriter.writeEnum("dest/CK_BITMAPDATA_FLAGS.py", single); -// CSharpWriter.writeEnum("dest/CK_BITMAPDATA_FLAGS.cs", single); -// CppWriter.writeAccVal("dest/CK_BITMAPDATA_FLAGS.AccVal.hpp", single, CommonHelper.CKParts.CK2); -// PythonWriter.writeAccVal("dest/CK_BITMAPDATA_FLAGS.AccVal.py", single); -// CSharpWriter.writeAccVal("dest/CK_BITMAPDATA_FLAGS.AccVal.cs", single); +// CppWriter.writeEnum("CK_BITMAPDATA_FLAGS.hpp", single); +// PythonWriter.writeEnum("CK_BITMAPDATA_FLAGS.py", single); +// CSharpWriter.writeEnum("CK_BITMAPDATA_FLAGS.cs", single); +// CppWriter.writeAccVal("CK_BITMAPDATA_FLAGS.AccVal.hpp", single, CommonHelper.CKParts.CK2); +// PythonWriter.writeAccVal("CK_BITMAPDATA_FLAGS.AccVal.py", single); +// CSharpWriter.writeAccVal("CK_BITMAPDATA_FLAGS.AccVal.cs", single); single = organiseDefines("CK_CAMERA_PROJECTION.txt", "CK_CAMERA_PROJECTION"); JsonWriter.writeEnum(CommonHelper.getOutputFilePath("CK_CAMERA_PROJECTION.json"), single); -// CppWriter.writeEnum("dest/CK_CAMERA_PROJECTION.hpp", single); -// PythonWriter.writeEnum("dest/CK_CAMERA_PROJECTION.py", single); -// CSharpWriter.writeEnum("dest/CK_CAMERA_PROJECTION.cs", single); -// CppWriter.writeAccVal("dest/CK_CAMERA_PROJECTION.AccVal.hpp", single, CommonHelper.CKParts.CK2); -// PythonWriter.writeAccVal("dest/CK_CAMERA_PROJECTION.AccVal.py", single); -// CSharpWriter.writeAccVal("dest/CK_CAMERA_PROJECTION.AccVal.cs", single); +// CppWriter.writeEnum("CK_CAMERA_PROJECTION.hpp", single); +// PythonWriter.writeEnum("CK_CAMERA_PROJECTION.py", single); +// CSharpWriter.writeEnum("CK_CAMERA_PROJECTION.cs", single); +// CppWriter.writeAccVal("CK_CAMERA_PROJECTION.AccVal.hpp", single, CommonHelper.CKParts.CK2); +// PythonWriter.writeAccVal("CK_CAMERA_PROJECTION.AccVal.py", single); +// CSharpWriter.writeAccVal("CK_CAMERA_PROJECTION.AccVal.cs", single); // print message. System.out.println("Done"); diff --git a/Assets/CodeGen/EnumsMigration/EnumsRender/json_loader.py b/Assets/CodeGen/EnumsMigration/EnumsRender/json_loader.py new file mode 100644 index 0000000..9b0dfd9 --- /dev/null +++ b/Assets/CodeGen/EnumsMigration/EnumsRender/json_loader.py @@ -0,0 +1,190 @@ +import json +import typing +import utils + + +class BEnumEntry: + """The struct to describe the entry of an enum.""" + + __entry_name: str + """The name of this entry.""" + __entry_value: str | None + """The value of this entry. None if this entry do not have explicit value.""" + __entry_comment: str | None + """The comment of this entry. None if no comment.""" + + def __init__( + self, entry_name: str, entry_value: str | None, entry_comment: str | None + ): + self.__entry_name = entry_name + self.__entry_value = entry_value + self.__entry_comment = entry_comment + + def get_entry_name(self) -> str: + """Get the name of this entry.""" + return self.__entry_name + + def get_entry_value(self) -> str | None: + """Get the value of this entry. None if this entry do not have explicit value.""" + return self.__entry_value + + def get_entry_comment(self) -> str | None: + """Get the comment of this entry. None if no comment.""" + return self.__entry_comment + + @staticmethod + def from_json(data: dict[str, typing.Any]) -> "BEnumEntry": + return BEnumEntry( + data["name"], + data.get("value", None), + data.get("comment", None), + ) + + +class BHierarchyEnumEntry(BEnumEntry): + """ + The specialized EnumEntry type which can store extra hierarchy info. + Used in CK_CLASSID parsing. + """ + + __hierarchy: list[str] + """ + The list to store this CK_CLASSID inheritance relationship. + The first item is the oldest parent in inheritance. + The last item is self. + """ + + def __init__( + self, + entry_name: str, + entry_value: str | None, + entry_comment: str | None, + hierarchy: list[str], + ): + super().__init__(entry_name, entry_value, entry_comment) + self.__hierarchy = hierarchy + + def iter_hierarchy(self, benum: "BEnum") -> typing.Iterator["BHierarchyEnumEntry"]: + return map( + lambda e: typing.cast(BHierarchyEnumEntry, benum.get_entry_by_name(e)), + self.__hierarchy, + ) + + @staticmethod + def from_json(data: dict[str, typing.Any]) -> "BHierarchyEnumEntry": + return BHierarchyEnumEntry( + data["name"], + data.get("value", None), + data.get("comment", None), + data["hierarchy"], + ) + + +class BEnum: + """The struct to describe an enum.""" + + __enum_name: str + """The name of this enum.""" + __enum_comment: str | None + """The comment of this enum. None if no comment.""" + __can_unsigned: bool + """True if this enum can use unsigned integer as its underlying type.""" + __use_flags: bool + """True if this enum will use flags feature (supporting OR, AND, operators).""" + __entries: list[BEnumEntry] + """The list to store entries of this enum.""" + + __entries_map: dict[str, BEnumEntry] + """The name map for `entries`.""" + + def __init__( + self, + enum_name: str, + enum_comment: str | None, + can_unsigned: bool, + use_flags: bool, + entries: list[BEnumEntry], + ): + self.__enum_name = enum_name + self.__enum_comment = enum_comment + self.__can_unsigned = can_unsigned + self.__use_flags = use_flags + self.__entries = entries + self.__entries_map = {e.get_entry_name(): e for e in entries} + + def get_enum_name(self) -> str: + """Get the name of this enum.""" + return self.__enum_name + + def get_enum_comment(self) -> str | None: + """Get the comment of this enum. None if no comment.""" + return self.__enum_comment + + def get_can_unsigned(self) -> bool: + """True if this enum can use unsigned integer as its underlying type.""" + return self.__can_unsigned + + def get_use_flags(self) -> bool: + """True if this enum will use flags feature (supporting OR, AND, operators).""" + return self.__use_flags + + def iter_entries(self) -> typing.Iterator[BEnumEntry]: + """Get the iterator of entries of this enum.""" + return iter(self.__entries) + + def get_entry_by_name(self, name: str) -> BEnumEntry: + return self.__entries_map[name] + + @staticmethod + def from_json(data: dict[str, typing.Any]) -> "BEnum": + return BEnum( + data["name"], + data.get("comment", None), + data["can_unsigned"], + data["use_flags"], + list(map(lambda i: BEnum.__create_entry_by_content(i), data["entries"])), + ) + + @staticmethod + def __create_entry_by_content(data: dict[str, typing.Any]) -> "BEnumEntry": + if "hierarchy" in data: + return BHierarchyEnumEntry.from_json(data) + else: + return BEnumEntry.from_json(data) + + +class BEnumCollection: + """The struct to describe a collection of enums.""" + + __enums: list[BEnum] + """The list to store enums.""" + + def __init__(self, enums: list[BEnum]): + self.__enums = enums + + def iter_enums(self) -> typing.Iterator[BEnum]: + """Get the iterator of enums.""" + return iter(self.__enums) + + def get_enums_count(self) -> int: + """Get the count of enums.""" + return len(self.__enums) + + def get_enum_by_index(self, index: int) -> BEnum: + """Get the enum by index.""" + return self.__enums[index] + + @staticmethod + def from_json(data: list[typing.Any]) -> "BEnumCollection": + return BEnumCollection(list(map(lambda i: BEnum.from_json(i), data))) + + +def load_enums(filename: str) -> BEnumCollection: + with open(utils.get_input_file_path(filename), "r", encoding="utf-8") as f: + return BEnumCollection.from_json(json.load(f)) + + +def load_enum(filename: str) -> BEnum: + collection = load_enums(filename) + assert collection.get_enums_count() == 1 + return collection.get_enum_by_index(0) diff --git a/Assets/CodeGen/EnumsMigration/EnumsRender/main.py b/Assets/CodeGen/EnumsMigration/EnumsRender/main.py index 8b5f1dd..7c89da5 100644 --- a/Assets/CodeGen/EnumsMigration/EnumsRender/main.py +++ b/Assets/CodeGen/EnumsMigration/EnumsRender/main.py @@ -1,5 +1,105 @@ +import json_loader +import template_render +import utils +from utils import CKParts + def main(): - print("Hello from enums-render!") + render = template_render.TemplateRender() + + # CKERROR + ckerror = json_loader.load_enum("CKERROR.json") + render.render_cpp_enum("CKERROR.hpp", ckerror) + render.render_py_enum("CKERROR.py", ckerror) + render.render_cs_enum("CKERROR.cs", ckerror) + render.render_cpp_ckerror_docstring("CKERROR.docstring.hpp", "CKERROR.docstring.cpp", ckerror) + render.render_py_enum_docstring("CKERROR.docstring.py", ckerror) + render.render_cs_enum_docstring("CKERROR.docstring.cs", ckerror) + render.render_rs_enum_docstring("CKERROR.docstring.rs", ckerror) + + # CK_CLASSID + classid = json_loader.load_enum("CK_CLASSID.json") + render.render_cpp_enum("CK_CLASSID.hpp", classid) + render.render_py_enum("CK_CLASSID.py", classid) + render.render_cs_enum("CK_CLASSID.cs", classid) + render.render_rs_enum("CK_CLASSID.rs", classid) + render.render_cpp_ckclassid_docstring("CK_CLASSID.docstring.hpp", "CK_CLASSID.docstring.cpp", classid) + render.render_py_enum_docstring("CK_CLASSID.docstring.py", classid) + render.render_cs_enum_docstring("CK_CLASSID.docstring.cs", classid) + render.render_rs_enum_docstring("CK_CLASSID.docstring.rs", classid) + + # Define2 + def2 = json_loader.load_enums("Defines2.json") + render.render_cpp_enums("Defines2.hpp", def2) + render.render_py_enums("Defines2.py", def2) + render.render_cs_enums("Defines2.cs", def2) + render.render_rs_enums("Defines2.rs", def2) + # Define2 do not need annotation output. + # Because they are CKStateChunk used value which are not exposed to outside. + + # Combined Enums + ck2Enums = json_loader.load_enums("CKEnums.json") + vxEnums = json_loader.load_enums("VxEnums.json") + + render.render_cpp_enums("CKEnums.hpp", ck2Enums) + render.render_py_enums("CKEnums.py", ck2Enums) + render.render_cs_enums("CKEnums.cs", ck2Enums) + render.render_rs_enums("CKEnums.rs", ck2Enums) + render.render_cpp_enum_docstrings("CKEnums.docstring.hpp", "CKEnums.docstring.cpp", ck2Enums, CKParts.CK2) + render.render_py_enum_docstrings("CKEnums.docstring.py", ck2Enums) + render.render_cs_enum_docstrings("CKEnums.docstring.cs", ck2Enums) + render.render_rs_enum_docstrings("CKEnums.docstring.rs", ck2Enums) + + render.render_cpp_enums("VxEnums.hpp", vxEnums) + render.render_py_enums("VxEnums.py", vxEnums) + render.render_cs_enums("VxEnums.cs", vxEnums) + render.render_rs_enums("VxEnums.rs", vxEnums) + render.render_cpp_enum_docstrings("VxEnums.docstring.hpp", "VxEnums.docstring.cpp", vxEnums, CKParts.VxMath) + render.render_py_enum_docstrings("VxEnums.docstring.py", vxEnums) + render.render_cs_enum_docstrings("VxEnums.docstring.cs", vxEnums) + render.render_rs_enum_docstrings("VxEnums.docstring.rs", vxEnums) + + # Single enums + single = json_loader.load_enum("CK_STATECHUNK_CHUNKVERSION.json") + render.render_cpp_enum("CK_STATECHUNK_CHUNKVERSION.hpp", single) + render.render_py_enum("CK_STATECHUNK_CHUNKVERSION.py", single) + render.render_cs_enum("CK_STATECHUNK_CHUNKVERSION.cs", single) + render.render_rs_enum("CK_STATECHUNK_CHUNKVERSION.rs", single) + render.render_cpp_enum_docstring("CK_STATECHUNK_CHUNKVERSION.docstring.hpp", "CK_STATECHUNK_CHUNKVERSION.docstring.cpp", single, CKParts.CK2) + render.render_py_enum_docstring("CK_STATECHUNK_CHUNKVERSION.docstring.py", single) + render.render_cs_enum_docstring("CK_STATECHUNK_CHUNKVERSION.docstring.cs", single) + render.render_rs_enum_docstring("CK_STATECHUNK_CHUNKVERSION.docstring.rs", single) + + single = json_loader.load_enum("CK_STATECHUNK_DATAVERSION.json") + render.render_cpp_enum("CK_STATECHUNK_DATAVERSION.hpp", single) + render.render_py_enum("CK_STATECHUNK_DATAVERSION.py", single) + render.render_cs_enum("CK_STATECHUNK_DATAVERSION.cs", single) + render.render_rs_enum("CK_STATECHUNK_DATAVERSION.rs", single) + render.render_cpp_enum_docstring("CK_STATECHUNK_DATAVERSION.docstring.hpp", "CK_STATECHUNK_DATAVERSION.docstring.cpp", single, CKParts.CK2) + render.render_py_enum_docstring("CK_STATECHUNK_DATAVERSION.docstring.py", single) + render.render_cs_enum_docstring("CK_STATECHUNK_DATAVERSION.docstring.cs", single) + render.render_rs_enum_docstring("CK_STATECHUNK_DATAVERSION.docstring.rs", single) + + single = json_loader.load_enum("CK_BITMAPDATA_FLAGS.json") + render.render_cpp_enum("CK_BITMAPDATA_FLAGS.hpp", single) + render.render_py_enum("CK_BITMAPDATA_FLAGS.py", single) + render.render_cs_enum("CK_BITMAPDATA_FLAGS.cs", single) + render.render_rs_enum("CK_BITMAPDATA_FLAGS.rs", single) + render.render_cpp_enum_docstring("CK_BITMAPDATA_FLAGS.docstring.hpp", "CK_BITMAPDATA_FLAGS.docstring.cpp", single, CKParts.CK2) + render.render_py_enum_docstring("CK_BITMAPDATA_FLAGS.docstring.py", single) + render.render_cs_enum_docstring("CK_BITMAPDATA_FLAGS.docstring.cs", single) + render.render_rs_enum_docstring("CK_BITMAPDATA_FLAGS.docstring.rs", single) + + single = json_loader.load_enum("CK_CAMERA_PROJECTION.json") + render.render_cpp_enum("CK_CAMERA_PROJECTION.hpp", single) + render.render_py_enum("CK_CAMERA_PROJECTION.py", single) + render.render_cs_enum("CK_CAMERA_PROJECTION.cs", single) + render.render_rs_enum("CK_CAMERA_PROJECTION.rs", single) + render.render_cpp_enum_docstring("CK_CAMERA_PROJECTION.docstring.hpp", "CK_CAMERA_PROJECTION.docstring.cpp", single, CKParts.CK2) + render.render_py_enum_docstring("CK_CAMERA_PROJECTION.docstring.py", single) + render.render_cs_enum_docstring("CK_CAMERA_PROJECTION.docstring.cs", single) + render.render_rs_enum_docstring("CK_CAMERA_PROJECTION.docstring.rs", single) + + print("Done") if __name__ == "__main__": diff --git a/Assets/CodeGen/EnumsMigration/EnumsRender/pyproject.toml b/Assets/CodeGen/EnumsMigration/EnumsRender/pyproject.toml index f03980a..34c45d3 100644 --- a/Assets/CodeGen/EnumsMigration/EnumsRender/pyproject.toml +++ b/Assets/CodeGen/EnumsMigration/EnumsRender/pyproject.toml @@ -4,4 +4,6 @@ version = "0.1.0" description = "Add your description here" readme = "README.md" requires-python = ">=3.11" -dependencies = [] +dependencies = [ + "jinja2==3.1.6", +] diff --git a/Assets/CodeGen/EnumsMigration/EnumsRender/template_render.py b/Assets/CodeGen/EnumsMigration/EnumsRender/template_render.py new file mode 100644 index 0000000..5bc8a92 --- /dev/null +++ b/Assets/CodeGen/EnumsMigration/EnumsRender/template_render.py @@ -0,0 +1,105 @@ +import jinja2 +import typing +from json_loader import BEnumCollection, BEnum +from utils import CKParts +import utils + +class RenderUtils: + pass + +class TemplateRender: + __loader: jinja2.BaseLoader + __environment: jinja2.Environment + + def __init__(self) -> None: + self.__loader = jinja2.FileSystemLoader(utils.get_template_directory()) + self.__environment = jinja2.Environment(loader=self.__loader) + + def __render(self, template_name: str, dest_filename: str, payload: BEnumCollection, extra: dict[str, typing.Any] = {}) -> None: + # prepare template argument + template_argument: dict[str, typing.Any] = { + 'payload': payload, + 'extra': extra, + 'utils': RenderUtils, + } + # fetch template + template = self.__environment.get_template(str(template_name)) + # render template and save + with open(utils.get_output_file_path(dest_filename), 'w', encoding='utf-8') as f: + f.write(template.render(**template_argument)) + + def __wrap_single_enum(self, benum: BEnum) -> BEnumCollection: + return BEnumCollection([benum, ]) + + # region: C++ Header and Sources + + def render_cpp_enums(self, filename: str, benums: BEnumCollection) -> None: + self.__render('generic.hpp.jinja', filename, benums) + + def render_cpp_enum(self, filename: str, benum: BEnum) -> None: + self.render_cpp_enums(filename, self.__wrap_single_enum(benum)) + + def render_cpp_enum_docstrings(self, hpp_filename: str, cpp_filename: str, benums: BEnumCollection, parts: CKParts) -> None: + self.__render('generic.docstring.hpp.jinja', hpp_filename, benums, {'parts': parts}) + self.__render('generic.docstring.cpp.jinja', cpp_filename, benums, {'parts': parts}) + + def render_cpp_enum_docstring(self, hpp_filename: str, cpp_filename: str, benum: BEnum, parts: CKParts) -> None: + self.render_cpp_enum_docstrings(hpp_filename, cpp_filename, self.__wrap_single_enum(benum), parts) + + def render_cpp_ckerror_docstring(self, hpp_filename: str, cpp_filename: str, benums: BEnum) -> None: + self.__render('CKERROR.docstring.hpp.jinja', hpp_filename, self.__wrap_single_enum(benums)) + self.__render('CKERROR.docstring.cpp.jinja', cpp_filename, self.__wrap_single_enum(benums)) + + def render_cpp_ckclassid_docstring(self, hpp_filename: str, cpp_filename: str, benums: BEnum) -> None: + self.__render('CK_CLASSID.docstring.hpp.jinja', hpp_filename, self.__wrap_single_enum(benums)) + self.__render('CK_CLASSID.docstring.cpp.jinja', cpp_filename, self.__wrap_single_enum(benums)) + + # endregion + + # region: Python + + def render_py_enums(self, filename: str, benums: BEnumCollection) -> None: + self.__render('generic.py.jinja', filename, benums) + + def render_py_enum(self, filename: str, benum: BEnum) -> None: + self.render_py_enums(filename, self.__wrap_single_enum(benum)) + + def render_py_enum_docstrings(self, filename: str, benums: BEnumCollection) -> None: + self.__render('generic.docstring.py.jinja', filename, benums) + + def render_py_enum_docstring(self, filename: str, benum: BEnum) -> None: + self.render_py_enum_docstrings(filename, self.__wrap_single_enum(benum)) + + # endregion + + # region: C# + + def render_cs_enums(self, filename: str, benums: BEnumCollection) -> None: + self.__render('generic.cs.jinja', filename, benums) + + def render_cs_enum(self, filename: str, benum: BEnum) -> None: + self.render_cs_enums(filename, self.__wrap_single_enum(benum)) + + def render_cs_enum_docstrings(self, filename: str, benums: BEnumCollection) -> None: + self.__render('generic.docstring.cs.jinja', filename, benums) + + def render_cs_enum_docstring(self, filename: str, benum: BEnum) -> None: + self.render_cs_enum_docstrings(filename, self.__wrap_single_enum(benum)) + + # endregion + + # region: Rust + + def render_rs_enums(self, filename: str, benums: BEnumCollection) -> None: + self.__render('generic.rs.jinja', filename, benums) + + def render_rs_enum(self, filename: str, benum: BEnum) -> None: + self.render_rs_enums(filename, self.__wrap_single_enum(benum)) + + def render_rs_enum_docstrings(self, filename: str, benums: BEnumCollection) -> None: + self.__render('generic.docstring.rs.jinja', filename, benums) + + def render_rs_enum_docstring(self, filename: str, benum: BEnum) -> None: + self.render_rs_enum_docstrings(filename, self.__wrap_single_enum(benum)) + + # endregion \ No newline at end of file diff --git a/Assets/CodeGen/EnumsMigration/EnumsRender/templates/CKERROR.docstring.cpp.jinja b/Assets/CodeGen/EnumsMigration/EnumsRender/templates/CKERROR.docstring.cpp.jinja new file mode 100644 index 0000000..248ca1e --- /dev/null +++ b/Assets/CodeGen/EnumsMigration/EnumsRender/templates/CKERROR.docstring.cpp.jinja @@ -0,0 +1,6 @@ +{%- set benum = first(payload.iter_enums()) %} +const CkErrorReflectionArray CKERROR { +{%- for entry in benum.iter_entries() %} + { LibCmo::CK2::CKERROR::{{ entry.get_entry_name() }}, { u8"{{ entry.get_entry_name() }}", u8"{{- entry.get_entry_comment() }}" } }, +{%- endfor %} +}; diff --git a/Assets/CodeGen/EnumsMigration/EnumsRender/templates/CKERROR.docstring.hpp.jinja b/Assets/CodeGen/EnumsMigration/EnumsRender/templates/CKERROR.docstring.hpp.jinja new file mode 100644 index 0000000..82b628c --- /dev/null +++ b/Assets/CodeGen/EnumsMigration/EnumsRender/templates/CKERROR.docstring.hpp.jinja @@ -0,0 +1,4 @@ +struct CkErrorReflection { const char8_t* mName; const char8_t* mDescription; }; +using CkErrorReflectionArray = std::vector>; + +const CkErrorReflectionArray CKERROR; diff --git a/Assets/CodeGen/EnumsMigration/EnumsRender/templates/CK_CLASSID.docstring.cpp.jinja b/Assets/CodeGen/EnumsMigration/EnumsRender/templates/CK_CLASSID.docstring.cpp.jinja new file mode 100644 index 0000000..12ce858 --- /dev/null +++ b/Assets/CodeGen/EnumsMigration/EnumsRender/templates/CK_CLASSID.docstring.cpp.jinja @@ -0,0 +1,6 @@ +{%- set benum = first(payload.iter_enums()) %} +const CkClassidReflectionArray CK_CLASSID { +{%- for entry in benum.iter_entries() %} + { LibCmo::CK2::CK_CLASSID::{{ entry.get_entry_name() }}, { { {% for item in entry.iter_hierarchy(benum) %} u8"{{ item.get_enum_name() }}" {%- if not loop.last -%}, {%- endif %} {%- endfor %} } } }, +{%- endfor %} +}; diff --git a/Assets/CodeGen/EnumsMigration/EnumsRender/templates/CK_CLASSID.docstring.hpp.jinja b/Assets/CodeGen/EnumsMigration/EnumsRender/templates/CK_CLASSID.docstring.hpp.jinja new file mode 100644 index 0000000..2e661b5 --- /dev/null +++ b/Assets/CodeGen/EnumsMigration/EnumsRender/templates/CK_CLASSID.docstring.hpp.jinja @@ -0,0 +1,5 @@ +struct CkClassidReflection { std::vector mHierarchy; }; +using CkClassidReflectionArray = std::vector>; + +const CkClassidReflectionArray CK_CLASSID; + diff --git a/Assets/CodeGen/EnumsMigration/EnumsRender/templates/generic.cs.jinja b/Assets/CodeGen/EnumsMigration/EnumsRender/templates/generic.cs.jinja new file mode 100644 index 0000000..375c769 --- /dev/null +++ b/Assets/CodeGen/EnumsMigration/EnumsRender/templates/generic.cs.jinja @@ -0,0 +1,13 @@ +{%- for benum in payload.iter_enums() %} +{%- if benum.get_enum_comment() is not none %} +/** +{{ benum.get_enum_comment() }} + */ +{%- endif %} +{% if benum.get_use_flags() %}[Flags]{%- endif %} +public enum {{ benum.get_enum_name() }} : {% if benum.get_can_unsigned() -%} uint {%- else -%} int {%- endif %} { +{%- for entry in benum.iter_entries() %} + {{ entry.get_entry_name() }} {%- if entry.get_entry_value() is not none %} = {{ entry.get_entry_value() }} {%- endif %}, {%- if entry.get_entry_comment() is not none %} /**< {{ entry.get_entry_comment() }} */ {%- endif %} +{%- endfor %} +}; +{%- endfor %} diff --git a/Assets/CodeGen/EnumsMigration/EnumsRender/templates/generic.docstring.cpp.jinja b/Assets/CodeGen/EnumsMigration/EnumsRender/templates/generic.docstring.cpp.jinja new file mode 100644 index 0000000..bea410b --- /dev/null +++ b/Assets/CodeGen/EnumsMigration/EnumsRender/templates/generic.docstring.cpp.jinja @@ -0,0 +1,7 @@ +{%- for benum in payload.iter_enums() %} +const GenericReflectionArray {{ benum.get_enum_name() }} { +{%- for entry in benum.iter_entries() %} + { LibCmo::{{ extra.parts }}::{{ benum.get_enum_name() }}::{{ entry.get_entry_name() }}, { u8"{{ entry.get_entry_name() }}" } }, +{%- endfor %} +}; +{%- endfor %} diff --git a/Assets/CodeGen/EnumsMigration/EnumsRender/templates/generic.docstring.cs.jinja b/Assets/CodeGen/EnumsMigration/EnumsRender/templates/generic.docstring.cs.jinja new file mode 100644 index 0000000..d81a437 --- /dev/null +++ b/Assets/CodeGen/EnumsMigration/EnumsRender/templates/generic.docstring.cs.jinja @@ -0,0 +1,7 @@ +{%- for benum in payload.iter_enums() %} +public static readonly System.Collections.Generic.Dictionary<{{ benum.get_enum_name() }}, string> {{ benum.get_enum_name() }} = new System.Collections.Generic.Dictionary<{{ benum.get_enum_name() }}, string>() { +{%- for entry in benum.iter_entries() %} + { {{ benum.get_enum_name() }}.{{ entry.get_entry_name() }}, "{{ entry.get_entry_name() }}" }, +{%- endfor %} +}; +{%- endfor %} diff --git a/Assets/CodeGen/EnumsMigration/EnumsRender/templates/generic.docstring.hpp.jinja b/Assets/CodeGen/EnumsMigration/EnumsRender/templates/generic.docstring.hpp.jinja new file mode 100644 index 0000000..68a1edb --- /dev/null +++ b/Assets/CodeGen/EnumsMigration/EnumsRender/templates/generic.docstring.hpp.jinja @@ -0,0 +1,8 @@ +struct GenericReflection { const char8_t* mName; }; +template + requires std::is_enum_v +using GenericReflectionArray = std::vector>; + +{% for benum in payload.iter_enums() %} +extern const GenericReflectionArray {{ benum.get_enum_name() }}; +{%- endfor %} diff --git a/Assets/CodeGen/EnumsMigration/EnumsRender/templates/generic.docstring.py.jinja b/Assets/CodeGen/EnumsMigration/EnumsRender/templates/generic.docstring.py.jinja new file mode 100644 index 0000000..e69de29 diff --git a/Assets/CodeGen/EnumsMigration/EnumsRender/templates/generic.docstring.rs.jinja b/Assets/CodeGen/EnumsMigration/EnumsRender/templates/generic.docstring.rs.jinja new file mode 100644 index 0000000..e69de29 diff --git a/Assets/CodeGen/EnumsMigration/EnumsRender/templates/generic.hpp.jinja b/Assets/CodeGen/EnumsMigration/EnumsRender/templates/generic.hpp.jinja new file mode 100644 index 0000000..c59f604 --- /dev/null +++ b/Assets/CodeGen/EnumsMigration/EnumsRender/templates/generic.hpp.jinja @@ -0,0 +1,12 @@ +{%- for benum in payload.iter_enums() %} +{%- if benum.get_enum_comment() is not none %} +/** +{{ benum.get_enum_comment() }} + */ +{%- endif %} +enum class {{ benum.get_enum_name() }} : {% if benum.get_can_unsigned() -%} CKDWORD {%- else -%} CKINT {%- endif %} { +{%- for entry in benum.iter_entries() %} + {{ entry.get_entry_name() }} {%- if entry.get_entry_value() is not none %} = {{ entry.get_entry_value() }} {%- endif %}, {%- if entry.get_entry_comment() is not none %} /**< {{ entry.get_entry_comment() }} */ {%- endif %} +{%- endfor %} +}; +{%- endfor %} diff --git a/Assets/CodeGen/EnumsMigration/EnumsRender/templates/generic.py.jinja b/Assets/CodeGen/EnumsMigration/EnumsRender/templates/generic.py.jinja new file mode 100644 index 0000000..f464b01 --- /dev/null +++ b/Assets/CodeGen/EnumsMigration/EnumsRender/templates/generic.py.jinja @@ -0,0 +1,14 @@ +import enum + +{% for benum in payload.iter_enums() %} +{%- if benum.get_enum_comment() is not none %} +/** +{{ benum.get_enum_comment() }} + */ +{%- endif %} +class {{ benum.get_enum_name() }}(enum.IntEnum): +{%- for entry in benum.iter_entries() %} + {{ entry.get_entry_name() }} = {% if entry.get_entry_value() is none -%} auto() {%- else -%} {{ entry.get_entry_value() }} {%- endif %} {%- if entry.get_entry_comment() is not none %} + """{{ entry.get_entry_comment() }}""" {%- endif %} +{%- endfor %} +{%- endfor %} diff --git a/Assets/CodeGen/EnumsMigration/EnumsRender/templates/generic.rs.jinja b/Assets/CodeGen/EnumsMigration/EnumsRender/templates/generic.rs.jinja new file mode 100644 index 0000000..e69de29 diff --git a/Assets/CodeGen/EnumsMigration/EnumsRender/utils.py b/Assets/CodeGen/EnumsMigration/EnumsRender/utils.py new file mode 100644 index 0000000..3e41260 --- /dev/null +++ b/Assets/CodeGen/EnumsMigration/EnumsRender/utils.py @@ -0,0 +1,28 @@ +import os +import enum +from pathlib import Path + + +class CKParts(enum.StrEnum): + CK2 = "CK2" + VxMath = "VxMath" + + +def _get_root_directory() -> Path: + enums_migration_root = os.environ.get("ENUMS_MIGRATION_ROOT", None) + if enums_migration_root is None: + return Path(__file__).resolve().parent.parent + else: + return Path(enums_migration_root).resolve() + + +def get_input_file_path(filename: str) -> Path: + return _get_root_directory() / "Intermediate" / filename + + +def get_output_file_path(filename: str) -> Path: + return _get_root_directory() / "Output" / filename + + +def get_template_directory() -> Path: + return Path(__file__).resolve().parent / "templates" diff --git a/Assets/CodeGen/EnumsMigration/EnumsRender/uv.lock b/Assets/CodeGen/EnumsMigration/EnumsRender/uv.lock new file mode 100644 index 0000000..93e2d1a --- /dev/null +++ b/Assets/CodeGen/EnumsMigration/EnumsRender/uv.lock @@ -0,0 +1,100 @@ +version = 1 +revision = 2 +requires-python = ">=3.11" + +[[package]] +name = "enums-render" +version = "0.1.0" +source = { virtual = "." } +dependencies = [ + { name = "jinja2" }, +] + +[package.metadata] +requires-dist = [{ name = "jinja2", specifier = "==3.1.6" }] + +[[package]] +name = "jinja2" +version = "3.1.6" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "markupsafe" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/df/bf/f7da0350254c0ed7c72f3e33cef02e048281fec7ecec5f032d4aac52226b/jinja2-3.1.6.tar.gz", hash = "sha256:0137fb05990d35f1275a587e9aee6d56da821fc83491a0fb838183be43f66d6d", size = 245115, upload-time = "2025-03-05T20:05:02.478Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/62/a1/3d680cbfd5f4b8f15abc1d571870c5fc3e594bb582bc3b64ea099db13e56/jinja2-3.1.6-py3-none-any.whl", hash = "sha256:85ece4451f492d0c13c5dd7c13a64681a86afae63a5f347908daf103ce6d2f67", size = 134899, upload-time = "2025-03-05T20:05:00.369Z" }, +] + +[[package]] +name = "markupsafe" +version = "3.0.3" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/7e/99/7690b6d4034fffd95959cbe0c02de8deb3098cc577c67bb6a24fe5d7caa7/markupsafe-3.0.3.tar.gz", hash = "sha256:722695808f4b6457b320fdc131280796bdceb04ab50fe1795cd540799ebe1698", size = 80313, upload-time = "2025-09-27T18:37:40.426Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/08/db/fefacb2136439fc8dd20e797950e749aa1f4997ed584c62cfb8ef7c2be0e/markupsafe-3.0.3-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:1cc7ea17a6824959616c525620e387f6dd30fec8cb44f649e31712db02123dad", size = 11631, upload-time = "2025-09-27T18:36:18.185Z" }, + { url = "https://files.pythonhosted.org/packages/e1/2e/5898933336b61975ce9dc04decbc0a7f2fee78c30353c5efba7f2d6ff27a/markupsafe-3.0.3-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:4bd4cd07944443f5a265608cc6aab442e4f74dff8088b0dfc8238647b8f6ae9a", size = 12058, upload-time = "2025-09-27T18:36:19.444Z" }, + { url = "https://files.pythonhosted.org/packages/1d/09/adf2df3699d87d1d8184038df46a9c80d78c0148492323f4693df54e17bb/markupsafe-3.0.3-cp311-cp311-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:6b5420a1d9450023228968e7e6a9ce57f65d148ab56d2313fcd589eee96a7a50", size = 24287, upload-time = "2025-09-27T18:36:20.768Z" }, + { url = "https://files.pythonhosted.org/packages/30/ac/0273f6fcb5f42e314c6d8cd99effae6a5354604d461b8d392b5ec9530a54/markupsafe-3.0.3-cp311-cp311-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:0bf2a864d67e76e5c9a34dc26ec616a66b9888e25e7b9460e1c76d3293bd9dbf", size = 22940, upload-time = "2025-09-27T18:36:22.249Z" }, + { url = "https://files.pythonhosted.org/packages/19/ae/31c1be199ef767124c042c6c3e904da327a2f7f0cd63a0337e1eca2967a8/markupsafe-3.0.3-cp311-cp311-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:bc51efed119bc9cfdf792cdeaa4d67e8f6fcccab66ed4bfdd6bde3e59bfcbb2f", size = 21887, upload-time = "2025-09-27T18:36:23.535Z" }, + { url = "https://files.pythonhosted.org/packages/b2/76/7edcab99d5349a4532a459e1fe64f0b0467a3365056ae550d3bcf3f79e1e/markupsafe-3.0.3-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:068f375c472b3e7acbe2d5318dea141359e6900156b5b2ba06a30b169086b91a", size = 23692, upload-time = "2025-09-27T18:36:24.823Z" }, + { url = "https://files.pythonhosted.org/packages/a4/28/6e74cdd26d7514849143d69f0bf2399f929c37dc2b31e6829fd2045b2765/markupsafe-3.0.3-cp311-cp311-musllinux_1_2_riscv64.whl", hash = "sha256:7be7b61bb172e1ed687f1754f8e7484f1c8019780f6f6b0786e76bb01c2ae115", size = 21471, upload-time = "2025-09-27T18:36:25.95Z" }, + { url = "https://files.pythonhosted.org/packages/62/7e/a145f36a5c2945673e590850a6f8014318d5577ed7e5920a4b3448e0865d/markupsafe-3.0.3-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:f9e130248f4462aaa8e2552d547f36ddadbeaa573879158d721bbd33dfe4743a", size = 22923, upload-time = "2025-09-27T18:36:27.109Z" }, + { url = "https://files.pythonhosted.org/packages/0f/62/d9c46a7f5c9adbeeeda52f5b8d802e1094e9717705a645efc71b0913a0a8/markupsafe-3.0.3-cp311-cp311-win32.whl", hash = "sha256:0db14f5dafddbb6d9208827849fad01f1a2609380add406671a26386cdf15a19", size = 14572, upload-time = "2025-09-27T18:36:28.045Z" }, + { url = "https://files.pythonhosted.org/packages/83/8a/4414c03d3f891739326e1783338e48fb49781cc915b2e0ee052aa490d586/markupsafe-3.0.3-cp311-cp311-win_amd64.whl", hash = "sha256:de8a88e63464af587c950061a5e6a67d3632e36df62b986892331d4620a35c01", size = 15077, upload-time = "2025-09-27T18:36:29.025Z" }, + { url = "https://files.pythonhosted.org/packages/35/73/893072b42e6862f319b5207adc9ae06070f095b358655f077f69a35601f0/markupsafe-3.0.3-cp311-cp311-win_arm64.whl", hash = "sha256:3b562dd9e9ea93f13d53989d23a7e775fdfd1066c33494ff43f5418bc8c58a5c", size = 13876, upload-time = "2025-09-27T18:36:29.954Z" }, + { url = "https://files.pythonhosted.org/packages/5a/72/147da192e38635ada20e0a2e1a51cf8823d2119ce8883f7053879c2199b5/markupsafe-3.0.3-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:d53197da72cc091b024dd97249dfc7794d6a56530370992a5e1a08983ad9230e", size = 11615, upload-time = "2025-09-27T18:36:30.854Z" }, + { url = "https://files.pythonhosted.org/packages/9a/81/7e4e08678a1f98521201c3079f77db69fb552acd56067661f8c2f534a718/markupsafe-3.0.3-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:1872df69a4de6aead3491198eaf13810b565bdbeec3ae2dc8780f14458ec73ce", size = 12020, upload-time = "2025-09-27T18:36:31.971Z" }, + { url = "https://files.pythonhosted.org/packages/1e/2c/799f4742efc39633a1b54a92eec4082e4f815314869865d876824c257c1e/markupsafe-3.0.3-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:3a7e8ae81ae39e62a41ec302f972ba6ae23a5c5396c8e60113e9066ef893da0d", size = 24332, upload-time = "2025-09-27T18:36:32.813Z" }, + { url = "https://files.pythonhosted.org/packages/3c/2e/8d0c2ab90a8c1d9a24f0399058ab8519a3279d1bd4289511d74e909f060e/markupsafe-3.0.3-cp312-cp312-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:d6dd0be5b5b189d31db7cda48b91d7e0a9795f31430b7f271219ab30f1d3ac9d", size = 22947, upload-time = "2025-09-27T18:36:33.86Z" }, + { url = "https://files.pythonhosted.org/packages/2c/54/887f3092a85238093a0b2154bd629c89444f395618842e8b0c41783898ea/markupsafe-3.0.3-cp312-cp312-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:94c6f0bb423f739146aec64595853541634bde58b2135f27f61c1ffd1cd4d16a", size = 21962, upload-time = "2025-09-27T18:36:35.099Z" }, + { url = "https://files.pythonhosted.org/packages/c9/2f/336b8c7b6f4a4d95e91119dc8521402461b74a485558d8f238a68312f11c/markupsafe-3.0.3-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:be8813b57049a7dc738189df53d69395eba14fb99345e0a5994914a3864c8a4b", size = 23760, upload-time = "2025-09-27T18:36:36.001Z" }, + { url = "https://files.pythonhosted.org/packages/32/43/67935f2b7e4982ffb50a4d169b724d74b62a3964bc1a9a527f5ac4f1ee2b/markupsafe-3.0.3-cp312-cp312-musllinux_1_2_riscv64.whl", hash = "sha256:83891d0e9fb81a825d9a6d61e3f07550ca70a076484292a70fde82c4b807286f", size = 21529, upload-time = "2025-09-27T18:36:36.906Z" }, + { url = "https://files.pythonhosted.org/packages/89/e0/4486f11e51bbba8b0c041098859e869e304d1c261e59244baa3d295d47b7/markupsafe-3.0.3-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:77f0643abe7495da77fb436f50f8dab76dbc6e5fd25d39589a0f1fe6548bfa2b", size = 23015, upload-time = "2025-09-27T18:36:37.868Z" }, + { url = "https://files.pythonhosted.org/packages/2f/e1/78ee7a023dac597a5825441ebd17170785a9dab23de95d2c7508ade94e0e/markupsafe-3.0.3-cp312-cp312-win32.whl", hash = "sha256:d88b440e37a16e651bda4c7c2b930eb586fd15ca7406cb39e211fcff3bf3017d", size = 14540, upload-time = "2025-09-27T18:36:38.761Z" }, + { url = "https://files.pythonhosted.org/packages/aa/5b/bec5aa9bbbb2c946ca2733ef9c4ca91c91b6a24580193e891b5f7dbe8e1e/markupsafe-3.0.3-cp312-cp312-win_amd64.whl", hash = "sha256:26a5784ded40c9e318cfc2bdb30fe164bdb8665ded9cd64d500a34fb42067b1c", size = 15105, upload-time = "2025-09-27T18:36:39.701Z" }, + { url = "https://files.pythonhosted.org/packages/e5/f1/216fc1bbfd74011693a4fd837e7026152e89c4bcf3e77b6692fba9923123/markupsafe-3.0.3-cp312-cp312-win_arm64.whl", hash = "sha256:35add3b638a5d900e807944a078b51922212fb3dedb01633a8defc4b01a3c85f", size = 13906, upload-time = "2025-09-27T18:36:40.689Z" }, + { url = "https://files.pythonhosted.org/packages/38/2f/907b9c7bbba283e68f20259574b13d005c121a0fa4c175f9bed27c4597ff/markupsafe-3.0.3-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:e1cf1972137e83c5d4c136c43ced9ac51d0e124706ee1c8aa8532c1287fa8795", size = 11622, upload-time = "2025-09-27T18:36:41.777Z" }, + { url = "https://files.pythonhosted.org/packages/9c/d9/5f7756922cdd676869eca1c4e3c0cd0df60ed30199ffd775e319089cb3ed/markupsafe-3.0.3-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:116bb52f642a37c115f517494ea5feb03889e04df47eeff5b130b1808ce7c219", size = 12029, upload-time = "2025-09-27T18:36:43.257Z" }, + { url = "https://files.pythonhosted.org/packages/00/07/575a68c754943058c78f30db02ee03a64b3c638586fba6a6dd56830b30a3/markupsafe-3.0.3-cp313-cp313-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:133a43e73a802c5562be9bbcd03d090aa5a1fe899db609c29e8c8d815c5f6de6", size = 24374, upload-time = "2025-09-27T18:36:44.508Z" }, + { url = "https://files.pythonhosted.org/packages/a9/21/9b05698b46f218fc0e118e1f8168395c65c8a2c750ae2bab54fc4bd4e0e8/markupsafe-3.0.3-cp313-cp313-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:ccfcd093f13f0f0b7fdd0f198b90053bf7b2f02a3927a30e63f3ccc9df56b676", size = 22980, upload-time = "2025-09-27T18:36:45.385Z" }, + { url = "https://files.pythonhosted.org/packages/7f/71/544260864f893f18b6827315b988c146b559391e6e7e8f7252839b1b846a/markupsafe-3.0.3-cp313-cp313-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:509fa21c6deb7a7a273d629cf5ec029bc209d1a51178615ddf718f5918992ab9", size = 21990, upload-time = "2025-09-27T18:36:46.916Z" }, + { url = "https://files.pythonhosted.org/packages/c2/28/b50fc2f74d1ad761af2f5dcce7492648b983d00a65b8c0e0cb457c82ebbe/markupsafe-3.0.3-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:a4afe79fb3de0b7097d81da19090f4df4f8d3a2b3adaa8764138aac2e44f3af1", size = 23784, upload-time = "2025-09-27T18:36:47.884Z" }, + { url = "https://files.pythonhosted.org/packages/ed/76/104b2aa106a208da8b17a2fb72e033a5a9d7073c68f7e508b94916ed47a9/markupsafe-3.0.3-cp313-cp313-musllinux_1_2_riscv64.whl", hash = "sha256:795e7751525cae078558e679d646ae45574b47ed6e7771863fcc079a6171a0fc", size = 21588, upload-time = "2025-09-27T18:36:48.82Z" }, + { url = "https://files.pythonhosted.org/packages/b5/99/16a5eb2d140087ebd97180d95249b00a03aa87e29cc224056274f2e45fd6/markupsafe-3.0.3-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:8485f406a96febb5140bfeca44a73e3ce5116b2501ac54fe953e488fb1d03b12", size = 23041, upload-time = "2025-09-27T18:36:49.797Z" }, + { url = "https://files.pythonhosted.org/packages/19/bc/e7140ed90c5d61d77cea142eed9f9c303f4c4806f60a1044c13e3f1471d0/markupsafe-3.0.3-cp313-cp313-win32.whl", hash = "sha256:bdd37121970bfd8be76c5fb069c7751683bdf373db1ed6c010162b2a130248ed", size = 14543, upload-time = "2025-09-27T18:36:51.584Z" }, + { url = "https://files.pythonhosted.org/packages/05/73/c4abe620b841b6b791f2edc248f556900667a5a1cf023a6646967ae98335/markupsafe-3.0.3-cp313-cp313-win_amd64.whl", hash = "sha256:9a1abfdc021a164803f4d485104931fb8f8c1efd55bc6b748d2f5774e78b62c5", size = 15113, upload-time = "2025-09-27T18:36:52.537Z" }, + { url = "https://files.pythonhosted.org/packages/f0/3a/fa34a0f7cfef23cf9500d68cb7c32dd64ffd58a12b09225fb03dd37d5b80/markupsafe-3.0.3-cp313-cp313-win_arm64.whl", hash = "sha256:7e68f88e5b8799aa49c85cd116c932a1ac15caaa3f5db09087854d218359e485", size = 13911, upload-time = "2025-09-27T18:36:53.513Z" }, + { url = "https://files.pythonhosted.org/packages/e4/d7/e05cd7efe43a88a17a37b3ae96e79a19e846f3f456fe79c57ca61356ef01/markupsafe-3.0.3-cp313-cp313t-macosx_10_13_x86_64.whl", hash = "sha256:218551f6df4868a8d527e3062d0fb968682fe92054e89978594c28e642c43a73", size = 11658, upload-time = "2025-09-27T18:36:54.819Z" }, + { url = "https://files.pythonhosted.org/packages/99/9e/e412117548182ce2148bdeacdda3bb494260c0b0184360fe0d56389b523b/markupsafe-3.0.3-cp313-cp313t-macosx_11_0_arm64.whl", hash = "sha256:3524b778fe5cfb3452a09d31e7b5adefeea8c5be1d43c4f810ba09f2ceb29d37", size = 12066, upload-time = "2025-09-27T18:36:55.714Z" }, + { url = "https://files.pythonhosted.org/packages/bc/e6/fa0ffcda717ef64a5108eaa7b4f5ed28d56122c9a6d70ab8b72f9f715c80/markupsafe-3.0.3-cp313-cp313t-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:4e885a3d1efa2eadc93c894a21770e4bc67899e3543680313b09f139e149ab19", size = 25639, upload-time = "2025-09-27T18:36:56.908Z" }, + { url = "https://files.pythonhosted.org/packages/96/ec/2102e881fe9d25fc16cb4b25d5f5cde50970967ffa5dddafdb771237062d/markupsafe-3.0.3-cp313-cp313t-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:8709b08f4a89aa7586de0aadc8da56180242ee0ada3999749b183aa23df95025", size = 23569, upload-time = "2025-09-27T18:36:57.913Z" }, + { url = "https://files.pythonhosted.org/packages/4b/30/6f2fce1f1f205fc9323255b216ca8a235b15860c34b6798f810f05828e32/markupsafe-3.0.3-cp313-cp313t-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:b8512a91625c9b3da6f127803b166b629725e68af71f8184ae7e7d54686a56d6", size = 23284, upload-time = "2025-09-27T18:36:58.833Z" }, + { url = "https://files.pythonhosted.org/packages/58/47/4a0ccea4ab9f5dcb6f79c0236d954acb382202721e704223a8aafa38b5c8/markupsafe-3.0.3-cp313-cp313t-musllinux_1_2_aarch64.whl", hash = "sha256:9b79b7a16f7fedff2495d684f2b59b0457c3b493778c9eed31111be64d58279f", size = 24801, upload-time = "2025-09-27T18:36:59.739Z" }, + { url = "https://files.pythonhosted.org/packages/6a/70/3780e9b72180b6fecb83a4814d84c3bf4b4ae4bf0b19c27196104149734c/markupsafe-3.0.3-cp313-cp313t-musllinux_1_2_riscv64.whl", hash = "sha256:12c63dfb4a98206f045aa9563db46507995f7ef6d83b2f68eda65c307c6829eb", size = 22769, upload-time = "2025-09-27T18:37:00.719Z" }, + { url = "https://files.pythonhosted.org/packages/98/c5/c03c7f4125180fc215220c035beac6b9cb684bc7a067c84fc69414d315f5/markupsafe-3.0.3-cp313-cp313t-musllinux_1_2_x86_64.whl", hash = "sha256:8f71bc33915be5186016f675cd83a1e08523649b0e33efdb898db577ef5bb009", size = 23642, upload-time = "2025-09-27T18:37:01.673Z" }, + { url = "https://files.pythonhosted.org/packages/80/d6/2d1b89f6ca4bff1036499b1e29a1d02d282259f3681540e16563f27ebc23/markupsafe-3.0.3-cp313-cp313t-win32.whl", hash = "sha256:69c0b73548bc525c8cb9a251cddf1931d1db4d2258e9599c28c07ef3580ef354", size = 14612, upload-time = "2025-09-27T18:37:02.639Z" }, + { url = "https://files.pythonhosted.org/packages/2b/98/e48a4bfba0a0ffcf9925fe2d69240bfaa19c6f7507b8cd09c70684a53c1e/markupsafe-3.0.3-cp313-cp313t-win_amd64.whl", hash = "sha256:1b4b79e8ebf6b55351f0d91fe80f893b4743f104bff22e90697db1590e47a218", size = 15200, upload-time = "2025-09-27T18:37:03.582Z" }, + { url = "https://files.pythonhosted.org/packages/0e/72/e3cc540f351f316e9ed0f092757459afbc595824ca724cbc5a5d4263713f/markupsafe-3.0.3-cp313-cp313t-win_arm64.whl", hash = "sha256:ad2cf8aa28b8c020ab2fc8287b0f823d0a7d8630784c31e9ee5edea20f406287", size = 13973, upload-time = "2025-09-27T18:37:04.929Z" }, + { url = "https://files.pythonhosted.org/packages/33/8a/8e42d4838cd89b7dde187011e97fe6c3af66d8c044997d2183fbd6d31352/markupsafe-3.0.3-cp314-cp314-macosx_10_13_x86_64.whl", hash = "sha256:eaa9599de571d72e2daf60164784109f19978b327a3910d3e9de8c97b5b70cfe", size = 11619, upload-time = "2025-09-27T18:37:06.342Z" }, + { url = "https://files.pythonhosted.org/packages/b5/64/7660f8a4a8e53c924d0fa05dc3a55c9cee10bbd82b11c5afb27d44b096ce/markupsafe-3.0.3-cp314-cp314-macosx_11_0_arm64.whl", hash = "sha256:c47a551199eb8eb2121d4f0f15ae0f923d31350ab9280078d1e5f12b249e0026", size = 12029, upload-time = "2025-09-27T18:37:07.213Z" }, + { url = "https://files.pythonhosted.org/packages/da/ef/e648bfd021127bef5fa12e1720ffed0c6cbb8310c8d9bea7266337ff06de/markupsafe-3.0.3-cp314-cp314-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:f34c41761022dd093b4b6896d4810782ffbabe30f2d443ff5f083e0cbbb8c737", size = 24408, upload-time = "2025-09-27T18:37:09.572Z" }, + { url = "https://files.pythonhosted.org/packages/41/3c/a36c2450754618e62008bf7435ccb0f88053e07592e6028a34776213d877/markupsafe-3.0.3-cp314-cp314-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:457a69a9577064c05a97c41f4e65148652db078a3a509039e64d3467b9e7ef97", size = 23005, upload-time = "2025-09-27T18:37:10.58Z" }, + { url = "https://files.pythonhosted.org/packages/bc/20/b7fdf89a8456b099837cd1dc21974632a02a999ec9bf7ca3e490aacd98e7/markupsafe-3.0.3-cp314-cp314-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:e8afc3f2ccfa24215f8cb28dcf43f0113ac3c37c2f0f0806d8c70e4228c5cf4d", size = 22048, upload-time = "2025-09-27T18:37:11.547Z" }, + { url = "https://files.pythonhosted.org/packages/9a/a7/591f592afdc734f47db08a75793a55d7fbcc6902a723ae4cfbab61010cc5/markupsafe-3.0.3-cp314-cp314-musllinux_1_2_aarch64.whl", hash = "sha256:ec15a59cf5af7be74194f7ab02d0f59a62bdcf1a537677ce67a2537c9b87fcda", size = 23821, upload-time = "2025-09-27T18:37:12.48Z" }, + { url = "https://files.pythonhosted.org/packages/7d/33/45b24e4f44195b26521bc6f1a82197118f74df348556594bd2262bda1038/markupsafe-3.0.3-cp314-cp314-musllinux_1_2_riscv64.whl", hash = "sha256:0eb9ff8191e8498cca014656ae6b8d61f39da5f95b488805da4bb029cccbfbaf", size = 21606, upload-time = "2025-09-27T18:37:13.485Z" }, + { url = "https://files.pythonhosted.org/packages/ff/0e/53dfaca23a69fbfbbf17a4b64072090e70717344c52eaaaa9c5ddff1e5f0/markupsafe-3.0.3-cp314-cp314-musllinux_1_2_x86_64.whl", hash = "sha256:2713baf880df847f2bece4230d4d094280f4e67b1e813eec43b4c0e144a34ffe", size = 23043, upload-time = "2025-09-27T18:37:14.408Z" }, + { url = "https://files.pythonhosted.org/packages/46/11/f333a06fc16236d5238bfe74daccbca41459dcd8d1fa952e8fbd5dccfb70/markupsafe-3.0.3-cp314-cp314-win32.whl", hash = "sha256:729586769a26dbceff69f7a7dbbf59ab6572b99d94576a5592625d5b411576b9", size = 14747, upload-time = "2025-09-27T18:37:15.36Z" }, + { url = "https://files.pythonhosted.org/packages/28/52/182836104b33b444e400b14f797212f720cbc9ed6ba34c800639d154e821/markupsafe-3.0.3-cp314-cp314-win_amd64.whl", hash = "sha256:bdc919ead48f234740ad807933cdf545180bfbe9342c2bb451556db2ed958581", size = 15341, upload-time = "2025-09-27T18:37:16.496Z" }, + { url = "https://files.pythonhosted.org/packages/6f/18/acf23e91bd94fd7b3031558b1f013adfa21a8e407a3fdb32745538730382/markupsafe-3.0.3-cp314-cp314-win_arm64.whl", hash = "sha256:5a7d5dc5140555cf21a6fefbdbf8723f06fcd2f63ef108f2854de715e4422cb4", size = 14073, upload-time = "2025-09-27T18:37:17.476Z" }, + { url = "https://files.pythonhosted.org/packages/3c/f0/57689aa4076e1b43b15fdfa646b04653969d50cf30c32a102762be2485da/markupsafe-3.0.3-cp314-cp314t-macosx_10_13_x86_64.whl", hash = "sha256:1353ef0c1b138e1907ae78e2f6c63ff67501122006b0f9abad68fda5f4ffc6ab", size = 11661, upload-time = "2025-09-27T18:37:18.453Z" }, + { url = "https://files.pythonhosted.org/packages/89/c3/2e67a7ca217c6912985ec766c6393b636fb0c2344443ff9d91404dc4c79f/markupsafe-3.0.3-cp314-cp314t-macosx_11_0_arm64.whl", hash = "sha256:1085e7fbddd3be5f89cc898938f42c0b3c711fdcb37d75221de2666af647c175", size = 12069, upload-time = "2025-09-27T18:37:19.332Z" }, + { url = "https://files.pythonhosted.org/packages/f0/00/be561dce4e6ca66b15276e184ce4b8aec61fe83662cce2f7d72bd3249d28/markupsafe-3.0.3-cp314-cp314t-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:1b52b4fb9df4eb9ae465f8d0c228a00624de2334f216f178a995ccdcf82c4634", size = 25670, upload-time = "2025-09-27T18:37:20.245Z" }, + { url = "https://files.pythonhosted.org/packages/50/09/c419f6f5a92e5fadde27efd190eca90f05e1261b10dbd8cbcb39cd8ea1dc/markupsafe-3.0.3-cp314-cp314t-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:fed51ac40f757d41b7c48425901843666a6677e3e8eb0abcff09e4ba6e664f50", size = 23598, upload-time = "2025-09-27T18:37:21.177Z" }, + { url = "https://files.pythonhosted.org/packages/22/44/a0681611106e0b2921b3033fc19bc53323e0b50bc70cffdd19f7d679bb66/markupsafe-3.0.3-cp314-cp314t-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:f190daf01f13c72eac4efd5c430a8de82489d9cff23c364c3ea822545032993e", size = 23261, upload-time = "2025-09-27T18:37:22.167Z" }, + { url = "https://files.pythonhosted.org/packages/5f/57/1b0b3f100259dc9fffe780cfb60d4be71375510e435efec3d116b6436d43/markupsafe-3.0.3-cp314-cp314t-musllinux_1_2_aarch64.whl", hash = "sha256:e56b7d45a839a697b5eb268c82a71bd8c7f6c94d6fd50c3d577fa39a9f1409f5", size = 24835, upload-time = "2025-09-27T18:37:23.296Z" }, + { url = "https://files.pythonhosted.org/packages/26/6a/4bf6d0c97c4920f1597cc14dd720705eca0bf7c787aebc6bb4d1bead5388/markupsafe-3.0.3-cp314-cp314t-musllinux_1_2_riscv64.whl", hash = "sha256:f3e98bb3798ead92273dc0e5fd0f31ade220f59a266ffd8a4f6065e0a3ce0523", size = 22733, upload-time = "2025-09-27T18:37:24.237Z" }, + { url = "https://files.pythonhosted.org/packages/14/c7/ca723101509b518797fedc2fdf79ba57f886b4aca8a7d31857ba3ee8281f/markupsafe-3.0.3-cp314-cp314t-musllinux_1_2_x86_64.whl", hash = "sha256:5678211cb9333a6468fb8d8be0305520aa073f50d17f089b5b4b477ea6e67fdc", size = 23672, upload-time = "2025-09-27T18:37:25.271Z" }, + { url = "https://files.pythonhosted.org/packages/fb/df/5bd7a48c256faecd1d36edc13133e51397e41b73bb77e1a69deab746ebac/markupsafe-3.0.3-cp314-cp314t-win32.whl", hash = "sha256:915c04ba3851909ce68ccc2b8e2cd691618c4dc4c4232fb7982bca3f41fd8c3d", size = 14819, upload-time = "2025-09-27T18:37:26.285Z" }, + { url = "https://files.pythonhosted.org/packages/1a/8a/0402ba61a2f16038b48b39bccca271134be00c5c9f0f623208399333c448/markupsafe-3.0.3-cp314-cp314t-win_amd64.whl", hash = "sha256:4faffd047e07c38848ce017e8725090413cd80cbc23d86e55c587bf979e579c9", size = 15426, upload-time = "2025-09-27T18:37:27.316Z" }, + { url = "https://files.pythonhosted.org/packages/70/bc/6f1c2f612465f5fa89b95bead1f44dcb607670fd42891d8fdcd5d039f4f4/markupsafe-3.0.3-cp314-cp314t-win_arm64.whl", hash = "sha256:32001d6a8fc98c8cb5c947787c5d08b0a50663d139f1305bac5885d98d9b40fa", size = 14146, upload-time = "2025-09-27T18:37:28.327Z" }, +]