From 61d0d55dab167c3fe7d5922dab7612bfd951af7d Mon Sep 17 00:00:00 2001 From: Thomas Ubensee <34603111+tomuben@users.noreply.github.com> Date: Tue, 3 Dec 2024 14:21:10 -0300 Subject: [PATCH 1/7] #245: Implemented support for Json for language definition --- doc/changes/changes_1.1.0.md | 2 +- .../tasks/build/docker_flavor_image_task.py | 36 ++- .../tasks/upload/language_def_parser.py | 3 +- .../slc/models/language_definition_common.py | 33 ++ .../models/language_definition_components.py | 34 +- .../slc/models/language_definition_model.py | 30 ++ poetry.lock | 295 +++++++++++++++++- pyproject.toml | 2 + .../real_flavor_base/build_steps.py | 3 + .../real_flavor_base/release/Dockerfile | 2 + test/test_docker_api_language_def_json.py | 162 ++++++++++ 11 files changed, 559 insertions(+), 43 deletions(-) create mode 100644 exasol/slc/models/language_definition_common.py create mode 100644 exasol/slc/models/language_definition_model.py create mode 100644 test/test_docker_api_language_def_json.py diff --git a/doc/changes/changes_1.1.0.md b/doc/changes/changes_1.1.0.md index 26647b92..07a10b4c 100644 --- a/doc/changes/changes_1.1.0.md +++ b/doc/changes/changes_1.1.0.md @@ -8,7 +8,7 @@ t.b.d. ## Features -n/a + - #245: Implemented support for Json for language definition ## Refactoring diff --git a/exasol/slc/internal/tasks/build/docker_flavor_image_task.py b/exasol/slc/internal/tasks/build/docker_flavor_image_task.py index bfc2ed6b..1812ea63 100644 --- a/exasol/slc/internal/tasks/build/docker_flavor_image_task.py +++ b/exasol/slc/internal/tasks/build/docker_flavor_image_task.py @@ -1,5 +1,5 @@ from pathlib import Path -from typing import Dict +from typing import Dict, Optional from exasol_integration_test_docker_environment.lib.base.flavor_task import ( FlavorBaseTask, @@ -15,6 +15,8 @@ DockerAnalyzeImageTask, ) +from exasol.slc.models.language_definition_model import LanguageDefinitionsModel + class DockerFlavorAnalyzeImageTask(DockerAnalyzeImageTask, FlavorBaseTask): # TODO change task inheritance with composition. @@ -56,7 +58,15 @@ def get_additional_build_directories_mapping(self) -> Dict[str, str]: """ return {} - def get_path_in_flavor(self): + def get_language_definition(self) -> str: + """ + Called by the constructor to get a language definition file which will be validated against to + the language definition JSON and (if validations succeeded) copied to the temporary build directory. + :return: string with source path of language definition JSON or an empty string + """ + return "" + + def get_path_in_flavor(self) -> Optional[Path]: """ Called by the constructor to get the path to the build context of the build step within the flavor path. Sub classes need to implement this method. @@ -94,18 +104,22 @@ def get_mapping_of_build_files_and_directories(self) -> Dict[str, str]: build_step_path = self.get_build_step_path() result = {self.build_step: str(build_step_path)} result.update(self.additional_build_directories_mapping) + if language_definition := self.get_language_definition(): + lang_def_path = self.get_path_in_flavor_path() / language_definition + LanguageDefinitionsModel.model_validate_json( + lang_def_path.read_text(), strict=True + ) + result.update({"language_definitions.json": str(lang_def_path)}) return result - def get_build_step_path(self): - path_in_flavor = ( # pylint: disable=assignment-from-none - self.get_path_in_flavor() - ) - if path_in_flavor is None: - build_step_path_in_flavor = Path(self.build_step) + def get_build_step_path(self) -> Path: + return self.get_path_in_flavor_path() / self.get_build_step() + + def get_path_in_flavor_path(self) -> Path: + if path_in_flavor := self.get_path_in_flavor(): + return Path(self.flavor_path) / path_in_flavor else: - build_step_path_in_flavor = Path(path_in_flavor).joinpath(self.build_step) - build_step_path = Path(self.flavor_path).joinpath(build_step_path_in_flavor) - return build_step_path + return Path(self.flavor_path) def get_dockerfile(self) -> str: return str(self.get_build_step_path().joinpath("Dockerfile")) diff --git a/exasol/slc/internal/tasks/upload/language_def_parser.py b/exasol/slc/internal/tasks/upload/language_def_parser.py index aa4a9b16..a6a4a97e 100644 --- a/exasol/slc/internal/tasks/upload/language_def_parser.py +++ b/exasol/slc/internal/tasks/upload/language_def_parser.py @@ -101,7 +101,8 @@ def parse_language_definition( if parsed_url.hostname: raise ValueError(f"Invalid language definition: '{lang_def}'") slc_parameters = [ - SLCParameter(key, value) for key, value in parse_qs(parsed_url.query).items() + SLCParameter(key=key, value=value) + for key, value in parse_qs(parsed_url.query).items() ] try: udf_client_path = _parse_udf_client_path(parsed_url.fragment) diff --git a/exasol/slc/models/language_definition_common.py b/exasol/slc/models/language_definition_common.py new file mode 100644 index 00000000..d26adb90 --- /dev/null +++ b/exasol/slc/models/language_definition_common.py @@ -0,0 +1,33 @@ +from dataclasses import dataclass +from enum import Enum +from pathlib import PurePosixPath +from typing import List + +from pydantic import BaseModel + + +class SLCLanguage(str, Enum): + Java = "java" + Python3 = "python" + R = "r" + + +class SLCParameter(BaseModel): + """ + Key value pair of a parameter passed to the Udf client. For example: `lang=java` + """ + + key: str + value: List[str] + + +class UdfClientRelativePath(BaseModel): + """ + Path to the udf client relative to the Script Languages Container root path. + For example `/exaudf/exaudfclient_py3` + """ + + executable: PurePosixPath + + def __str__(self) -> str: + return str(self.executable) diff --git a/exasol/slc/models/language_definition_components.py b/exasol/slc/models/language_definition_components.py index c373dfc6..1a724ad8 100644 --- a/exasol/slc/models/language_definition_components.py +++ b/exasol/slc/models/language_definition_components.py @@ -1,24 +1,13 @@ from dataclasses import dataclass -from enum import Enum from pathlib import PurePosixPath from typing import List, Optional, Union from urllib.parse import ParseResult, urlencode, urlunparse - -class SLCLanguage(Enum): - Java = "java" - Python3 = "python" - R = "r" - - -@dataclass -class SLCParameter: - """ - Key value pair of a parameter passed to the Udf client. For example: `lang=java` - """ - - key: str - value: List[str] +from exasol.slc.models.language_definition_common import ( + SLCLanguage, + SLCParameter, + UdfClientRelativePath, +) @dataclass @@ -35,19 +24,6 @@ def __str__(self) -> str: return f"buckets/{self.bucketfs_name}/{self.bucket_name}/" f"{self.executable}" -@dataclass -class UdfClientRelativePath: - """ - Path to the udf client relative to the Script Languages Container root path. - For example `/exaudf/exaudfclient_py3` - """ - - executable: PurePosixPath - - def __str__(self) -> str: - return str(self.executable) - - @dataclass class ChrootPath: """ diff --git a/exasol/slc/models/language_definition_model.py b/exasol/slc/models/language_definition_model.py new file mode 100644 index 00000000..9ed8e838 --- /dev/null +++ b/exasol/slc/models/language_definition_model.py @@ -0,0 +1,30 @@ +from dataclasses import dataclass +from typing import Annotated, List + +from pydantic import BaseModel, Field + +from exasol.slc.models.language_definition_common import ( + SLCLanguage, + SLCParameter, + UdfClientRelativePath, +) + + +class LanguageDefinition(BaseModel): + """ + Contains information about a supported language and the respective path of the UDF client of an Script-Languages-Container. + """ + + protocol: str + default_alias: str + language: SLCLanguage + parameters: List[SLCParameter] + udf_client_path: UdfClientRelativePath + + +class LanguageDefinitionsModel(BaseModel): + """ + Contains information about all supported languages and the respective path of the UDF client of an Script-Languages-Container. + """ + + language_definitions: List[LanguageDefinition] diff --git a/poetry.lock b/poetry.lock index 5469cf8b..de058abc 100644 --- a/poetry.lock +++ b/poetry.lock @@ -11,6 +11,17 @@ files = [ {file = "alabaster-0.7.16.tar.gz", hash = "sha256:75a8b99c28a5dad50dd7f8ccdd447a121ddb3892da9e53d1ca5cca3106d58d65"}, ] +[[package]] +name = "annotated-types" +version = "0.7.0" +description = "Reusable constraint types to use with typing.Annotated" +optional = false +python-versions = ">=3.8" +files = [ + {file = "annotated_types-0.7.0-py3-none-any.whl", hash = "sha256:1f02e8b43a8fbbc3f3e0d4f0f4bfc8131bcb4eebe8849b8e5c773f3a1c582a53"}, + {file = "annotated_types-0.7.0.tar.gz", hash = "sha256:aff07c09a53a08bc8cfccb9c85b05f1aa9a2a6f23728d790723543408344ce89"}, +] + [[package]] name = "anyio" version = "4.6.0" @@ -450,6 +461,7 @@ description = "Config file reading, writing and validation." optional = false python-versions = ">=3.7" files = [ + {file = "configobj-5.0.9-py2.py3-none-any.whl", hash = "sha256:1ba10c5b6ee16229c79a05047aeda2b55eb4e80d7c7d8ecf17ec1ca600c79882"}, {file = "configobj-5.0.9.tar.gz", hash = "sha256:03c881bbf23aa07bccf1b837005975993c4ab4427ba57f959afdd9d1a2386848"}, ] @@ -1128,6 +1140,41 @@ docs = ["furo", "rst.linker (>=1.9)", "sphinx"] packaging = ["build", "twine"] testing = ["bson", "ecdsa", "feedparser", "gmpy2", "numpy", "pandas", "pymongo", "pytest (>=3.5,!=3.7.3)", "pytest-benchmark", "pytest-benchmark[histogram]", "pytest-checkdocs (>=1.2.3)", "pytest-cov", "pytest-enabler (>=1.0.1)", "pytest-ruff (>=0.2.1)", "scikit-learn", "scipy", "scipy (>=1.9.3)", "simplejson", "sqlalchemy", "ujson"] +[[package]] +name = "jsonschema" +version = "4.23.0" +description = "An implementation of JSON Schema validation for Python" +optional = false +python-versions = ">=3.8" +files = [ + {file = "jsonschema-4.23.0-py3-none-any.whl", hash = "sha256:fbadb6f8b144a8f8cf9f0b89ba94501d143e50411a1278633f56a7acf7fd5566"}, + {file = "jsonschema-4.23.0.tar.gz", hash = "sha256:d71497fef26351a33265337fa77ffeb82423f3ea21283cd9467bb03999266bc4"}, +] + +[package.dependencies] +attrs = ">=22.2.0" +jsonschema-specifications = ">=2023.03.6" +referencing = ">=0.28.4" +rpds-py = ">=0.7.1" + +[package.extras] +format = ["fqdn", "idna", "isoduration", "jsonpointer (>1.13)", "rfc3339-validator", "rfc3987", "uri-template", "webcolors (>=1.11)"] +format-nongpl = ["fqdn", "idna", "isoduration", "jsonpointer (>1.13)", "rfc3339-validator", "rfc3986-validator (>0.1.0)", "uri-template", "webcolors (>=24.6.0)"] + +[[package]] +name = "jsonschema-specifications" +version = "2024.10.1" +description = "The JSON Schema meta-schemas and vocabularies, exposed as a Registry" +optional = false +python-versions = ">=3.9" +files = [ + {file = "jsonschema_specifications-2024.10.1-py3-none-any.whl", hash = "sha256:a09a0680616357d9a0ecf05c12ad234479f549239d0f5b55f3deea67475da9bf"}, + {file = "jsonschema_specifications-2024.10.1.tar.gz", hash = "sha256:0f38b83639958ce1152d02a7f062902c41c8fd20d558b0c34344292d417ae272"}, +] + +[package.dependencies] +referencing = ">=0.31.0" + [[package]] name = "lockfile" version = "0.12.2" @@ -1583,6 +1630,138 @@ files = [ {file = "pycparser-2.22.tar.gz", hash = "sha256:491c8be9c040f5390f5bf44a5b07752bd07f56edf992381b05c701439eec10f6"}, ] +[[package]] +name = "pydantic" +version = "2.10.2" +description = "Data validation using Python type hints" +optional = false +python-versions = ">=3.8" +files = [ + {file = "pydantic-2.10.2-py3-none-any.whl", hash = "sha256:cfb96e45951117c3024e6b67b25cdc33a3cb7b2fa62e239f7af1378358a1d99e"}, + {file = "pydantic-2.10.2.tar.gz", hash = "sha256:2bc2d7f17232e0841cbba4641e65ba1eb6fafb3a08de3a091ff3ce14a197c4fa"}, +] + +[package.dependencies] +annotated-types = ">=0.6.0" +pydantic-core = "2.27.1" +typing-extensions = ">=4.12.2" + +[package.extras] +email = ["email-validator (>=2.0.0)"] +timezone = ["tzdata"] + +[[package]] +name = "pydantic-core" +version = "2.27.1" +description = "Core functionality for Pydantic validation and serialization" +optional = false +python-versions = ">=3.8" +files = [ + {file = "pydantic_core-2.27.1-cp310-cp310-macosx_10_12_x86_64.whl", hash = "sha256:71a5e35c75c021aaf400ac048dacc855f000bdfed91614b4a726f7432f1f3d6a"}, + {file = "pydantic_core-2.27.1-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:f82d068a2d6ecfc6e054726080af69a6764a10015467d7d7b9f66d6ed5afa23b"}, + {file = "pydantic_core-2.27.1-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:121ceb0e822f79163dd4699e4c54f5ad38b157084d97b34de8b232bcaad70278"}, + {file = "pydantic_core-2.27.1-cp310-cp310-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:4603137322c18eaf2e06a4495f426aa8d8388940f3c457e7548145011bb68e05"}, + {file = "pydantic_core-2.27.1-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:a33cd6ad9017bbeaa9ed78a2e0752c5e250eafb9534f308e7a5f7849b0b1bfb4"}, + {file = "pydantic_core-2.27.1-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:15cc53a3179ba0fcefe1e3ae50beb2784dede4003ad2dfd24f81bba4b23a454f"}, + {file = "pydantic_core-2.27.1-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:45d9c5eb9273aa50999ad6adc6be5e0ecea7e09dbd0d31bd0c65a55a2592ca08"}, + {file = "pydantic_core-2.27.1-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:8bf7b66ce12a2ac52d16f776b31d16d91033150266eb796967a7e4621707e4f6"}, + {file = "pydantic_core-2.27.1-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:655d7dd86f26cb15ce8a431036f66ce0318648f8853d709b4167786ec2fa4807"}, + {file = "pydantic_core-2.27.1-cp310-cp310-musllinux_1_1_armv7l.whl", hash = "sha256:5556470f1a2157031e676f776c2bc20acd34c1990ca5f7e56f1ebf938b9ab57c"}, + {file = "pydantic_core-2.27.1-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:f69ed81ab24d5a3bd93861c8c4436f54afdf8e8cc421562b0c7504cf3be58206"}, + {file = "pydantic_core-2.27.1-cp310-none-win32.whl", hash = "sha256:f5a823165e6d04ccea61a9f0576f345f8ce40ed533013580e087bd4d7442b52c"}, + {file = "pydantic_core-2.27.1-cp310-none-win_amd64.whl", hash = "sha256:57866a76e0b3823e0b56692d1a0bf722bffb324839bb5b7226a7dbd6c9a40b17"}, + {file = "pydantic_core-2.27.1-cp311-cp311-macosx_10_12_x86_64.whl", hash = "sha256:ac3b20653bdbe160febbea8aa6c079d3df19310d50ac314911ed8cc4eb7f8cb8"}, + {file = "pydantic_core-2.27.1-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:a5a8e19d7c707c4cadb8c18f5f60c843052ae83c20fa7d44f41594c644a1d330"}, + {file = "pydantic_core-2.27.1-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:7f7059ca8d64fea7f238994c97d91f75965216bcbe5f695bb44f354893f11d52"}, + {file = "pydantic_core-2.27.1-cp311-cp311-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:bed0f8a0eeea9fb72937ba118f9db0cb7e90773462af7962d382445f3005e5a4"}, + {file = "pydantic_core-2.27.1-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:a3cb37038123447cf0f3ea4c74751f6a9d7afef0eb71aa07bf5f652b5e6a132c"}, + {file = "pydantic_core-2.27.1-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:84286494f6c5d05243456e04223d5a9417d7f443c3b76065e75001beb26f88de"}, + {file = "pydantic_core-2.27.1-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:acc07b2cfc5b835444b44a9956846b578d27beeacd4b52e45489e93276241025"}, + {file = "pydantic_core-2.27.1-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:4fefee876e07a6e9aad7a8c8c9f85b0cdbe7df52b8a9552307b09050f7512c7e"}, + {file = "pydantic_core-2.27.1-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:258c57abf1188926c774a4c94dd29237e77eda19462e5bb901d88adcab6af919"}, + {file = "pydantic_core-2.27.1-cp311-cp311-musllinux_1_1_armv7l.whl", hash = "sha256:35c14ac45fcfdf7167ca76cc80b2001205a8d5d16d80524e13508371fb8cdd9c"}, + {file = "pydantic_core-2.27.1-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:d1b26e1dff225c31897696cab7d4f0a315d4c0d9e8666dbffdb28216f3b17fdc"}, + {file = "pydantic_core-2.27.1-cp311-none-win32.whl", hash = "sha256:2cdf7d86886bc6982354862204ae3b2f7f96f21a3eb0ba5ca0ac42c7b38598b9"}, + {file = "pydantic_core-2.27.1-cp311-none-win_amd64.whl", hash = "sha256:3af385b0cee8df3746c3f406f38bcbfdc9041b5c2d5ce3e5fc6637256e60bbc5"}, + {file = "pydantic_core-2.27.1-cp311-none-win_arm64.whl", hash = "sha256:81f2ec23ddc1b476ff96563f2e8d723830b06dceae348ce02914a37cb4e74b89"}, + {file = "pydantic_core-2.27.1-cp312-cp312-macosx_10_12_x86_64.whl", hash = "sha256:9cbd94fc661d2bab2bc702cddd2d3370bbdcc4cd0f8f57488a81bcce90c7a54f"}, + {file = "pydantic_core-2.27.1-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:5f8c4718cd44ec1580e180cb739713ecda2bdee1341084c1467802a417fe0f02"}, + {file = "pydantic_core-2.27.1-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:15aae984e46de8d376df515f00450d1522077254ef6b7ce189b38ecee7c9677c"}, + {file = "pydantic_core-2.27.1-cp312-cp312-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:1ba5e3963344ff25fc8c40da90f44b0afca8cfd89d12964feb79ac1411a260ac"}, + {file = "pydantic_core-2.27.1-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:992cea5f4f3b29d6b4f7f1726ed8ee46c8331c6b4eed6db5b40134c6fe1768bb"}, + {file = "pydantic_core-2.27.1-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:0325336f348dbee6550d129b1627cb8f5351a9dc91aad141ffb96d4937bd9529"}, + {file = "pydantic_core-2.27.1-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7597c07fbd11515f654d6ece3d0e4e5093edc30a436c63142d9a4b8e22f19c35"}, + {file = "pydantic_core-2.27.1-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:3bbd5d8cc692616d5ef6fbbbd50dbec142c7e6ad9beb66b78a96e9c16729b089"}, + {file = "pydantic_core-2.27.1-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:dc61505e73298a84a2f317255fcc72b710b72980f3a1f670447a21efc88f8381"}, + {file = "pydantic_core-2.27.1-cp312-cp312-musllinux_1_1_armv7l.whl", hash = "sha256:e1f735dc43da318cad19b4173dd1ffce1d84aafd6c9b782b3abc04a0d5a6f5bb"}, + {file = "pydantic_core-2.27.1-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:f4e5658dbffe8843a0f12366a4c2d1c316dbe09bb4dfbdc9d2d9cd6031de8aae"}, + {file = "pydantic_core-2.27.1-cp312-none-win32.whl", hash = "sha256:672ebbe820bb37988c4d136eca2652ee114992d5d41c7e4858cdd90ea94ffe5c"}, + {file = "pydantic_core-2.27.1-cp312-none-win_amd64.whl", hash = "sha256:66ff044fd0bb1768688aecbe28b6190f6e799349221fb0de0e6f4048eca14c16"}, + {file = "pydantic_core-2.27.1-cp312-none-win_arm64.whl", hash = "sha256:9a3b0793b1bbfd4146304e23d90045f2a9b5fd5823aa682665fbdaf2a6c28f3e"}, + {file = "pydantic_core-2.27.1-cp313-cp313-macosx_10_12_x86_64.whl", hash = "sha256:f216dbce0e60e4d03e0c4353c7023b202d95cbaeff12e5fd2e82ea0a66905073"}, + {file = "pydantic_core-2.27.1-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:a2e02889071850bbfd36b56fd6bc98945e23670773bc7a76657e90e6b6603c08"}, + {file = "pydantic_core-2.27.1-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:42b0e23f119b2b456d07ca91b307ae167cc3f6c846a7b169fca5326e32fdc6cf"}, + {file = "pydantic_core-2.27.1-cp313-cp313-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:764be71193f87d460a03f1f7385a82e226639732214b402f9aa61f0d025f0737"}, + {file = "pydantic_core-2.27.1-cp313-cp313-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:1c00666a3bd2f84920a4e94434f5974d7bbc57e461318d6bb34ce9cdbbc1f6b2"}, + {file = "pydantic_core-2.27.1-cp313-cp313-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:3ccaa88b24eebc0f849ce0a4d09e8a408ec5a94afff395eb69baf868f5183107"}, + {file = "pydantic_core-2.27.1-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:c65af9088ac534313e1963443d0ec360bb2b9cba6c2909478d22c2e363d98a51"}, + {file = "pydantic_core-2.27.1-cp313-cp313-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:206b5cf6f0c513baffaeae7bd817717140770c74528f3e4c3e1cec7871ddd61a"}, + {file = "pydantic_core-2.27.1-cp313-cp313-musllinux_1_1_aarch64.whl", hash = "sha256:062f60e512fc7fff8b8a9d680ff0ddaaef0193dba9fa83e679c0c5f5fbd018bc"}, + {file = "pydantic_core-2.27.1-cp313-cp313-musllinux_1_1_armv7l.whl", hash = "sha256:a0697803ed7d4af5e4c1adf1670af078f8fcab7a86350e969f454daf598c4960"}, + {file = "pydantic_core-2.27.1-cp313-cp313-musllinux_1_1_x86_64.whl", hash = "sha256:58ca98a950171f3151c603aeea9303ef6c235f692fe555e883591103da709b23"}, + {file = "pydantic_core-2.27.1-cp313-none-win32.whl", hash = "sha256:8065914ff79f7eab1599bd80406681f0ad08f8e47c880f17b416c9f8f7a26d05"}, + {file = "pydantic_core-2.27.1-cp313-none-win_amd64.whl", hash = "sha256:ba630d5e3db74c79300d9a5bdaaf6200172b107f263c98a0539eeecb857b2337"}, + {file = "pydantic_core-2.27.1-cp313-none-win_arm64.whl", hash = "sha256:45cf8588c066860b623cd11c4ba687f8d7175d5f7ef65f7129df8a394c502de5"}, + {file = "pydantic_core-2.27.1-cp38-cp38-macosx_10_12_x86_64.whl", hash = "sha256:5897bec80a09b4084aee23f9b73a9477a46c3304ad1d2d07acca19723fb1de62"}, + {file = "pydantic_core-2.27.1-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:d0165ab2914379bd56908c02294ed8405c252250668ebcb438a55494c69f44ab"}, + {file = "pydantic_core-2.27.1-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6b9af86e1d8e4cfc82c2022bfaa6f459381a50b94a29e95dcdda8442d6d83864"}, + {file = "pydantic_core-2.27.1-cp38-cp38-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:5f6c8a66741c5f5447e047ab0ba7a1c61d1e95580d64bce852e3df1f895c4067"}, + {file = "pydantic_core-2.27.1-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:9a42d6a8156ff78981f8aa56eb6394114e0dedb217cf8b729f438f643608cbcd"}, + {file = "pydantic_core-2.27.1-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:64c65f40b4cd8b0e049a8edde07e38b476da7e3aaebe63287c899d2cff253fa5"}, + {file = "pydantic_core-2.27.1-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:9fdcf339322a3fae5cbd504edcefddd5a50d9ee00d968696846f089b4432cf78"}, + {file = "pydantic_core-2.27.1-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:bf99c8404f008750c846cb4ac4667b798a9f7de673ff719d705d9b2d6de49c5f"}, + {file = "pydantic_core-2.27.1-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:8f1edcea27918d748c7e5e4d917297b2a0ab80cad10f86631e488b7cddf76a36"}, + {file = "pydantic_core-2.27.1-cp38-cp38-musllinux_1_1_armv7l.whl", hash = "sha256:159cac0a3d096f79ab6a44d77a961917219707e2a130739c64d4dd46281f5c2a"}, + {file = "pydantic_core-2.27.1-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:029d9757eb621cc6e1848fa0b0310310de7301057f623985698ed7ebb014391b"}, + {file = "pydantic_core-2.27.1-cp38-none-win32.whl", hash = "sha256:a28af0695a45f7060e6f9b7092558a928a28553366519f64083c63a44f70e618"}, + {file = "pydantic_core-2.27.1-cp38-none-win_amd64.whl", hash = "sha256:2d4567c850905d5eaaed2f7a404e61012a51caf288292e016360aa2b96ff38d4"}, + {file = "pydantic_core-2.27.1-cp39-cp39-macosx_10_12_x86_64.whl", hash = "sha256:e9386266798d64eeb19dd3677051f5705bf873e98e15897ddb7d76f477131967"}, + {file = "pydantic_core-2.27.1-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:4228b5b646caa73f119b1ae756216b59cc6e2267201c27d3912b592c5e323b60"}, + {file = "pydantic_core-2.27.1-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:0b3dfe500de26c52abe0477dde16192ac39c98f05bf2d80e76102d394bd13854"}, + {file = "pydantic_core-2.27.1-cp39-cp39-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:aee66be87825cdf72ac64cb03ad4c15ffef4143dbf5c113f64a5ff4f81477bf9"}, + {file = "pydantic_core-2.27.1-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:3b748c44bb9f53031c8cbc99a8a061bc181c1000c60a30f55393b6e9c45cc5bd"}, + {file = "pydantic_core-2.27.1-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:5ca038c7f6a0afd0b2448941b6ef9d5e1949e999f9e5517692eb6da58e9d44be"}, + {file = "pydantic_core-2.27.1-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6e0bd57539da59a3e4671b90a502da9a28c72322a4f17866ba3ac63a82c4498e"}, + {file = "pydantic_core-2.27.1-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:ac6c2c45c847bbf8f91930d88716a0fb924b51e0c6dad329b793d670ec5db792"}, + {file = "pydantic_core-2.27.1-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:b94d4ba43739bbe8b0ce4262bcc3b7b9f31459ad120fb595627eaeb7f9b9ca01"}, + {file = "pydantic_core-2.27.1-cp39-cp39-musllinux_1_1_armv7l.whl", hash = "sha256:00e6424f4b26fe82d44577b4c842d7df97c20be6439e8e685d0d715feceb9fb9"}, + {file = "pydantic_core-2.27.1-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:38de0a70160dd97540335b7ad3a74571b24f1dc3ed33f815f0880682e6880131"}, + {file = "pydantic_core-2.27.1-cp39-none-win32.whl", hash = "sha256:7ccebf51efc61634f6c2344da73e366c75e735960b5654b63d7e6f69a5885fa3"}, + {file = "pydantic_core-2.27.1-cp39-none-win_amd64.whl", hash = "sha256:a57847b090d7892f123726202b7daa20df6694cbd583b67a592e856bff603d6c"}, + {file = "pydantic_core-2.27.1-pp310-pypy310_pp73-macosx_10_12_x86_64.whl", hash = "sha256:3fa80ac2bd5856580e242dbc202db873c60a01b20309c8319b5c5986fbe53ce6"}, + {file = "pydantic_core-2.27.1-pp310-pypy310_pp73-macosx_11_0_arm64.whl", hash = "sha256:d950caa237bb1954f1b8c9227b5065ba6875ac9771bb8ec790d956a699b78676"}, + {file = "pydantic_core-2.27.1-pp310-pypy310_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:0e4216e64d203e39c62df627aa882f02a2438d18a5f21d7f721621f7a5d3611d"}, + {file = "pydantic_core-2.27.1-pp310-pypy310_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:02a3d637bd387c41d46b002f0e49c52642281edacd2740e5a42f7017feea3f2c"}, + {file = "pydantic_core-2.27.1-pp310-pypy310_pp73-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:161c27ccce13b6b0c8689418da3885d3220ed2eae2ea5e9b2f7f3d48f1d52c27"}, + {file = "pydantic_core-2.27.1-pp310-pypy310_pp73-musllinux_1_1_aarch64.whl", hash = "sha256:19910754e4cc9c63bc1c7f6d73aa1cfee82f42007e407c0f413695c2f7ed777f"}, + {file = "pydantic_core-2.27.1-pp310-pypy310_pp73-musllinux_1_1_armv7l.whl", hash = "sha256:e173486019cc283dc9778315fa29a363579372fe67045e971e89b6365cc035ed"}, + {file = "pydantic_core-2.27.1-pp310-pypy310_pp73-musllinux_1_1_x86_64.whl", hash = "sha256:af52d26579b308921b73b956153066481f064875140ccd1dfd4e77db89dbb12f"}, + {file = "pydantic_core-2.27.1-pp310-pypy310_pp73-win_amd64.whl", hash = "sha256:981fb88516bd1ae8b0cbbd2034678a39dedc98752f264ac9bc5839d3923fa04c"}, + {file = "pydantic_core-2.27.1-pp39-pypy39_pp73-macosx_10_12_x86_64.whl", hash = "sha256:5fde892e6c697ce3e30c61b239330fc5d569a71fefd4eb6512fc6caec9dd9e2f"}, + {file = "pydantic_core-2.27.1-pp39-pypy39_pp73-macosx_11_0_arm64.whl", hash = "sha256:816f5aa087094099fff7edabb5e01cc370eb21aa1a1d44fe2d2aefdfb5599b31"}, + {file = "pydantic_core-2.27.1-pp39-pypy39_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:9c10c309e18e443ddb108f0ef64e8729363adbfd92d6d57beec680f6261556f3"}, + {file = "pydantic_core-2.27.1-pp39-pypy39_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:98476c98b02c8e9b2eec76ac4156fd006628b1b2d0ef27e548ffa978393fd154"}, + {file = "pydantic_core-2.27.1-pp39-pypy39_pp73-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:c3027001c28434e7ca5a6e1e527487051136aa81803ac812be51802150d880dd"}, + {file = "pydantic_core-2.27.1-pp39-pypy39_pp73-musllinux_1_1_aarch64.whl", hash = "sha256:7699b1df36a48169cdebda7ab5a2bac265204003f153b4bd17276153d997670a"}, + {file = "pydantic_core-2.27.1-pp39-pypy39_pp73-musllinux_1_1_armv7l.whl", hash = "sha256:1c39b07d90be6b48968ddc8c19e7585052088fd7ec8d568bb31ff64c70ae3c97"}, + {file = "pydantic_core-2.27.1-pp39-pypy39_pp73-musllinux_1_1_x86_64.whl", hash = "sha256:46ccfe3032b3915586e469d4972973f893c0a2bb65669194a5bdea9bacc088c2"}, + {file = "pydantic_core-2.27.1-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:62ba45e21cf6571d7f716d903b5b7b6d2617e2d5d67c0923dc47b9d41369f840"}, + {file = "pydantic_core-2.27.1.tar.gz", hash = "sha256:62a763352879b84aa31058fc931884055fd75089cccbd9d58bb6afd01141b235"}, +] + +[package.dependencies] +typing-extensions = ">=4.6.0,<4.7.0 || >4.7.0" + [[package]] name = "pydot" version = "3.0.2" @@ -1869,6 +2048,21 @@ files = [ {file = "pyyaml-6.0.2.tar.gz", hash = "sha256:d584d9ec91ad65861cc08d42e834324ef890a082e591037abe114850ff7bbc3e"}, ] +[[package]] +name = "referencing" +version = "0.35.1" +description = "JSON Referencing + Python" +optional = false +python-versions = ">=3.8" +files = [ + {file = "referencing-0.35.1-py3-none-any.whl", hash = "sha256:eda6d3234d62814d1c64e305c1331c9a3a6132da475ab6382eaa997b21ee75de"}, + {file = "referencing-0.35.1.tar.gz", hash = "sha256:25b42124a6c8b632a425174f24087783efb348a6f1e0008e63cd4466fedf703c"}, +] + +[package.dependencies] +attrs = ">=22.2.0" +rpds-py = ">=0.7.0" + [[package]] name = "requests" version = "2.32.3" @@ -1908,6 +2102,105 @@ pygments = ">=2.13.0,<3.0.0" [package.extras] jupyter = ["ipywidgets (>=7.5.1,<9)"] +[[package]] +name = "rpds-py" +version = "0.21.0" +description = "Python bindings to Rust's persistent data structures (rpds)" +optional = false +python-versions = ">=3.9" +files = [ + {file = "rpds_py-0.21.0-cp310-cp310-macosx_10_12_x86_64.whl", hash = "sha256:a017f813f24b9df929674d0332a374d40d7f0162b326562daae8066b502d0590"}, + {file = "rpds_py-0.21.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:20cc1ed0bcc86d8e1a7e968cce15be45178fd16e2ff656a243145e0b439bd250"}, + {file = "rpds_py-0.21.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ad116dda078d0bc4886cb7840e19811562acdc7a8e296ea6ec37e70326c1b41c"}, + {file = "rpds_py-0.21.0-cp310-cp310-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:808f1ac7cf3b44f81c9475475ceb221f982ef548e44e024ad5f9e7060649540e"}, + {file = "rpds_py-0.21.0-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:de552f4a1916e520f2703ec474d2b4d3f86d41f353e7680b597512ffe7eac5d0"}, + {file = "rpds_py-0.21.0-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:efec946f331349dfc4ae9d0e034c263ddde19414fe5128580f512619abed05f1"}, + {file = "rpds_py-0.21.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b80b4690bbff51a034bfde9c9f6bf9357f0a8c61f548942b80f7b66356508bf5"}, + {file = "rpds_py-0.21.0-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:085ed25baac88953d4283e5b5bd094b155075bb40d07c29c4f073e10623f9f2e"}, + {file = "rpds_py-0.21.0-cp310-cp310-musllinux_1_2_aarch64.whl", hash = "sha256:daa8efac2a1273eed2354397a51216ae1e198ecbce9036fba4e7610b308b6153"}, + {file = "rpds_py-0.21.0-cp310-cp310-musllinux_1_2_i686.whl", hash = "sha256:95a5bad1ac8a5c77b4e658671642e4af3707f095d2b78a1fdd08af0dfb647624"}, + {file = "rpds_py-0.21.0-cp310-cp310-musllinux_1_2_x86_64.whl", hash = "sha256:3e53861b29a13d5b70116ea4230b5f0f3547b2c222c5daa090eb7c9c82d7f664"}, + {file = "rpds_py-0.21.0-cp310-none-win32.whl", hash = "sha256:ea3a6ac4d74820c98fcc9da4a57847ad2cc36475a8bd9683f32ab6d47a2bd682"}, + {file = "rpds_py-0.21.0-cp310-none-win_amd64.whl", hash = "sha256:b8f107395f2f1d151181880b69a2869c69e87ec079c49c0016ab96860b6acbe5"}, + {file = "rpds_py-0.21.0-cp311-cp311-macosx_10_12_x86_64.whl", hash = "sha256:5555db3e618a77034954b9dc547eae94166391a98eb867905ec8fcbce1308d95"}, + {file = "rpds_py-0.21.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:97ef67d9bbc3e15584c2f3c74bcf064af36336c10d2e21a2131e123ce0f924c9"}, + {file = "rpds_py-0.21.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:4ab2c2a26d2f69cdf833174f4d9d86118edc781ad9a8fa13970b527bf8236027"}, + {file = "rpds_py-0.21.0-cp311-cp311-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:4e8921a259f54bfbc755c5bbd60c82bb2339ae0324163f32868f63f0ebb873d9"}, + {file = "rpds_py-0.21.0-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:8a7ff941004d74d55a47f916afc38494bd1cfd4b53c482b77c03147c91ac0ac3"}, + {file = "rpds_py-0.21.0-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:5145282a7cd2ac16ea0dc46b82167754d5e103a05614b724457cffe614f25bd8"}, + {file = "rpds_py-0.21.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:de609a6f1b682f70bb7163da745ee815d8f230d97276db049ab447767466a09d"}, + {file = "rpds_py-0.21.0-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:40c91c6e34cf016fa8e6b59d75e3dbe354830777fcfd74c58b279dceb7975b75"}, + {file = "rpds_py-0.21.0-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:d2132377f9deef0c4db89e65e8bb28644ff75a18df5293e132a8d67748397b9f"}, + {file = "rpds_py-0.21.0-cp311-cp311-musllinux_1_2_i686.whl", hash = "sha256:0a9e0759e7be10109645a9fddaaad0619d58c9bf30a3f248a2ea57a7c417173a"}, + {file = "rpds_py-0.21.0-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:9e20da3957bdf7824afdd4b6eeb29510e83e026473e04952dca565170cd1ecc8"}, + {file = "rpds_py-0.21.0-cp311-none-win32.whl", hash = "sha256:f71009b0d5e94c0e86533c0b27ed7cacc1239cb51c178fd239c3cfefefb0400a"}, + {file = "rpds_py-0.21.0-cp311-none-win_amd64.whl", hash = "sha256:e168afe6bf6ab7ab46c8c375606298784ecbe3ba31c0980b7dcbb9631dcba97e"}, + {file = "rpds_py-0.21.0-cp312-cp312-macosx_10_12_x86_64.whl", hash = "sha256:30b912c965b2aa76ba5168fd610087bad7fcde47f0a8367ee8f1876086ee6d1d"}, + {file = "rpds_py-0.21.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:ca9989d5d9b1b300bc18e1801c67b9f6d2c66b8fd9621b36072ed1df2c977f72"}, + {file = "rpds_py-0.21.0-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6f54e7106f0001244a5f4cf810ba8d3f9c542e2730821b16e969d6887b664266"}, + {file = "rpds_py-0.21.0-cp312-cp312-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:fed5dfefdf384d6fe975cc026886aece4f292feaf69d0eeb716cfd3c5a4dd8be"}, + {file = "rpds_py-0.21.0-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:590ef88db231c9c1eece44dcfefd7515d8bf0d986d64d0caf06a81998a9e8cab"}, + {file = "rpds_py-0.21.0-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:f983e4c2f603c95dde63df633eec42955508eefd8d0f0e6d236d31a044c882d7"}, + {file = "rpds_py-0.21.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b229ce052ddf1a01c67d68166c19cb004fb3612424921b81c46e7ea7ccf7c3bf"}, + {file = "rpds_py-0.21.0-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:ebf64e281a06c904a7636781d2e973d1f0926a5b8b480ac658dc0f556e7779f4"}, + {file = "rpds_py-0.21.0-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:998a8080c4495e4f72132f3d66ff91f5997d799e86cec6ee05342f8f3cda7dca"}, + {file = "rpds_py-0.21.0-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:98486337f7b4f3c324ab402e83453e25bb844f44418c066623db88e4c56b7c7b"}, + {file = "rpds_py-0.21.0-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:a78d8b634c9df7f8d175451cfeac3810a702ccb85f98ec95797fa98b942cea11"}, + {file = "rpds_py-0.21.0-cp312-none-win32.whl", hash = "sha256:a58ce66847711c4aa2ecfcfaff04cb0327f907fead8945ffc47d9407f41ff952"}, + {file = "rpds_py-0.21.0-cp312-none-win_amd64.whl", hash = "sha256:e860f065cc4ea6f256d6f411aba4b1251255366e48e972f8a347cf88077b24fd"}, + {file = "rpds_py-0.21.0-cp313-cp313-macosx_10_12_x86_64.whl", hash = "sha256:ee4eafd77cc98d355a0d02f263efc0d3ae3ce4a7c24740010a8b4012bbb24937"}, + {file = "rpds_py-0.21.0-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:688c93b77e468d72579351a84b95f976bd7b3e84aa6686be6497045ba84be560"}, + {file = "rpds_py-0.21.0-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:c38dbf31c57032667dd5a2f0568ccde66e868e8f78d5a0d27dcc56d70f3fcd3b"}, + {file = "rpds_py-0.21.0-cp313-cp313-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:2d6129137f43f7fa02d41542ffff4871d4aefa724a5fe38e2c31a4e0fd343fb0"}, + {file = "rpds_py-0.21.0-cp313-cp313-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:520ed8b99b0bf86a176271f6fe23024323862ac674b1ce5b02a72bfeff3fff44"}, + {file = "rpds_py-0.21.0-cp313-cp313-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:aaeb25ccfb9b9014a10eaf70904ebf3f79faaa8e60e99e19eef9f478651b9b74"}, + {file = "rpds_py-0.21.0-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:af04ac89c738e0f0f1b913918024c3eab6e3ace989518ea838807177d38a2e94"}, + {file = "rpds_py-0.21.0-cp313-cp313-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:b9b76e2afd585803c53c5b29e992ecd183f68285b62fe2668383a18e74abe7a3"}, + {file = "rpds_py-0.21.0-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:5afb5efde74c54724e1a01118c6e5c15e54e642c42a1ba588ab1f03544ac8c7a"}, + {file = "rpds_py-0.21.0-cp313-cp313-musllinux_1_2_i686.whl", hash = "sha256:52c041802a6efa625ea18027a0723676a778869481d16803481ef6cc02ea8cb3"}, + {file = "rpds_py-0.21.0-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:ee1e4fc267b437bb89990b2f2abf6c25765b89b72dd4a11e21934df449e0c976"}, + {file = "rpds_py-0.21.0-cp313-none-win32.whl", hash = "sha256:0c025820b78817db6a76413fff6866790786c38f95ea3f3d3c93dbb73b632202"}, + {file = "rpds_py-0.21.0-cp313-none-win_amd64.whl", hash = "sha256:320c808df533695326610a1b6a0a6e98f033e49de55d7dc36a13c8a30cfa756e"}, + {file = "rpds_py-0.21.0-cp39-cp39-macosx_10_12_x86_64.whl", hash = "sha256:2c51d99c30091f72a3c5d126fad26236c3f75716b8b5e5cf8effb18889ced928"}, + {file = "rpds_py-0.21.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:cbd7504a10b0955ea287114f003b7ad62330c9e65ba012c6223dba646f6ffd05"}, + {file = "rpds_py-0.21.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6dcc4949be728ede49e6244eabd04064336012b37f5c2200e8ec8eb2988b209c"}, + {file = "rpds_py-0.21.0-cp39-cp39-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:f414da5c51bf350e4b7960644617c130140423882305f7574b6cf65a3081cecb"}, + {file = "rpds_py-0.21.0-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:9afe42102b40007f588666bc7de82451e10c6788f6f70984629db193849dced1"}, + {file = "rpds_py-0.21.0-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:3b929c2bb6e29ab31f12a1117c39f7e6d6450419ab7464a4ea9b0b417174f044"}, + {file = "rpds_py-0.21.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8404b3717da03cbf773a1d275d01fec84ea007754ed380f63dfc24fb76ce4592"}, + {file = "rpds_py-0.21.0-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:e12bb09678f38b7597b8346983d2323a6482dcd59e423d9448108c1be37cac9d"}, + {file = "rpds_py-0.21.0-cp39-cp39-musllinux_1_2_aarch64.whl", hash = "sha256:58a0e345be4b18e6b8501d3b0aa540dad90caeed814c515e5206bb2ec26736fd"}, + {file = "rpds_py-0.21.0-cp39-cp39-musllinux_1_2_i686.whl", hash = "sha256:c3761f62fcfccf0864cc4665b6e7c3f0c626f0380b41b8bd1ce322103fa3ef87"}, + {file = "rpds_py-0.21.0-cp39-cp39-musllinux_1_2_x86_64.whl", hash = "sha256:c2b2f71c6ad6c2e4fc9ed9401080badd1469fa9889657ec3abea42a3d6b2e1ed"}, + {file = "rpds_py-0.21.0-cp39-none-win32.whl", hash = "sha256:b21747f79f360e790525e6f6438c7569ddbfb1b3197b9e65043f25c3c9b489d8"}, + {file = "rpds_py-0.21.0-cp39-none-win_amd64.whl", hash = "sha256:0626238a43152918f9e72ede9a3b6ccc9e299adc8ade0d67c5e142d564c9a83d"}, + {file = "rpds_py-0.21.0-pp310-pypy310_pp73-macosx_10_12_x86_64.whl", hash = "sha256:6b4ef7725386dc0762857097f6b7266a6cdd62bfd209664da6712cb26acef035"}, + {file = "rpds_py-0.21.0-pp310-pypy310_pp73-macosx_11_0_arm64.whl", hash = "sha256:6bc0e697d4d79ab1aacbf20ee5f0df80359ecf55db33ff41481cf3e24f206919"}, + {file = "rpds_py-0.21.0-pp310-pypy310_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:da52d62a96e61c1c444f3998c434e8b263c384f6d68aca8274d2e08d1906325c"}, + {file = "rpds_py-0.21.0-pp310-pypy310_pp73-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:98e4fe5db40db87ce1c65031463a760ec7906ab230ad2249b4572c2fc3ef1f9f"}, + {file = "rpds_py-0.21.0-pp310-pypy310_pp73-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:30bdc973f10d28e0337f71d202ff29345320f8bc49a31c90e6c257e1ccef4333"}, + {file = "rpds_py-0.21.0-pp310-pypy310_pp73-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:faa5e8496c530f9c71f2b4e1c49758b06e5f4055e17144906245c99fa6d45356"}, + {file = "rpds_py-0.21.0-pp310-pypy310_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:32eb88c30b6a4f0605508023b7141d043a79b14acb3b969aa0b4f99b25bc7d4a"}, + {file = "rpds_py-0.21.0-pp310-pypy310_pp73-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:a89a8ce9e4e75aeb7fa5d8ad0f3fecdee813802592f4f46a15754dcb2fd6b061"}, + {file = "rpds_py-0.21.0-pp310-pypy310_pp73-musllinux_1_2_aarch64.whl", hash = "sha256:241e6c125568493f553c3d0fdbb38c74babf54b45cef86439d4cd97ff8feb34d"}, + {file = "rpds_py-0.21.0-pp310-pypy310_pp73-musllinux_1_2_i686.whl", hash = "sha256:3b766a9f57663396e4f34f5140b3595b233a7b146e94777b97a8413a1da1be18"}, + {file = "rpds_py-0.21.0-pp310-pypy310_pp73-musllinux_1_2_x86_64.whl", hash = "sha256:af4a644bf890f56e41e74be7d34e9511e4954894d544ec6b8efe1e21a1a8da6c"}, + {file = "rpds_py-0.21.0-pp310-pypy310_pp73-win_amd64.whl", hash = "sha256:3e30a69a706e8ea20444b98a49f386c17b26f860aa9245329bab0851ed100677"}, + {file = "rpds_py-0.21.0-pp39-pypy39_pp73-macosx_10_12_x86_64.whl", hash = "sha256:031819f906bb146561af051c7cef4ba2003d28cff07efacef59da973ff7969ba"}, + {file = "rpds_py-0.21.0-pp39-pypy39_pp73-macosx_11_0_arm64.whl", hash = "sha256:b876f2bc27ab5954e2fd88890c071bd0ed18b9c50f6ec3de3c50a5ece612f7a6"}, + {file = "rpds_py-0.21.0-pp39-pypy39_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:dc5695c321e518d9f03b7ea6abb5ea3af4567766f9852ad1560f501b17588c7b"}, + {file = "rpds_py-0.21.0-pp39-pypy39_pp73-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:b4de1da871b5c0fd5537b26a6fc6814c3cc05cabe0c941db6e9044ffbb12f04a"}, + {file = "rpds_py-0.21.0-pp39-pypy39_pp73-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:878f6fea96621fda5303a2867887686d7a198d9e0f8a40be100a63f5d60c88c9"}, + {file = "rpds_py-0.21.0-pp39-pypy39_pp73-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:a8eeec67590e94189f434c6d11c426892e396ae59e4801d17a93ac96b8c02a6c"}, + {file = "rpds_py-0.21.0-pp39-pypy39_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:1ff2eba7f6c0cb523d7e9cff0903f2fe1feff8f0b2ceb6bd71c0e20a4dcee271"}, + {file = "rpds_py-0.21.0-pp39-pypy39_pp73-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:a429b99337062877d7875e4ff1a51fe788424d522bd64a8c0a20ef3021fdb6ed"}, + {file = "rpds_py-0.21.0-pp39-pypy39_pp73-musllinux_1_2_aarch64.whl", hash = "sha256:d167e4dbbdac48bd58893c7e446684ad5d425b407f9336e04ab52e8b9194e2ed"}, + {file = "rpds_py-0.21.0-pp39-pypy39_pp73-musllinux_1_2_i686.whl", hash = "sha256:4eb2de8a147ffe0626bfdc275fc6563aa7bf4b6db59cf0d44f0ccd6ca625a24e"}, + {file = "rpds_py-0.21.0-pp39-pypy39_pp73-musllinux_1_2_x86_64.whl", hash = "sha256:e78868e98f34f34a88e23ee9ccaeeec460e4eaf6db16d51d7a9b883e5e785a5e"}, + {file = "rpds_py-0.21.0-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:4991ca61656e3160cdaca4851151fd3f4a92e9eba5c7a530ab030d6aee96ec89"}, + {file = "rpds_py-0.21.0.tar.gz", hash = "sha256:ed6378c9d66d0de903763e7706383d60c33829581f0adff47b6535f1802fa6db"}, +] + [[package]] name = "setuptools" version = "75.1.0" @@ -2601,4 +2894,4 @@ type = ["pytest-mypy"] [metadata] lock-version = "2.0" python-versions = ">=3.10,<4" -content-hash = "3602b6be3165e9dcb3f3fc2701add89c2701df5e26ee6bdd23542ecda0c0118c" +content-hash = "808435a06df610ccaa0f40211492d06c5b117265bf52691899a3b0ab07000c40" diff --git a/pyproject.toml b/pyproject.toml index 91715907..be78e75c 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -33,6 +33,8 @@ importlib-resources = ">=6.4.0" networkx = "^3.3.0" exasol-integration-test-docker-environment = "^3.1.0" exasol-bucketfs="^0.13.0" +pydantic="^2.10.2" +jsonschema="^4.23.0" [build-system] requires = ["poetry_core>=1.0.0"] diff --git a/test/resources/flavors/real-test-flavor/real_flavor_base/build_steps.py b/test/resources/flavors/real-test-flavor/real_flavor_base/build_steps.py index b053fef5..ecd574cd 100644 --- a/test/resources/flavors/real-test-flavor/real_flavor_base/build_steps.py +++ b/test/resources/flavors/real-test-flavor/real_flavor_base/build_steps.py @@ -143,6 +143,9 @@ def requires_tasks(self): def get_path_in_flavor(self): return "flavor_base" + def get_language_definition(self) -> str: + return "language_definitions.json" + class SecurityScan(DockerFlavorAnalyzeImageTask): def get_build_step(self) -> str: diff --git a/test/resources/flavors/real-test-flavor/real_flavor_base/release/Dockerfile b/test/resources/flavors/real-test-flavor/real_flavor_base/release/Dockerfile index d4612ad2..a768ea52 100644 --- a/test/resources/flavors/real-test-flavor/real_flavor_base/release/Dockerfile +++ b/test/resources/flavors/real-test-flavor/real_flavor_base/release/Dockerfile @@ -42,3 +42,5 @@ RUN true # workaround for https://github.com/moby/moby/issues/37965 COPY --from={{build_run}} /build_info /build_info RUN true # workaround for https://github.com/moby/moby/issues/37965 +COPY language_definitions.json /build_info/language_definitions.json +RUN true # workaround for https://github.com/moby/moby/issues/37965 diff --git a/test/test_docker_api_language_def_json.py b/test/test_docker_api_language_def_json.py new file mode 100644 index 00000000..eaa68033 --- /dev/null +++ b/test/test_docker_api_language_def_json.py @@ -0,0 +1,162 @@ +import json +import shutil +import unittest +from pathlib import Path +from tempfile import TemporaryDirectory, tempdir + +import docker # type: ignore +import pydantic_core +import utils as exaslct_utils # type: ignore # pylint: disable=import-error +from exasol_integration_test_docker_environment.lib.docker.images.image_info import ( + ImageInfo, +) +from exasol_integration_test_docker_environment.testing import utils # type: ignore + +from exasol.slc.api import build +from exasol.slc.internal.utils.docker_utils import find_images_by_tag +from exasol.slc.models.language_definition_common import ( + SLCLanguage, + UdfClientRelativePath, +) +from exasol.slc.models.language_definition_model import ( + LanguageDefinition, + LanguageDefinitionsModel, +) + + +class ApiDockerBuildLangDefJsonTest(unittest.TestCase): + + def setUp(self): + print(f"SetUp {self.__class__.__name__}") + self.test_environment = exaslct_utils.ExaslctApiTestEnvironmentWithCleanup( + self, True + ) + self.docker_client = docker.from_env() + self.test_environment.clean_all_images() + + def tearDown(self): + try: + self.docker_client.close() + except Exception as e: + print(e) + + utils.close_environments(self.test_environment) + + def read_file_from_docker_image(self, image_name, file_path): + # Create a container from the image + container = self.docker_client.containers.create( + image_name, command="sleep infinity", detach=True + ) + + try: + # Start the container + container.start() + + # Copy the file from the container to host + tar_stream, _ = container.get_archive(file_path) + + # Extract the file content from the tar stream + import io + import tarfile + + file_content = None + + with TemporaryDirectory() as d: + tar_file_path = Path(d) / "tmp.tar" + with open(tar_file_path, "wb") as f: + for chunk in tar_stream: + f.write(chunk) + with tarfile.open(tar_file_path) as tar: + for member in tar.getmembers(): + f = tar.extractfile(member) + if f: + file_content = f.read() + break + + return file_content.decode("utf-8") if file_content else None + + finally: + # Stop and remove the container + container.stop() + container.remove() + + def test_docker_build(self): + flavor_path = exaslct_utils.get_test_flavor() + image_infos = build( + flavor_path=(str(flavor_path),), + source_docker_repository_name=self.test_environment.docker_repository_name, + target_docker_repository_name=self.test_environment.docker_repository_name, + ) + assert len(image_infos) == 1 + images = find_images_by_tag( + self.docker_client, + lambda tag: tag.startswith(self.test_environment.docker_repository_name), + ) + self.assertTrue( + len(images) > 0, + f"Did not found images for repository " + f"{self.test_environment.docker_repository_name} in list {images}", + ) + print("image_infos", image_infos.keys()) + image_infos_for_test_flavor = image_infos[str(flavor_path)] + image_info: ImageInfo = image_infos_for_test_flavor["release"] + + expected_prefix = f"{image_info.target_repository_name}:{image_info.target_tag}" + images = find_images_by_tag( + self.docker_client, lambda tag: tag.startswith(expected_prefix) + ) + self.assertTrue( + len(images) == 1, + f"Did not found image for goal 'release' with prefix {expected_prefix} in list {images}", + ) + + lang_def_json = self.read_file_from_docker_image( + images[0].id, "build_info/language_definitions.json" + ) + model = LanguageDefinitionsModel.model_validate_json(lang_def_json) + self.assertEqual( + model, + LanguageDefinitionsModel( + language_definitions=[ + LanguageDefinition( + protocol="localzmq+protobuf", + default_alias="JAVA", + language=SLCLanguage.Java, + udf_client_path=UdfClientRelativePath( + executable="/exaudf/exaudfclient" + ), + parameters=[], + ) + ] + ), + ) + + def test_docker_build_invalid_lang_def_json(self): + flavor_path = exaslct_utils.get_test_flavor() + with TemporaryDirectory() as d: + temp_flavor_path = Path(d) / "test_flavor" + shutil.copytree(flavor_path, temp_flavor_path) + + lang_def_json_path = ( + temp_flavor_path / "flavor_base" / "language_definitions.json" + ) + orig_lang_def_json = lang_def_json_path.read_text() + lang_def_invalid = json.loads(orig_lang_def_json) + lang_def_invalid.update({"language_definitions": "abc"}) + with open(lang_def_json_path, "w") as f: + f.write(json.dumps(lang_def_invalid)) + + throwed_exception = False + try: + build( + flavor_path=(str(temp_flavor_path),), + source_docker_repository_name=self.test_environment.docker_repository_name, + target_docker_repository_name=self.test_environment.docker_repository_name, + ) + except pydantic_core._pydantic_core.ValidationError as e: + throwed_exception = True + self.assertTrue(throwed_exception) + + +if __name__ == "__main__": + unittest.main() From dea3e0159ac3508ab1be6ea61eb764bec74e5162 Mon Sep 17 00:00:00 2001 From: Thomas Ubensee <34603111+tomuben@users.noreply.github.com> Date: Tue, 3 Dec 2024 14:44:10 -0300 Subject: [PATCH 2/7] Added language_definitions.json --- .../real_flavor_base/language_definitions.json | 13 +++++++++++++ 1 file changed, 13 insertions(+) create mode 100644 test/resources/flavors/real-test-flavor/real_flavor_base/language_definitions.json diff --git a/test/resources/flavors/real-test-flavor/real_flavor_base/language_definitions.json b/test/resources/flavors/real-test-flavor/real_flavor_base/language_definitions.json new file mode 100644 index 00000000..902b86b1 --- /dev/null +++ b/test/resources/flavors/real-test-flavor/real_flavor_base/language_definitions.json @@ -0,0 +1,13 @@ +{ + "language_definitions": [ + { + "protocol": "localzmq+protobuf", + "default_alias": "JAVA", + "language": "java", + "parameters": [], + "udf_client_path": { + "executable": "/exaudf/exaudfclient" + } + } + ] +} From 73ddc4312c62b6cb54f2df2053d75b8035a57cde Mon Sep 17 00:00:00 2001 From: Thomas Ubensee <34603111+tomuben@users.noreply.github.com> Date: Tue, 3 Dec 2024 15:43:54 -0300 Subject: [PATCH 3/7] Changed to list of aliases --- exasol/slc/models/language_definition_model.py | 2 +- .../real-test-flavor/real_flavor_base/language_definitions.json | 2 +- test/test_docker_api_language_def_json.py | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/exasol/slc/models/language_definition_model.py b/exasol/slc/models/language_definition_model.py index 9ed8e838..c9e6f22d 100644 --- a/exasol/slc/models/language_definition_model.py +++ b/exasol/slc/models/language_definition_model.py @@ -16,7 +16,7 @@ class LanguageDefinition(BaseModel): """ protocol: str - default_alias: str + aliases: List[str] language: SLCLanguage parameters: List[SLCParameter] udf_client_path: UdfClientRelativePath diff --git a/test/resources/flavors/real-test-flavor/real_flavor_base/language_definitions.json b/test/resources/flavors/real-test-flavor/real_flavor_base/language_definitions.json index 902b86b1..debc4904 100644 --- a/test/resources/flavors/real-test-flavor/real_flavor_base/language_definitions.json +++ b/test/resources/flavors/real-test-flavor/real_flavor_base/language_definitions.json @@ -2,7 +2,7 @@ "language_definitions": [ { "protocol": "localzmq+protobuf", - "default_alias": "JAVA", + "aliases": ["JAVA"], "language": "java", "parameters": [], "udf_client_path": { diff --git a/test/test_docker_api_language_def_json.py b/test/test_docker_api_language_def_json.py index eaa68033..700031b0 100644 --- a/test/test_docker_api_language_def_json.py +++ b/test/test_docker_api_language_def_json.py @@ -120,7 +120,7 @@ def test_docker_build(self): language_definitions=[ LanguageDefinition( protocol="localzmq+protobuf", - default_alias="JAVA", + aliases=["JAVA"], language=SLCLanguage.Java, udf_client_path=UdfClientRelativePath( executable="/exaudf/exaudfclient" From 652dd9e6758f3c6962e6f9c2bf7218b23d9f7c10 Mon Sep 17 00:00:00 2001 From: Thomas Ubensee <34603111+tomuben@users.noreply.github.com> Date: Thu, 5 Dec 2024 09:32:08 -0300 Subject: [PATCH 4/7] Apply suggestions from code review Co-authored-by: Torsten Kilias --- exasol/slc/internal/tasks/build/docker_flavor_image_task.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/exasol/slc/internal/tasks/build/docker_flavor_image_task.py b/exasol/slc/internal/tasks/build/docker_flavor_image_task.py index 1812ea63..fd81a548 100644 --- a/exasol/slc/internal/tasks/build/docker_flavor_image_task.py +++ b/exasol/slc/internal/tasks/build/docker_flavor_image_task.py @@ -60,8 +60,8 @@ def get_additional_build_directories_mapping(self) -> Dict[str, str]: def get_language_definition(self) -> str: """ - Called by the constructor to get a language definition file which will be validated against to - the language definition JSON and (if validations succeeded) copied to the temporary build directory. + Called by the constructor to get a language definition file which will be validated against + the language definition JSON Schema and (if validations succeeded) copied to the temporary build directory. :return: string with source path of language definition JSON or an empty string """ return "" From 3f2345bf6c8660a0ef4976538236e7b37ed25da0 Mon Sep 17 00:00:00 2001 From: Thomas Ubensee <34603111+tomuben@users.noreply.github.com> Date: Thu, 5 Dec 2024 10:32:47 -0300 Subject: [PATCH 5/7] 1. Added "schema_version" 2. Fixes from review --- .../slc/models/language_definition_model.py | 2 + .../language_definitions.json | 1 + test/test_docker_api_language_def_json.py | 47 ++++++++++--------- 3 files changed, 29 insertions(+), 21 deletions(-) diff --git a/exasol/slc/models/language_definition_model.py b/exasol/slc/models/language_definition_model.py index c9e6f22d..ac5af382 100644 --- a/exasol/slc/models/language_definition_model.py +++ b/exasol/slc/models/language_definition_model.py @@ -27,4 +27,6 @@ class LanguageDefinitionsModel(BaseModel): Contains information about all supported languages and the respective path of the UDF client of an Script-Languages-Container. """ + schema_version: int + language_definitions: List[LanguageDefinition] diff --git a/test/resources/flavors/real-test-flavor/real_flavor_base/language_definitions.json b/test/resources/flavors/real-test-flavor/real_flavor_base/language_definitions.json index debc4904..b2f9b079 100644 --- a/test/resources/flavors/real-test-flavor/real_flavor_base/language_definitions.json +++ b/test/resources/flavors/real-test-flavor/real_flavor_base/language_definitions.json @@ -1,4 +1,5 @@ { + "schema_version": 1, "language_definitions": [ { "protocol": "localzmq+protobuf", diff --git a/test/test_docker_api_language_def_json.py b/test/test_docker_api_language_def_json.py index 700031b0..567e2f42 100644 --- a/test/test_docker_api_language_def_json.py +++ b/test/test_docker_api_language_def_json.py @@ -1,5 +1,6 @@ import json import shutil +import tarfile import unittest from pathlib import Path from tempfile import TemporaryDirectory, tempdir @@ -57,21 +58,13 @@ def read_file_from_docker_image(self, image_name, file_path): # Extract the file content from the tar stream import io - import tarfile file_content = None with TemporaryDirectory() as d: tar_file_path = Path(d) / "tmp.tar" - with open(tar_file_path, "wb") as f: - for chunk in tar_stream: - f.write(chunk) - with tarfile.open(tar_file_path) as tar: - for member in tar.getmembers(): - f = tar.extractfile(member) - if f: - file_content = f.read() - break + self.write_tar_file(tar_file_path, tar_stream) + file_content = self.read_tar_file_content(file_content, tar_file_path) return file_content.decode("utf-8") if file_content else None @@ -80,6 +73,20 @@ def read_file_from_docker_image(self, image_name, file_path): container.stop() container.remove() + def write_tar_file(self, tar_file_path, tar_stream): + with open(tar_file_path, "wb") as f: + for chunk in tar_stream: + f.write(chunk) + + def read_tar_file_content(self, file_content, tar_file_path): + with tarfile.open(tar_file_path) as tar: + for member in tar.getmembers(): + f = tar.extractfile(member) + if f: + file_content = f.read() + break + return file_content + def test_docker_build(self): flavor_path = exaslct_utils.get_test_flavor() image_infos = build( @@ -117,6 +124,7 @@ def test_docker_build(self): self.assertEqual( model, LanguageDefinitionsModel( + schema_version=1, language_definitions=[ LanguageDefinition( protocol="localzmq+protobuf", @@ -127,7 +135,7 @@ def test_docker_build(self): ), parameters=[], ) - ] + ], ), ) @@ -146,16 +154,13 @@ def test_docker_build_invalid_lang_def_json(self): with open(lang_def_json_path, "w") as f: f.write(json.dumps(lang_def_invalid)) - throwed_exception = False - try: - build( - flavor_path=(str(temp_flavor_path),), - source_docker_repository_name=self.test_environment.docker_repository_name, - target_docker_repository_name=self.test_environment.docker_repository_name, - ) - except pydantic_core._pydantic_core.ValidationError as e: - throwed_exception = True - self.assertTrue(throwed_exception) + self.assertRaises( + pydantic_core._pydantic_core.ValidationError, + build, + flavor_path=(str(temp_flavor_path),), + source_docker_repository_name=self.test_environment.docker_repository_name, + target_docker_repository_name=self.test_environment.docker_repository_name, + ) if __name__ == "__main__": From 38c1c88f73040be3cba0ae8bcf39e0860ce829ad Mon Sep 17 00:00:00 2001 From: Thomas Ubensee <34603111+tomuben@users.noreply.github.com> Date: Thu, 5 Dec 2024 12:40:40 -0300 Subject: [PATCH 6/7] Added a check for the schema version --- .../internal/tasks/build/docker_flavor_image_task.py | 11 +++++++++-- exasol/slc/models/language_definition_model.py | 6 ++++-- 2 files changed, 13 insertions(+), 4 deletions(-) diff --git a/exasol/slc/internal/tasks/build/docker_flavor_image_task.py b/exasol/slc/internal/tasks/build/docker_flavor_image_task.py index fd81a548..47b39e95 100644 --- a/exasol/slc/internal/tasks/build/docker_flavor_image_task.py +++ b/exasol/slc/internal/tasks/build/docker_flavor_image_task.py @@ -15,7 +15,10 @@ DockerAnalyzeImageTask, ) -from exasol.slc.models.language_definition_model import LanguageDefinitionsModel +from exasol.slc.models.language_definition_model import ( + LANGUAGE_DEFINITON_SCHEMA_VERSION, + LanguageDefinitionsModel, +) class DockerFlavorAnalyzeImageTask(DockerAnalyzeImageTask, FlavorBaseTask): @@ -106,9 +109,13 @@ def get_mapping_of_build_files_and_directories(self) -> Dict[str, str]: result.update(self.additional_build_directories_mapping) if language_definition := self.get_language_definition(): lang_def_path = self.get_path_in_flavor_path() / language_definition - LanguageDefinitionsModel.model_validate_json( + model = LanguageDefinitionsModel.model_validate_json( lang_def_path.read_text(), strict=True ) + if model.schema_version != LANGUAGE_DEFINITON_SCHEMA_VERSION: + raise RuntimeError( + f"Unsupported schema version. Version from JSON: {model.schema_version}. Expected: {LANGUAGE_DEFINITON_SCHEMA_VERSION}" + ) result.update({"language_definitions.json": str(lang_def_path)}) return result diff --git a/exasol/slc/models/language_definition_model.py b/exasol/slc/models/language_definition_model.py index ac5af382..90105251 100644 --- a/exasol/slc/models/language_definition_model.py +++ b/exasol/slc/models/language_definition_model.py @@ -1,7 +1,7 @@ from dataclasses import dataclass -from typing import Annotated, List +from typing import List -from pydantic import BaseModel, Field +from pydantic import BaseModel from exasol.slc.models.language_definition_common import ( SLCLanguage, @@ -9,6 +9,8 @@ UdfClientRelativePath, ) +LANGUAGE_DEFINITON_SCHEMA_VERSION = 1 + class LanguageDefinition(BaseModel): """ From 3bc5760d7cab00e259cb80dbe64315d436dae557 Mon Sep 17 00:00:00 2001 From: Thomas Ubensee <34603111+tomuben@users.noreply.github.com> Date: Fri, 6 Dec 2024 07:46:56 -0300 Subject: [PATCH 7/7] Fixed import path of ValidationError --- test/test_docker_api_language_def_json.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/test/test_docker_api_language_def_json.py b/test/test_docker_api_language_def_json.py index 567e2f42..0dfd095e 100644 --- a/test/test_docker_api_language_def_json.py +++ b/test/test_docker_api_language_def_json.py @@ -6,12 +6,12 @@ from tempfile import TemporaryDirectory, tempdir import docker # type: ignore -import pydantic_core import utils as exaslct_utils # type: ignore # pylint: disable=import-error from exasol_integration_test_docker_environment.lib.docker.images.image_info import ( ImageInfo, ) from exasol_integration_test_docker_environment.testing import utils # type: ignore +from pydantic import ValidationError from exasol.slc.api import build from exasol.slc.internal.utils.docker_utils import find_images_by_tag @@ -155,7 +155,7 @@ def test_docker_build_invalid_lang_def_json(self): f.write(json.dumps(lang_def_invalid)) self.assertRaises( - pydantic_core._pydantic_core.ValidationError, + ValidationError, build, flavor_path=(str(temp_flavor_path),), source_docker_repository_name=self.test_environment.docker_repository_name,