From b16db6ab4780882df526343fdc8f59a4759540a5 Mon Sep 17 00:00:00 2001 From: Matthew Shipton Date: Sat, 21 Sep 2024 12:08:14 +0100 Subject: [PATCH 01/10] feat(bigquery): non-nullable schema support for embedded fields in struct (#10189) --- .../bigquery/tests/unit/test_datatypes.py | 5 +++++ ibis/backends/sql/datatypes.py | 18 ++++++++++++++++++ 2 files changed, 23 insertions(+) diff --git a/ibis/backends/bigquery/tests/unit/test_datatypes.py b/ibis/backends/bigquery/tests/unit/test_datatypes.py index 4cd17094c7a8..a0ab8a334760 100644 --- a/ibis/backends/bigquery/tests/unit/test_datatypes.py +++ b/ibis/backends/bigquery/tests/unit/test_datatypes.py @@ -24,6 +24,11 @@ param(dt.string, "STRING", id="string"), param(dt.Array(dt.int64), "ARRAY", id="array"), param(dt.Array(dt.string), "ARRAY", id="array"), + param( + dt.Struct({"a": dt.String(nullable=False)}), + "STRUCT<`a` STRING NOT NULL>", + id="struct", + ), param( dt.Struct.from_tuples( [("a", dt.int64), ("b", dt.string), ("c", dt.Array(dt.string))] diff --git a/ibis/backends/sql/datatypes.py b/ibis/backends/sql/datatypes.py index d955a853d7b3..b76630b75900 100644 --- a/ibis/backends/sql/datatypes.py +++ b/ibis/backends/sql/datatypes.py @@ -944,6 +944,24 @@ def _from_ibis_GeoSpatial(cls, dtype: dt.GeoSpatial) -> sge.DataType: f"Current geotype: {dtype.geotype}, Current srid: {dtype.srid}" ) + @classmethod + def _from_ibis_Struct(cls, dtype: dt.Struct) -> sge.DataType: + fields = [ + sge.ColumnDef( + # always quote struct fields to allow reserved words as field names + this=sg.to_identifier(name, quoted=True), + # Bigquery supports embeddable nulls + kind=cls.from_ibis(field), + constraints=( + None + if field.nullable + else [sge.ColumnConstraint(kind=sge.NotNullColumnConstraint())] + ), + ) + for name, field in dtype.items() + ] + return sge.DataType(this=typecode.STRUCT, expressions=fields, nested=True) + class BigQueryUDFType(BigQueryType): @classmethod From 1aa050b1a9d63fec0c2d0e7738d78b3b9963e161 Mon Sep 17 00:00:00 2001 From: Phillip Cloud <417981+cpcloud@users.noreply.github.com> Date: Mon, 23 Sep 2024 05:18:30 -0400 Subject: [PATCH 02/10] chore: remove `poetry.lock` from nix label patterns (#10192) --- .github/labeler.yml | 1 - 1 file changed, 1 deletion(-) diff --git a/.github/labeler.yml b/.github/labeler.yml index d49a58f89f5f..e95e39d2c387 100644 --- a/.github/labeler.yml +++ b/.github/labeler.yml @@ -79,7 +79,6 @@ tests: nix: - changed-files: - any-glob-to-any-file: "**/*.nix" - - any-glob-to-any-file: "poetry.lock" datatypes: - changed-files: From 63dc764430ba0c37ff811749e422376634f60d2c Mon Sep 17 00:00:00 2001 From: "renovate[bot]" <29139614+renovate[bot]@users.noreply.github.com> Date: Mon, 23 Sep 2024 09:51:43 +0000 Subject: [PATCH 03/10] chore(deps): lock file maintenance (#10191) Co-authored-by: renovate[bot] <29139614+renovate[bot]@users.noreply.github.com> Co-authored-by: Phillip Cloud <417981+cpcloud@users.noreply.github.com> --- poetry.lock | 712 ++++++++++++++++++++++--------------------- requirements-dev.txt | 42 +-- 2 files changed, 387 insertions(+), 367 deletions(-) diff --git a/poetry.lock b/poetry.lock index e2b70943c21c..eff58ca98c73 100644 --- a/poetry.lock +++ b/poetry.lock @@ -173,13 +173,13 @@ files = [ [[package]] name = "anyio" -version = "4.4.0" +version = "4.6.0" description = "High level compatibility layer for multiple asynchronous event loop implementations" optional = false -python-versions = ">=3.8" +python-versions = ">=3.9" files = [ - {file = "anyio-4.4.0-py3-none-any.whl", hash = "sha256:c1b2d8f46a8a812513012e1107cb0e68c17159a7a594208005a57dc776e1bdc7"}, - {file = "anyio-4.4.0.tar.gz", hash = "sha256:5aadc6a1bbb7cdb0bede386cac5e2940f5e2ff3aa20277e991cf028e0585ce94"}, + {file = "anyio-4.6.0-py3-none-any.whl", hash = "sha256:c7d2e9d63e31599eeb636c8c5c03a7e108d73b345f064f1c19fdc87b79036a9a"}, + {file = "anyio-4.6.0.tar.gz", hash = "sha256:137b4559cbb034c477165047febb6ff83f390fc3b20bf181c1fc0a728cb8beeb"}, ] [package.dependencies] @@ -189,9 +189,9 @@ sniffio = ">=1.1" typing-extensions = {version = ">=4.1", markers = "python_version < \"3.11\""} [package.extras] -doc = ["Sphinx (>=7)", "packaging", "sphinx-autodoc-typehints (>=1.2.0)", "sphinx-rtd-theme"] -test = ["anyio[trio]", "coverage[toml] (>=7)", "exceptiongroup (>=1.2.0)", "hypothesis (>=4.0)", "psutil (>=5.9)", "pytest (>=7.0)", "pytest-mock (>=3.6.1)", "trustme", "uvloop (>=0.17)"] -trio = ["trio (>=0.23)"] +doc = ["Sphinx (>=7.4,<8.0)", "packaging", "sphinx-autodoc-typehints (>=1.2.0)", "sphinx-rtd-theme"] +test = ["anyio[trio]", "coverage[toml] (>=7)", "exceptiongroup (>=1.2.0)", "hypothesis (>=4.0)", "psutil (>=5.9)", "pytest (>=7.0)", "pytest-mock (>=3.6.1)", "trustme", "uvloop (>=0.21.0b1)"] +trio = ["trio (>=0.26.1)"] [[package]] name = "anywidget" @@ -1430,17 +1430,17 @@ files = [ [[package]] name = "deltalake" -version = "0.19.2" +version = "0.20.0" description = "Native Delta Lake Python binding based on delta-rs with Pandas integration" optional = true python-versions = ">=3.8" files = [ - {file = "deltalake-0.19.2-cp38-abi3-macosx_10_12_x86_64.whl", hash = "sha256:471add9cbde0b3c6e4d0fa4f752fa0fbd661039dc59c3c0c986fd40beb9f6958"}, - {file = "deltalake-0.19.2-cp38-abi3-macosx_11_0_arm64.whl", hash = "sha256:2ef9abe0a5b33dff307eb43051d0e24bdf35e029decf1467ff2d9a1c103c1672"}, - {file = "deltalake-0.19.2-cp38-abi3-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:7ba87501b8d4b589caf256b7676f5cf77494aa37a2e7dc82771ceb3ec032e02f"}, - {file = "deltalake-0.19.2-cp38-abi3-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:972d1d0a800bc8d1111a3fd0db242ed8406e2a2b4ce0777111d483eccca71c2b"}, - {file = "deltalake-0.19.2-cp38-abi3-win_amd64.whl", hash = "sha256:badab7d9aa1797a77b941b37894407d9a43719e46c5f0c91ee0ffacd182bb6c1"}, - {file = "deltalake-0.19.2.tar.gz", hash = "sha256:551026f6f50d265102f5698d389cbbe9163a0de0618bee1380aeed667b5a28cb"}, + {file = "deltalake-0.20.0-cp38-abi3-macosx_10_12_x86_64.whl", hash = "sha256:34302a239e2268d666dff1f6809d07116fe0a327d47fd260a2bfd6e1b43e273b"}, + {file = "deltalake-0.20.0-cp38-abi3-macosx_11_0_arm64.whl", hash = "sha256:06160050b6472c3c5125369738e6af8f31cc476682d4a3a60a2998acdbc999e6"}, + {file = "deltalake-0.20.0-cp38-abi3-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:dabafe00c4fba0fb10e46d5df064b8b6cd0d770217dd358bc5ac41e0cb3c416f"}, + {file = "deltalake-0.20.0-cp38-abi3-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:70ae5fabf56e91a0eae2f1995a6f79f7c37865f4ca3d058a031294cb0e051ab8"}, + {file = "deltalake-0.20.0-cp38-abi3-win_amd64.whl", hash = "sha256:c5a008330071c891946b9f344e8243dd657ce14c091653cbd78b1d4c94beb3e6"}, + {file = "deltalake-0.20.0.tar.gz", hash = "sha256:edb2d440ac5f8df6ae1514812f53edb027c70b316aa1753767620fa0d1fe25f1"}, ] [package.dependencies] @@ -1694,18 +1694,18 @@ devel = ["colorama", "json-spec", "jsonschema", "pylint", "pytest", "pytest-benc [[package]] name = "filelock" -version = "3.16.0" +version = "3.16.1" description = "A platform independent file lock." optional = false python-versions = ">=3.8" files = [ - {file = "filelock-3.16.0-py3-none-any.whl", hash = "sha256:f6ed4c963184f4c84dd5557ce8fece759a3724b37b80c6c4f20a2f63a4dc6609"}, - {file = "filelock-3.16.0.tar.gz", hash = "sha256:81de9eb8453c769b63369f87f11131a7ab04e367f8d97ad39dc230daa07e3bec"}, + {file = "filelock-3.16.1-py3-none-any.whl", hash = "sha256:2082e5703d51fbf98ea75855d9d5527e33d8ff23099bec374a134febee6946b0"}, + {file = "filelock-3.16.1.tar.gz", hash = "sha256:c249fbfcd5db47e5e2d6d62198e565475ee65e4831e2561c8e313fa7eb961435"}, ] [package.extras] -docs = ["furo (>=2024.8.6)", "sphinx (>=8.0.2)", "sphinx-autodoc-typehints (>=2.4)"] -testing = ["covdefaults (>=2.3)", "coverage (>=7.6.1)", "diff-cover (>=9.1.1)", "pytest (>=8.3.2)", "pytest-asyncio (>=0.24)", "pytest-cov (>=5)", "pytest-mock (>=3.14)", "pytest-timeout (>=2.3.1)", "virtualenv (>=20.26.3)"] +docs = ["furo (>=2024.8.6)", "sphinx (>=8.0.2)", "sphinx-autodoc-typehints (>=2.4.1)"] +testing = ["covdefaults (>=2.3)", "coverage (>=7.6.1)", "diff-cover (>=9.2)", "pytest (>=8.3.3)", "pytest-asyncio (>=0.24)", "pytest-cov (>=5)", "pytest-mock (>=3.14)", "pytest-timeout (>=2.3.1)", "virtualenv (>=20.26.4)"] typing = ["typing-extensions (>=4.12.2)"] [[package]] @@ -1972,13 +1972,13 @@ dev = ["black", "codecov", "pre-commit", "pytest (>=3.1.0)", "pytest-cov", "pyte [[package]] name = "google-api-core" -version = "2.19.2" +version = "2.20.0" description = "Google API client core library" optional = false python-versions = ">=3.7" files = [ - {file = "google_api_core-2.19.2-py3-none-any.whl", hash = "sha256:53ec0258f2837dd53bbd3d3df50f5359281b3cc13f800c941dd15a9b5a415af4"}, - {file = "google_api_core-2.19.2.tar.gz", hash = "sha256:ca07de7e8aa1c98a8bfca9321890ad2340ef7f2eb136e558cee68f24b94b0a8f"}, + {file = "google_api_core-2.20.0-py3-none-any.whl", hash = "sha256:ef0591ef03c30bb83f79b3d0575c3f31219001fc9c5cf37024d08310aeffed8a"}, + {file = "google_api_core-2.20.0.tar.gz", hash = "sha256:f74dff1889ba291a4b76c5079df0711810e2d9da81abfdc99957bc961c1eb28f"}, ] [package.dependencies] @@ -2003,13 +2003,13 @@ grpcio-gcp = ["grpcio-gcp (>=0.2.2,<1.0.dev0)"] [[package]] name = "google-auth" -version = "2.34.0" +version = "2.35.0" description = "Google Authentication Library" optional = false python-versions = ">=3.7" files = [ - {file = "google_auth-2.34.0-py2.py3-none-any.whl", hash = "sha256:72fd4733b80b6d777dcde515628a9eb4a577339437012874ea286bca7261ee65"}, - {file = "google_auth-2.34.0.tar.gz", hash = "sha256:8eb87396435c19b20d32abd2f984e31c191a15284af72eb922f10e5bde9c04cc"}, + {file = "google_auth-2.35.0-py2.py3-none-any.whl", hash = "sha256:25df55f327ef021de8be50bad0dfd4a916ad0de96da86cd05661c9297723ad3f"}, + {file = "google_auth-2.35.0.tar.gz", hash = "sha256:f4c64ed4e01e8e8b646ef34c018f8bf3338df0c8e37d8b3bba40e7f574a3278a"}, ] [package.dependencies] @@ -2231,77 +2231,84 @@ test = ["coverage", "pytest (>=7,<8.1)", "pytest-cov", "pytest-mock (>=3)"] [[package]] name = "greenlet" -version = "3.1.0" +version = "3.1.1" description = "Lightweight in-process concurrent programming" optional = false python-versions = ">=3.7" files = [ - {file = "greenlet-3.1.0-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:a814dc3100e8a046ff48faeaa909e80cdb358411a3d6dd5293158425c684eda8"}, - {file = "greenlet-3.1.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a771dc64fa44ebe58d65768d869fcfb9060169d203446c1d446e844b62bdfdca"}, - {file = "greenlet-3.1.0-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:0e49a65d25d7350cca2da15aac31b6f67a43d867448babf997fe83c7505f57bc"}, - {file = "greenlet-3.1.0-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:2cd8518eade968bc52262d8c46727cfc0826ff4d552cf0430b8d65aaf50bb91d"}, - {file = "greenlet-3.1.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:76dc19e660baea5c38e949455c1181bc018893f25372d10ffe24b3ed7341fb25"}, - {file = "greenlet-3.1.0-cp310-cp310-manylinux_2_24_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:c0a5b1c22c82831f56f2f7ad9bbe4948879762fe0d59833a4a71f16e5fa0f682"}, - {file = "greenlet-3.1.0-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:2651dfb006f391bcb240635079a68a261b227a10a08af6349cba834a2141efa1"}, - {file = "greenlet-3.1.0-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:3e7e6ef1737a819819b1163116ad4b48d06cfdd40352d813bb14436024fcda99"}, - {file = "greenlet-3.1.0-cp310-cp310-win_amd64.whl", hash = "sha256:ffb08f2a1e59d38c7b8b9ac8083c9c8b9875f0955b1e9b9b9a965607a51f8e54"}, - {file = "greenlet-3.1.0-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:9730929375021ec90f6447bff4f7f5508faef1c02f399a1953870cdb78e0c345"}, - {file = "greenlet-3.1.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:713d450cf8e61854de9420fb7eea8ad228df4e27e7d4ed465de98c955d2b3fa6"}, - {file = "greenlet-3.1.0-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:4c3446937be153718250fe421da548f973124189f18fe4575a0510b5c928f0cc"}, - {file = "greenlet-3.1.0-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:1ddc7bcedeb47187be74208bc652d63d6b20cb24f4e596bd356092d8000da6d6"}, - {file = "greenlet-3.1.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:44151d7b81b9391ed759a2f2865bbe623ef00d648fed59363be2bbbd5154656f"}, - {file = "greenlet-3.1.0-cp311-cp311-manylinux_2_24_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:6cea1cca3be76c9483282dc7760ea1cc08a6ecec1f0b6ca0a94ea0d17432da19"}, - {file = "greenlet-3.1.0-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:619935a44f414274a2c08c9e74611965650b730eb4efe4b2270f91df5e4adf9a"}, - {file = "greenlet-3.1.0-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:221169d31cada333a0c7fd087b957c8f431c1dba202c3a58cf5a3583ed973e9b"}, - {file = "greenlet-3.1.0-cp311-cp311-win_amd64.whl", hash = "sha256:01059afb9b178606b4b6e92c3e710ea1635597c3537e44da69f4531e111dd5e9"}, - {file = "greenlet-3.1.0-cp312-cp312-macosx_11_0_universal2.whl", hash = "sha256:24fc216ec7c8be9becba8b64a98a78f9cd057fd2dc75ae952ca94ed8a893bf27"}, - {file = "greenlet-3.1.0-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:3d07c28b85b350564bdff9f51c1c5007dfb2f389385d1bc23288de51134ca303"}, - {file = "greenlet-3.1.0-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:243a223c96a4246f8a30ea470c440fe9db1f5e444941ee3c3cd79df119b8eebf"}, - {file = "greenlet-3.1.0-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:26811df4dc81271033a7836bc20d12cd30938e6bd2e9437f56fa03da81b0f8fc"}, - {file = "greenlet-3.1.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:c9d86401550b09a55410f32ceb5fe7efcd998bd2dad9e82521713cb148a4a15f"}, - {file = "greenlet-3.1.0-cp312-cp312-manylinux_2_24_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:26d9c1c4f1748ccac0bae1dbb465fb1a795a75aba8af8ca871503019f4285e2a"}, - {file = "greenlet-3.1.0-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:cd468ec62257bb4544989402b19d795d2305eccb06cde5da0eb739b63dc04665"}, - {file = "greenlet-3.1.0-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:a53dfe8f82b715319e9953330fa5c8708b610d48b5c59f1316337302af5c0811"}, - {file = "greenlet-3.1.0-cp312-cp312-win_amd64.whl", hash = "sha256:28fe80a3eb673b2d5cc3b12eea468a5e5f4603c26aa34d88bf61bba82ceb2f9b"}, - {file = "greenlet-3.1.0-cp313-cp313-macosx_11_0_universal2.whl", hash = "sha256:76b3e3976d2a452cba7aa9e453498ac72240d43030fdc6d538a72b87eaff52fd"}, - {file = "greenlet-3.1.0-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:655b21ffd37a96b1e78cc48bf254f5ea4b5b85efaf9e9e2a526b3c9309d660ca"}, - {file = "greenlet-3.1.0-cp313-cp313-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:c6f4c2027689093775fd58ca2388d58789009116844432d920e9147f91acbe64"}, - {file = "greenlet-3.1.0-cp313-cp313-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:76e5064fd8e94c3f74d9fd69b02d99e3cdb8fc286ed49a1f10b256e59d0d3a0b"}, - {file = "greenlet-3.1.0-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6a4bf607f690f7987ab3291406e012cd8591a4f77aa54f29b890f9c331e84989"}, - {file = "greenlet-3.1.0-cp313-cp313-manylinux_2_24_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:037d9ac99540ace9424cb9ea89f0accfaff4316f149520b4ae293eebc5bded17"}, - {file = "greenlet-3.1.0-cp313-cp313-musllinux_1_1_aarch64.whl", hash = "sha256:90b5bbf05fe3d3ef697103850c2ce3374558f6fe40fd57c9fac1bf14903f50a5"}, - {file = "greenlet-3.1.0-cp313-cp313-musllinux_1_1_x86_64.whl", hash = "sha256:726377bd60081172685c0ff46afbc600d064f01053190e4450857483c4d44484"}, - {file = "greenlet-3.1.0-cp313-cp313-win_amd64.whl", hash = "sha256:d46d5069e2eeda111d6f71970e341f4bd9aeeee92074e649ae263b834286ecc0"}, - {file = "greenlet-3.1.0-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:81eeec4403a7d7684b5812a8aaa626fa23b7d0848edb3a28d2eb3220daddcbd0"}, - {file = "greenlet-3.1.0-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:4a3dae7492d16e85ea6045fd11cb8e782b63eac8c8d520c3a92c02ac4573b0a6"}, - {file = "greenlet-3.1.0-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:4b5ea3664eed571779403858d7cd0a9b0ebf50d57d2cdeafc7748e09ef8cd81a"}, - {file = "greenlet-3.1.0-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a22f4e26400f7f48faef2d69c20dc055a1f3043d330923f9abe08ea0aecc44df"}, - {file = "greenlet-3.1.0-cp37-cp37m-manylinux_2_24_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:13ff8c8e54a10472ce3b2a2da007f915175192f18e6495bad50486e87c7f6637"}, - {file = "greenlet-3.1.0-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:f9671e7282d8c6fcabc32c0fb8d7c0ea8894ae85cee89c9aadc2d7129e1a9954"}, - {file = "greenlet-3.1.0-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:184258372ae9e1e9bddce6f187967f2e08ecd16906557c4320e3ba88a93438c3"}, - {file = "greenlet-3.1.0-cp37-cp37m-win32.whl", hash = "sha256:a0409bc18a9f85321399c29baf93545152d74a49d92f2f55302f122007cfda00"}, - {file = "greenlet-3.1.0-cp37-cp37m-win_amd64.whl", hash = "sha256:9eb4a1d7399b9f3c7ac68ae6baa6be5f9195d1d08c9ddc45ad559aa6b556bce6"}, - {file = "greenlet-3.1.0-cp38-cp38-macosx_11_0_universal2.whl", hash = "sha256:a8870983af660798dc1b529e1fd6f1cefd94e45135a32e58bd70edd694540f33"}, - {file = "greenlet-3.1.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:cfcfb73aed40f550a57ea904629bdaf2e562c68fa1164fa4588e752af6efdc3f"}, - {file = "greenlet-3.1.0-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:f9482c2ed414781c0af0b35d9d575226da6b728bd1a720668fa05837184965b7"}, - {file = "greenlet-3.1.0-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:d58ec349e0c2c0bc6669bf2cd4982d2f93bf067860d23a0ea1fe677b0f0b1e09"}, - {file = "greenlet-3.1.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:dd65695a8df1233309b701dec2539cc4b11e97d4fcc0f4185b4a12ce54db0491"}, - {file = "greenlet-3.1.0-cp38-cp38-manylinux_2_24_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:665b21e95bc0fce5cab03b2e1d90ba9c66c510f1bb5fdc864f3a377d0f553f6b"}, - {file = "greenlet-3.1.0-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:d3c59a06c2c28a81a026ff11fbf012081ea34fb9b7052f2ed0366e14896f0a1d"}, - {file = "greenlet-3.1.0-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:5415b9494ff6240b09af06b91a375731febe0090218e2898d2b85f9b92abcda0"}, - {file = "greenlet-3.1.0-cp38-cp38-win32.whl", hash = "sha256:1544b8dd090b494c55e60c4ff46e238be44fdc472d2589e943c241e0169bcea2"}, - {file = "greenlet-3.1.0-cp38-cp38-win_amd64.whl", hash = "sha256:7f346d24d74c00b6730440f5eb8ec3fe5774ca8d1c9574e8e57c8671bb51b910"}, - {file = "greenlet-3.1.0-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:db1b3ccb93488328c74e97ff888604a8b95ae4f35f4f56677ca57a4fc3a4220b"}, - {file = "greenlet-3.1.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:44cd313629ded43bb3b98737bba2f3e2c2c8679b55ea29ed73daea6b755fe8e7"}, - {file = "greenlet-3.1.0-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:fad7a051e07f64e297e6e8399b4d6a3bdcad3d7297409e9a06ef8cbccff4f501"}, - {file = "greenlet-3.1.0-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:c3967dcc1cd2ea61b08b0b276659242cbce5caca39e7cbc02408222fb9e6ff39"}, - {file = "greenlet-3.1.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d45b75b0f3fd8d99f62eb7908cfa6d727b7ed190737dec7fe46d993da550b81a"}, - {file = "greenlet-3.1.0-cp39-cp39-manylinux_2_24_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:2d004db911ed7b6218ec5c5bfe4cf70ae8aa2223dffbb5b3c69e342bb253cb28"}, - {file = "greenlet-3.1.0-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:b9505a0c8579899057cbefd4ec34d865ab99852baf1ff33a9481eb3924e2da0b"}, - {file = "greenlet-3.1.0-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:5fd6e94593f6f9714dbad1aaba734b5ec04593374fa6638df61592055868f8b8"}, - {file = "greenlet-3.1.0-cp39-cp39-win32.whl", hash = "sha256:d0dd943282231480aad5f50f89bdf26690c995e8ff555f26d8a5b9887b559bcc"}, - {file = "greenlet-3.1.0-cp39-cp39-win_amd64.whl", hash = "sha256:ac0adfdb3a21dc2a24ed728b61e72440d297d0fd3a577389df566651fcd08f97"}, - {file = "greenlet-3.1.0.tar.gz", hash = "sha256:b395121e9bbe8d02a750886f108d540abe66075e61e22f7353d9acb0b81be0f0"}, + {file = "greenlet-3.1.1-cp310-cp310-macosx_11_0_universal2.whl", hash = "sha256:0bbae94a29c9e5c7e4a2b7f0aae5c17e8e90acbfd3bf6270eeba60c39fce3563"}, + {file = "greenlet-3.1.1-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:0fde093fb93f35ca72a556cf72c92ea3ebfda3d79fc35bb19fbe685853869a83"}, + {file = "greenlet-3.1.1-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:36b89d13c49216cadb828db8dfa6ce86bbbc476a82d3a6c397f0efae0525bdd0"}, + {file = "greenlet-3.1.1-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:94b6150a85e1b33b40b1464a3f9988dcc5251d6ed06842abff82e42632fac120"}, + {file = "greenlet-3.1.1-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:93147c513fac16385d1036b7e5b102c7fbbdb163d556b791f0f11eada7ba65dc"}, + {file = "greenlet-3.1.1-cp310-cp310-manylinux_2_24_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:da7a9bff22ce038e19bf62c4dd1ec8391062878710ded0a845bcf47cc0200617"}, + {file = "greenlet-3.1.1-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:b2795058c23988728eec1f36a4e5e4ebad22f8320c85f3587b539b9ac84128d7"}, + {file = "greenlet-3.1.1-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:ed10eac5830befbdd0c32f83e8aa6288361597550ba669b04c48f0f9a2c843c6"}, + {file = "greenlet-3.1.1-cp310-cp310-win_amd64.whl", hash = "sha256:77c386de38a60d1dfb8e55b8c1101d68c79dfdd25c7095d51fec2dd800892b80"}, + {file = "greenlet-3.1.1-cp311-cp311-macosx_11_0_universal2.whl", hash = "sha256:e4d333e558953648ca09d64f13e6d8f0523fa705f51cae3f03b5983489958c70"}, + {file = "greenlet-3.1.1-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:09fc016b73c94e98e29af67ab7b9a879c307c6731a2c9da0db5a7d9b7edd1159"}, + {file = "greenlet-3.1.1-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:d5e975ca70269d66d17dd995dafc06f1b06e8cb1ec1e9ed54c1d1e4a7c4cf26e"}, + {file = "greenlet-3.1.1-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:3b2813dc3de8c1ee3f924e4d4227999285fd335d1bcc0d2be6dc3f1f6a318ec1"}, + {file = "greenlet-3.1.1-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e347b3bfcf985a05e8c0b7d462ba6f15b1ee1c909e2dcad795e49e91b152c383"}, + {file = "greenlet-3.1.1-cp311-cp311-manylinux_2_24_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:9e8f8c9cb53cdac7ba9793c276acd90168f416b9ce36799b9b885790f8ad6c0a"}, + {file = "greenlet-3.1.1-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:62ee94988d6b4722ce0028644418d93a52429e977d742ca2ccbe1c4f4a792511"}, + {file = "greenlet-3.1.1-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:1776fd7f989fc6b8d8c8cb8da1f6b82c5814957264d1f6cf818d475ec2bf6395"}, + {file = "greenlet-3.1.1-cp311-cp311-win_amd64.whl", hash = "sha256:48ca08c771c268a768087b408658e216133aecd835c0ded47ce955381105ba39"}, + {file = "greenlet-3.1.1-cp312-cp312-macosx_11_0_universal2.whl", hash = "sha256:4afe7ea89de619adc868e087b4d2359282058479d7cfb94970adf4b55284574d"}, + {file = "greenlet-3.1.1-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:f406b22b7c9a9b4f8aa9d2ab13d6ae0ac3e85c9a809bd590ad53fed2bf70dc79"}, + {file = "greenlet-3.1.1-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:c3a701fe5a9695b238503ce5bbe8218e03c3bcccf7e204e455e7462d770268aa"}, + {file = "greenlet-3.1.1-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:2846930c65b47d70b9d178e89c7e1a69c95c1f68ea5aa0a58646b7a96df12441"}, + {file = "greenlet-3.1.1-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:99cfaa2110534e2cf3ba31a7abcac9d328d1d9f1b95beede58294a60348fba36"}, + {file = "greenlet-3.1.1-cp312-cp312-manylinux_2_24_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:1443279c19fca463fc33e65ef2a935a5b09bb90f978beab37729e1c3c6c25fe9"}, + {file = "greenlet-3.1.1-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:b7cede291382a78f7bb5f04a529cb18e068dd29e0fb27376074b6d0317bf4dd0"}, + {file = "greenlet-3.1.1-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:23f20bb60ae298d7d8656c6ec6db134bca379ecefadb0b19ce6f19d1f232a942"}, + {file = "greenlet-3.1.1-cp312-cp312-win_amd64.whl", hash = "sha256:7124e16b4c55d417577c2077be379514321916d5790fa287c9ed6f23bd2ffd01"}, + {file = "greenlet-3.1.1-cp313-cp313-macosx_11_0_universal2.whl", hash = "sha256:05175c27cb459dcfc05d026c4232f9de8913ed006d42713cb8a5137bd49375f1"}, + {file = "greenlet-3.1.1-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:935e943ec47c4afab8965954bf49bfa639c05d4ccf9ef6e924188f762145c0ff"}, + {file = "greenlet-3.1.1-cp313-cp313-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:667a9706c970cb552ede35aee17339a18e8f2a87a51fba2ed39ceeeb1004798a"}, + {file = "greenlet-3.1.1-cp313-cp313-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:b8a678974d1f3aa55f6cc34dc480169d58f2e6d8958895d68845fa4ab566509e"}, + {file = "greenlet-3.1.1-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:efc0f674aa41b92da8c49e0346318c6075d734994c3c4e4430b1c3f853e498e4"}, + {file = "greenlet-3.1.1-cp313-cp313-manylinux_2_24_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:0153404a4bb921f0ff1abeb5ce8a5131da56b953eda6e14b88dc6bbc04d2049e"}, + {file = "greenlet-3.1.1-cp313-cp313-musllinux_1_1_aarch64.whl", hash = "sha256:275f72decf9932639c1c6dd1013a1bc266438eb32710016a1c742df5da6e60a1"}, + {file = "greenlet-3.1.1-cp313-cp313-musllinux_1_1_x86_64.whl", hash = "sha256:c4aab7f6381f38a4b42f269057aee279ab0fc7bf2e929e3d4abfae97b682a12c"}, + {file = "greenlet-3.1.1-cp313-cp313-win_amd64.whl", hash = "sha256:b42703b1cf69f2aa1df7d1030b9d77d3e584a70755674d60e710f0af570f3761"}, + {file = "greenlet-3.1.1-cp313-cp313t-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:f1695e76146579f8c06c1509c7ce4dfe0706f49c6831a817ac04eebb2fd02011"}, + {file = "greenlet-3.1.1-cp313-cp313t-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:7876452af029456b3f3549b696bb36a06db7c90747740c5302f74a9e9fa14b13"}, + {file = "greenlet-3.1.1-cp313-cp313t-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:4ead44c85f8ab905852d3de8d86f6f8baf77109f9da589cb4fa142bd3b57b475"}, + {file = "greenlet-3.1.1-cp313-cp313t-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8320f64b777d00dd7ccdade271eaf0cad6636343293a25074cc5566160e4de7b"}, + {file = "greenlet-3.1.1-cp313-cp313t-manylinux_2_24_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:6510bf84a6b643dabba74d3049ead221257603a253d0a9873f55f6a59a65f822"}, + {file = "greenlet-3.1.1-cp313-cp313t-musllinux_1_1_aarch64.whl", hash = "sha256:04b013dc07c96f83134b1e99888e7a79979f1a247e2a9f59697fa14b5862ed01"}, + {file = "greenlet-3.1.1-cp313-cp313t-musllinux_1_1_x86_64.whl", hash = "sha256:411f015496fec93c1c8cd4e5238da364e1da7a124bcb293f085bf2860c32c6f6"}, + {file = "greenlet-3.1.1-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:47da355d8687fd65240c364c90a31569a133b7b60de111c255ef5b606f2ae291"}, + {file = "greenlet-3.1.1-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:98884ecf2ffb7d7fe6bd517e8eb99d31ff7855a840fa6d0d63cd07c037f6a981"}, + {file = "greenlet-3.1.1-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:f1d4aeb8891338e60d1ab6127af1fe45def5259def8094b9c7e34690c8858803"}, + {file = "greenlet-3.1.1-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:db32b5348615a04b82240cc67983cb315309e88d444a288934ee6ceaebcad6cc"}, + {file = "greenlet-3.1.1-cp37-cp37m-manylinux_2_24_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:dcc62f31eae24de7f8dce72134c8651c58000d3b1868e01392baea7c32c247de"}, + {file = "greenlet-3.1.1-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:1d3755bcb2e02de341c55b4fca7a745a24a9e7212ac953f6b3a48d117d7257aa"}, + {file = "greenlet-3.1.1-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:b8da394b34370874b4572676f36acabac172602abf054cbc4ac910219f3340af"}, + {file = "greenlet-3.1.1-cp37-cp37m-win32.whl", hash = "sha256:a0dfc6c143b519113354e780a50381508139b07d2177cb6ad6a08278ec655798"}, + {file = "greenlet-3.1.1-cp37-cp37m-win_amd64.whl", hash = "sha256:54558ea205654b50c438029505def3834e80f0869a70fb15b871c29b4575ddef"}, + {file = "greenlet-3.1.1-cp38-cp38-macosx_11_0_universal2.whl", hash = "sha256:346bed03fe47414091be4ad44786d1bd8bef0c3fcad6ed3dee074a032ab408a9"}, + {file = "greenlet-3.1.1-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:dfc59d69fc48664bc693842bd57acfdd490acafda1ab52c7836e3fc75c90a111"}, + {file = "greenlet-3.1.1-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:d21e10da6ec19b457b82636209cbe2331ff4306b54d06fa04b7c138ba18c8a81"}, + {file = "greenlet-3.1.1-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:37b9de5a96111fc15418819ab4c4432e4f3c2ede61e660b1e33971eba26ef9ba"}, + {file = "greenlet-3.1.1-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6ef9ea3f137e5711f0dbe5f9263e8c009b7069d8a1acea822bd5e9dae0ae49c8"}, + {file = "greenlet-3.1.1-cp38-cp38-manylinux_2_24_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:85f3ff71e2e60bd4b4932a043fbbe0f499e263c628390b285cb599154a3b03b1"}, + {file = "greenlet-3.1.1-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:95ffcf719966dd7c453f908e208e14cde192e09fde6c7186c8f1896ef778d8cd"}, + {file = "greenlet-3.1.1-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:03a088b9de532cbfe2ba2034b2b85e82df37874681e8c470d6fb2f8c04d7e4b7"}, + {file = "greenlet-3.1.1-cp38-cp38-win32.whl", hash = "sha256:8b8b36671f10ba80e159378df9c4f15c14098c4fd73a36b9ad715f057272fbef"}, + {file = "greenlet-3.1.1-cp38-cp38-win_amd64.whl", hash = "sha256:7017b2be767b9d43cc31416aba48aab0d2309ee31b4dbf10a1d38fb7972bdf9d"}, + {file = "greenlet-3.1.1-cp39-cp39-macosx_11_0_universal2.whl", hash = "sha256:396979749bd95f018296af156201d6211240e7a23090f50a8d5d18c370084dc3"}, + {file = "greenlet-3.1.1-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ca9d0ff5ad43e785350894d97e13633a66e2b50000e8a183a50a88d834752d42"}, + {file = "greenlet-3.1.1-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:f6ff3b14f2df4c41660a7dec01045a045653998784bf8cfcb5a525bdffffbc8f"}, + {file = "greenlet-3.1.1-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:94ebba31df2aa506d7b14866fed00ac141a867e63143fe5bca82a8e503b36437"}, + {file = "greenlet-3.1.1-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:73aaad12ac0ff500f62cebed98d8789198ea0e6f233421059fa68a5aa7220145"}, + {file = "greenlet-3.1.1-cp39-cp39-manylinux_2_24_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:63e4844797b975b9af3a3fb8f7866ff08775f5426925e1e0bbcfe7932059a12c"}, + {file = "greenlet-3.1.1-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:7939aa3ca7d2a1593596e7ac6d59391ff30281ef280d8632fa03d81f7c5f955e"}, + {file = "greenlet-3.1.1-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:d0028e725ee18175c6e422797c407874da24381ce0690d6b9396c204c7f7276e"}, + {file = "greenlet-3.1.1-cp39-cp39-win32.whl", hash = "sha256:5e06afd14cbaf9e00899fae69b24a32f2196c19de08fcb9f4779dd4f004e5e7c"}, + {file = "greenlet-3.1.1-cp39-cp39-win_amd64.whl", hash = "sha256:3319aa75e0e0639bc15ff54ca327e8dc7a6fe404003496e3c6925cd3142e0e22"}, + {file = "greenlet-3.1.1.tar.gz", hash = "sha256:4ce3ac6cdb6adf7946475d7ef31777c26d94bccc377e070a7986bd2d5c515467"}, ] [package.extras] @@ -2915,13 +2922,13 @@ testing = ["coverage", "ipykernel", "jupytext", "matplotlib", "nbdime", "nbforma [[package]] name = "jupyter-client" -version = "8.6.2" +version = "8.6.3" description = "Jupyter protocol implementation and client libraries" optional = false python-versions = ">=3.8" files = [ - {file = "jupyter_client-8.6.2-py3-none-any.whl", hash = "sha256:50cbc5c66fd1b8f65ecb66bc490ab73217993632809b6e505687de18e9dea39f"}, - {file = "jupyter_client-8.6.2.tar.gz", hash = "sha256:2bda14d55ee5ba58552a8c53ae43d215ad9868853489213f37da060ced54d8df"}, + {file = "jupyter_client-8.6.3-py3-none-any.whl", hash = "sha256:e8a19cc986cc45905ac3362915f410f3af85424b4c0905e94fa5f2cb08e8f23f"}, + {file = "jupyter_client-8.6.3.tar.gz", hash = "sha256:35b3a0947c4a6e9d589eb97d7d4cd5e90f910ee73101611f01283732bd6d9419"}, ] [package.dependencies] @@ -3845,13 +3852,13 @@ files = [ [[package]] name = "narwhals" -version = "1.8.1" +version = "1.8.2" description = "Extremely lightweight compatibility layer between dataframe libraries" optional = false python-versions = ">=3.8" files = [ - {file = "narwhals-1.8.1-py3-none-any.whl", hash = "sha256:91a3af813733df39a74f590fdd1bb0d2d6d8a33e32aa409f56d941c0a29f8cdd"}, - {file = "narwhals-1.8.1.tar.gz", hash = "sha256:97527778e11f39a1e5e2113b8fbb9ead788be41c0337f21852e684e378f583e8"}, + {file = "narwhals-1.8.2-py3-none-any.whl", hash = "sha256:08554e9e46411b9dba0deffe8510b948baad603270697414aebd2e5694e35a52"}, + {file = "narwhals-1.8.2.tar.gz", hash = "sha256:82bc3c630dd48c30dd87e3762e785bd0ae07a6b9039fff8a8004bb452f8c7db6"}, ] [package.extras] @@ -4137,40 +4144,53 @@ files = [ [[package]] name = "pandas" -version = "2.2.2" +version = "2.2.3" description = "Powerful data structures for data analysis, time series, and statistics" optional = false python-versions = ">=3.9" files = [ - {file = "pandas-2.2.2-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:90c6fca2acf139569e74e8781709dccb6fe25940488755716d1d354d6bc58bce"}, - {file = "pandas-2.2.2-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:c7adfc142dac335d8c1e0dcbd37eb8617eac386596eb9e1a1b77791cf2498238"}, - {file = "pandas-2.2.2-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:4abfe0be0d7221be4f12552995e58723c7422c80a659da13ca382697de830c08"}, - {file = "pandas-2.2.2-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8635c16bf3d99040fdf3ca3db669a7250ddf49c55dc4aa8fe0ae0fa8d6dcc1f0"}, - {file = "pandas-2.2.2-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:40ae1dffb3967a52203105a077415a86044a2bea011b5f321c6aa64b379a3f51"}, - {file = "pandas-2.2.2-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:8e5a0b00e1e56a842f922e7fae8ae4077aee4af0acb5ae3622bd4b4c30aedf99"}, - {file = "pandas-2.2.2-cp310-cp310-win_amd64.whl", hash = "sha256:ddf818e4e6c7c6f4f7c8a12709696d193976b591cc7dc50588d3d1a6b5dc8772"}, - {file = "pandas-2.2.2-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:696039430f7a562b74fa45f540aca068ea85fa34c244d0deee539cb6d70aa288"}, - {file = "pandas-2.2.2-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:8e90497254aacacbc4ea6ae5e7a8cd75629d6ad2b30025a4a8b09aa4faf55151"}, - {file = "pandas-2.2.2-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:58b84b91b0b9f4bafac2a0ac55002280c094dfc6402402332c0913a59654ab2b"}, - {file = "pandas-2.2.2-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6d2123dc9ad6a814bcdea0f099885276b31b24f7edf40f6cdbc0912672e22eee"}, - {file = "pandas-2.2.2-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:2925720037f06e89af896c70bca73459d7e6a4be96f9de79e2d440bd499fe0db"}, - {file = "pandas-2.2.2-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:0cace394b6ea70c01ca1595f839cf193df35d1575986e484ad35c4aeae7266c1"}, - {file = "pandas-2.2.2-cp311-cp311-win_amd64.whl", hash = "sha256:873d13d177501a28b2756375d59816c365e42ed8417b41665f346289adc68d24"}, - {file = "pandas-2.2.2-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:9dfde2a0ddef507a631dc9dc4af6a9489d5e2e740e226ad426a05cabfbd7c8ef"}, - {file = "pandas-2.2.2-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:e9b79011ff7a0f4b1d6da6a61aa1aa604fb312d6647de5bad20013682d1429ce"}, - {file = "pandas-2.2.2-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1cb51fe389360f3b5a4d57dbd2848a5f033350336ca3b340d1c53a1fad33bcad"}, - {file = "pandas-2.2.2-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:eee3a87076c0756de40b05c5e9a6069c035ba43e8dd71c379e68cab2c20f16ad"}, - {file = "pandas-2.2.2-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:3e374f59e440d4ab45ca2fffde54b81ac3834cf5ae2cdfa69c90bc03bde04d76"}, - {file = "pandas-2.2.2-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:43498c0bdb43d55cb162cdc8c06fac328ccb5d2eabe3cadeb3529ae6f0517c32"}, - {file = "pandas-2.2.2-cp312-cp312-win_amd64.whl", hash = "sha256:d187d355ecec3629624fccb01d104da7d7f391db0311145817525281e2804d23"}, - {file = "pandas-2.2.2-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:0ca6377b8fca51815f382bd0b697a0814c8bda55115678cbc94c30aacbb6eff2"}, - {file = "pandas-2.2.2-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:9057e6aa78a584bc93a13f0a9bf7e753a5e9770a30b4d758b8d5f2a62a9433cd"}, - {file = "pandas-2.2.2-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:001910ad31abc7bf06f49dcc903755d2f7f3a9186c0c040b827e522e9cef0863"}, - {file = "pandas-2.2.2-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:66b479b0bd07204e37583c191535505410daa8df638fd8e75ae1b383851fe921"}, - {file = "pandas-2.2.2-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:a77e9d1c386196879aa5eb712e77461aaee433e54c68cf253053a73b7e49c33a"}, - {file = "pandas-2.2.2-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:92fd6b027924a7e178ac202cfbe25e53368db90d56872d20ffae94b96c7acc57"}, - {file = "pandas-2.2.2-cp39-cp39-win_amd64.whl", hash = "sha256:640cef9aa381b60e296db324337a554aeeb883ead99dc8f6c18e81a93942f5f4"}, - {file = "pandas-2.2.2.tar.gz", hash = "sha256:9e79019aba43cb4fda9e4d983f8e88ca0373adbb697ae9c6c43093218de28b54"}, + {file = "pandas-2.2.3-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:1948ddde24197a0f7add2bdc4ca83bf2b1ef84a1bc8ccffd95eda17fd836ecb5"}, + {file = "pandas-2.2.3-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:381175499d3802cde0eabbaf6324cce0c4f5d52ca6f8c377c29ad442f50f6348"}, + {file = "pandas-2.2.3-cp310-cp310-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:d9c45366def9a3dd85a6454c0e7908f2b3b8e9c138f5dc38fed7ce720d8453ed"}, + {file = "pandas-2.2.3-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:86976a1c5b25ae3f8ccae3a5306e443569ee3c3faf444dfd0f41cda24667ad57"}, + {file = "pandas-2.2.3-cp310-cp310-musllinux_1_2_aarch64.whl", hash = "sha256:b8661b0238a69d7aafe156b7fa86c44b881387509653fdf857bebc5e4008ad42"}, + {file = "pandas-2.2.3-cp310-cp310-musllinux_1_2_x86_64.whl", hash = "sha256:37e0aced3e8f539eccf2e099f65cdb9c8aa85109b0be6e93e2baff94264bdc6f"}, + {file = "pandas-2.2.3-cp310-cp310-win_amd64.whl", hash = "sha256:56534ce0746a58afaf7942ba4863e0ef81c9c50d3f0ae93e9497d6a41a057645"}, + {file = "pandas-2.2.3-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:66108071e1b935240e74525006034333f98bcdb87ea116de573a6a0dccb6c039"}, + {file = "pandas-2.2.3-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:7c2875855b0ff77b2a64a0365e24455d9990730d6431b9e0ee18ad8acee13dbd"}, + {file = "pandas-2.2.3-cp311-cp311-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:cd8d0c3be0515c12fed0bdbae072551c8b54b7192c7b1fda0ba56059a0179698"}, + {file = "pandas-2.2.3-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:c124333816c3a9b03fbeef3a9f230ba9a737e9e5bb4060aa2107a86cc0a497fc"}, + {file = "pandas-2.2.3-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:63cc132e40a2e084cf01adf0775b15ac515ba905d7dcca47e9a251819c575ef3"}, + {file = "pandas-2.2.3-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:29401dbfa9ad77319367d36940cd8a0b3a11aba16063e39632d98b0e931ddf32"}, + {file = "pandas-2.2.3-cp311-cp311-win_amd64.whl", hash = "sha256:3fc6873a41186404dad67245896a6e440baacc92f5b716ccd1bc9ed2995ab2c5"}, + {file = "pandas-2.2.3-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:b1d432e8d08679a40e2a6d8b2f9770a5c21793a6f9f47fdd52c5ce1948a5a8a9"}, + {file = "pandas-2.2.3-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:a5a1595fe639f5988ba6a8e5bc9649af3baf26df3998a0abe56c02609392e0a4"}, + {file = "pandas-2.2.3-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:5de54125a92bb4d1c051c0659e6fcb75256bf799a732a87184e5ea503965bce3"}, + {file = "pandas-2.2.3-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:fffb8ae78d8af97f849404f21411c95062db1496aeb3e56f146f0355c9989319"}, + {file = "pandas-2.2.3-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:6dfcb5ee8d4d50c06a51c2fffa6cff6272098ad6540aed1a76d15fb9318194d8"}, + {file = "pandas-2.2.3-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:062309c1b9ea12a50e8ce661145c6aab431b1e99530d3cd60640e255778bd43a"}, + {file = "pandas-2.2.3-cp312-cp312-win_amd64.whl", hash = "sha256:59ef3764d0fe818125a5097d2ae867ca3fa64df032331b7e0917cf5d7bf66b13"}, + {file = "pandas-2.2.3-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:f00d1345d84d8c86a63e476bb4955e46458b304b9575dcf71102b5c705320015"}, + {file = "pandas-2.2.3-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:3508d914817e153ad359d7e069d752cdd736a247c322d932eb89e6bc84217f28"}, + {file = "pandas-2.2.3-cp313-cp313-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:22a9d949bfc9a502d320aa04e5d02feab689d61da4e7764b62c30b991c42c5f0"}, + {file = "pandas-2.2.3-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:f3a255b2c19987fbbe62a9dfd6cff7ff2aa9ccab3fc75218fd4b7530f01efa24"}, + {file = "pandas-2.2.3-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:800250ecdadb6d9c78eae4990da62743b857b470883fa27f652db8bdde7f6659"}, + {file = "pandas-2.2.3-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:6374c452ff3ec675a8f46fd9ab25c4ad0ba590b71cf0656f8b6daa5202bca3fb"}, + {file = "pandas-2.2.3-cp313-cp313-win_amd64.whl", hash = "sha256:61c5ad4043f791b61dd4752191d9f07f0ae412515d59ba8f005832a532f8736d"}, + {file = "pandas-2.2.3-cp313-cp313t-macosx_10_13_x86_64.whl", hash = "sha256:3b71f27954685ee685317063bf13c7709a7ba74fc996b84fc6821c59b0f06468"}, + {file = "pandas-2.2.3-cp313-cp313t-macosx_11_0_arm64.whl", hash = "sha256:38cf8125c40dae9d5acc10fa66af8ea6fdf760b2714ee482ca691fc66e6fcb18"}, + {file = "pandas-2.2.3-cp313-cp313t-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:ba96630bc17c875161df3818780af30e43be9b166ce51c9a18c1feae342906c2"}, + {file = "pandas-2.2.3-cp313-cp313t-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:1db71525a1538b30142094edb9adc10be3f3e176748cd7acc2240c2f2e5aa3a4"}, + {file = "pandas-2.2.3-cp313-cp313t-musllinux_1_2_aarch64.whl", hash = "sha256:15c0e1e02e93116177d29ff83e8b1619c93ddc9c49083f237d4312337a61165d"}, + {file = "pandas-2.2.3-cp313-cp313t-musllinux_1_2_x86_64.whl", hash = "sha256:ad5b65698ab28ed8d7f18790a0dc58005c7629f227be9ecc1072aa74c0c1d43a"}, + {file = "pandas-2.2.3-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:bc6b93f9b966093cb0fd62ff1a7e4c09e6d546ad7c1de191767baffc57628f39"}, + {file = "pandas-2.2.3-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:5dbca4c1acd72e8eeef4753eeca07de9b1db4f398669d5994086f788a5d7cc30"}, + {file = "pandas-2.2.3-cp39-cp39-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:8cd6d7cc958a3910f934ea8dbdf17b2364827bb4dafc38ce6eef6bb3d65ff09c"}, + {file = "pandas-2.2.3-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:99df71520d25fade9db7c1076ac94eb994f4d2673ef2aa2e86ee039b6746d20c"}, + {file = "pandas-2.2.3-cp39-cp39-musllinux_1_2_aarch64.whl", hash = "sha256:31d0ced62d4ea3e231a9f228366919a5ea0b07440d9d4dac345376fd8e1477ea"}, + {file = "pandas-2.2.3-cp39-cp39-musllinux_1_2_x86_64.whl", hash = "sha256:7eee9e7cea6adf3e3d24e304ac6b8300646e2a5d1cd3a3c2abed9101b0846761"}, + {file = "pandas-2.2.3-cp39-cp39-win_amd64.whl", hash = "sha256:4850ba03528b6dd51d6c5d273c46f183f39a9baf3f0143e566b89450965b105e"}, + {file = "pandas-2.2.3.tar.gz", hash = "sha256:4f18ba62b61d7e192368b84517265a99b4d7ee8912f8708660fb4a366cc82667"}, ] [package.dependencies] @@ -4433,13 +4453,13 @@ testing = ["pytest", "pytest-cov", "wheel"] [[package]] name = "platformdirs" -version = "4.3.3" +version = "4.3.6" description = "A small Python package for determining appropriate platform-specific dirs, e.g. a `user data dir`." optional = false python-versions = ">=3.8" files = [ - {file = "platformdirs-4.3.3-py3-none-any.whl", hash = "sha256:50a5450e2e84f44539718293cbb1da0a0885c9d14adf21b77bae4e66fc99d9b5"}, - {file = "platformdirs-4.3.3.tar.gz", hash = "sha256:d4e0b7d8ec176b341fb03cb11ca12d0276faa8c485f9cd218f613840463fc2c0"}, + {file = "platformdirs-4.3.6-py3-none-any.whl", hash = "sha256:73e575e1408ab8103900836b97580d5307456908a03e92031bab39e4554cc3fb"}, + {file = "platformdirs-4.3.6.tar.gz", hash = "sha256:357fb2acbc885b0419afd3ce3ed34564c13c9b95c89360cd9563f73aa5e2b907"}, ] [package.extras] @@ -4678,13 +4698,13 @@ virtualenv = ">=20.10.0" [[package]] name = "prometheus-client" -version = "0.20.0" +version = "0.21.0" description = "Python client for the Prometheus monitoring system." optional = false python-versions = ">=3.8" files = [ - {file = "prometheus_client-0.20.0-py3-none-any.whl", hash = "sha256:cde524a85bce83ca359cc837f28b8c0db5cac7aa653a588fd7e84ba061c329e7"}, - {file = "prometheus_client-0.20.0.tar.gz", hash = "sha256:287629d00b147a32dcb2be0b9df905da599b2d82f80377083ec8463309a4bb89"}, + {file = "prometheus_client-0.21.0-py3-none-any.whl", hash = "sha256:4fa6b4dd0ac16d58bb587c04b1caae65b8c5043e85f778f42f5f632f6af2e166"}, + {file = "prometheus_client-0.21.0.tar.gz", hash = "sha256:96c83c606b71ff2b0a433c98889d275f51ffec6c5e267de37c7a2b5c9aa9233e"}, ] [package.extras] @@ -4723,22 +4743,22 @@ testing = ["google-api-core (>=1.31.5)"] [[package]] name = "protobuf" -version = "5.28.1" +version = "5.28.2" description = "" optional = false python-versions = ">=3.8" files = [ - {file = "protobuf-5.28.1-cp310-abi3-win32.whl", hash = "sha256:fc063acaf7a3d9ca13146fefb5b42ac94ab943ec6e978f543cd5637da2d57957"}, - {file = "protobuf-5.28.1-cp310-abi3-win_amd64.whl", hash = "sha256:4c7f5cb38c640919791c9f74ea80c5b82314c69a8409ea36f2599617d03989af"}, - {file = "protobuf-5.28.1-cp38-abi3-macosx_10_9_universal2.whl", hash = "sha256:4304e4fceb823d91699e924a1fdf95cde0e066f3b1c28edb665bda762ecde10f"}, - {file = "protobuf-5.28.1-cp38-abi3-manylinux2014_aarch64.whl", hash = "sha256:0dfd86d2b5edf03d91ec2a7c15b4e950258150f14f9af5f51c17fa224ee1931f"}, - {file = "protobuf-5.28.1-cp38-abi3-manylinux2014_x86_64.whl", hash = "sha256:51f09caab818707ab91cf09cc5c156026599cf05a4520779ccbf53c1b352fb25"}, - {file = "protobuf-5.28.1-cp38-cp38-win32.whl", hash = "sha256:1b04bde117a10ff9d906841a89ec326686c48ececeb65690f15b8cabe7149495"}, - {file = "protobuf-5.28.1-cp38-cp38-win_amd64.whl", hash = "sha256:cabfe43044ee319ad6832b2fda332646f9ef1636b0130186a3ae0a52fc264bb4"}, - {file = "protobuf-5.28.1-cp39-cp39-win32.whl", hash = "sha256:4b4b9a0562a35773ff47a3df823177ab71a1f5eb1ff56d8f842b7432ecfd7fd2"}, - {file = "protobuf-5.28.1-cp39-cp39-win_amd64.whl", hash = "sha256:f24e5d70e6af8ee9672ff605d5503491635f63d5db2fffb6472be78ba62efd8f"}, - {file = "protobuf-5.28.1-py3-none-any.whl", hash = "sha256:c529535e5c0effcf417682563719e5d8ac8d2b93de07a56108b4c2d436d7a29a"}, - {file = "protobuf-5.28.1.tar.gz", hash = "sha256:42597e938f83bb7f3e4b35f03aa45208d49ae8d5bcb4bc10b9fc825e0ab5e423"}, + {file = "protobuf-5.28.2-cp310-abi3-win32.whl", hash = "sha256:eeea10f3dc0ac7e6b4933d32db20662902b4ab81bf28df12218aa389e9c2102d"}, + {file = "protobuf-5.28.2-cp310-abi3-win_amd64.whl", hash = "sha256:2c69461a7fcc8e24be697624c09a839976d82ae75062b11a0972e41fd2cd9132"}, + {file = "protobuf-5.28.2-cp38-abi3-macosx_10_9_universal2.whl", hash = "sha256:a8b9403fc70764b08d2f593ce44f1d2920c5077bf7d311fefec999f8c40f78b7"}, + {file = "protobuf-5.28.2-cp38-abi3-manylinux2014_aarch64.whl", hash = "sha256:35cfcb15f213449af7ff6198d6eb5f739c37d7e4f1c09b5d0641babf2cc0c68f"}, + {file = "protobuf-5.28.2-cp38-abi3-manylinux2014_x86_64.whl", hash = "sha256:5e8a95246d581eef20471b5d5ba010d55f66740942b95ba9b872d918c459452f"}, + {file = "protobuf-5.28.2-cp38-cp38-win32.whl", hash = "sha256:87317e9bcda04a32f2ee82089a204d3a2f0d3c8aeed16568c7daf4756e4f1fe0"}, + {file = "protobuf-5.28.2-cp38-cp38-win_amd64.whl", hash = "sha256:c0ea0123dac3399a2eeb1a1443d82b7afc9ff40241433296769f7da42d142ec3"}, + {file = "protobuf-5.28.2-cp39-cp39-win32.whl", hash = "sha256:ca53faf29896c526863366a52a8f4d88e69cd04ec9571ed6082fa117fac3ab36"}, + {file = "protobuf-5.28.2-cp39-cp39-win_amd64.whl", hash = "sha256:8ddc60bf374785fb7cb12510b267f59067fa10087325b8e1855b898a0d81d276"}, + {file = "protobuf-5.28.2-py3-none-any.whl", hash = "sha256:52235802093bd8a2811abbe8bf0ab9c5f54cca0a751fdd3f6ac2a21438bffece"}, + {file = "protobuf-5.28.2.tar.gz", hash = "sha256:59379674ff119717404f7454647913787034f03fe7049cbef1d74a97bb4593f0"}, ] [[package]] @@ -4992,18 +5012,18 @@ files = [ [[package]] name = "pydantic" -version = "2.9.1" +version = "2.9.2" description = "Data validation using Python type hints" optional = false python-versions = ">=3.8" files = [ - {file = "pydantic-2.9.1-py3-none-any.whl", hash = "sha256:7aff4db5fdf3cf573d4b3c30926a510a10e19a0774d38fc4967f78beb6deb612"}, - {file = "pydantic-2.9.1.tar.gz", hash = "sha256:1363c7d975c7036df0db2b4a61f2e062fbc0aa5ab5f2772e0ffc7191a4f4bce2"}, + {file = "pydantic-2.9.2-py3-none-any.whl", hash = "sha256:f048cec7b26778210e28a0459867920654d48e5e62db0958433636cde4254f12"}, + {file = "pydantic-2.9.2.tar.gz", hash = "sha256:d155cef71265d1e9807ed1c32b4c8deec042a44a50a4188b25ac67ecd81a9c0f"}, ] [package.dependencies] annotated-types = ">=0.6.0" -pydantic-core = "2.23.3" +pydantic-core = "2.23.4" typing-extensions = {version = ">=4.6.1", markers = "python_version < \"3.13\""} [package.extras] @@ -5012,100 +5032,100 @@ timezone = ["tzdata"] [[package]] name = "pydantic-core" -version = "2.23.3" +version = "2.23.4" description = "Core functionality for Pydantic validation and serialization" optional = false python-versions = ">=3.8" files = [ - {file = "pydantic_core-2.23.3-cp310-cp310-macosx_10_12_x86_64.whl", hash = "sha256:7f10a5d1b9281392f1bf507d16ac720e78285dfd635b05737c3911637601bae6"}, - {file = "pydantic_core-2.23.3-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:3c09a7885dd33ee8c65266e5aa7fb7e2f23d49d8043f089989726391dd7350c5"}, - {file = "pydantic_core-2.23.3-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6470b5a1ec4d1c2e9afe928c6cb37eb33381cab99292a708b8cb9aa89e62429b"}, - {file = "pydantic_core-2.23.3-cp310-cp310-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:9172d2088e27d9a185ea0a6c8cebe227a9139fd90295221d7d495944d2367700"}, - {file = "pydantic_core-2.23.3-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:86fc6c762ca7ac8fbbdff80d61b2c59fb6b7d144aa46e2d54d9e1b7b0e780e01"}, - {file = "pydantic_core-2.23.3-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:f0cb80fd5c2df4898693aa841425ea1727b1b6d2167448253077d2a49003e0ed"}, - {file = "pydantic_core-2.23.3-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:03667cec5daf43ac4995cefa8aaf58f99de036204a37b889c24a80927b629cec"}, - {file = "pydantic_core-2.23.3-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:047531242f8e9c2db733599f1c612925de095e93c9cc0e599e96cf536aaf56ba"}, - {file = "pydantic_core-2.23.3-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:5499798317fff7f25dbef9347f4451b91ac2a4330c6669821c8202fd354c7bee"}, - {file = "pydantic_core-2.23.3-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:bbb5e45eab7624440516ee3722a3044b83fff4c0372efe183fd6ba678ff681fe"}, - {file = "pydantic_core-2.23.3-cp310-none-win32.whl", hash = "sha256:8b5b3ed73abb147704a6e9f556d8c5cb078f8c095be4588e669d315e0d11893b"}, - {file = "pydantic_core-2.23.3-cp310-none-win_amd64.whl", hash = "sha256:2b603cde285322758a0279995b5796d64b63060bfbe214b50a3ca23b5cee3e83"}, - {file = "pydantic_core-2.23.3-cp311-cp311-macosx_10_12_x86_64.whl", hash = "sha256:c889fd87e1f1bbeb877c2ee56b63bb297de4636661cc9bbfcf4b34e5e925bc27"}, - {file = "pydantic_core-2.23.3-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:ea85bda3189fb27503af4c45273735bcde3dd31c1ab17d11f37b04877859ef45"}, - {file = "pydantic_core-2.23.3-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a7f7f72f721223f33d3dc98a791666ebc6a91fa023ce63733709f4894a7dc611"}, - {file = "pydantic_core-2.23.3-cp311-cp311-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:2b2b55b0448e9da68f56b696f313949cda1039e8ec7b5d294285335b53104b61"}, - {file = "pydantic_core-2.23.3-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:c24574c7e92e2c56379706b9a3f07c1e0c7f2f87a41b6ee86653100c4ce343e5"}, - {file = "pydantic_core-2.23.3-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:f2b05e6ccbee333a8f4b8f4d7c244fdb7a979e90977ad9c51ea31261e2085ce0"}, - {file = "pydantic_core-2.23.3-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e2c409ce1c219c091e47cb03feb3c4ed8c2b8e004efc940da0166aaee8f9d6c8"}, - {file = "pydantic_core-2.23.3-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:d965e8b325f443ed3196db890d85dfebbb09f7384486a77461347f4adb1fa7f8"}, - {file = "pydantic_core-2.23.3-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:f56af3a420fb1ffaf43ece3ea09c2d27c444e7c40dcb7c6e7cf57aae764f2b48"}, - {file = "pydantic_core-2.23.3-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:5b01a078dd4f9a52494370af21aa52964e0a96d4862ac64ff7cea06e0f12d2c5"}, - {file = "pydantic_core-2.23.3-cp311-none-win32.whl", hash = "sha256:560e32f0df04ac69b3dd818f71339983f6d1f70eb99d4d1f8e9705fb6c34a5c1"}, - {file = "pydantic_core-2.23.3-cp311-none-win_amd64.whl", hash = "sha256:c744fa100fdea0d000d8bcddee95213d2de2e95b9c12be083370b2072333a0fa"}, - {file = "pydantic_core-2.23.3-cp312-cp312-macosx_10_12_x86_64.whl", hash = "sha256:e0ec50663feedf64d21bad0809f5857bac1ce91deded203efc4a84b31b2e4305"}, - {file = "pydantic_core-2.23.3-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:db6e6afcb95edbe6b357786684b71008499836e91f2a4a1e55b840955b341dbb"}, - {file = "pydantic_core-2.23.3-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:98ccd69edcf49f0875d86942f4418a4e83eb3047f20eb897bffa62a5d419c8fa"}, - {file = "pydantic_core-2.23.3-cp312-cp312-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:a678c1ac5c5ec5685af0133262103defb427114e62eafeda12f1357a12140162"}, - {file = "pydantic_core-2.23.3-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:01491d8b4d8db9f3391d93b0df60701e644ff0894352947f31fff3e52bd5c801"}, - {file = "pydantic_core-2.23.3-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:fcf31facf2796a2d3b7fe338fe8640aa0166e4e55b4cb108dbfd1058049bf4cb"}, - {file = "pydantic_core-2.23.3-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7200fd561fb3be06827340da066df4311d0b6b8eb0c2116a110be5245dceb326"}, - {file = "pydantic_core-2.23.3-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:dc1636770a809dee2bd44dd74b89cc80eb41172bcad8af75dd0bc182c2666d4c"}, - {file = "pydantic_core-2.23.3-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:67a5def279309f2e23014b608c4150b0c2d323bd7bccd27ff07b001c12c2415c"}, - {file = "pydantic_core-2.23.3-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:748bdf985014c6dd3e1e4cc3db90f1c3ecc7246ff5a3cd4ddab20c768b2f1dab"}, - {file = "pydantic_core-2.23.3-cp312-none-win32.whl", hash = "sha256:255ec6dcb899c115f1e2a64bc9ebc24cc0e3ab097775755244f77360d1f3c06c"}, - {file = "pydantic_core-2.23.3-cp312-none-win_amd64.whl", hash = "sha256:40b8441be16c1e940abebed83cd006ddb9e3737a279e339dbd6d31578b802f7b"}, - {file = "pydantic_core-2.23.3-cp313-cp313-macosx_10_12_x86_64.whl", hash = "sha256:6daaf5b1ba1369a22c8b050b643250e3e5efc6a78366d323294aee54953a4d5f"}, - {file = "pydantic_core-2.23.3-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:d015e63b985a78a3d4ccffd3bdf22b7c20b3bbd4b8227809b3e8e75bc37f9cb2"}, - {file = "pydantic_core-2.23.3-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a3fc572d9b5b5cfe13f8e8a6e26271d5d13f80173724b738557a8c7f3a8a3791"}, - {file = "pydantic_core-2.23.3-cp313-cp313-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:f6bd91345b5163ee7448bee201ed7dd601ca24f43f439109b0212e296eb5b423"}, - {file = "pydantic_core-2.23.3-cp313-cp313-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:fc379c73fd66606628b866f661e8785088afe2adaba78e6bbe80796baf708a63"}, - {file = "pydantic_core-2.23.3-cp313-cp313-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:fbdce4b47592f9e296e19ac31667daed8753c8367ebb34b9a9bd89dacaa299c9"}, - {file = "pydantic_core-2.23.3-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:fc3cf31edf405a161a0adad83246568647c54404739b614b1ff43dad2b02e6d5"}, - {file = "pydantic_core-2.23.3-cp313-cp313-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:8e22b477bf90db71c156f89a55bfe4d25177b81fce4aa09294d9e805eec13855"}, - {file = "pydantic_core-2.23.3-cp313-cp313-musllinux_1_1_aarch64.whl", hash = "sha256:0a0137ddf462575d9bce863c4c95bac3493ba8e22f8c28ca94634b4a1d3e2bb4"}, - {file = "pydantic_core-2.23.3-cp313-cp313-musllinux_1_1_x86_64.whl", hash = "sha256:203171e48946c3164fe7691fc349c79241ff8f28306abd4cad5f4f75ed80bc8d"}, - {file = "pydantic_core-2.23.3-cp313-none-win32.whl", hash = "sha256:76bdab0de4acb3f119c2a4bff740e0c7dc2e6de7692774620f7452ce11ca76c8"}, - {file = "pydantic_core-2.23.3-cp313-none-win_amd64.whl", hash = "sha256:37ba321ac2a46100c578a92e9a6aa33afe9ec99ffa084424291d84e456f490c1"}, - {file = "pydantic_core-2.23.3-cp38-cp38-macosx_10_12_x86_64.whl", hash = "sha256:d063c6b9fed7d992bcbebfc9133f4c24b7a7f215d6b102f3e082b1117cddb72c"}, - {file = "pydantic_core-2.23.3-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:6cb968da9a0746a0cf521b2b5ef25fc5a0bee9b9a1a8214e0a1cfaea5be7e8a4"}, - {file = "pydantic_core-2.23.3-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:edbefe079a520c5984e30e1f1f29325054b59534729c25b874a16a5048028d16"}, - {file = "pydantic_core-2.23.3-cp38-cp38-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:cbaaf2ef20d282659093913da9d402108203f7cb5955020bd8d1ae5a2325d1c4"}, - {file = "pydantic_core-2.23.3-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:fb539d7e5dc4aac345846f290cf504d2fd3c1be26ac4e8b5e4c2b688069ff4cf"}, - {file = "pydantic_core-2.23.3-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:7e6f33503c5495059148cc486867e1d24ca35df5fc064686e631e314d959ad5b"}, - {file = "pydantic_core-2.23.3-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:04b07490bc2f6f2717b10c3969e1b830f5720b632f8ae2f3b8b1542394c47a8e"}, - {file = "pydantic_core-2.23.3-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:03795b9e8a5d7fda05f3873efc3f59105e2dcff14231680296b87b80bb327295"}, - {file = "pydantic_core-2.23.3-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:c483dab0f14b8d3f0df0c6c18d70b21b086f74c87ab03c59250dbf6d3c89baba"}, - {file = "pydantic_core-2.23.3-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:8b2682038e255e94baf2c473dca914a7460069171ff5cdd4080be18ab8a7fd6e"}, - {file = "pydantic_core-2.23.3-cp38-none-win32.whl", hash = "sha256:f4a57db8966b3a1d1a350012839c6a0099f0898c56512dfade8a1fe5fb278710"}, - {file = "pydantic_core-2.23.3-cp38-none-win_amd64.whl", hash = "sha256:13dd45ba2561603681a2676ca56006d6dee94493f03d5cadc055d2055615c3ea"}, - {file = "pydantic_core-2.23.3-cp39-cp39-macosx_10_12_x86_64.whl", hash = "sha256:82da2f4703894134a9f000e24965df73cc103e31e8c31906cc1ee89fde72cbd8"}, - {file = "pydantic_core-2.23.3-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:dd9be0a42de08f4b58a3cc73a123f124f65c24698b95a54c1543065baca8cf0e"}, - {file = "pydantic_core-2.23.3-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:89b731f25c80830c76fdb13705c68fef6a2b6dc494402987c7ea9584fe189f5d"}, - {file = "pydantic_core-2.23.3-cp39-cp39-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:c6de1ec30c4bb94f3a69c9f5f2182baeda5b809f806676675e9ef6b8dc936f28"}, - {file = "pydantic_core-2.23.3-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:bb68b41c3fa64587412b104294b9cbb027509dc2f6958446c502638d481525ef"}, - {file = "pydantic_core-2.23.3-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:1c3980f2843de5184656aab58698011b42763ccba11c4a8c35936c8dd6c7068c"}, - {file = "pydantic_core-2.23.3-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:94f85614f2cba13f62c3c6481716e4adeae48e1eaa7e8bac379b9d177d93947a"}, - {file = "pydantic_core-2.23.3-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:510b7fb0a86dc8f10a8bb43bd2f97beb63cffad1203071dc434dac26453955cd"}, - {file = "pydantic_core-2.23.3-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:1eba2f7ce3e30ee2170410e2171867ea73dbd692433b81a93758ab2de6c64835"}, - {file = "pydantic_core-2.23.3-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:4b259fd8409ab84b4041b7b3f24dcc41e4696f180b775961ca8142b5b21d0e70"}, - {file = "pydantic_core-2.23.3-cp39-none-win32.whl", hash = "sha256:40d9bd259538dba2f40963286009bf7caf18b5112b19d2b55b09c14dde6db6a7"}, - {file = "pydantic_core-2.23.3-cp39-none-win_amd64.whl", hash = "sha256:5a8cd3074a98ee70173a8633ad3c10e00dcb991ecec57263aacb4095c5efb958"}, - {file = "pydantic_core-2.23.3-pp310-pypy310_pp73-macosx_10_12_x86_64.whl", hash = "sha256:f399e8657c67313476a121a6944311fab377085ca7f490648c9af97fc732732d"}, - {file = "pydantic_core-2.23.3-pp310-pypy310_pp73-macosx_11_0_arm64.whl", hash = "sha256:6b5547d098c76e1694ba85f05b595720d7c60d342f24d5aad32c3049131fa5c4"}, - {file = "pydantic_core-2.23.3-pp310-pypy310_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:0dda0290a6f608504882d9f7650975b4651ff91c85673341789a476b1159f211"}, - {file = "pydantic_core-2.23.3-pp310-pypy310_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:65b6e5da855e9c55a0c67f4db8a492bf13d8d3316a59999cfbaf98cc6e401961"}, - {file = "pydantic_core-2.23.3-pp310-pypy310_pp73-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:09e926397f392059ce0afdcac920df29d9c833256354d0c55f1584b0b70cf07e"}, - {file = "pydantic_core-2.23.3-pp310-pypy310_pp73-musllinux_1_1_aarch64.whl", hash = "sha256:87cfa0ed6b8c5bd6ae8b66de941cece179281239d482f363814d2b986b79cedc"}, - {file = "pydantic_core-2.23.3-pp310-pypy310_pp73-musllinux_1_1_x86_64.whl", hash = "sha256:e61328920154b6a44d98cabcb709f10e8b74276bc709c9a513a8c37a18786cc4"}, - {file = "pydantic_core-2.23.3-pp310-pypy310_pp73-win_amd64.whl", hash = "sha256:ce3317d155628301d649fe5e16a99528d5680af4ec7aa70b90b8dacd2d725c9b"}, - {file = "pydantic_core-2.23.3-pp39-pypy39_pp73-macosx_10_12_x86_64.whl", hash = "sha256:e89513f014c6be0d17b00a9a7c81b1c426f4eb9224b15433f3d98c1a071f8433"}, - {file = "pydantic_core-2.23.3-pp39-pypy39_pp73-macosx_11_0_arm64.whl", hash = "sha256:4f62c1c953d7ee375df5eb2e44ad50ce2f5aff931723b398b8bc6f0ac159791a"}, - {file = "pydantic_core-2.23.3-pp39-pypy39_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2718443bc671c7ac331de4eef9b673063b10af32a0bb385019ad61dcf2cc8f6c"}, - {file = "pydantic_core-2.23.3-pp39-pypy39_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a0d90e08b2727c5d01af1b5ef4121d2f0c99fbee692c762f4d9d0409c9da6541"}, - {file = "pydantic_core-2.23.3-pp39-pypy39_pp73-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:2b676583fc459c64146debea14ba3af54e540b61762dfc0613dc4e98c3f66eeb"}, - {file = "pydantic_core-2.23.3-pp39-pypy39_pp73-musllinux_1_1_aarch64.whl", hash = "sha256:50e4661f3337977740fdbfbae084ae5693e505ca2b3130a6d4eb0f2281dc43b8"}, - {file = "pydantic_core-2.23.3-pp39-pypy39_pp73-musllinux_1_1_x86_64.whl", hash = "sha256:68f4cf373f0de6abfe599a38307f4417c1c867ca381c03df27c873a9069cda25"}, - {file = "pydantic_core-2.23.3-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:59d52cf01854cb26c46958552a21acb10dd78a52aa34c86f284e66b209db8cab"}, - {file = "pydantic_core-2.23.3.tar.gz", hash = "sha256:3cb0f65d8b4121c1b015c60104a685feb929a29d7cf204387c7f2688c7974690"}, + {file = "pydantic_core-2.23.4-cp310-cp310-macosx_10_12_x86_64.whl", hash = "sha256:b10bd51f823d891193d4717448fab065733958bdb6a6b351967bd349d48d5c9b"}, + {file = "pydantic_core-2.23.4-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:4fc714bdbfb534f94034efaa6eadd74e5b93c8fa6315565a222f7b6f42ca1166"}, + {file = "pydantic_core-2.23.4-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:63e46b3169866bd62849936de036f901a9356e36376079b05efa83caeaa02ceb"}, + {file = "pydantic_core-2.23.4-cp310-cp310-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:ed1a53de42fbe34853ba90513cea21673481cd81ed1be739f7f2efb931b24916"}, + {file = "pydantic_core-2.23.4-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:cfdd16ab5e59fc31b5e906d1a3f666571abc367598e3e02c83403acabc092e07"}, + {file = "pydantic_core-2.23.4-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:255a8ef062cbf6674450e668482456abac99a5583bbafb73f9ad469540a3a232"}, + {file = "pydantic_core-2.23.4-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:4a7cd62e831afe623fbb7aabbb4fe583212115b3ef38a9f6b71869ba644624a2"}, + {file = "pydantic_core-2.23.4-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:f09e2ff1f17c2b51f2bc76d1cc33da96298f0a036a137f5440ab3ec5360b624f"}, + {file = "pydantic_core-2.23.4-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:e38e63e6f3d1cec5a27e0afe90a085af8b6806ee208b33030e65b6516353f1a3"}, + {file = "pydantic_core-2.23.4-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:0dbd8dbed2085ed23b5c04afa29d8fd2771674223135dc9bc937f3c09284d071"}, + {file = "pydantic_core-2.23.4-cp310-none-win32.whl", hash = "sha256:6531b7ca5f951d663c339002e91aaebda765ec7d61b7d1e3991051906ddde119"}, + {file = "pydantic_core-2.23.4-cp310-none-win_amd64.whl", hash = "sha256:7c9129eb40958b3d4500fa2467e6a83356b3b61bfff1b414c7361d9220f9ae8f"}, + {file = "pydantic_core-2.23.4-cp311-cp311-macosx_10_12_x86_64.whl", hash = "sha256:77733e3892bb0a7fa797826361ce8a9184d25c8dffaec60b7ffe928153680ba8"}, + {file = "pydantic_core-2.23.4-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:1b84d168f6c48fabd1f2027a3d1bdfe62f92cade1fb273a5d68e621da0e44e6d"}, + {file = "pydantic_core-2.23.4-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:df49e7a0861a8c36d089c1ed57d308623d60416dab2647a4a17fe050ba85de0e"}, + {file = "pydantic_core-2.23.4-cp311-cp311-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:ff02b6d461a6de369f07ec15e465a88895f3223eb75073ffea56b84d9331f607"}, + {file = "pydantic_core-2.23.4-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:996a38a83508c54c78a5f41456b0103c30508fed9abcad0a59b876d7398f25fd"}, + {file = "pydantic_core-2.23.4-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:d97683ddee4723ae8c95d1eddac7c192e8c552da0c73a925a89fa8649bf13eea"}, + {file = "pydantic_core-2.23.4-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:216f9b2d7713eb98cb83c80b9c794de1f6b7e3145eef40400c62e86cee5f4e1e"}, + {file = "pydantic_core-2.23.4-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:6f783e0ec4803c787bcea93e13e9932edab72068f68ecffdf86a99fd5918878b"}, + {file = "pydantic_core-2.23.4-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:d0776dea117cf5272382634bd2a5c1b6eb16767c223c6a5317cd3e2a757c61a0"}, + {file = "pydantic_core-2.23.4-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:d5f7a395a8cf1621939692dba2a6b6a830efa6b3cee787d82c7de1ad2930de64"}, + {file = "pydantic_core-2.23.4-cp311-none-win32.whl", hash = "sha256:74b9127ffea03643e998e0c5ad9bd3811d3dac8c676e47db17b0ee7c3c3bf35f"}, + {file = "pydantic_core-2.23.4-cp311-none-win_amd64.whl", hash = "sha256:98d134c954828488b153d88ba1f34e14259284f256180ce659e8d83e9c05eaa3"}, + {file = "pydantic_core-2.23.4-cp312-cp312-macosx_10_12_x86_64.whl", hash = "sha256:f3e0da4ebaef65158d4dfd7d3678aad692f7666877df0002b8a522cdf088f231"}, + {file = "pydantic_core-2.23.4-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:f69a8e0b033b747bb3e36a44e7732f0c99f7edd5cea723d45bc0d6e95377ffee"}, + {file = "pydantic_core-2.23.4-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:723314c1d51722ab28bfcd5240d858512ffd3116449c557a1336cbe3919beb87"}, + {file = "pydantic_core-2.23.4-cp312-cp312-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:bb2802e667b7051a1bebbfe93684841cc9351004e2badbd6411bf357ab8d5ac8"}, + {file = "pydantic_core-2.23.4-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:d18ca8148bebe1b0a382a27a8ee60350091a6ddaf475fa05ef50dc35b5df6327"}, + {file = "pydantic_core-2.23.4-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:33e3d65a85a2a4a0dc3b092b938a4062b1a05f3a9abde65ea93b233bca0e03f2"}, + {file = "pydantic_core-2.23.4-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:128585782e5bfa515c590ccee4b727fb76925dd04a98864182b22e89a4e6ed36"}, + {file = "pydantic_core-2.23.4-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:68665f4c17edcceecc112dfed5dbe6f92261fb9d6054b47d01bf6371a6196126"}, + {file = "pydantic_core-2.23.4-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:20152074317d9bed6b7a95ade3b7d6054845d70584216160860425f4fbd5ee9e"}, + {file = "pydantic_core-2.23.4-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:9261d3ce84fa1d38ed649c3638feefeae23d32ba9182963e465d58d62203bd24"}, + {file = "pydantic_core-2.23.4-cp312-none-win32.whl", hash = "sha256:4ba762ed58e8d68657fc1281e9bb72e1c3e79cc5d464be146e260c541ec12d84"}, + {file = "pydantic_core-2.23.4-cp312-none-win_amd64.whl", hash = "sha256:97df63000f4fea395b2824da80e169731088656d1818a11b95f3b173747b6cd9"}, + {file = "pydantic_core-2.23.4-cp313-cp313-macosx_10_12_x86_64.whl", hash = "sha256:7530e201d10d7d14abce4fb54cfe5b94a0aefc87da539d0346a484ead376c3cc"}, + {file = "pydantic_core-2.23.4-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:df933278128ea1cd77772673c73954e53a1c95a4fdf41eef97c2b779271bd0bd"}, + {file = "pydantic_core-2.23.4-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:0cb3da3fd1b6a5d0279a01877713dbda118a2a4fc6f0d821a57da2e464793f05"}, + {file = "pydantic_core-2.23.4-cp313-cp313-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:42c6dcb030aefb668a2b7009c85b27f90e51e6a3b4d5c9bc4c57631292015b0d"}, + {file = "pydantic_core-2.23.4-cp313-cp313-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:696dd8d674d6ce621ab9d45b205df149399e4bb9aa34102c970b721554828510"}, + {file = "pydantic_core-2.23.4-cp313-cp313-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:2971bb5ffe72cc0f555c13e19b23c85b654dd2a8f7ab493c262071377bfce9f6"}, + {file = "pydantic_core-2.23.4-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8394d940e5d400d04cad4f75c0598665cbb81aecefaca82ca85bd28264af7f9b"}, + {file = "pydantic_core-2.23.4-cp313-cp313-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:0dff76e0602ca7d4cdaacc1ac4c005e0ce0dcfe095d5b5259163a80d3a10d327"}, + {file = "pydantic_core-2.23.4-cp313-cp313-musllinux_1_1_aarch64.whl", hash = "sha256:7d32706badfe136888bdea71c0def994644e09fff0bfe47441deaed8e96fdbc6"}, + {file = "pydantic_core-2.23.4-cp313-cp313-musllinux_1_1_x86_64.whl", hash = "sha256:ed541d70698978a20eb63d8c5d72f2cc6d7079d9d90f6b50bad07826f1320f5f"}, + {file = "pydantic_core-2.23.4-cp313-none-win32.whl", hash = "sha256:3d5639516376dce1940ea36edf408c554475369f5da2abd45d44621cb616f769"}, + {file = "pydantic_core-2.23.4-cp313-none-win_amd64.whl", hash = "sha256:5a1504ad17ba4210df3a045132a7baeeba5a200e930f57512ee02909fc5c4cb5"}, + {file = "pydantic_core-2.23.4-cp38-cp38-macosx_10_12_x86_64.whl", hash = "sha256:d4488a93b071c04dc20f5cecc3631fc78b9789dd72483ba15d423b5b3689b555"}, + {file = "pydantic_core-2.23.4-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:81965a16b675b35e1d09dd14df53f190f9129c0202356ed44ab2728b1c905658"}, + {file = "pydantic_core-2.23.4-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:4ffa2ebd4c8530079140dd2d7f794a9d9a73cbb8e9d59ffe24c63436efa8f271"}, + {file = "pydantic_core-2.23.4-cp38-cp38-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:61817945f2fe7d166e75fbfb28004034b48e44878177fc54d81688e7b85a3665"}, + {file = "pydantic_core-2.23.4-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:29d2c342c4bc01b88402d60189f3df065fb0dda3654744d5a165a5288a657368"}, + {file = "pydantic_core-2.23.4-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:5e11661ce0fd30a6790e8bcdf263b9ec5988e95e63cf901972107efc49218b13"}, + {file = "pydantic_core-2.23.4-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:9d18368b137c6295db49ce7218b1a9ba15c5bc254c96d7c9f9e924a9bc7825ad"}, + {file = "pydantic_core-2.23.4-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:ec4e55f79b1c4ffb2eecd8a0cfba9955a2588497d96851f4c8f99aa4a1d39b12"}, + {file = "pydantic_core-2.23.4-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:374a5e5049eda9e0a44c696c7ade3ff355f06b1fe0bb945ea3cac2bc336478a2"}, + {file = "pydantic_core-2.23.4-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:5c364564d17da23db1106787675fc7af45f2f7b58b4173bfdd105564e132e6fb"}, + {file = "pydantic_core-2.23.4-cp38-none-win32.whl", hash = "sha256:d7a80d21d613eec45e3d41eb22f8f94ddc758a6c4720842dc74c0581f54993d6"}, + {file = "pydantic_core-2.23.4-cp38-none-win_amd64.whl", hash = "sha256:5f5ff8d839f4566a474a969508fe1c5e59c31c80d9e140566f9a37bba7b8d556"}, + {file = "pydantic_core-2.23.4-cp39-cp39-macosx_10_12_x86_64.whl", hash = "sha256:a4fa4fc04dff799089689f4fd502ce7d59de529fc2f40a2c8836886c03e0175a"}, + {file = "pydantic_core-2.23.4-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:0a7df63886be5e270da67e0966cf4afbae86069501d35c8c1b3b6c168f42cb36"}, + {file = "pydantic_core-2.23.4-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:dcedcd19a557e182628afa1d553c3895a9f825b936415d0dbd3cd0bbcfd29b4b"}, + {file = "pydantic_core-2.23.4-cp39-cp39-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:5f54b118ce5de9ac21c363d9b3caa6c800341e8c47a508787e5868c6b79c9323"}, + {file = "pydantic_core-2.23.4-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:86d2f57d3e1379a9525c5ab067b27dbb8a0642fb5d454e17a9ac434f9ce523e3"}, + {file = "pydantic_core-2.23.4-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:de6d1d1b9e5101508cb37ab0d972357cac5235f5c6533d1071964c47139257df"}, + {file = "pydantic_core-2.23.4-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:1278e0d324f6908e872730c9102b0112477a7f7cf88b308e4fc36ce1bdb6d58c"}, + {file = "pydantic_core-2.23.4-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:9a6b5099eeec78827553827f4c6b8615978bb4b6a88e5d9b93eddf8bb6790f55"}, + {file = "pydantic_core-2.23.4-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:e55541f756f9b3ee346b840103f32779c695a19826a4c442b7954550a0972040"}, + {file = "pydantic_core-2.23.4-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:a5c7ba8ffb6d6f8f2ab08743be203654bb1aaa8c9dcb09f82ddd34eadb695605"}, + {file = "pydantic_core-2.23.4-cp39-none-win32.whl", hash = "sha256:37b0fe330e4a58d3c58b24d91d1eb102aeec675a3db4c292ec3928ecd892a9a6"}, + {file = "pydantic_core-2.23.4-cp39-none-win_amd64.whl", hash = "sha256:1498bec4c05c9c787bde9125cfdcc63a41004ff167f495063191b863399b1a29"}, + {file = "pydantic_core-2.23.4-pp310-pypy310_pp73-macosx_10_12_x86_64.whl", hash = "sha256:f455ee30a9d61d3e1a15abd5068827773d6e4dc513e795f380cdd59932c782d5"}, + {file = "pydantic_core-2.23.4-pp310-pypy310_pp73-macosx_11_0_arm64.whl", hash = "sha256:1e90d2e3bd2c3863d48525d297cd143fe541be8bbf6f579504b9712cb6b643ec"}, + {file = "pydantic_core-2.23.4-pp310-pypy310_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2e203fdf807ac7e12ab59ca2bfcabb38c7cf0b33c41efeb00f8e5da1d86af480"}, + {file = "pydantic_core-2.23.4-pp310-pypy310_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e08277a400de01bc72436a0ccd02bdf596631411f592ad985dcee21445bd0068"}, + {file = "pydantic_core-2.23.4-pp310-pypy310_pp73-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:f220b0eea5965dec25480b6333c788fb72ce5f9129e8759ef876a1d805d00801"}, + {file = "pydantic_core-2.23.4-pp310-pypy310_pp73-musllinux_1_1_aarch64.whl", hash = "sha256:d06b0c8da4f16d1d1e352134427cb194a0a6e19ad5db9161bf32b2113409e728"}, + {file = "pydantic_core-2.23.4-pp310-pypy310_pp73-musllinux_1_1_x86_64.whl", hash = "sha256:ba1a0996f6c2773bd83e63f18914c1de3c9dd26d55f4ac302a7efe93fb8e7433"}, + {file = "pydantic_core-2.23.4-pp310-pypy310_pp73-win_amd64.whl", hash = "sha256:9a5bce9d23aac8f0cf0836ecfc033896aa8443b501c58d0602dbfd5bd5b37753"}, + {file = "pydantic_core-2.23.4-pp39-pypy39_pp73-macosx_10_12_x86_64.whl", hash = "sha256:78ddaaa81421a29574a682b3179d4cf9e6d405a09b99d93ddcf7e5239c742e21"}, + {file = "pydantic_core-2.23.4-pp39-pypy39_pp73-macosx_11_0_arm64.whl", hash = "sha256:883a91b5dd7d26492ff2f04f40fbb652de40fcc0afe07e8129e8ae779c2110eb"}, + {file = "pydantic_core-2.23.4-pp39-pypy39_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:88ad334a15b32a791ea935af224b9de1bf99bcd62fabf745d5f3442199d86d59"}, + {file = "pydantic_core-2.23.4-pp39-pypy39_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:233710f069d251feb12a56da21e14cca67994eab08362207785cf8c598e74577"}, + {file = "pydantic_core-2.23.4-pp39-pypy39_pp73-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:19442362866a753485ba5e4be408964644dd6a09123d9416c54cd49171f50744"}, + {file = "pydantic_core-2.23.4-pp39-pypy39_pp73-musllinux_1_1_aarch64.whl", hash = "sha256:624e278a7d29b6445e4e813af92af37820fafb6dcc55c012c834f9e26f9aaaef"}, + {file = "pydantic_core-2.23.4-pp39-pypy39_pp73-musllinux_1_1_x86_64.whl", hash = "sha256:f5ef8f42bec47f21d07668a043f077d507e5bf4e668d5c6dfe6aaba89de1a5b8"}, + {file = "pydantic_core-2.23.4-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:aea443fffa9fbe3af1a9ba721a87f926fe548d32cab71d188a6ede77d0ff244e"}, + {file = "pydantic_core-2.23.4.tar.gz", hash = "sha256:2584f7cf844ac4d970fba483a717dbe10c1c1c96a969bf65d61ffe94df1b2863"}, ] [package.dependencies] @@ -6422,29 +6442,29 @@ pyasn1 = ">=0.1.3" [[package]] name = "ruff" -version = "0.6.5" +version = "0.6.7" description = "An extremely fast Python linter and code formatter, written in Rust." optional = false python-versions = ">=3.7" files = [ - {file = "ruff-0.6.5-py3-none-linux_armv6l.whl", hash = "sha256:7e4e308f16e07c95fc7753fc1aaac690a323b2bb9f4ec5e844a97bb7fbebd748"}, - {file = "ruff-0.6.5-py3-none-macosx_10_12_x86_64.whl", hash = "sha256:932cd69eefe4daf8c7d92bd6689f7e8182571cb934ea720af218929da7bd7d69"}, - {file = "ruff-0.6.5-py3-none-macosx_11_0_arm64.whl", hash = "sha256:3a8d42d11fff8d3143ff4da41742a98f8f233bf8890e9fe23077826818f8d680"}, - {file = "ruff-0.6.5-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a50af6e828ee692fb10ff2dfe53f05caecf077f4210fae9677e06a808275754f"}, - {file = "ruff-0.6.5-py3-none-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:794ada3400a0d0b89e3015f1a7e01f4c97320ac665b7bc3ade24b50b54cb2972"}, - {file = "ruff-0.6.5-py3-none-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:381413ec47f71ce1d1c614f7779d88886f406f1fd53d289c77e4e533dc6ea200"}, - {file = "ruff-0.6.5-py3-none-manylinux_2_17_ppc64.manylinux2014_ppc64.whl", hash = "sha256:52e75a82bbc9b42e63c08d22ad0ac525117e72aee9729a069d7c4f235fc4d276"}, - {file = "ruff-0.6.5-py3-none-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:09c72a833fd3551135ceddcba5ebdb68ff89225d30758027280968c9acdc7810"}, - {file = "ruff-0.6.5-py3-none-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:800c50371bdcb99b3c1551d5691e14d16d6f07063a518770254227f7f6e8c178"}, - {file = "ruff-0.6.5-py3-none-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8e25ddd9cd63ba1f3bd51c1f09903904a6adf8429df34f17d728a8fa11174253"}, - {file = "ruff-0.6.5-py3-none-musllinux_1_2_aarch64.whl", hash = "sha256:7291e64d7129f24d1b0c947ec3ec4c0076e958d1475c61202497c6aced35dd19"}, - {file = "ruff-0.6.5-py3-none-musllinux_1_2_armv7l.whl", hash = "sha256:9ad7dfbd138d09d9a7e6931e6a7e797651ce29becd688be8a0d4d5f8177b4b0c"}, - {file = "ruff-0.6.5-py3-none-musllinux_1_2_i686.whl", hash = "sha256:005256d977021790cc52aa23d78f06bb5090dc0bfbd42de46d49c201533982ae"}, - {file = "ruff-0.6.5-py3-none-musllinux_1_2_x86_64.whl", hash = "sha256:482c1e6bfeb615eafc5899127b805d28e387bd87db38b2c0c41d271f5e58d8cc"}, - {file = "ruff-0.6.5-py3-none-win32.whl", hash = "sha256:cf4d3fa53644137f6a4a27a2b397381d16454a1566ae5335855c187fbf67e4f5"}, - {file = "ruff-0.6.5-py3-none-win_amd64.whl", hash = "sha256:3e42a57b58e3612051a636bc1ac4e6b838679530235520e8f095f7c44f706ff9"}, - {file = "ruff-0.6.5-py3-none-win_arm64.whl", hash = "sha256:51935067740773afdf97493ba9b8231279e9beef0f2a8079188c4776c25688e0"}, - {file = "ruff-0.6.5.tar.gz", hash = "sha256:4d32d87fab433c0cf285c3683dd4dae63be05fd7a1d65b3f5bf7cdd05a6b96fb"}, + {file = "ruff-0.6.7-py3-none-linux_armv6l.whl", hash = "sha256:08277b217534bfdcc2e1377f7f933e1c7957453e8a79764d004e44c40db923f2"}, + {file = "ruff-0.6.7-py3-none-macosx_10_12_x86_64.whl", hash = "sha256:c6707a32e03b791f4448dc0dce24b636cbcdee4dd5607adc24e5ee73fd86c00a"}, + {file = "ruff-0.6.7-py3-none-macosx_11_0_arm64.whl", hash = "sha256:533d66b7774ef224e7cf91506a7dafcc9e8ec7c059263ec46629e54e7b1f90ab"}, + {file = "ruff-0.6.7-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:17a86aac6f915932d259f7bec79173e356165518859f94649d8c50b81ff087e9"}, + {file = "ruff-0.6.7-py3-none-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:b3f8822defd260ae2460ea3832b24d37d203c3577f48b055590a426a722d50ef"}, + {file = "ruff-0.6.7-py3-none-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:9ba4efe5c6dbbb58be58dd83feedb83b5e95c00091bf09987b4baf510fee5c99"}, + {file = "ruff-0.6.7-py3-none-manylinux_2_17_ppc64.manylinux2014_ppc64.whl", hash = "sha256:525201b77f94d2b54868f0cbe5edc018e64c22563da6c5c2e5c107a4e85c1c0d"}, + {file = "ruff-0.6.7-py3-none-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:8854450839f339e1049fdbe15d875384242b8e85d5c6947bb2faad33c651020b"}, + {file = "ruff-0.6.7-py3-none-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:2f0b62056246234d59cbf2ea66e84812dc9ec4540518e37553513392c171cb18"}, + {file = "ruff-0.6.7-py3-none-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6b1462fa56c832dc0cea5b4041cfc9c97813505d11cce74ebc6d1aae068de36b"}, + {file = "ruff-0.6.7-py3-none-musllinux_1_2_aarch64.whl", hash = "sha256:02b083770e4cdb1495ed313f5694c62808e71764ec6ee5db84eedd82fd32d8f5"}, + {file = "ruff-0.6.7-py3-none-musllinux_1_2_armv7l.whl", hash = "sha256:0c05fd37013de36dfa883a3854fae57b3113aaa8abf5dea79202675991d48624"}, + {file = "ruff-0.6.7-py3-none-musllinux_1_2_i686.whl", hash = "sha256:f49c9caa28d9bbfac4a637ae10327b3db00f47d038f3fbb2195c4d682e925b14"}, + {file = "ruff-0.6.7-py3-none-musllinux_1_2_x86_64.whl", hash = "sha256:a0e1655868164e114ba43a908fd2d64a271a23660195017c17691fb6355d59bb"}, + {file = "ruff-0.6.7-py3-none-win32.whl", hash = "sha256:a939ca435b49f6966a7dd64b765c9df16f1faed0ca3b6f16acdf7731969deb35"}, + {file = "ruff-0.6.7-py3-none-win_amd64.whl", hash = "sha256:590445eec5653f36248584579c06252ad2e110a5d1f32db5420de35fb0e1c977"}, + {file = "ruff-0.6.7-py3-none-win_arm64.whl", hash = "sha256:b28f0d5e2f771c1fe3c7a45d3f53916fc74a480698c4b5731f0bea61e52137c8"}, + {file = "ruff-0.6.7.tar.gz", hash = "sha256:44e52129d82266fa59b587e2cd74def5637b730a69c4542525dfdecfaae38bd5"}, ] [[package]] @@ -6596,13 +6616,13 @@ win32 = ["pywin32"] [[package]] name = "setuptools" -version = "75.0.0" +version = "75.1.0" description = "Easily download, build, install, upgrade, and uninstall Python packages" optional = false python-versions = ">=3.8" files = [ - {file = "setuptools-75.0.0-py3-none-any.whl", hash = "sha256:791ae94f04f78c880b5e614e560dd32d4b4af5d151bd9e7483e3377846caf90a"}, - {file = "setuptools-75.0.0.tar.gz", hash = "sha256:25af69c809d9334cd8e653d385277abeb5a102dca255954005a7092d282575ea"}, + {file = "setuptools-75.1.0-py3-none-any.whl", hash = "sha256:35ab7fd3bcd95e6b7fd704e4a1539513edad446c097797f2985e0e4b960772f2"}, + {file = "setuptools-75.1.0.tar.gz", hash = "sha256:d59a21b17a275fb872a9c3dae73963160ae079f1049ed956880cd7c09b120538"}, ] [package.extras] @@ -6803,60 +6823,60 @@ jsonschema = ">=3.0" [[package]] name = "sqlalchemy" -version = "2.0.34" +version = "2.0.35" description = "Database Abstraction Library" optional = false python-versions = ">=3.7" files = [ - {file = "SQLAlchemy-2.0.34-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:95d0b2cf8791ab5fb9e3aa3d9a79a0d5d51f55b6357eecf532a120ba3b5524db"}, - {file = "SQLAlchemy-2.0.34-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:243f92596f4fd4c8bd30ab8e8dd5965afe226363d75cab2468f2c707f64cd83b"}, - {file = "SQLAlchemy-2.0.34-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:9ea54f7300553af0a2a7235e9b85f4204e1fc21848f917a3213b0e0818de9a24"}, - {file = "SQLAlchemy-2.0.34-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:173f5f122d2e1bff8fbd9f7811b7942bead1f5e9f371cdf9e670b327e6703ebd"}, - {file = "SQLAlchemy-2.0.34-cp310-cp310-musllinux_1_2_aarch64.whl", hash = "sha256:196958cde924a00488e3e83ff917be3b73cd4ed8352bbc0f2989333176d1c54d"}, - {file = "SQLAlchemy-2.0.34-cp310-cp310-musllinux_1_2_x86_64.whl", hash = "sha256:bd90c221ed4e60ac9d476db967f436cfcecbd4ef744537c0f2d5291439848768"}, - {file = "SQLAlchemy-2.0.34-cp310-cp310-win32.whl", hash = "sha256:3166dfff2d16fe9be3241ee60ece6fcb01cf8e74dd7c5e0b64f8e19fab44911b"}, - {file = "SQLAlchemy-2.0.34-cp310-cp310-win_amd64.whl", hash = "sha256:6831a78bbd3c40f909b3e5233f87341f12d0b34a58f14115c9e94b4cdaf726d3"}, - {file = "SQLAlchemy-2.0.34-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:c7db3db284a0edaebe87f8f6642c2b2c27ed85c3e70064b84d1c9e4ec06d5d84"}, - {file = "SQLAlchemy-2.0.34-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:430093fce0efc7941d911d34f75a70084f12f6ca5c15d19595c18753edb7c33b"}, - {file = "SQLAlchemy-2.0.34-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:79cb400c360c7c210097b147c16a9e4c14688a6402445ac848f296ade6283bbc"}, - {file = "SQLAlchemy-2.0.34-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:fb1b30f31a36c7f3fee848391ff77eebdd3af5750bf95fbf9b8b5323edfdb4ec"}, - {file = "SQLAlchemy-2.0.34-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:8fddde2368e777ea2a4891a3fb4341e910a056be0bb15303bf1b92f073b80c02"}, - {file = "SQLAlchemy-2.0.34-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:80bd73ea335203b125cf1d8e50fef06be709619eb6ab9e7b891ea34b5baa2287"}, - {file = "SQLAlchemy-2.0.34-cp311-cp311-win32.whl", hash = "sha256:6daeb8382d0df526372abd9cb795c992e18eed25ef2c43afe518c73f8cccb721"}, - {file = "SQLAlchemy-2.0.34-cp311-cp311-win_amd64.whl", hash = "sha256:5bc08e75ed11693ecb648b7a0a4ed80da6d10845e44be0c98c03f2f880b68ff4"}, - {file = "SQLAlchemy-2.0.34-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:53e68b091492c8ed2bd0141e00ad3089bcc6bf0e6ec4142ad6505b4afe64163e"}, - {file = "SQLAlchemy-2.0.34-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:bcd18441a49499bf5528deaa9dee1f5c01ca491fc2791b13604e8f972877f812"}, - {file = "SQLAlchemy-2.0.34-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:165bbe0b376541092bf49542bd9827b048357f4623486096fc9aaa6d4e7c59a2"}, - {file = "SQLAlchemy-2.0.34-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:c3330415cd387d2b88600e8e26b510d0370db9b7eaf984354a43e19c40df2e2b"}, - {file = "SQLAlchemy-2.0.34-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:97b850f73f8abbffb66ccbab6e55a195a0eb655e5dc74624d15cff4bfb35bd74"}, - {file = "SQLAlchemy-2.0.34-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:7cee4c6917857fd6121ed84f56d1dc78eb1d0e87f845ab5a568aba73e78adf83"}, - {file = "SQLAlchemy-2.0.34-cp312-cp312-win32.whl", hash = "sha256:fbb034f565ecbe6c530dff948239377ba859420d146d5f62f0271407ffb8c580"}, - {file = "SQLAlchemy-2.0.34-cp312-cp312-win_amd64.whl", hash = "sha256:707c8f44931a4facd4149b52b75b80544a8d824162602b8cd2fe788207307f9a"}, - {file = "SQLAlchemy-2.0.34-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:24af3dc43568f3780b7e1e57c49b41d98b2d940c1fd2e62d65d3928b6f95f021"}, - {file = "SQLAlchemy-2.0.34-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:e60ed6ef0a35c6b76b7640fe452d0e47acc832ccbb8475de549a5cc5f90c2c06"}, - {file = "SQLAlchemy-2.0.34-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:413c85cd0177c23e32dee6898c67a5f49296640041d98fddb2c40888fe4daa2e"}, - {file = "SQLAlchemy-2.0.34-cp37-cp37m-musllinux_1_2_aarch64.whl", hash = "sha256:25691f4adfb9d5e796fd48bf1432272f95f4bbe5f89c475a788f31232ea6afba"}, - {file = "SQLAlchemy-2.0.34-cp37-cp37m-musllinux_1_2_x86_64.whl", hash = "sha256:526ce723265643dbc4c7efb54f56648cc30e7abe20f387d763364b3ce7506c82"}, - {file = "SQLAlchemy-2.0.34-cp37-cp37m-win32.whl", hash = "sha256:13be2cc683b76977a700948411a94c67ad8faf542fa7da2a4b167f2244781cf3"}, - {file = "SQLAlchemy-2.0.34-cp37-cp37m-win_amd64.whl", hash = "sha256:e54ef33ea80d464c3dcfe881eb00ad5921b60f8115ea1a30d781653edc2fd6a2"}, - {file = "SQLAlchemy-2.0.34-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:43f28005141165edd11fbbf1541c920bd29e167b8bbc1fb410d4fe2269c1667a"}, - {file = "SQLAlchemy-2.0.34-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:b68094b165a9e930aedef90725a8fcfafe9ef95370cbb54abc0464062dbf808f"}, - {file = "SQLAlchemy-2.0.34-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6a1e03db964e9d32f112bae36f0cc1dcd1988d096cfd75d6a588a3c3def9ab2b"}, - {file = "SQLAlchemy-2.0.34-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:203d46bddeaa7982f9c3cc693e5bc93db476ab5de9d4b4640d5c99ff219bee8c"}, - {file = "SQLAlchemy-2.0.34-cp38-cp38-musllinux_1_2_aarch64.whl", hash = "sha256:ae92bebca3b1e6bd203494e5ef919a60fb6dfe4d9a47ed2453211d3bd451b9f5"}, - {file = "SQLAlchemy-2.0.34-cp38-cp38-musllinux_1_2_x86_64.whl", hash = "sha256:9661268415f450c95f72f0ac1217cc6f10256f860eed85c2ae32e75b60278ad8"}, - {file = "SQLAlchemy-2.0.34-cp38-cp38-win32.whl", hash = "sha256:895184dfef8708e15f7516bd930bda7e50ead069280d2ce09ba11781b630a434"}, - {file = "SQLAlchemy-2.0.34-cp38-cp38-win_amd64.whl", hash = "sha256:6e7cde3a2221aa89247944cafb1b26616380e30c63e37ed19ff0bba5e968688d"}, - {file = "SQLAlchemy-2.0.34-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:dbcdf987f3aceef9763b6d7b1fd3e4ee210ddd26cac421d78b3c206d07b2700b"}, - {file = "SQLAlchemy-2.0.34-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:ce119fc4ce0d64124d37f66a6f2a584fddc3c5001755f8a49f1ca0a177ef9796"}, - {file = "SQLAlchemy-2.0.34-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a17d8fac6df9835d8e2b4c5523666e7051d0897a93756518a1fe101c7f47f2f0"}, - {file = "SQLAlchemy-2.0.34-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:9ebc11c54c6ecdd07bb4efbfa1554538982f5432dfb8456958b6d46b9f834bb7"}, - {file = "SQLAlchemy-2.0.34-cp39-cp39-musllinux_1_2_aarch64.whl", hash = "sha256:2e6965346fc1491a566e019a4a1d3dfc081ce7ac1a736536367ca305da6472a8"}, - {file = "SQLAlchemy-2.0.34-cp39-cp39-musllinux_1_2_x86_64.whl", hash = "sha256:220574e78ad986aea8e81ac68821e47ea9202b7e44f251b7ed8c66d9ae3f4278"}, - {file = "SQLAlchemy-2.0.34-cp39-cp39-win32.whl", hash = "sha256:b75b00083e7fe6621ce13cfce9d4469c4774e55e8e9d38c305b37f13cf1e874c"}, - {file = "SQLAlchemy-2.0.34-cp39-cp39-win_amd64.whl", hash = "sha256:c29d03e0adf3cc1a8c3ec62d176824972ae29b67a66cbb18daff3062acc6faa8"}, - {file = "SQLAlchemy-2.0.34-py3-none-any.whl", hash = "sha256:7286c353ee6475613d8beff83167374006c6b3e3f0e6491bfe8ca610eb1dec0f"}, - {file = "sqlalchemy-2.0.34.tar.gz", hash = "sha256:10d8f36990dd929690666679b0f42235c159a7051534adb135728ee52828dd22"}, + {file = "SQLAlchemy-2.0.35-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:67219632be22f14750f0d1c70e62f204ba69d28f62fd6432ba05ab295853de9b"}, + {file = "SQLAlchemy-2.0.35-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:4668bd8faf7e5b71c0319407b608f278f279668f358857dbfd10ef1954ac9f90"}, + {file = "SQLAlchemy-2.0.35-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:cb8bea573863762bbf45d1e13f87c2d2fd32cee2dbd50d050f83f87429c9e1ea"}, + {file = "SQLAlchemy-2.0.35-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:f552023710d4b93d8fb29a91fadf97de89c5926c6bd758897875435f2a939f33"}, + {file = "SQLAlchemy-2.0.35-cp310-cp310-musllinux_1_2_aarch64.whl", hash = "sha256:016b2e665f778f13d3c438651dd4de244214b527a275e0acf1d44c05bc6026a9"}, + {file = "SQLAlchemy-2.0.35-cp310-cp310-musllinux_1_2_x86_64.whl", hash = "sha256:7befc148de64b6060937231cbff8d01ccf0bfd75aa26383ffdf8d82b12ec04ff"}, + {file = "SQLAlchemy-2.0.35-cp310-cp310-win32.whl", hash = "sha256:22b83aed390e3099584b839b93f80a0f4a95ee7f48270c97c90acd40ee646f0b"}, + {file = "SQLAlchemy-2.0.35-cp310-cp310-win_amd64.whl", hash = "sha256:a29762cd3d116585278ffb2e5b8cc311fb095ea278b96feef28d0b423154858e"}, + {file = "SQLAlchemy-2.0.35-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:e21f66748ab725ade40fa7af8ec8b5019c68ab00b929f6643e1b1af461eddb60"}, + {file = "SQLAlchemy-2.0.35-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:8a6219108a15fc6d24de499d0d515c7235c617b2540d97116b663dade1a54d62"}, + {file = "SQLAlchemy-2.0.35-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:042622a5306c23b972192283f4e22372da3b8ddf5f7aac1cc5d9c9b222ab3ff6"}, + {file = "SQLAlchemy-2.0.35-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:627dee0c280eea91aed87b20a1f849e9ae2fe719d52cbf847c0e0ea34464b3f7"}, + {file = "SQLAlchemy-2.0.35-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:4fdcd72a789c1c31ed242fd8c1bcd9ea186a98ee8e5408a50e610edfef980d71"}, + {file = "SQLAlchemy-2.0.35-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:89b64cd8898a3a6f642db4eb7b26d1b28a497d4022eccd7717ca066823e9fb01"}, + {file = "SQLAlchemy-2.0.35-cp311-cp311-win32.whl", hash = "sha256:6a93c5a0dfe8d34951e8a6f499a9479ffb9258123551fa007fc708ae2ac2bc5e"}, + {file = "SQLAlchemy-2.0.35-cp311-cp311-win_amd64.whl", hash = "sha256:c68fe3fcde03920c46697585620135b4ecfdfc1ed23e75cc2c2ae9f8502c10b8"}, + {file = "SQLAlchemy-2.0.35-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:eb60b026d8ad0c97917cb81d3662d0b39b8ff1335e3fabb24984c6acd0c900a2"}, + {file = "SQLAlchemy-2.0.35-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:6921ee01caf375363be5e9ae70d08ce7ca9d7e0e8983183080211a062d299468"}, + {file = "SQLAlchemy-2.0.35-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8cdf1a0dbe5ced887a9b127da4ffd7354e9c1a3b9bb330dce84df6b70ccb3a8d"}, + {file = "SQLAlchemy-2.0.35-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:93a71c8601e823236ac0e5d087e4f397874a421017b3318fd92c0b14acf2b6db"}, + {file = "SQLAlchemy-2.0.35-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:e04b622bb8a88f10e439084486f2f6349bf4d50605ac3e445869c7ea5cf0fa8c"}, + {file = "SQLAlchemy-2.0.35-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:1b56961e2d31389aaadf4906d453859f35302b4eb818d34a26fab72596076bb8"}, + {file = "SQLAlchemy-2.0.35-cp312-cp312-win32.whl", hash = "sha256:0f9f3f9a3763b9c4deb8c5d09c4cc52ffe49f9876af41cc1b2ad0138878453cf"}, + {file = "SQLAlchemy-2.0.35-cp312-cp312-win_amd64.whl", hash = "sha256:25b0f63e7fcc2a6290cb5f7f5b4fc4047843504983a28856ce9b35d8f7de03cc"}, + {file = "SQLAlchemy-2.0.35-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:f021d334f2ca692523aaf7bbf7592ceff70c8594fad853416a81d66b35e3abf9"}, + {file = "SQLAlchemy-2.0.35-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:05c3f58cf91683102f2f0265c0db3bd3892e9eedabe059720492dbaa4f922da1"}, + {file = "SQLAlchemy-2.0.35-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:032d979ce77a6c2432653322ba4cbeabf5a6837f704d16fa38b5a05d8e21fa00"}, + {file = "SQLAlchemy-2.0.35-cp37-cp37m-musllinux_1_2_aarch64.whl", hash = "sha256:2e795c2f7d7249b75bb5f479b432a51b59041580d20599d4e112b5f2046437a3"}, + {file = "SQLAlchemy-2.0.35-cp37-cp37m-musllinux_1_2_x86_64.whl", hash = "sha256:cc32b2990fc34380ec2f6195f33a76b6cdaa9eecf09f0c9404b74fc120aef36f"}, + {file = "SQLAlchemy-2.0.35-cp37-cp37m-win32.whl", hash = "sha256:9509c4123491d0e63fb5e16199e09f8e262066e58903e84615c301dde8fa2e87"}, + {file = "SQLAlchemy-2.0.35-cp37-cp37m-win_amd64.whl", hash = "sha256:3655af10ebcc0f1e4e06c5900bb33e080d6a1fa4228f502121f28a3b1753cde5"}, + {file = "SQLAlchemy-2.0.35-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:4c31943b61ed8fdd63dfd12ccc919f2bf95eefca133767db6fbbd15da62078ec"}, + {file = "SQLAlchemy-2.0.35-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:a62dd5d7cc8626a3634208df458c5fe4f21200d96a74d122c83bc2015b333bc1"}, + {file = "SQLAlchemy-2.0.35-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:0630774b0977804fba4b6bbea6852ab56c14965a2b0c7fc7282c5f7d90a1ae72"}, + {file = "SQLAlchemy-2.0.35-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8d625eddf7efeba2abfd9c014a22c0f6b3796e0ffb48f5d5ab106568ef01ff5a"}, + {file = "SQLAlchemy-2.0.35-cp38-cp38-musllinux_1_2_aarch64.whl", hash = "sha256:ada603db10bb865bbe591939de854faf2c60f43c9b763e90f653224138f910d9"}, + {file = "SQLAlchemy-2.0.35-cp38-cp38-musllinux_1_2_x86_64.whl", hash = "sha256:c41411e192f8d3ea39ea70e0fae48762cd11a2244e03751a98bd3c0ca9a4e936"}, + {file = "SQLAlchemy-2.0.35-cp38-cp38-win32.whl", hash = "sha256:d299797d75cd747e7797b1b41817111406b8b10a4f88b6e8fe5b5e59598b43b0"}, + {file = "SQLAlchemy-2.0.35-cp38-cp38-win_amd64.whl", hash = "sha256:0375a141e1c0878103eb3d719eb6d5aa444b490c96f3fedab8471c7f6ffe70ee"}, + {file = "SQLAlchemy-2.0.35-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:ccae5de2a0140d8be6838c331604f91d6fafd0735dbdcee1ac78fc8fbaba76b4"}, + {file = "SQLAlchemy-2.0.35-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:2a275a806f73e849e1c309ac11108ea1a14cd7058577aba962cd7190e27c9e3c"}, + {file = "SQLAlchemy-2.0.35-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:732e026240cdd1c1b2e3ac515c7a23820430ed94292ce33806a95869c46bd139"}, + {file = "SQLAlchemy-2.0.35-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:890da8cd1941fa3dab28c5bac3b9da8502e7e366f895b3b8e500896f12f94d11"}, + {file = "SQLAlchemy-2.0.35-cp39-cp39-musllinux_1_2_aarch64.whl", hash = "sha256:c0d8326269dbf944b9201911b0d9f3dc524d64779a07518199a58384c3d37a44"}, + {file = "SQLAlchemy-2.0.35-cp39-cp39-musllinux_1_2_x86_64.whl", hash = "sha256:b76d63495b0508ab9fc23f8152bac63205d2a704cd009a2b0722f4c8e0cba8e0"}, + {file = "SQLAlchemy-2.0.35-cp39-cp39-win32.whl", hash = "sha256:69683e02e8a9de37f17985905a5eca18ad651bf592314b4d3d799029797d0eb3"}, + {file = "SQLAlchemy-2.0.35-cp39-cp39-win_amd64.whl", hash = "sha256:aee110e4ef3c528f3abbc3c2018c121e708938adeeff9006428dd7c8555e9b3f"}, + {file = "SQLAlchemy-2.0.35-py3-none-any.whl", hash = "sha256:2ab3f0336c0387662ce6221ad30ab3a5e6499aab01b9790879b6578fd9b8faa1"}, + {file = "sqlalchemy-2.0.35.tar.gz", hash = "sha256:e11d7ea4d24f0a262bccf9a7cd6284c976c5369dac21db237cff59586045ab9f"}, ] [package.dependencies] @@ -6924,47 +6944,47 @@ tests = ["cython", "littleutils", "pygments", "pytest", "typeguard"] [[package]] name = "statsmodels" -version = "0.14.2" +version = "0.14.3" description = "Statistical computations and models for Python" optional = false python-versions = ">=3.9" files = [ - {file = "statsmodels-0.14.2-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:df5d6f95c46f0341da6c79ee7617e025bf2b371e190a8e60af1ae9cabbdb7a97"}, - {file = "statsmodels-0.14.2-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:a87ef21fadb445b650f327340dde703f13aec1540f3d497afb66324499dea97a"}, - {file = "statsmodels-0.14.2-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:5827a12e3ede2b98a784476d61d6bec43011fedb64aa815f2098e0573bece257"}, - {file = "statsmodels-0.14.2-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:10f2b7611a61adb7d596a6d239abdf1a4d5492b931b00d5ed23d32844d40e48e"}, - {file = "statsmodels-0.14.2-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:c254c66142f1167b4c7d031cf8db55294cc62ff3280e090fc45bd10a7f5fd029"}, - {file = "statsmodels-0.14.2-cp310-cp310-win_amd64.whl", hash = "sha256:0e46e9d59293c1af4cc1f4e5248f17e7e7bc596bfce44d327c789ac27f09111b"}, - {file = "statsmodels-0.14.2-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:50fcb633987779e795142f51ba49fb27648d46e8a1382b32ebe8e503aaabaa9e"}, - {file = "statsmodels-0.14.2-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:876794068abfaeed41df71b7887000031ecf44fbfa6b50d53ccb12ebb4ab747a"}, - {file = "statsmodels-0.14.2-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:7a91f6c4943de13e3ce2e20ee3b5d26d02bd42300616a421becd53756f5deb37"}, - {file = "statsmodels-0.14.2-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:4864a1c4615c5ea5f2e3b078a75bdedc90dd9da210a37e0738e064b419eccee2"}, - {file = "statsmodels-0.14.2-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:afbd92410e0df06f3d8c4e7c0e2e71f63f4969531f280fb66059e2ecdb6e0415"}, - {file = "statsmodels-0.14.2-cp311-cp311-win_amd64.whl", hash = "sha256:8e004cfad0e46ce73fe3f3812010c746f0d4cfd48e307b45c14e9e360f3d2510"}, - {file = "statsmodels-0.14.2-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:eb0ba1ad3627705f5ae20af6b2982f500546d43892543b36c7bca3e2f87105e7"}, - {file = "statsmodels-0.14.2-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:90fd2f0110b73fc3fa5a2f21c3ca99b0e22285cccf38e56b5b8fd8ce28791b0f"}, - {file = "statsmodels-0.14.2-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ac780ad9ff552773798829a0b9c46820b0faa10e6454891f5e49a845123758ab"}, - {file = "statsmodels-0.14.2-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:55d1742778400ae67acb04b50a2c7f5804182f8a874bd09ca397d69ed159a751"}, - {file = "statsmodels-0.14.2-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:f870d14a587ea58a3b596aa994c2ed889cc051f9e450e887d2c83656fc6a64bf"}, - {file = "statsmodels-0.14.2-cp312-cp312-win_amd64.whl", hash = "sha256:f450fcbae214aae66bd9d2b9af48e0f8ba1cb0e8596c6ebb34e6e3f0fec6542c"}, - {file = "statsmodels-0.14.2-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:201c3d00929c4a67cda1fe05b098c8dcf1b1eeefa88e80a8f963a844801ed59f"}, - {file = "statsmodels-0.14.2-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:9edefa4ce08e40bc1d67d2f79bc686ee5e238e801312b5a029ee7786448c389a"}, - {file = "statsmodels-0.14.2-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:29c78a7601fdae1aa32104c5ebff2e0b72c26f33e870e2f94ab1bcfd927ece9b"}, - {file = "statsmodels-0.14.2-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:f36494df7c03d63168fccee5038a62f469469ed6a4dd6eaeb9338abedcd0d5f5"}, - {file = "statsmodels-0.14.2-cp39-cp39-win_amd64.whl", hash = "sha256:8875823bdd41806dc853333cc4e1b7ef9481bad2380a999e66ea42382cf2178d"}, - {file = "statsmodels-0.14.2.tar.gz", hash = "sha256:890550147ad3a81cda24f0ba1a5c4021adc1601080bd00e191ae7cd6feecd6ad"}, -] - -[package.dependencies] -numpy = ">=1.22.3" + {file = "statsmodels-0.14.3-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:7372c92f18b8afb06355e067285abb94e8b214afd9f2fda6d3c26f3ea004cbdf"}, + {file = "statsmodels-0.14.3-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:42459cdaafe217f455e6b95c05d9e089caf02dd53295aebe63bc1e0206f83176"}, + {file = "statsmodels-0.14.3-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6a72d3d9fe61f70baf18667bc9cf2e68b6bdd8f5cce4f7b21f9e662e19d2ffdf"}, + {file = "statsmodels-0.14.3-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:9050e5817f23a5adcb87822406b5260758795c42c41fa2fa60816023f0a0d8ef"}, + {file = "statsmodels-0.14.3-cp310-cp310-musllinux_1_2_x86_64.whl", hash = "sha256:f12d74743936323165dae648f75193ee4a47381a85610be661d34de56c7634e0"}, + {file = "statsmodels-0.14.3-cp310-cp310-win_amd64.whl", hash = "sha256:53212f597747534bed475bbd89f4bc39a3757c20692bb7664021e30fbd967c53"}, + {file = "statsmodels-0.14.3-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:e49a63757e12269ef02841f05906e91bdb70f5bc358cbaca97f171f4a4de09c4"}, + {file = "statsmodels-0.14.3-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:de4b989f0fea684f89bdf5ff641f9acb7acddfd712459f28365904a974afaeff"}, + {file = "statsmodels-0.14.3-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:45a5ae177e92348532bf2522f27feecd0589b88b243709b28e2b068631c9c181"}, + {file = "statsmodels-0.14.3-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a736ac24da1388e444bb2b0d381a7307b29074b237acef040a793cfdd508e160"}, + {file = "statsmodels-0.14.3-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:ea8491b6a36fca738403037709e9469412a9d3e8a8e54db482c20e8dd70efa1f"}, + {file = "statsmodels-0.14.3-cp311-cp311-win_amd64.whl", hash = "sha256:efb946ced8243923eb78909834699be55442172cea3dc37158e3e1c5370e4189"}, + {file = "statsmodels-0.14.3-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:9bf3690f71ebacff0c976c1584994174bc1bb72785b5a35645b385a00a5107e0"}, + {file = "statsmodels-0.14.3-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:197bcb1aeaaa5c7e9ba4ad87c2369f9600c6cd69d6e2db829eb46d3d9fe534c9"}, + {file = "statsmodels-0.14.3-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:492b8fd867687f9539b1f7f111dafb2464e04f65fa834585c08725b8aa1a3d98"}, + {file = "statsmodels-0.14.3-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a849e78dcb3ed6416bb9043b9549415f1f8cd00426deb467ff4dfe0acbaaad8e"}, + {file = "statsmodels-0.14.3-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:e8a82aa8a99a428f39a9ead1b03fbd2339e40908412371abe089239d21467fd5"}, + {file = "statsmodels-0.14.3-cp312-cp312-win_amd64.whl", hash = "sha256:5724e51a370227655679f1a487f429919f03de325d7b5702e919526353d0cb1d"}, + {file = "statsmodels-0.14.3-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:280e69721925a936493153dba692b53a2fe4e3f46e5fafd32a453f5d9fa2a344"}, + {file = "statsmodels-0.14.3-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:97f28958e456aea788d4ffd83d7ade82d2a4a3bd5c7e8eabf791f224cddef2bf"}, + {file = "statsmodels-0.14.3-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2ef24d6350a15f5d25f7c6cb774fce89dff77e3687181ce4410cafd6a4004f04"}, + {file = "statsmodels-0.14.3-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:1ddbd07b7d05e16d1a2ea6df3d7e2255dfb3e0363b91d859623d9fc3aff32b4a"}, + {file = "statsmodels-0.14.3-cp39-cp39-win_amd64.whl", hash = "sha256:42dfb9084a5520342248441904357bd5d7fcf01ec05c9bdc7dd764a88e15a9c4"}, + {file = "statsmodels-0.14.3.tar.gz", hash = "sha256:ecf3502643fa93aabe5f0bdf238efb59609517c4d60a811632d31fcdce86c2d2"}, +] + +[package.dependencies] +numpy = ">=1.22.3,<3" packaging = ">=21.3" pandas = ">=1.4,<2.1.0 || >2.1.0" patsy = ">=0.5.6" scipy = ">=1.8,<1.9.2 || >1.9.2" [package.extras] -build = ["cython (>=0.29.33)"] -develop = ["colorama", "cython (>=0.29.33)", "cython (>=3.0.10,<4)", "flake8", "isort", "joblib", "matplotlib (>=3)", "pytest (>=7.3.0,<8)", "pytest-cov", "pytest-randomly", "pytest-xdist", "pywinpty", "setuptools-scm[toml] (>=8.0,<9.0)"] +build = ["cython (>=3.0.10)"] +develop = ["colorama", "cython (>=3.0.10)", "cython (>=3.0.10,<4)", "flake8", "isort", "joblib", "matplotlib (>=3)", "pytest (>=7.3.0,<8)", "pytest-cov", "pytest-randomly", "pytest-xdist", "pywinpty", "setuptools-scm[toml] (>=8.0,<9.0)"] docs = ["ipykernel", "jupyter-client", "matplotlib", "nbconvert", "nbformat", "numpydoc", "pandas-datareader", "sphinx"] [[package]] @@ -7287,13 +7307,13 @@ zstd = ["zstandard (>=0.18.0)"] [[package]] name = "virtualenv" -version = "20.26.4" +version = "20.26.5" description = "Virtual Python Environment builder" optional = false python-versions = ">=3.7" files = [ - {file = "virtualenv-20.26.4-py3-none-any.whl", hash = "sha256:48f2695d9809277003f30776d155615ffc11328e6a0a8c1f0ec80188d7874a55"}, - {file = "virtualenv-20.26.4.tar.gz", hash = "sha256:c17f4e0f3e6036e9f26700446f85c76ab11df65ff6d8a9cbfad9f71aabfcf23c"}, + {file = "virtualenv-20.26.5-py3-none-any.whl", hash = "sha256:4f3ac17b81fba3ce3bd6f4ead2749a72da5929c01774948e243db9ba41df4ff6"}, + {file = "virtualenv-20.26.5.tar.gz", hash = "sha256:ce489cac131aa58f4b25e321d6d186171f78e6cb13fafbf32a840cee67733ff4"}, ] [package.dependencies] diff --git a/requirements-dev.txt b/requirements-dev.txt index c2528c4bf870..da98a6c55f0e 100644 --- a/requirements-dev.txt +++ b/requirements-dev.txt @@ -3,7 +3,7 @@ aiohttp==3.10.5 ; python_version >= "3.10" and python_version < "4.0" aiosignal==1.3.1 ; python_version >= "3.10" and python_version < "4.0" altair==5.4.1 ; python_version >= "3.10" and python_version < "3.13" annotated-types==0.7.0 ; python_version >= "3.10" and python_version < "3.13" -anyio==4.4.0 ; python_version >= "3.10" and python_version < "3.13" +anyio==4.6.0 ; python_version >= "3.10" and python_version < "3.13" anywidget==0.9.13 ; python_version >= "3.10" and python_version < "3.13" appdirs==1.4.4 ; python_version >= "3.10" and python_version < "4.0" appnope==0.1.4 ; python_version >= "3.10" and python_version < "3.13" and platform_system == "Darwin" @@ -47,7 +47,7 @@ db-dtypes==1.3.0 ; python_version >= "3.10" and python_version < "4.0" debugpy==1.8.5 ; python_version >= "3.10" and python_version < "3.13" decorator==5.1.1 ; python_version >= "3.10" and python_version < "4.0" defusedxml==0.7.1 ; python_version >= "3.10" and python_version < "3.13" -deltalake==0.19.2 ; python_version >= "3.10" and python_version < "4.0" +deltalake==0.20.0 ; python_version >= "3.10" and python_version < "4.0" distlib==0.3.8 ; python_version >= "3.10" and python_version < "4.0" doit==0.36.0 ; python_version >= "3.10" and python_version < "3.13" duckdb==1.1.0 ; python_version >= "3.10" and python_version < "4.0" @@ -57,7 +57,7 @@ exceptiongroup==1.2.2 ; python_version >= "3.10" and python_version < "3.11" execnet==2.1.1 ; python_version >= "3.10" and python_version < "4.0" executing==2.1.0 ; python_version >= "3.10" and python_version < "4.0" fastjsonschema==2.20.0 ; python_version >= "3.10" and python_version < "4.0" -filelock==3.16.0 ; python_version >= "3.10" and python_version < "4.0" +filelock==3.16.1 ; python_version >= "3.10" and python_version < "4.0" fonttools==4.53.1 ; python_version >= "3.10" and python_version < "3.13" fqdn==1.5.1 ; python_version >= "3.10" and python_version < "3.13" frozenlist==1.4.1 ; python_version >= "3.10" and python_version < "4.0" @@ -65,10 +65,10 @@ fsspec==2024.9.0 ; python_version >= "3.10" and python_version < "4.0" gcsfs==2024.9.0.post1 ; python_version >= "3.10" and python_version < "4.0" geoarrow-types==0.2.0 ; python_version >= "3.10" and python_version < "4.0" geopandas==1.0.1 ; python_version >= "3.10" and python_version < "4.0" -google-api-core==2.19.2 ; python_version >= "3.10" and python_version < "4.0" -google-api-core[grpc]==2.19.2 ; python_version >= "3.10" and python_version < "4.0" +google-api-core==2.20.0 ; python_version >= "3.10" and python_version < "4.0" +google-api-core[grpc]==2.20.0 ; python_version >= "3.10" and python_version < "4.0" google-auth-oauthlib==1.2.1 ; python_version >= "3.10" and python_version < "4.0" -google-auth==2.34.0 ; python_version >= "3.10" and python_version < "4.0" +google-auth==2.35.0 ; python_version >= "3.10" and python_version < "4.0" google-cloud-bigquery-storage==2.26.0 ; python_version >= "3.10" and python_version < "4.0" google-cloud-bigquery==3.25.0 ; python_version >= "3.10" and python_version < "4.0" google-cloud-core==2.4.1 ; python_version >= "3.10" and python_version < "4.0" @@ -77,7 +77,7 @@ google-crc32c==1.6.0 ; python_version >= "3.10" and python_version < "4.0" google-resumable-media==2.7.2 ; python_version >= "3.10" and python_version < "4.0" googleapis-common-protos==1.65.0 ; python_version >= "3.10" and python_version < "4.0" graphviz==0.20.3 ; python_version >= "3.10" and python_version < "4.0" -greenlet==3.1.0 ; python_version < "3.13" and (platform_machine == "aarch64" or platform_machine == "ppc64le" or platform_machine == "x86_64" or platform_machine == "amd64" or platform_machine == "AMD64" or platform_machine == "win32" or platform_machine == "WIN32") and python_version >= "3.10" +greenlet==3.1.1 ; python_version < "3.13" and (platform_machine == "aarch64" or platform_machine == "ppc64le" or platform_machine == "x86_64" or platform_machine == "amd64" or platform_machine == "AMD64" or platform_machine == "win32" or platform_machine == "WIN32") and python_version >= "3.10" griffe==0.49.0 ; python_version >= "3.10" and python_version < "3.13" grpcio-status==1.66.1 ; python_version >= "3.10" and python_version < "4.0" grpcio==1.66.1 ; python_version >= "3.10" and python_version < "4.0" @@ -109,7 +109,7 @@ jsonschema-specifications==2023.12.1 ; python_version >= "3.10" and python_versi jsonschema==4.23.0 ; python_version >= "3.10" and python_version < "3.13" jsonschema[format-nongpl]==4.23.0 ; python_version >= "3.10" and python_version < "3.13" jupyter-cache==1.0.0 ; python_version >= "3.10" and python_version < "3.13" -jupyter-client==8.6.2 ; python_version >= "3.10" and python_version < "3.13" +jupyter-client==8.6.3 ; python_version >= "3.10" and python_version < "3.13" jupyter-core==5.7.2 ; python_version >= "3.10" and python_version < "3.13" jupyter-events==0.10.0 ; python_version >= "3.10" and python_version < "3.13" jupyter-lsp==2.2.5 ; python_version >= "3.10" and python_version < "3.13" @@ -138,7 +138,7 @@ msgpack==1.1.0 ; python_version >= "3.10" and python_version < "4.0" multidict==6.1.0 ; python_version >= "3.10" and python_version < "4.0" mypy-extensions==1.0.0 ; python_version >= "3.10" and python_version < "4.0" mysqlclient==2.2.4 ; python_version >= "3.10" and python_version < "4.0" -narwhals==1.8.1 ; python_version >= "3.10" and python_version < "3.13" +narwhals==1.8.2 ; python_version >= "3.10" and python_version < "3.13" nbclient==0.10.0 ; python_version >= "3.10" and python_version < "3.13" nbconvert==7.16.4 ; python_version >= "3.10" and python_version < "3.13" nbformat==5.10.4 ; python_version >= "3.10" and python_version < "3.13" @@ -151,8 +151,8 @@ oracledb==2.4.1 ; python_version >= "3.10" and python_version < "4.0" overrides==7.7.0 ; python_version >= "3.10" and python_version < "3.13" packaging==24.1 ; python_version >= "3.10" and python_version < "4.0" palettable==3.3.3 ; python_version >= "3.10" and python_version < "3.13" -pandas==2.2.2 ; python_version >= "3.10" and python_version < "4.0" -pandas[numpy]==2.2.2 ; python_version >= "3.10" and python_version < "4.0" +pandas==2.2.3 ; python_version >= "3.10" and python_version < "4.0" +pandas[numpy]==2.2.3 ; python_version >= "3.10" and python_version < "4.0" pandocfilters==1.5.1 ; python_version >= "3.10" and python_version < "3.13" parso==0.8.4 ; python_version >= "3.10" and python_version < "4.0" parsy==2.1 ; python_version >= "3.10" and python_version < "4.0" @@ -162,7 +162,7 @@ pexpect==4.9.0 ; python_version >= "3.10" and python_version < "4.0" pillow==10.4.0 ; python_version >= "3.10" and python_version < "3.13" pins[gcs]==0.8.6 ; python_version >= "3.10" and python_version < "4.0" pkginfo==1.11.1 ; python_version >= "3.10" and python_version < "4.0" -platformdirs==4.3.3 ; python_version >= "3.10" and python_version < "4.0" +platformdirs==4.3.6 ; python_version >= "3.10" and python_version < "4.0" plotly==5.24.1 ; python_version >= "3.10" and python_version < "3.13" plotnine==0.13.6 ; python_version >= "3.10" and python_version < "3.13" pluggy==1.5.0 ; python_version >= "3.10" and python_version < "4.0" @@ -174,10 +174,10 @@ poetry==1.8.3 ; python_version >= "3.10" and python_version < "4.0" polars==1.7.1 ; python_version >= "3.10" and python_version < "4.0" pprintpp==0.4.0 ; python_version >= "3.10" and python_version < "4.0" pre-commit==3.8.0 ; python_version >= "3.10" and python_version < "4.0" -prometheus-client==0.20.0 ; python_version >= "3.10" and python_version < "3.13" +prometheus-client==0.21.0 ; python_version >= "3.10" and python_version < "3.13" prompt-toolkit==3.0.47 ; python_version >= "3.10" and python_version < "4.0" proto-plus==1.24.0 ; python_version >= "3.10" and python_version < "4.0" -protobuf==5.28.1 ; python_version >= "3.10" and python_version < "4.0" +protobuf==5.28.2 ; python_version >= "3.10" and python_version < "4.0" psutil==6.0.0 ; python_version >= "3.10" and python_version < "3.13" psycopg2==2.9.9 ; python_version >= "3.10" and python_version < "4.0" psygnal==0.11.1 ; python_version >= "3.10" and python_version < "3.13" @@ -191,8 +191,8 @@ pyarrow==17.0.0 ; python_version >= "3.10" and python_version < "4.0" pyasn1-modules==0.4.1 ; python_version >= "3.10" and python_version < "4.0" pyasn1==0.6.1 ; python_version >= "3.10" and python_version < "4" pycparser==2.22 ; python_version >= "3.10" and python_version < "4.0" -pydantic-core==2.23.3 ; python_version >= "3.10" and python_version < "3.13" -pydantic==2.9.1 ; python_version >= "3.10" and python_version < "3.13" +pydantic-core==2.23.4 ; python_version >= "3.10" and python_version < "3.13" +pydantic==2.9.2 ; python_version >= "3.10" and python_version < "3.13" pydata-google-auth==1.8.2 ; python_version >= "3.10" and python_version < "4.0" pydruid==0.6.9 ; python_version >= "3.10" and python_version < "4.0" pyexasol[pandas]==0.27.0 ; python_version >= "3.10" and python_version < "4.0" @@ -238,13 +238,13 @@ rfc3986-validator==0.1.1 ; python_version >= "3.10" and python_version < "3.13" rich==13.8.1 ; python_version >= "3.10" and python_version < "4.0" rpds-py==0.20.0 ; python_version >= "3.10" and python_version < "3.13" rsa==4.9 ; python_version >= "3.10" and python_version < "4" -ruff==0.6.5 ; python_version >= "3.10" and python_version < "4.0" +ruff==0.6.7 ; python_version >= "3.10" and python_version < "4.0" scikit-learn==1.5.2 ; python_version >= "3.10" and python_version < "3.13" scipy==1.14.1 ; python_version >= "3.10" and python_version < "3.13" seaborn==0.13.2 ; python_version >= "3.10" and python_version < "3.13" secretstorage==3.3.3 ; python_version >= "3.10" and python_version < "4.0" and sys_platform == "linux" send2trash==1.8.3 ; python_version >= "3.10" and python_version < "3.13" -setuptools==75.0.0 ; python_version >= "3.10" and python_version < "4.0" +setuptools==75.1.0 ; python_version >= "3.10" and python_version < "4.0" shapely==2.0.6 ; python_version >= "3.10" and python_version < "4.0" shellingham==1.5.4 ; python_version >= "3.10" and python_version < "4.0" six==1.16.0 ; python_version >= "3.10" and python_version < "4.0" @@ -253,10 +253,10 @@ snowflake-connector-python==3.12.2 ; python_version >= "3.10" and python_version sortedcontainers==2.4.0 ; python_version >= "3.10" and python_version < "4.0" soupsieve==2.6 ; python_version >= "3.10" and python_version < "3.13" sphobjinv==2.3.1.1 ; python_version >= "3.10" and python_version < "3.13" -sqlalchemy==2.0.34 ; python_version >= "3.10" and python_version < "3.13" +sqlalchemy==2.0.35 ; python_version >= "3.10" and python_version < "3.13" sqlglot==25.22.0 ; python_version >= "3.10" and python_version < "4.0" stack-data==0.6.3 ; python_version >= "3.10" and python_version < "4.0" -statsmodels==0.14.2 ; python_version >= "3.10" and python_version < "3.13" +statsmodels==0.14.3 ; python_version >= "3.10" and python_version < "3.13" tabulate==0.9.0 ; python_version >= "3.10" and python_version < "3.13" tenacity==9.0.0 ; python_version >= "3.10" and python_version < "3.13" terminado==0.18.1 ; python_version >= "3.10" and python_version < "3.13" @@ -278,7 +278,7 @@ tzdata==2024.1 ; python_version >= "3.10" and python_version < "4.0" tzlocal==5.2 ; python_version >= "3.10" and python_version < "4.0" uri-template==1.3.0 ; python_version >= "3.10" and python_version < "3.13" urllib3==2.2.3 ; python_version >= "3.10" and python_version < "4.0" -virtualenv==20.26.4 ; python_version >= "3.10" and python_version < "4.0" +virtualenv==20.26.5 ; python_version >= "3.10" and python_version < "4.0" watchdog==5.0.2 ; python_version >= "3.10" and python_version < "3.13" wcwidth==0.2.13 ; python_version >= "3.10" and python_version < "4.0" webcolors==24.8.0 ; python_version >= "3.10" and python_version < "3.13" From 4487ff4f0c3c6df06119a900c5ff64f61f79bdb0 Mon Sep 17 00:00:00 2001 From: "renovate[bot]" <29139614+renovate[bot]@users.noreply.github.com> Date: Mon, 23 Sep 2024 05:52:25 -0400 Subject: [PATCH 04/10] chore(deps): update dependency itables to >=1.6.3,<2.3 (#10190) Co-authored-by: renovate[bot] <29139614+renovate[bot]@users.noreply.github.com> Co-authored-by: Phillip Cloud <417981+cpcloud@users.noreply.github.com> --- poetry.lock | 14 ++++++++------ pyproject.toml | 2 +- requirements-dev.txt | 2 +- 3 files changed, 10 insertions(+), 8 deletions(-) diff --git a/poetry.lock b/poetry.lock index eff58ca98c73..577ee5e1068f 100644 --- a/poetry.lock +++ b/poetry.lock @@ -2727,13 +2727,13 @@ arrow = ">=0.15.0" [[package]] name = "itables" -version = "2.1.5" +version = "2.2.1" description = "Pandas and Polar DataFrames as interactive DataTables" optional = false python-versions = ">=3.7" files = [ - {file = "itables-2.1.5-py3-none-any.whl", hash = "sha256:842e7e31f3f6ea5e4a085d189dcb6ee38e20c0750e4322a31eccb895e5a00a76"}, - {file = "itables-2.1.5.tar.gz", hash = "sha256:ed96c98cdd91a91c004c1a0150d572a6b1021a1311bc94e8bfa2e68ba42d44fb"}, + {file = "itables-2.2.1-py3-none-any.whl", hash = "sha256:6d66eeb6cd19b10a2458c5de5913d34785e709bf2dcb85e6652546f541274581"}, + {file = "itables-2.2.1.tar.gz", hash = "sha256:96fcb51ece4afc546f7682b2bc094d5aa9cb2cea0568edb5cd7e51e41f7275b7"}, ] [package.dependencies] @@ -2742,11 +2742,13 @@ numpy = "*" pandas = "*" [package.extras] -all = ["matplotlib", "polars", "pyarrow", "pytz", "world-bank-data"] +all = ["anywidget", "matplotlib", "polars", "pyarrow", "pytz", "traitlets", "world-bank-data"] +dev = ["anywidget", "ipykernel", "jupyterlab", "jupytext", "matplotlib", "nbconvert", "polars", "pyarrow", "pytest", "pytest-cov", "pytz", "requests", "shiny", "traitlets", "watchfiles", "world-bank-data"] polars = ["polars", "pyarrow"] samples = ["pytz", "world-bank-data"] style = ["matplotlib"] -test = ["ipykernel", "jupytext", "matplotlib", "nbconvert", "polars", "pyarrow", "pytest", "pytest-cov", "pytz", "requests", "shiny", "world-bank-data"] +test = ["anywidget", "ipykernel", "jupytext", "matplotlib", "nbconvert", "polars", "pyarrow", "pytest", "pytest-cov", "pytz", "requests", "shiny", "traitlets", "world-bank-data"] +widget = ["anywidget", "traitlets"] [[package]] name = "jaraco-classes" @@ -7917,4 +7919,4 @@ visualization = ["graphviz"] [metadata] lock-version = "2.0" python-versions = "^3.10" -content-hash = "3a61ef75272dd8c592f94823bba3865b7ef5f53ec8bd3ba0263dff7ec9de07ee" +content-hash = "33c56e8ced59cb731699d9ae515d503f113d8e6a885d5eb5687c4b40a369327b" diff --git a/pyproject.toml b/pyproject.toml index 1f5b99d89d92..ca83f20f971d 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -125,7 +125,7 @@ altair = { version = ">=5.0.1,<6", python = ">=3.10,<3.13" } # pin griffe until https://github.com/ibis-project/ibis/pull/9865 is addressed griffe = { version = "<1", python = ">=3.10,<3.13" } ipykernel = { version = ">=6.25.1,<7", python = ">=3.10,<3.13" } -itables = { version = ">=1.6.3,<2.2", python = ">=3.10,<3.13" } +itables = { version = ">=1.6.3,<2.3", python = ">=3.10,<3.13" } nbclient = { version = ">=0.8.0,<1", python = ">=3.10,<3.13" } plotly = { version = ">=5.16.1,<6", python = ">=3.10,<3.13" } plotnine = { version = ">=0.12.2,<1", python = ">=3.10,<3.13" } diff --git a/requirements-dev.txt b/requirements-dev.txt index da98a6c55f0e..aef0726a39f7 100644 --- a/requirements-dev.txt +++ b/requirements-dev.txt @@ -97,7 +97,7 @@ ipykernel==6.29.5 ; python_version >= "3.10" and python_version < "3.13" ipython==8.27.0 ; python_version >= "3.10" and python_version < "4.0" ipywidgets==8.1.5 ; python_version >= "3.10" and python_version < "3.13" isoduration==20.11.0 ; python_version >= "3.10" and python_version < "3.13" -itables==2.1.5 ; python_version >= "3.10" and python_version < "3.13" +itables==2.2.1 ; python_version >= "3.10" and python_version < "3.13" jaraco-classes==3.4.0 ; python_version >= "3.10" and python_version < "4.0" jedi==0.19.1 ; python_version >= "3.10" and python_version < "4.0" jeepney==0.8.0 ; python_version >= "3.10" and python_version < "4.0" and sys_platform == "linux" From 3877d6d46b218a3b683ce2c3175ac72183cecf06 Mon Sep 17 00:00:00 2001 From: Deepyaman Datta Date: Mon, 23 Sep 2024 04:41:50 -0600 Subject: [PATCH 05/10] feat(polars): allow user to specify "engine" kwarg (#10151) Co-authored-by: Phillip Cloud <417981+cpcloud@users.noreply.github.com> Co-authored-by: Jim Crist-Harif --- ibis/backends/polars/__init__.py | 29 +++++++++++++++++++---- ibis/backends/polars/tests/test_client.py | 16 +++++++++++++ 2 files changed, 40 insertions(+), 5 deletions(-) diff --git a/ibis/backends/polars/__init__.py b/ibis/backends/polars/__init__.py index 9e9c25ceb40a..21e322207159 100644 --- a/ibis/backends/polars/__init__.py +++ b/ibis/backends/polars/__init__.py @@ -3,7 +3,7 @@ from collections.abc import Iterable, Mapping from functools import lru_cache from pathlib import Path -from typing import TYPE_CHECKING, Any +from typing import TYPE_CHECKING, Any, Literal import polars as pl @@ -492,6 +492,7 @@ def _to_dataframe( params: Mapping[ir.Expr, object] | None = None, limit: int | None = None, streaming: bool = False, + engine: Literal["cpu", "gpu"] | pl.GPUEngine = "cpu", **kwargs: Any, ) -> pl.DataFrame: self._run_pre_execute_hooks(expr) @@ -501,7 +502,7 @@ def _to_dataframe( limit = ibis.options.sql.default_limit if limit is not None: lf = lf.limit(limit) - df = lf.collect(streaming=streaming) + df = lf.collect(streaming=streaming, engine=engine) # XXX: Polars sometimes returns data with the incorrect column names. # For now we catch this case and rename them here if needed. expected_cols = tuple(table_expr.columns) @@ -515,10 +516,16 @@ def execute( params: Mapping[ir.Expr, object] | None = None, limit: int | None = None, streaming: bool = False, + engine: Literal["cpu", "gpu"] | pl.GPUEngine = "cpu", **kwargs: Any, ): df = self._to_dataframe( - expr, params=params, limit=limit, streaming=streaming, **kwargs + expr, + params=params, + limit=limit, + streaming=streaming, + engine=engine, + **kwargs, ) if isinstance(expr, (ir.Table, ir.Scalar)): return expr.__pandas_result__(df.to_pandas()) @@ -540,10 +547,16 @@ def to_polars( params: Mapping[ir.Expr, object] | None = None, limit: int | None = None, streaming: bool = False, + engine: Literal["cpu", "gpu"] | pl.GPUEngine = "cpu", **kwargs: Any, ): df = self._to_dataframe( - expr, params=params, limit=limit, streaming=streaming, **kwargs + expr, + params=params, + limit=limit, + streaming=streaming, + engine=engine, + **kwargs, ) return expr.__polars_result__(df) @@ -553,12 +566,18 @@ def _to_pyarrow_table( params: Mapping[ir.Expr, object] | None = None, limit: int | None = None, streaming: bool = False, + engine: Literal["cpu", "gpu"] | pl.GPUEngine = "cpu", **kwargs: Any, ): from ibis.formats.pyarrow import PyArrowData df = self._to_dataframe( - expr, params=params, limit=limit, streaming=streaming, **kwargs + expr, + params=params, + limit=limit, + streaming=streaming, + engine=engine, + **kwargs, ) return PyArrowData.convert_table(df.to_arrow(), expr.as_table().schema()) diff --git a/ibis/backends/polars/tests/test_client.py b/ibis/backends/polars/tests/test_client.py index e0d4c698c103..d7f3a829bb64 100644 --- a/ibis/backends/polars/tests/test_client.py +++ b/ibis/backends/polars/tests/test_client.py @@ -60,3 +60,19 @@ def test_memtable_polars_types(con): res = con.to_polars((t.x + t.y + t.z).name("test")) sol = (df["x"] + df["y"] + df["z"]).rename("test") pl.testing.assert_series_equal(res, sol) + + +@pytest.mark.parametrize("to_method", ["to_pyarrow", "to_polars"]) +def test_streaming(con, mocker, to_method): + t = con.table("functional_alltypes") + mocked_collect = mocker.patch("polars.LazyFrame.collect") + getattr(con, to_method)(t, streaming=True) + mocked_collect.assert_called_once_with(streaming=True, engine="cpu") + + +@pytest.mark.parametrize("to_method", ["to_pyarrow", "to_polars"]) +def test_engine(con, mocker, to_method): + t = con.table("functional_alltypes") + mocked_collect = mocker.patch("polars.LazyFrame.collect") + getattr(con, to_method)(t, engine="gpu") + mocked_collect.assert_called_once_with(streaming=False, engine="gpu") From f97bb779a75257b2aa819f79cf3f77354b3908a1 Mon Sep 17 00:00:00 2001 From: Phillip Cloud <417981+cpcloud@users.noreply.github.com> Date: Mon, 23 Sep 2024 09:27:26 -0400 Subject: [PATCH 06/10] fix(polars): use elementwise flatten to flatten nested arrays (#10168) Fixes #10135. --- ibis/backends/polars/compiler.py | 6 +++++- ibis/backends/tests/errors.py | 3 ++- ibis/backends/tests/test_array.py | 22 ++++++++++++++++++++++ 3 files changed, 29 insertions(+), 2 deletions(-) diff --git a/ibis/backends/polars/compiler.py b/ibis/backends/polars/compiler.py index 7bc9c9f14715..20f5ff15749b 100644 --- a/ibis/backends/polars/compiler.py +++ b/ibis/backends/polars/compiler.py @@ -1021,7 +1021,11 @@ def array_flatten(op, **kw): .then(None) .when(result.list.len() == 0) .then([]) - .otherwise(result.flatten()) + # polars doesn't have an efficient API (yet?) for removing one level of + # nesting from an array so we use elementwise evaluation + # + # https://github.com/ibis-project/ibis/issues/10135 + .otherwise(result.list.eval(pl.element().flatten())) ) diff --git a/ibis/backends/tests/errors.py b/ibis/backends/tests/errors.py index cc21b1582850..436f74dab71e 100644 --- a/ibis/backends/tests/errors.py +++ b/ibis/backends/tests/errors.py @@ -103,6 +103,7 @@ TrinoUserError = None try: + from psycopg2.errors import ArraySubscriptError as PsycoPg2ArraySubscriptError from psycopg2.errors import DivisionByZero as PsycoPg2DivisionByZero from psycopg2.errors import IndeterminateDatatype as PsycoPg2IndeterminateDatatype from psycopg2.errors import InternalError_ as PsycoPg2InternalError @@ -118,7 +119,7 @@ PsycoPg2InvalidTextRepresentation ) = PsycoPg2DivisionByZero = PsycoPg2InternalError = PsycoPg2ProgrammingError = ( PsycoPg2OperationalError - ) = PsycoPg2UndefinedObject = None + ) = PsycoPg2UndefinedObject = PsycoPg2ArraySubscriptError = None try: from MySQLdb import NotSupportedError as MySQLNotSupportedError diff --git a/ibis/backends/tests/test_array.py b/ibis/backends/tests/test_array.py index 3b440feb2293..cba8ae64d1aa 100644 --- a/ibis/backends/tests/test_array.py +++ b/ibis/backends/tests/test_array.py @@ -21,6 +21,7 @@ GoogleBadRequest, MySQLOperationalError, PolarsComputeError, + PsycoPg2ArraySubscriptError, PsycoPg2IndeterminateDatatype, PsycoPg2InternalError, PsycoPg2ProgrammingError, @@ -1006,6 +1007,11 @@ def flatten_data(): reason="Arrays are never nullable", raises=AssertionError, ), + pytest.mark.notyet( + ["polars"], + reason="flattened empty arrays incorrectly insert a null", + raises=AssertionError, + ), ], ), ], @@ -1557,3 +1563,19 @@ def test_array_agg_bool(con, data, agg, baseline_func): result = [x if pd.notna(x) else None for x in result] expected = [baseline_func(x) for x in df.x] assert result == expected + + +@pytest.mark.notyet( + ["postgres"], + raises=PsycoPg2ArraySubscriptError, + reason="all dimensions must match in size", +) +@pytest.mark.notimpl(["risingwave", "flink"], raises=com.OperationNotDefinedError) +def test_flatten(con): + t = ibis.memtable( + [{"arr": [[1, 5, 7], [3, 4]]}], schema={"arr": "array>"} + ) + expr = t.arr.flatten().name("result") + result = con.execute(expr) + expected = pd.Series([[1, 5, 7, 3, 4]], name="result") + tm.assert_series_equal(result, expected) From ea6ef626b3c3fab322f31b7b4bcbf490bb3c0ab2 Mon Sep 17 00:00:00 2001 From: Phillip Cloud <417981+cpcloud@users.noreply.github.com> Date: Mon, 23 Sep 2024 11:00:26 -0400 Subject: [PATCH 07/10] test(bigquery): xfail nested flatten test (#10194) Fix failing test on BigQuery CI. --- ibis/backends/tests/test_array.py | 3 +++ 1 file changed, 3 insertions(+) diff --git a/ibis/backends/tests/test_array.py b/ibis/backends/tests/test_array.py index cba8ae64d1aa..123aa60409a3 100644 --- a/ibis/backends/tests/test_array.py +++ b/ibis/backends/tests/test_array.py @@ -1571,6 +1571,9 @@ def test_array_agg_bool(con, data, agg, baseline_func): reason="all dimensions must match in size", ) @pytest.mark.notimpl(["risingwave", "flink"], raises=com.OperationNotDefinedError) +@pytest.mark.notyet( + ["bigquery"], raises=TypeError, reason="nested arrays aren't supported" +) def test_flatten(con): t = ibis.memtable( [{"arr": [[1, 5, 7], [3, 4]]}], schema={"arr": "array>"} From 10b38ee7e87ceb045955ceb196bf4f0e508a91d5 Mon Sep 17 00:00:00 2001 From: Phillip Cloud <417981+cpcloud@users.noreply.github.com> Date: Mon, 23 Sep 2024 11:44:06 -0400 Subject: [PATCH 08/10] fix(api): use `to_pyarrow()` instead of `execute()` when pretty printing scalars --- ibis/backends/tests/test_interactive.py | 19 +++++++++++++++++++ ibis/expr/types/pretty.py | 2 +- 2 files changed, 20 insertions(+), 1 deletion(-) diff --git a/ibis/backends/tests/test_interactive.py b/ibis/backends/tests/test_interactive.py index 05b40a3f72fd..74219cf0a5af 100644 --- a/ibis/backends/tests/test_interactive.py +++ b/ibis/backends/tests/test_interactive.py @@ -105,3 +105,22 @@ def test_no_recursion_error(con, monkeypatch): exc.RelationError, match="The scalar expression cannot be converted" ): repr(expr) + + +@pytest.mark.notimpl( + ["impala", "flink", "pyspark"], + reason="backend calls `execute` as part of pyarrow conversion", +) +def test_scalar_uses_pyarrow(con, table, monkeypatch, mocker): + monkeypatch.setattr(ibis.options, "interactive", True) + + execute_spy = mocker.spy(con, "execute") + to_pyarrow_spy = mocker.spy(con, "to_pyarrow") + + repr(table.limit(1).string_col) + + # pyarrow does get called + to_pyarrow_spy.assert_called_once() + + # execute doesn't get called + execute_spy.assert_not_called() diff --git a/ibis/expr/types/pretty.py b/ibis/expr/types/pretty.py index acb378cb843f..88f12170d6ea 100644 --- a/ibis/expr/types/pretty.py +++ b/ibis/expr/types/pretty.py @@ -291,7 +291,7 @@ def _to_rich_scalar( ) -> Pretty: value = format_values( expr.type(), - [expr.execute()], + [expr.to_pyarrow().as_py()], max_length=max_length or ibis.options.repr.interactive.max_length, max_string=max_string or ibis.options.repr.interactive.max_string, max_depth=max_depth or ibis.options.repr.interactive.max_depth, From 8166717f7d586979b130c005f455ce0bd3c99495 Mon Sep 17 00:00:00 2001 From: Gil Forsyth Date: Mon, 23 Sep 2024 11:54:46 -0400 Subject: [PATCH 09/10] refactor(joins): require explicit abstract table as RHS of joins (#9661) ## Description of changes We have (had) limited support for passing in in-memory objects as the RHS of a join, where we would create a memtable for the user and then use that. For backends where memtable creation is expensive, or for queries where there may be multiple calls to the same in-memory data, it is better to be explicit and first register the in-memory data with the backend using either `memtable` or `create_table`. BREAKING CHANGE: Passing a `pyarrow.Table` or a `pandas.DataFrame` as the right-hand-side of a join is no longer supported. To join against in-memory data, you can pass the in-memory object to `ibis.memtable` or `con.create_table` and use the resulting table object instead. ## Issues closed * Resolves #9571 --- ibis/backends/tests/test_join.py | 7 +++++-- ibis/expr/types/joins.py | 29 +++++------------------------ ibis/tests/expr/test_table.py | 2 +- 3 files changed, 11 insertions(+), 27 deletions(-) diff --git a/ibis/backends/tests/test_join.py b/ibis/backends/tests/test_join.py index 88fdd5489384..b81b1878a882 100644 --- a/ibis/backends/tests/test_join.py +++ b/ibis/backends/tests/test_join.py @@ -188,7 +188,8 @@ def test_join_with_pandas(batting, awards_players): batting_filt = batting.filter(lambda t: t.yearID < 1900) awards_players_filt = awards_players.filter(lambda t: t.yearID < 1900).execute() assert isinstance(awards_players_filt, pd.DataFrame) - expr = batting_filt.join(awards_players_filt, "yearID") + t = ibis.memtable(awards_players_filt) + expr = batting_filt.join(t, "yearID") df = expr.execute() assert df.yearID.nunique() == 7 @@ -206,7 +207,9 @@ def test_join_with_pandas_non_null_typed_columns(batting, awards_players): assert sch.infer(awards_players_filt) == sch.Schema(dict(yearID="int")) assert isinstance(awards_players_filt, pd.DataFrame) - expr = batting_filt.join(awards_players_filt, "yearID") + + t = ibis.memtable(awards_players_filt) + expr = batting_filt.join(t, "yearID") df = expr.execute() assert df.yearID.nunique() == 7 diff --git a/ibis/expr/types/joins.py b/ibis/expr/types/joins.py index 2987848802af..c0b9a7a8b817 100644 --- a/ibis/expr/types/joins.py +++ b/ibis/expr/types/joins.py @@ -5,7 +5,6 @@ from public import public -import ibis import ibis.expr.operations as ops from ibis import util from ibis.common.deferred import Deferred @@ -13,6 +12,7 @@ from ibis.common.exceptions import ( ExpressionError, IbisInputError, + IbisTypeError, InputTypeError, IntegrityError, ) @@ -31,28 +31,6 @@ from ibis.expr.operations.relations import JoinKind -def coerce_to_table(data): - try: - import pandas as pd - except ImportError: - pass - else: - if isinstance(data, pd.DataFrame): - return ibis.memtable(data) - - try: - import pyarrow as pa - except ImportError: - pass - else: - if isinstance(data, pa.Table): - return ibis.memtable(data) - - if not isinstance(data, Table): - raise TypeError(f"right operand must be a Table, got {type(data).__name__}") - return data - - def disambiguate_fields( how, predicates, @@ -254,7 +232,10 @@ def join( lname: str = "", rname: str = "{name}_right", ): - right = coerce_to_table(right) + if not isinstance(right, Table): + raise IbisTypeError( + f"Right side of join must be an Ibis table, got {type(right)}." + ) if how == "left_semi": how = "semi" diff --git a/ibis/tests/expr/test_table.py b/ibis/tests/expr/test_table.py index 2b117cbb93ab..b0ebe5b76355 100644 --- a/ibis/tests/expr/test_table.py +++ b/ibis/tests/expr/test_table.py @@ -1315,7 +1315,7 @@ def test_join_invalid_expr_type(con): invalid_right = left.foo_id join_key = ["bar_id"] - with pytest.raises(TypeError): + with pytest.raises(com.IbisTypeError): left.inner_join(invalid_right, join_key) From abb55939c284fee6478b75100922074907c0e706 Mon Sep 17 00:00:00 2001 From: Phillip Cloud <417981+cpcloud@users.noreply.github.com> Date: Mon, 23 Sep 2024 17:02:00 -0400 Subject: [PATCH 10/10] feat(pyspark): add official support and ci testing with spark connect (#10187) ## Description of changes This PR adds testing for using the pyspark Ibis backend with spark-connect. The way this is done is running a Spark connect instance as a docker compose service, similar to our other client-server model backends. The primary bit of functionality that isn't tested is UDFs (which means JSON unwrapping is also not tested, because that's implemented as a UDF). These effectively require a clone of the Python environment on the server, and that seems out of scope for initial support of spark connect. --- .env | 1 + .github/workflows/ibis-backends.yml | 33 +- compose.yaml | 20 + docker/spark-connect/conf.properties | 12 + docker/spark-connect/log4j2.properties | 68 +++ ibis/backends/pyspark/__init__.py | 38 +- ibis/backends/pyspark/tests/conftest.py | 483 +++++++++++---------- ibis/backends/pyspark/tests/test_array.py | 174 -------- ibis/backends/pyspark/tests/test_basic.py | 20 +- ibis/backends/pyspark/tests/test_ddl.py | 3 +- ibis/backends/pyspark/tests/test_udf.py | 5 + ibis/backends/tests/errors.py | 15 +- ibis/backends/tests/test_aggregation.py | 31 +- ibis/backends/tests/test_array.py | 39 +- ibis/backends/tests/test_client.py | 7 + ibis/backends/tests/test_export.py | 3 + ibis/backends/tests/test_expr_caching.py | 2 + ibis/backends/tests/test_json.py | 8 + ibis/backends/tests/test_register.py | 9 +- ibis/backends/tests/test_temporal.py | 19 +- ibis/backends/tests/test_udf.py | 9 +- ibis/backends/tests/test_vectorized_udf.py | 15 +- ibis/backends/tests/test_window.py | 26 ++ ibis/conftest.py | 2 + pyproject.toml | 1 + 25 files changed, 581 insertions(+), 462 deletions(-) create mode 100644 docker/spark-connect/conf.properties create mode 100644 docker/spark-connect/log4j2.properties delete mode 100644 ibis/backends/pyspark/tests/test_array.py diff --git a/.env b/.env index 05c6411ce4e2..70f4abe8c082 100644 --- a/.env +++ b/.env @@ -5,3 +5,4 @@ PGPASSWORD="postgres" MYSQL_PWD="ibis" MSSQL_SA_PASSWORD="1bis_Testing!" DRUID_URL="druid://localhost:8082/druid/v2/sql" +SPARK_CONFIG=./docker/spark-connect/conf.properties diff --git a/.github/workflows/ibis-backends.yml b/.github/workflows/ibis-backends.yml index d44fbd0b0c48..70ffa1ab4af4 100644 --- a/.github/workflows/ibis-backends.yml +++ b/.github/workflows/ibis-backends.yml @@ -442,13 +442,9 @@ jobs: - name: download backend data run: just download-data - - name: show docker compose version - if: matrix.backend.services != null - run: docker compose version - - name: start services if: matrix.backend.services != null - run: docker compose up --wait ${{ join(matrix.backend.services, ' ') }} + run: just up ${{ join(matrix.backend.services, ' ') }} - name: install python uses: actions/setup-python@v5 @@ -600,7 +596,7 @@ jobs: - name: start services if: matrix.backend.services != null - run: docker compose up --wait ${{ join(matrix.backend.services, ' ') }} + run: just up ${{ join(matrix.backend.services, ' ') }} - name: install python uses: actions/setup-python@v5 @@ -653,7 +649,7 @@ jobs: run: docker compose logs test_pyspark: - name: PySpark ${{ matrix.pyspark-minor-version }} ubuntu-latest python-${{ matrix.python-version }} + name: PySpark ${{ matrix.tag }} ${{ matrix.pyspark-minor-version }} ubuntu-latest python-${{ matrix.python-version }} runs-on: ubuntu-latest strategy: fail-fast: false @@ -665,12 +661,14 @@ jobs: deps: - "'pandas@<2'" - "'numpy@<1.24'" + tag: local - python-version: "3.11" pyspark-version: "3.5.2" pyspark-minor-version: "3.5" deps: - "'pandas@>2'" - "'numpy@>1.24'" + tag: local - python-version: "3.12" pyspark-version: "3.5.2" pyspark-minor-version: "3.5" @@ -678,6 +676,14 @@ jobs: - "'pandas@>2'" - "'numpy@>1.24'" - setuptools + tag: local + - python-version: "3.12" + pyspark-version: "3.5.2" + pyspark-minor-version: "3.5" + deps: + - setuptools + tag: remote + SPARK_REMOTE: "sc://localhost:15002" steps: - name: checkout uses: actions/checkout@v4 @@ -691,6 +697,10 @@ jobs: env: GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} + - name: start services + if: matrix.tag == 'remote' + run: just up spark-connect + - name: download backend data run: just download-data @@ -730,7 +740,14 @@ jobs: shell: bash run: just download-iceberg-jar ${{ matrix.pyspark-minor-version }} - - name: run tests + - name: run spark connect tests + if: matrix.tag == 'remote' + run: just ci-check -m pyspark + env: + SPARK_REMOTE: ${{ matrix.SPARK_REMOTE }} + + - name: run spark tests + if: matrix.tag == 'local' run: just ci-check -m pyspark - name: check that no untracked files were produced diff --git a/compose.yaml b/compose.yaml index d82c4bc5be33..91107904d7c4 100644 --- a/compose.yaml +++ b/compose.yaml @@ -589,6 +589,24 @@ services: networks: - risingwave + spark-connect: + image: bitnami/spark:3.5.2 + ports: + - 15002:15002 + command: /opt/bitnami/spark/sbin/start-connect-server.sh --name ibis_testing --packages org.apache.spark:spark-connect_2.12:3.5.2,org.apache.iceberg:iceberg-spark-runtime-3.5_2.12:1.5.2 + healthcheck: + test: + - CMD-SHELL + - bash -c 'printf \"GET / HTTP/1.1\n\n\" > /dev/tcp/127.0.0.1/15002; exit $$?;' + interval: 5s + retries: 6 + volumes: + - spark-connect:/data + - $PWD/docker/spark-connect/conf.properties:/opt/bitnami/spark/conf/spark-defaults.conf:ro + # - $PWD/docker/spark-connect/log4j2.properties:/opt/bitnami/spark/conf/log4j2.properties:ro + networks: + - spark-connect + networks: impala: # docker defaults to naming networks "$PROJECT_$NETWORK" but the Java Hive @@ -606,6 +624,7 @@ networks: exasol: flink: risingwave: + spark-connect: volumes: clickhouse: @@ -617,3 +636,4 @@ volumes: exasol: impala: risingwave: + spark-connect: diff --git a/docker/spark-connect/conf.properties b/docker/spark-connect/conf.properties new file mode 100644 index 000000000000..ffc1a253def2 --- /dev/null +++ b/docker/spark-connect/conf.properties @@ -0,0 +1,12 @@ +spark.driver.extraJavaOptions=-Duser.timezone=GMT +spark.executor.extraJavaOptions=-Duser.timezone=GMT +spark.jars.packages=org.apache.iceberg:iceberg-spark-runtime-3.5_2.12:1.5.2 +spark.sql.catalog.local.type=hadoop +spark.sql.catalog.local.warehouse=warehouse +spark.sql.catalog.local=org.apache.iceberg.spark.SparkCatalog +spark.sql.extensions=org.apache.iceberg.spark.extensions.IcebergSparkSessionExtensions +spark.sql.legacy.timeParserPolicy=LEGACY +spark.sql.session.timeZone=UTC +spark.sql.streaming.schemaInference=true +spark.ui.enabled=false +spark.ui.showConsoleProgress=false diff --git a/docker/spark-connect/log4j2.properties b/docker/spark-connect/log4j2.properties new file mode 100644 index 000000000000..deab3438d4dd --- /dev/null +++ b/docker/spark-connect/log4j2.properties @@ -0,0 +1,68 @@ +# Licensed to the Apache Software Foundation (ASF) under one or more +# contributor license agreements. See the NOTICE file distributed with +# this work for additional information regarding copyright ownership. +# The ASF licenses this file to You under the Apache License, Version 2.0 +# (the "License"); you may not use this file except in compliance with +# the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# + +# Set everything to be logged to the console +rootLogger.level = error +rootLogger.appenderRef.stdout.ref = console + +# In the pattern layout configuration below, we specify an explicit `%ex` conversion +# pattern for logging Throwables. If this was omitted, then (by default) Log4J would +# implicitly add an `%xEx` conversion pattern which logs stacktraces with additional +# class packaging information. That extra information can sometimes add a substantial +# performance overhead, so we disable it in our default logging config. +# For more information, see SPARK-39361. +appender.console.type = Console +appender.console.name = console +appender.console.target = SYSTEM_ERR +appender.console.layout.type = PatternLayout +appender.console.layout.pattern = %d{yy/MM/dd HH:mm:ss} %p %c{1}: %m%n%ex + +# Set the default spark-shell/spark-sql log level to WARN. When running the +# spark-shell/spark-sql, the log level for these classes is used to overwrite +# the root logger's log level, so that the user can have different defaults +# for the shell and regular Spark apps. +logger.repl.name = org.apache.spark.repl.Main +logger.repl.level = error + +logger.thriftserver.name = org.apache.spark.sql.hive.thriftserver.SparkSQLCLIDriver +logger.thriftserver.level = error + +# Settings to quiet third party logs that are too verbose +logger.jetty1.name = org.sparkproject.jetty +logger.jetty1.level = error +logger.jetty2.name = org.sparkproject.jetty.util.component.AbstractLifeCycle +logger.jetty2.level = error +logger.replexprTyper.name = org.apache.spark.repl.SparkIMain$exprTyper +logger.replexprTyper.level = error +logger.replSparkILoopInterpreter.name = org.apache.spark.repl.SparkILoop$SparkILoopInterpreter +logger.replSparkILoopInterpreter.level = error +logger.parquet1.name = org.apache.parquet +logger.parquet1.level = error +logger.parquet2.name = parquet +logger.parquet2.level = error + +# SPARK-9183: Settings to avoid annoying messages when looking up nonexistent UDFs in SparkSQL with Hive support +logger.RetryingHMSHandler.name = org.apache.hadoop.hive.metastore.RetryingHMSHandler +logger.RetryingHMSHandler.level = fatal +logger.FunctionRegistry.name = org.apache.hadoop.hive.ql.exec.FunctionRegistry +logger.FunctionRegistry.level = error + +# For deploying Spark ThriftServer +# SPARK-34128: Suppress undesirable TTransportException warnings involved in THRIFT-4805 +appender.console.filter.1.type = RegexFilter +appender.console.filter.1.regex = .*Thrift error occurred during processing of message.* +appender.console.filter.1.onMatch = deny +appender.console.filter.1.onMismatch = neutral diff --git a/ibis/backends/pyspark/__init__.py b/ibis/backends/pyspark/__init__.py index babbd124eeb6..c5f921381139 100644 --- a/ibis/backends/pyspark/__init__.py +++ b/ibis/backends/pyspark/__init__.py @@ -31,9 +31,15 @@ from ibis.util import deprecated try: - from pyspark.errors import AnalysisException, ParseException + from pyspark.errors import ParseException + from pyspark.errors.exceptions.connect import SparkConnectGrpcException except ImportError: - from pyspark.sql.utils import AnalysisException, ParseException + from pyspark.sql.utils import ParseException + + # Use a dummy class for when spark connect is not available + class SparkConnectGrpcException(Exception): + pass + if TYPE_CHECKING: from collections.abc import Mapping, Sequence @@ -186,13 +192,6 @@ def do_connect( # Databricks Serverless compute only supports limited properties # and any attempt to set unsupported properties will result in an error. # https://docs.databricks.com/en/spark/conf.html - try: - from pyspark.errors.exceptions.connect import SparkConnectGrpcException - except ImportError: - # Use a dummy class for when spark connect is not available - class SparkConnectGrpcException(Exception): - pass - with contextlib.suppress(SparkConnectGrpcException): self._session.conf.set("spark.sql.mapKeyDedupPolicy", "LAST_WIN") @@ -456,7 +455,9 @@ def _register_in_memory_table(self, op: ops.InMemoryTable) -> None: df.createTempView(op.name) def _finalize_memtable(self, name: str) -> None: - self._session.catalog.dropTempView(name) + """No-op, otherwise a deadlock can occur when using Spark Connect.""" + if isinstance(session := self._session, pyspark.sql.SparkSession): + session.catalog.dropTempView(name) @contextlib.contextmanager def _safe_raw_sql(self, query: str) -> Any: @@ -579,16 +580,20 @@ def get_schema( table_loc = self._to_sqlglot_table((catalog, database)) catalog, db = self._to_catalog_db_tuple(table_loc) + session = self._session with self._active_catalog_database(catalog, db): try: - df = self._session.table(table_name) - except AnalysisException as e: - if not self._session.catalog.tableExists(table_name): + df = session.table(table_name) + # this is intentionally included in the try block because when + # using spark connect, the table-not-found exception coming + # from the server will *NOT* be raised until the schema + # property is accessed + struct = PySparkType.to_ibis(df.schema) + except Exception as e: + if not session.catalog.tableExists(table_name): raise com.TableNotFound(table_name) from e raise - struct = PySparkType.to_ibis(df.schema) - return sch.Schema(struct) def create_table( @@ -752,7 +757,7 @@ def _create_cached_table(self, name, expr): query = self.compile(expr) t = self._session.sql(query).cache() assert t.is_cached - t.createOrReplaceTempView(name) + t.createTempView(name) # store the underlying spark dataframe so we can release memory when # asked to, instead of when the session ends self._cached_dataframes[name] = t @@ -761,7 +766,6 @@ def _create_cached_table(self, name, expr): def _drop_cached_table(self, name): self._session.catalog.dropTempView(name) t = self._cached_dataframes.pop(name) - assert t.is_cached t.unpersist() assert not t.is_cached diff --git a/ibis/backends/pyspark/tests/conftest.py b/ibis/backends/pyspark/tests/conftest.py index 3d53e52e24be..65df70f78142 100644 --- a/ibis/backends/pyspark/tests/conftest.py +++ b/ibis/backends/pyspark/tests/conftest.py @@ -1,7 +1,9 @@ from __future__ import annotations +import abc import os from datetime import datetime, timedelta, timezone +from pathlib import Path from typing import TYPE_CHECKING, Any from unittest import mock @@ -9,28 +11,33 @@ import pandas as pd import pytest from filelock import FileLock -from packaging.version import parse as vparse import ibis from ibis import util from ibis.backends.conftest import TEST_TABLES from ibis.backends.pyspark import Backend from ibis.backends.pyspark.datatypes import PySparkSchema -from ibis.backends.tests.base import BackendTest +from ibis.backends.tests.base import BackendTest, ServiceBackendTest from ibis.backends.tests.data import json_types, topk, win +from ibis.conftest import IS_SPARK_REMOTE, SPARK_REMOTE if TYPE_CHECKING: - from pathlib import Path + from collections.abc import Iterable def set_pyspark_database(con, database): con._session.catalog.setCurrentDatabase(database) -class TestConf(BackendTest): - deps = ("pyspark",) +class BaseSparkTestConf(abc.ABC): + @property + @abc.abstractmethod + def parquet_dir(self) -> str: + """Directory containing Parquet files.""" def _load_data(self, **_: Any) -> None: + import pyspark.sql.functions as F + import pyspark.sql.types as pt from pyspark.sql import Row s = self.connection._session @@ -39,7 +46,7 @@ def _load_data(self, **_: Any) -> None: sort_cols = {"functional_alltypes": "id"} for name in TEST_TABLES: - path = str(self.data_dir / "parquet" / f"{name}.parquet") + path = os.path.join(self.parquet_dir, f"{name}.parquet") t = s.read.parquet(path).repartition(num_partitions) if (sort_col := sort_cols.get(name)) is not None: t = t.sort(sort_col) @@ -138,246 +145,264 @@ def _load_data(self, **_: Any) -> None: s.createDataFrame(win).createOrReplaceTempView("win") s.createDataFrame(topk.to_pandas()).createOrReplaceTempView("topk") - @staticmethod - def connect(*, tmpdir, worker_id, **kw): - # Spark internally stores timestamps as UTC values, and timestamp - # data that is brought in without a specified time zone is - # converted as local time to UTC with microsecond resolution. - # https://spark.apache.org/docs/latest/sql-pyspark-pandas-with-arrow.html#timestamp-with-time-zone-semantics - - import pyspark - from pyspark.sql import SparkSession - - pyspark_version = vparse(pyspark.__version__) - pyspark_minor_version = f"{pyspark_version.major:d}.{pyspark_version.minor:d}" - - config = ( - SparkSession.builder.appName("ibis_testing") - .master("local[1]") - .config("spark.cores.max", 1) - .config("spark.default.parallelism", 1) - .config("spark.driver.extraJavaOptions", "-Duser.timezone=GMT") - .config("spark.dynamicAllocation.enabled", False) - .config("spark.executor.extraJavaOptions", "-Duser.timezone=GMT") - .config("spark.executor.heartbeatInterval", "3600s") - .config("spark.executor.instances", 1) - .config("spark.network.timeout", "4200s") - .config("spark.rdd.compress", False) - .config("spark.serializer", "org.apache.spark.serializer.KryoSerializer") - .config("spark.shuffle.compress", False) - .config("spark.shuffle.spill.compress", False) - .config("spark.sql.legacy.timeParserPolicy", "LEGACY") - .config("spark.sql.session.timeZone", "UTC") - .config("spark.sql.shuffle.partitions", 1) - .config("spark.storage.blockManagerSlaveTimeoutMs", "4200s") - .config("spark.ui.enabled", False) - .config("spark.ui.showConsoleProgress", False) - .config("spark.sql.execution.arrow.pyspark.enabled", False) - .config("spark.sql.streaming.schemaInference", True) - .config( - "spark.sql.extensions", - "org.apache.iceberg.spark.extensions.IcebergSparkSessionExtensions", - ) - .config( - "spark.jars.packages", - f"org.apache.iceberg:iceberg-spark-runtime-{pyspark_minor_version}_2.12:1.5.2", - ) - .config("spark.sql.catalog.local", "org.apache.iceberg.spark.SparkCatalog") - .config("spark.sql.catalog.local.type", "hadoop") - .config("spark.sql.catalog.local.warehouse", "icehouse") + s.range(0, 10).withColumn("str_col", F.lit("value")).createTempView( + "basic_table" ) - try: - from delta.pip_utils import configure_spark_with_delta_pip - except ImportError: - configure_spark_with_delta_pip = lambda cfg: cfg - else: - config = config.config( - "spark.sql.catalog.spark_catalog", - "org.apache.spark.sql.delta.catalog.DeltaCatalog", - ).config("spark.sql.extensions", "io.delta.sql.DeltaSparkSessionExtension") + df_nulls = s.createDataFrame( + [ + ["k1", np.nan, "Alfred", None], + ["k1", 3.0, None, "joker"], + ["k2", 27.0, "Batman", "batmobile"], + ["k2", None, "Catwoman", "motorcycle"], + ], + ["key", "age", "user", "toy"], + ) + df_nulls.createTempView("null_table") - spark = configure_spark_with_delta_pip(config).getOrCreate() - return ibis.pyspark.connect(spark, **kw) + df_dates = s.createDataFrame( + [["2018-01-02"], ["2018-01-03"], ["2018-01-04"]], ["date_str"] + ) + df_dates.createTempView("date_table") + df_time_indexed = s.createDataFrame( + [ + [datetime(2017, 1, 2, 5, tzinfo=timezone.utc), 1, 1.0], + [datetime(2017, 1, 2, 5, tzinfo=timezone.utc), 2, 2.0], + [datetime(2017, 1, 2, 6, tzinfo=timezone.utc), 1, 3.0], + [datetime(2017, 1, 2, 6, tzinfo=timezone.utc), 2, 4.0], + [datetime(2017, 1, 2, 7, tzinfo=timezone.utc), 1, 5.0], + [datetime(2017, 1, 2, 7, tzinfo=timezone.utc), 2, 6.0], + [datetime(2017, 1, 4, 8, tzinfo=timezone.utc), 1, 7.0], + [datetime(2017, 1, 4, 8, tzinfo=timezone.utc), 2, 8.0], + ], + ["time", "key", "value"], + ) -class TestConfForStreaming(BackendTest): - deps = ("pyspark",) + df_time_indexed.createTempView("time_indexed_table") + + if not IS_SPARK_REMOTE: + # TODO(cpcloud): understand why this doesn't work with spark connect + df_interval = s.createDataFrame( + [ + [ + timedelta(days=10), + timedelta(hours=10), + timedelta(minutes=10), + timedelta(seconds=10), + ] + ], + pt.StructType( + [ + pt.StructField( + "interval_day", + pt.DayTimeIntervalType( + pt.DayTimeIntervalType.DAY, pt.DayTimeIntervalType.DAY + ), + ), + pt.StructField( + "interval_hour", + pt.DayTimeIntervalType( + pt.DayTimeIntervalType.HOUR, pt.DayTimeIntervalType.HOUR + ), + ), + pt.StructField( + "interval_minute", + pt.DayTimeIntervalType( + pt.DayTimeIntervalType.MINUTE, + pt.DayTimeIntervalType.MINUTE, + ), + ), + pt.StructField( + "interval_second", + pt.DayTimeIntervalType( + pt.DayTimeIntervalType.SECOND, + pt.DayTimeIntervalType.SECOND, + ), + ), + ] + ), + ) - def _load_data(self, **_: Any) -> None: - s = self.connection._session - num_partitions = 4 + df_interval.createTempView("interval_table") + + +if IS_SPARK_REMOTE: + + class TestConf(BaseSparkTestConf, ServiceBackendTest): + deps = ("pyspark",) + data_volume = "/data" + service_name = "spark-connect" - watermark_cols = {"functional_alltypes": "timestamp_col"} + @property + def parquet_dir(self) -> str: + return self.data_volume - for name, schema in TEST_TABLES.items(): - path = str(self.data_dir / "directory" / "parquet" / name) - t = ( - s.readStream.schema(PySparkSchema.from_ibis(schema)) - .parquet(path) - .repartition(num_partitions) + @property + def test_files(self) -> Iterable[Path]: + return self.data_dir.joinpath("parquet").glob("*.parquet") + + @staticmethod + def connect(*, tmpdir, worker_id, **kw): + from pyspark.sql import SparkSession + + spark = ( + SparkSession.builder.appName("ibis_testing") + .remote(SPARK_REMOTE) + .getOrCreate() ) - if (watermark_col := watermark_cols.get(name)) is not None: - t = t.withWatermark(watermark_col, "10 seconds") - t.createOrReplaceTempView(name) + return ibis.pyspark.connect(spark, **kw) - @classmethod - def load_data( - cls, data_dir: Path, tmpdir: Path, worker_id: str, **kw: Any - ) -> BackendTest: - """Load testdata from `data_dir`.""" - # handling for multi-processes pytest + @pytest.fixture(scope="session") + def con_streaming(data_dir, tmp_path_factory, worker_id): + pytest.skip("Streaming tests are not supported in remote mode") - # get the temp directory shared by all workers - root_tmp_dir = tmpdir.getbasetemp() / "streaming" - if worker_id != "master": - root_tmp_dir = root_tmp_dir.parent + def write_to_memory(self, expr, table_name): + assert self.mode == "batch" + raise NotImplementedError +else: - fn = root_tmp_dir / cls.name() - with FileLock(f"{fn}.lock"): - cls.skip_if_missing_deps() + class TestConf(BaseSparkTestConf, BackendTest): + deps = ("pyspark",) - inst = cls(data_dir=data_dir, tmpdir=tmpdir, worker_id=worker_id, **kw) + @property + def parquet_dir(self) -> str: + return str(self.data_dir / "parquet") - if inst.stateful: - inst.stateful_load(fn, **kw) - else: - inst.stateless_load(**kw) - inst.postload(tmpdir=tmpdir, worker_id=worker_id, **kw) - return inst + @staticmethod + def connect(*, tmpdir, worker_id, **kw): + from pyspark.sql import SparkSession - @staticmethod - def connect(*, tmpdir, worker_id, **kw): - from pyspark.sql import SparkSession + config = SparkSession.builder.appName("ibis_testing") - # SparkContext is shared globally; only one SparkContext should be active - # per JVM. We need to create a new SparkSession for streaming tests but - # this session shares the same SparkContext. - spark = SparkSession.getActiveSession().newSession() - con = ibis.pyspark.connect(spark, mode="streaming", **kw) - return con + # load from properties file, yuck + with Path( + os.environ.get( + "SPARK_CONFIG", + Path(ibis.__file__) + .parents[1] + .joinpath("docker", "spark-connect", "conf.properties"), + ) + ).open(mode="r") as config_file: + for line in config_file: + config = config.config(*map(str.strip, line.strip().split("=", 1))) + + config = ( + config.config("spark.cores.max", "1") + .config("spark.default.parallelism", "1") + .config("spark.dynamicAllocation.enabled", "false") + .config("spark.executor.heartbeatInterval", "3600s") + .config("spark.executor.instances", "1") + .config("spark.network.timeout", "4200s") + .config("spark.rdd.compress", "false") + .config( + "spark.serializer", "org.apache.spark.serializer.KryoSerializer" + ) + .config("spark.shuffle.compress", "false") + .config("spark.shuffle.spill.compress", "false") + .config("spark.sql.execution.arrow.pyspark.enabled", "false") + .config("spark.sql.shuffle.partitions", "1") + .config("spark.storage.blockManagerSlaveTimeoutMs", "4200s") + ) + try: + from delta.pip_utils import configure_spark_with_delta_pip + except ImportError: + configure_spark_with_delta_pip = lambda cfg: cfg + else: + config = config.config( + "spark.sql.catalog.spark_catalog", + "org.apache.spark.sql.delta.catalog.DeltaCatalog", + ).config( + "spark.sql.extensions", + "io.delta.sql.DeltaSparkSessionExtension", + ) -@pytest.fixture(scope="session") -def con(data_dir, tmp_path_factory, worker_id): - import pyspark.sql.functions as F - import pyspark.sql.types as pt + spark = configure_spark_with_delta_pip(config).getOrCreate() + return ibis.pyspark.connect(spark, **kw) - backend_test = TestConf.load_data(data_dir, tmp_path_factory, worker_id) - con = backend_test.connection + class TestConfForStreaming(BackendTest): + deps = ("pyspark",) - df = con._session.range(0, 10) - df = df.withColumn("str_col", F.lit("value")) - df.createTempView("basic_table") - - df_nulls = con._session.createDataFrame( - [ - ["k1", np.nan, "Alfred", None], - ["k1", 3.0, None, "joker"], - ["k2", 27.0, "Batman", "batmobile"], - ["k2", None, "Catwoman", "motorcycle"], - ], - ["key", "age", "user", "toy"], - ) - df_nulls.createTempView("null_table") - - df_dates = con._session.createDataFrame( - [["2018-01-02"], ["2018-01-03"], ["2018-01-04"]], ["date_str"] - ) - df_dates.createTempView("date_table") - - df_arrays = con._session.createDataFrame( - [ - ["k1", [1, 2, 3], ["a"]], - ["k2", [4, 5], ["test1", "test2", "test3"]], - ["k3", [6], ["w", "x", "y", "z"]], - ["k1", [], ["cat", "dog"]], - ["k1", [7, 8], []], - ], - ["key", "array_int", "array_str"], - ) - df_arrays.createTempView("array_table") - - df_time_indexed = con._session.createDataFrame( - [ - [datetime(2017, 1, 2, 5, tzinfo=timezone.utc), 1, 1.0], - [datetime(2017, 1, 2, 5, tzinfo=timezone.utc), 2, 2.0], - [datetime(2017, 1, 2, 6, tzinfo=timezone.utc), 1, 3.0], - [datetime(2017, 1, 2, 6, tzinfo=timezone.utc), 2, 4.0], - [datetime(2017, 1, 2, 7, tzinfo=timezone.utc), 1, 5.0], - [datetime(2017, 1, 2, 7, tzinfo=timezone.utc), 2, 6.0], - [datetime(2017, 1, 4, 8, tzinfo=timezone.utc), 1, 7.0], - [datetime(2017, 1, 4, 8, tzinfo=timezone.utc), 2, 8.0], - ], - ["time", "key", "value"], - ) - - df_time_indexed.createTempView("time_indexed_table") - - df_interval = con._session.createDataFrame( - [ - [ - timedelta(days=10), - timedelta(hours=10), - timedelta(minutes=10), - timedelta(seconds=10), - ] - ], - pt.StructType( - [ - pt.StructField( - "interval_day", - pt.DayTimeIntervalType( - pt.DayTimeIntervalType.DAY, pt.DayTimeIntervalType.DAY - ), - ), - pt.StructField( - "interval_hour", - pt.DayTimeIntervalType( - pt.DayTimeIntervalType.HOUR, pt.DayTimeIntervalType.HOUR - ), - ), - pt.StructField( - "interval_minute", - pt.DayTimeIntervalType( - pt.DayTimeIntervalType.MINUTE, pt.DayTimeIntervalType.MINUTE - ), - ), - pt.StructField( - "interval_second", - pt.DayTimeIntervalType( - pt.DayTimeIntervalType.SECOND, pt.DayTimeIntervalType.SECOND - ), - ), - ] - ), - ) + def _load_data(self, **_: Any) -> None: + s = self.connection._session + num_partitions = 4 - df_interval.createTempView("interval_table") + watermark_cols = {"functional_alltypes": "timestamp_col"} - df_interval_invalid = con._session.createDataFrame( - [[timedelta(days=10, hours=10, minutes=10, seconds=10)]], - pt.StructType( - [ - pt.StructField( - "interval_day_hour", - pt.DayTimeIntervalType( - pt.DayTimeIntervalType.DAY, pt.DayTimeIntervalType.HOUR - ), + for name, schema in TEST_TABLES.items(): + path = str(self.data_dir / "directory" / "parquet" / name) + t = ( + s.readStream.schema(PySparkSchema.from_ibis(schema)) + .parquet(path) + .repartition(num_partitions) ) - ] - ), - ) + if (watermark_col := watermark_cols.get(name)) is not None: + t = t.withWatermark(watermark_col, "10 seconds") + t.createOrReplaceTempView(name) + + @classmethod + def load_data( + cls, data_dir: Path, tmpdir: Path, worker_id: str, **kw: Any + ) -> BackendTest: + """Load testdata from `data_dir`.""" + # handling for multi-processes pytest + + # get the temp directory shared by all workers + root_tmp_dir = tmpdir.getbasetemp() / "streaming" + if worker_id != "master": + root_tmp_dir = root_tmp_dir.parent + + fn = root_tmp_dir / cls.name() + with FileLock(f"{fn}.lock"): + cls.skip_if_missing_deps() + + inst = cls(data_dir=data_dir, tmpdir=tmpdir, worker_id=worker_id, **kw) + + if inst.stateful: + inst.stateful_load(fn, **kw) + else: + inst.stateless_load(**kw) + inst.postload(tmpdir=tmpdir, worker_id=worker_id, **kw) + return inst + + @staticmethod + def connect(*, tmpdir, worker_id, **kw): + from pyspark.sql import SparkSession + + # SparkContext is shared globally; only one SparkContext should be active + # per JVM. We need to create a new SparkSession for streaming tests but + # this session shares the same SparkContext. + spark = SparkSession.getActiveSession().newSession() + con = ibis.pyspark.connect(spark, mode="streaming", **kw) + return con + + @pytest.fixture(scope="session") + def con_streaming(data_dir, tmp_path_factory, worker_id): + backend_test = TestConfForStreaming.load_data( + data_dir, tmp_path_factory, worker_id + ) + return backend_test.connection - df_interval_invalid.createTempView("invalid_interval_table") + @pytest.fixture(autouse=True, scope="function") + def stop_active_jobs(con_streaming): + yield + for sq in con_streaming._session.streams.active: + sq.stop() + sq.awaitTermination() - return con + def write_to_memory(self, expr, table_name): + assert self.mode == "streaming" + df = self._session.sql(expr.compile()) + df.writeStream.format("memory").queryName(table_name).start() @pytest.fixture(scope="session") -def con_streaming(data_dir, tmp_path_factory, worker_id): - backend_test = TestConfForStreaming.load_data(data_dir, tmp_path_factory, worker_id) - return backend_test.connection +def con(data_dir, tmp_path_factory, worker_id): + backend_test = TestConf.load_data(data_dir, tmp_path_factory, worker_id) + con = backend_test.connection + + return con class IbisWindow: @@ -388,12 +413,7 @@ def __init__(self, windows): def get_windows(self): # Return a list of Ibis windows return [ - ibis.window( - preceding=w[0], - following=w[1], - order_by="time", - group_by="key", - ) + ibis.window(preceding=w[0], following=w[1], order_by="time", group_by="key") for w in self.windows ] @@ -440,21 +460,6 @@ def default_session_fixture(): yield -def write_to_memory(self, expr, table_name): - if self.mode == "batch": - raise NotImplementedError - df = self._session.sql(expr.compile()) - df.writeStream.format("memory").queryName(table_name).start() - - -@pytest.fixture(autouse=True, scope="function") -def stop_active_jobs(con_streaming): - yield - for sq in con_streaming._session.streams.active: - sq.stop() - sq.awaitTermination() - - @pytest.fixture def awards_players_schema(): return TEST_TABLES["awards_players"] diff --git a/ibis/backends/pyspark/tests/test_array.py b/ibis/backends/pyspark/tests/test_array.py deleted file mode 100644 index b253b084cc70..000000000000 --- a/ibis/backends/pyspark/tests/test_array.py +++ /dev/null @@ -1,174 +0,0 @@ -from __future__ import annotations - -import numpy as np -import pandas as pd -import pandas.testing as tm -import pytest - -import ibis - -pytest.importorskip("pyspark") - - -@pytest.fixture -def t(con): - return con.table("array_table") - - -@pytest.fixture -def df(con): - return con._session.table("array_table").toPandas() - - -def test_array_length(t, df): - result = t.mutate(length=t.array_int.length()).execute() - expected = df.assign(length=df.array_int.map(lambda a: len(a))) - tm.assert_frame_equal(result, expected) - - -def test_array_length_scalar(con): - raw_value = [1, 2, 3] - value = ibis.literal(raw_value) - expr = value.length() - result = con.execute(expr) - expected = len(raw_value) - assert result == expected - - -@pytest.mark.parametrize( - ["start", "stop"], - [ - (1, 3), - (1, 1), - (2, 3), - (2, 5), - (None, 3), - (None, None), - (3, None), - (-3, None), - (None, -3), - (-3, -1), - ], -) -def test_array_slice(t, df, start, stop): - result = t.mutate(sliced=t.array_int[start:stop]).execute() - expected = df.assign(sliced=df.array_int.map(lambda a: a[start:stop])) - tm.assert_frame_equal(result, expected) - - -@pytest.mark.parametrize( - ["start", "stop"], - [ - (1, 3), - (1, 1), - (2, 3), - (2, 5), - (None, 3), - (None, None), - (3, None), - (-3, None), - (None, -3), - (-3, -1), - ], -) -def test_array_slice_scalar(con, start, stop): - raw_value = [-11, 42, 10] - value = ibis.literal(raw_value) - expr = value[start:stop] - result = con.execute(expr) - expected = raw_value[start:stop] - assert result == expected - - -@pytest.mark.parametrize("index", [1, 3, 4, 11, -11]) -def test_array_index(t, df, index): - expr = t.select(t.array_int[index].name("indexed")) - result = expr.execute() - - expected = pd.DataFrame( - { - "indexed": df.array_int.apply( - lambda x: x[index] if -len(x) <= index < len(x) else np.nan - ) - } - ) - tm.assert_frame_equal(result, expected) - - -@pytest.mark.parametrize("index", [1, 3, 4, 11]) -def test_array_index_scalar(con, index): - raw_value = [-10, 1, 2, 42] - value = ibis.literal(raw_value) - expr = value[index] - result = con.execute(expr) - expected = raw_value[index] if index < len(raw_value) else np.nan - assert result == expected or (np.isnan(result) and np.isnan(expected)) - - -@pytest.mark.parametrize("op", [lambda x, y: x + y, lambda x, y: y + x]) -def test_array_concat(t, df, op): - x = t.array_int.cast("array") - y = t.array_str - expr = op(x, y).name("array_result") - result = expr.execute() - - expected = op(df.array_int.apply(lambda x: list(map(str, x))), df.array_str).rename( - "array_result" - ) - tm.assert_series_equal(result, expected) - - -@pytest.mark.parametrize("op", [lambda x, y: x + y, lambda x, y: y + x]) -def test_array_concat_scalar(con, op): - raw_left = [1, 2, 3] - raw_right = [3, 4] - left = ibis.literal(raw_left) - right = ibis.literal(raw_right) - expr = op(left, right) - result = con.execute(expr) - assert result == op(raw_left, raw_right) - - -@pytest.mark.parametrize("n", [1, 3, 4, 7, -2]) # negative returns empty list -@pytest.mark.parametrize("mul", [lambda x, n: x * n, lambda x, n: n * x]) -def test_array_repeat(t, df, n, mul): - expr = t.select(mul(t.array_int, n).name("repeated")) - result = expr.execute() - - expected = pd.DataFrame({"repeated": df.array_int * n}) - tm.assert_frame_equal(result, expected) - - -@pytest.mark.parametrize("n", [1, 3, 4, 7, -2]) # negative returns empty list -@pytest.mark.parametrize("mul", [lambda x, n: x * n, lambda x, n: n * x]) -def test_array_repeat_scalar(con, n, mul): - raw_array = [1, 2] - array = ibis.literal(raw_array) - expr = mul(array, n) - result = con.execute(expr) - expected = mul(raw_array, n) - assert result == expected - - -def test_array_collect(t, df): - expr = t.group_by(t.key).aggregate(collected=t.array_int.collect()) - result = expr.execute().sort_values("key").reset_index(drop=True) - - expected = ( - df.groupby("key") - .array_int.apply(list) - .reset_index() - .rename(columns={"array_int": "collected"}) - ) - tm.assert_frame_equal(result, expected) - - -def test_array_filter(t, df): - expr = t.select(t.array_int.filter(lambda item: item != 3).name("array_int")) - result = expr.execute() - - df["array_int"] = df["array_int"].apply( - lambda ar: [item for item in ar if item != 3] - ) - expected = df[["array_int"]] - tm.assert_frame_equal(result, expected) diff --git a/ibis/backends/pyspark/tests/test_basic.py b/ibis/backends/pyspark/tests/test_basic.py index 192d06e4f081..95bafb4690dd 100644 --- a/ibis/backends/pyspark/tests/test_basic.py +++ b/ibis/backends/pyspark/tests/test_basic.py @@ -1,7 +1,10 @@ from __future__ import annotations +from datetime import timedelta + import pandas as pd import pandas.testing as tm +import pyspark.sql.types as pt import pytest from pytest import param @@ -116,7 +119,22 @@ def test_alias_after_select(t, df): def test_interval_columns_invalid(con): - msg = r"DayTimeIntervalType\(0, 1\) couldn't be converted to Interval" + df_interval_invalid = con._session.createDataFrame( + [[timedelta(days=10, hours=10, minutes=10, seconds=10)]], + pt.StructType( + [ + pt.StructField( + "interval_day_hour", + pt.DayTimeIntervalType( + pt.DayTimeIntervalType.DAY, pt.DayTimeIntervalType.SECOND + ), + ) + ] + ), + ) + + df_interval_invalid.createTempView("invalid_interval_table") + msg = r"DayTimeIntervalType.+ couldn't be converted to Interval" with pytest.raises(IbisTypeError, match=msg): con.table("invalid_interval_table") diff --git a/ibis/backends/pyspark/tests/test_ddl.py b/ibis/backends/pyspark/tests/test_ddl.py index 975d2840ca0e..cb00750a0003 100644 --- a/ibis/backends/pyspark/tests/test_ddl.py +++ b/ibis/backends/pyspark/tests/test_ddl.py @@ -59,10 +59,9 @@ def temp_db(con, temp_base): def test_create_database_with_location(con, temp_db): - base = os.path.dirname(temp_db) name = os.path.basename(temp_db) con.create_database(name, path=temp_db) - assert os.path.exists(base) + assert name in con.list_databases() def test_drop_table_not_exist(con): diff --git a/ibis/backends/pyspark/tests/test_udf.py b/ibis/backends/pyspark/tests/test_udf.py index d5c80ac27c35..e6b8789f8fce 100644 --- a/ibis/backends/pyspark/tests/test_udf.py +++ b/ibis/backends/pyspark/tests/test_udf.py @@ -5,6 +5,7 @@ import ibis from ibis.backends.pyspark import PYSPARK_LT_35 +from ibis.conftest import IS_SPARK_REMOTE pytest.importorskip("pyspark") @@ -46,6 +47,10 @@ def test_python_udf(t, df): @pytest.mark.xfail(PYSPARK_LT_35, reason="pyarrow UDFs require PySpark 3.5+") +@pytest.mark.xfail( + IS_SPARK_REMOTE, + reason="pyarrow UDFs aren't tested with spark remote due to environment setup complexities", +) def test_pyarrow_udf(t, df): result = t.mutate(repeated=pyarrow_repeat(t.str_col, 2)).execute() expected = df.assign(repeated=df.str_col * 2) diff --git a/ibis/backends/tests/errors.py b/ibis/backends/tests/errors.py index 436f74dab71e..17bb81b97849 100644 --- a/ibis/backends/tests/errors.py +++ b/ibis/backends/tests/errors.py @@ -27,7 +27,6 @@ None ) - try: from pyexasol.exceptions import ExaQueryError except ImportError: @@ -47,11 +46,21 @@ try: # PySpark 3.5.0 - from pyspark.errors.exceptions.captured import ( + from pyspark.errors.exceptions.base import ( + AnalysisException as PySparkAnalysisException, + ) + from pyspark.errors.exceptions.base import ( ArithmeticException as PySparkArithmeticException, ) + from pyspark.errors.exceptions.base import ParseException as PySparkParseException + from pyspark.errors.exceptions.base import PythonException as PySparkPythonException + from pyspark.errors.exceptions.connect import ( + SparkConnectGrpcException as PySparkConnectGrpcException, + ) except ImportError: - PySparkArithmeticException = None + PySparkParseException = PySparkAnalysisException = PySparkArithmeticException = ( + PySparkPythonException + ) = PySparkConnectGrpcException = None try: from google.api_core.exceptions import BadRequest as GoogleBadRequest diff --git a/ibis/backends/tests/test_aggregation.py b/ibis/backends/tests/test_aggregation.py index 1db71e2eccc9..62b2142d19a3 100644 --- a/ibis/backends/tests/test_aggregation.py +++ b/ibis/backends/tests/test_aggregation.py @@ -26,9 +26,11 @@ PyDruidProgrammingError, PyODBCProgrammingError, PySparkAnalysisException, + PySparkPythonException, SnowflakeProgrammingError, TrinoUserError, ) +from ibis.conftest import IS_SPARK_REMOTE from ibis.legacy.udf.vectorized import reduction np = pytest.importorskip("numpy") @@ -73,6 +75,12 @@ def mean_udf(s): reason="no udf support", raises=com.OperationNotDefinedError, ), + pytest.mark.notyet( + ["pyspark"], + condition=IS_SPARK_REMOTE, + raises=PySparkPythonException, + reason="remote udfs not yet tested due to environment complexities", + ), ], ), param(lambda t: t.double_col.min(), lambda t: t.double_col.min(), id="min"), @@ -211,14 +219,9 @@ def test_aggregate_grouped(backend, alltypes, df, result_fn, expected_fn): ], raises=com.OperationNotDefinedError, ) -@pytest.mark.notimpl( +@pytest.mark.notyet( ["pyspark"], - raises=NotImplementedError, - reason=( - "Invalid return type with grouped aggregate Pandas UDFs: " - "StructType([StructField('mean', DoubleType(), True), " - "StructField('std', DoubleType(), True)]) is not supported" - ), + raises=(PySparkPythonException, NotImplementedError), ) def test_aggregate_multikey_group_reduction_udf(backend, alltypes, df): """Tests .aggregate() on a multi-key group_by with a reduction @@ -1426,7 +1429,7 @@ def test_topk_filter_op(con, alltypes, df, result_fn, expected_fn): @pytest.mark.parametrize( - "agg_fn", [lambda s: list(s), lambda s: np.array(s)], ids=lambda obj: obj.__name__ + "agg_fn", [lambda s: list(s), lambda s: np.array(s)], ids=["list", "ndarray"] ) @pytest.mark.notimpl( [ @@ -1450,6 +1453,12 @@ def test_topk_filter_op(con, alltypes, df, result_fn, expected_fn): ], raises=com.OperationNotDefinedError, ) +@pytest.mark.notyet( + ["pyspark"], + condition=IS_SPARK_REMOTE, + raises=PySparkPythonException, + reason="remote udfs not yet tested due to environment complexities", +) def test_aggregate_list_like(backend, alltypes, df, agg_fn): """Tests .aggregate() where the result of an aggregation is a list-like. @@ -1492,6 +1501,12 @@ def test_aggregate_list_like(backend, alltypes, df, agg_fn): ], raises=com.OperationNotDefinedError, ) +@pytest.mark.notyet( + ["pyspark"], + condition=IS_SPARK_REMOTE, + raises=PySparkPythonException, + reason="remote udfs not yet tested due to environment complexities", +) def test_aggregate_mixed_udf(backend, alltypes, df): """Tests .aggregate() with multiple aggregations with mixed result types. diff --git a/ibis/backends/tests/test_array.py b/ibis/backends/tests/test_array.py index 123aa60409a3..c49d808412c9 100644 --- a/ibis/backends/tests/test_array.py +++ b/ibis/backends/tests/test_array.py @@ -31,6 +31,7 @@ TrinoUserError, ) from ibis.common.collections import frozendict +from ibis.conftest import IS_SPARK_REMOTE np = pytest.importorskip("numpy") pd = pytest.importorskip("pandas") @@ -170,6 +171,17 @@ def test_array_radd_concat(con): assert np.array_equal(result, expected) +@pytest.mark.parametrize("op", [lambda x, y: x + y, lambda x, y: y + x]) +def test_array_concat_scalar(con, op): + raw_left = [1, 2, 3] + raw_right = [3, 4] + left = ibis.literal(raw_left) + right = ibis.literal(raw_right) + expr = op(left, right) + result = con.execute(expr) + assert result == op(raw_left, raw_right) + + def test_array_length(con): expr = ibis.literal([1, 2, 3]).length() assert con.execute(expr.name("tmp")) == 3 @@ -432,7 +444,13 @@ def test_array_slice(backend, start, stop): ["bigquery"], raises=GoogleBadRequest, reason="BigQuery doesn't support arrays with null elements", - ) + ), + pytest.mark.notyet( + ["pyspark"], + condition=IS_SPARK_REMOTE, + raises=AssertionError, + reason="somehow, transformed results are different types", + ), ], id="nulls", ), @@ -444,11 +462,6 @@ def test_array_slice(backend, start, stop): [lambda x: x + 1, partial(lambda x, y: x + y, y=1), ibis._ + 1], ids=["lambda", "partial", "deferred"], ) -@pytest.mark.notimpl( - ["risingwave"], - raises=PsycoPg2InternalError, - reason="TODO(Kexiang): seems a bug", -) def test_array_map(con, input, output, func): t = ibis.memtable(input, schema=ibis.schema(dict(a="!array"))) t = ibis.memtable(input, schema=ibis.schema(dict(a="!array"))) @@ -672,6 +685,14 @@ def test_array_remove(con, input, expected): {"a": [[1, 3, 3], [], [42, 42], [], [None], None]}, [{3, 1}, set(), {42}, set(), {None}, None], id="null", + marks=[ + pytest.mark.notyet( + ["pyspark"], + condition=IS_SPARK_REMOTE, + raises=AssertionError, + reason="somehow, transformed results are different types", + ), + ], ), param( {"a": [[1, 3, 3], [], [42, 42], [], None]}, @@ -743,6 +764,12 @@ def test_array_sort(con, data): raises=AssertionError, reason="DataFusion transforms null elements to NAN", ), + pytest.mark.notyet( + ["pyspark"], + condition=IS_SPARK_REMOTE, + raises=AssertionError, + reason="somehow, transformed results are different types", + ), ], ), param( diff --git a/ibis/backends/tests/test_client.py b/ibis/backends/tests/test_client.py index 069d6a525b2a..7d1ae751e58b 100644 --- a/ibis/backends/tests/test_client.py +++ b/ibis/backends/tests/test_client.py @@ -35,6 +35,7 @@ PyODBCProgrammingError, SnowflakeProgrammingError, ) +from ibis.conftest import IS_SPARK_REMOTE from ibis.util import gen_name if TYPE_CHECKING: @@ -1686,6 +1687,12 @@ def test_insert_into_table_missing_columns(con, temp_table): @pytest.mark.notyet( ["bigquery"], raises=AssertionError, reason="test is flaky", strict=False ) +@pytest.mark.notyet( + ["pyspark"], + condition=IS_SPARK_REMOTE, + raises=AssertionError, + reason="likely deadlock when using spark connect", +) def test_memtable_cleanup(con): name = ibis.util.gen_name("temp_memtable") t = ibis.memtable({"a": [1, 2, 3], "b": list("def")}, name=name) diff --git a/ibis/backends/tests/test_export.py b/ibis/backends/tests/test_export.py index 112bcbf3f128..977243519862 100644 --- a/ibis/backends/tests/test_export.py +++ b/ibis/backends/tests/test_export.py @@ -23,6 +23,7 @@ SnowflakeProgrammingError, TrinoUserError, ) +from ibis.conftest import IS_SPARK_REMOTE pd = pytest.importorskip("pandas") pa = pytest.importorskip("pyarrow") @@ -215,6 +216,8 @@ def test_table_to_parquet_dir(tmp_path, backend, awards_players): outparquet_dir = tmp_path / "out" if backend.name() == "pyspark": + if IS_SPARK_REMOTE: + pytest.skip("writes to remote output directory") # pyspark already writes more than one file awards_players.to_parquet_dir(outparquet_dir) else: diff --git a/ibis/backends/tests/test_expr_caching.py b/ibis/backends/tests/test_expr_caching.py index 75aba279553e..03c73e3e9772 100644 --- a/ibis/backends/tests/test_expr_caching.py +++ b/ibis/backends/tests/test_expr_caching.py @@ -5,6 +5,7 @@ import ibis import ibis.common.exceptions as com +from ibis.conftest import IS_SPARK_REMOTE pa = pytest.importorskip("pyarrow") ds = pytest.importorskip("pyarrow.dataset") @@ -56,6 +57,7 @@ def test_persist_expression_contextmanager(backend, con, alltypes): @mark.notimpl(["exasol"], reason="Exasol does not support temporary tables") @mark.notyet( ["pyspark"], + condition=not IS_SPARK_REMOTE, raises=AssertionError, reason=( "PySpark holds on to `cached_table` in the stack frame of an internal function. " diff --git a/ibis/backends/tests/test_json.py b/ibis/backends/tests/test_json.py index ae2374d8dcd1..d4d772beda26 100644 --- a/ibis/backends/tests/test_json.py +++ b/ibis/backends/tests/test_json.py @@ -8,6 +8,8 @@ from packaging.version import parse as vparse import ibis.expr.types as ir +from ibis.backends.tests.errors import PySparkPythonException +from ibis.conftest import IS_SPARK_REMOTE np = pytest.importorskip("numpy") pd = pytest.importorskip("pandas") @@ -109,6 +111,12 @@ def test_json_array(backend, json_t): ) @pytest.mark.notimpl(["risingwave"]) @pytest.mark.notyet(["flink"], reason="should work but doesn't deserialize JSON") +@pytest.mark.notyet( + ["pyspark"], + condition=IS_SPARK_REMOTE, + raises=PySparkPythonException, + reason="environment issues", +) @pytest.mark.parametrize( ("typ", "expected_data"), [ diff --git a/ibis/backends/tests/test_register.py b/ibis/backends/tests/test_register.py index 02df85ef1cb0..0ed6925a2ce7 100644 --- a/ibis/backends/tests/test_register.py +++ b/ibis/backends/tests/test_register.py @@ -12,13 +12,20 @@ import ibis from ibis.backends.conftest import TEST_TABLES +from ibis.backends.tests.errors import PySparkAnalysisException +from ibis.conftest import IS_SPARK_REMOTE if TYPE_CHECKING: from collections.abc import Iterator import pyarrow as pa -pytestmark = pytest.mark.notimpl(["druid", "exasol", "oracle"]) +pytestmark = [ + pytest.mark.notimpl(["druid", "exasol", "oracle"]), + pytest.mark.notyet( + ["pyspark"], condition=IS_SPARK_REMOTE, raises=PySparkAnalysisException + ), +] @contextlib.contextmanager diff --git a/ibis/backends/tests/test_temporal.py b/ibis/backends/tests/test_temporal.py index 0bc48c3038b9..9c08d7fe3245 100644 --- a/ibis/backends/tests/test_temporal.py +++ b/ibis/backends/tests/test_temporal.py @@ -33,10 +33,12 @@ Py4JJavaError, PyDruidProgrammingError, PyODBCProgrammingError, + PySparkConnectGrpcException, SnowflakeProgrammingError, TrinoUserError, ) from ibis.common.annotations import ValidationError +from ibis.conftest import IS_SPARK_REMOTE np = pytest.importorskip("numpy") pd = pytest.importorskip("pandas") @@ -743,6 +745,12 @@ def convert_to_offset(x): ["bigquery", "snowflake", "sqlite", "exasol", "mssql"], raises=com.OperationNotDefinedError, ), + pytest.mark.notyet( + ["pyspark"], + condition=IS_SPARK_REMOTE, + raises=PySparkConnectGrpcException, + reason="arrow conversion breaks", + ), pytest.mark.notimpl(["druid"], raises=PyDruidProgrammingError), pytest.mark.notimpl( ["duckdb"], @@ -792,6 +800,12 @@ def convert_to_offset(x): raises=com.OperationNotDefinedError, reason="Some wonkiness in sqlglot generation.", ), + pytest.mark.notyet( + ["pyspark"], + condition=IS_SPARK_REMOTE, + raises=PySparkConnectGrpcException, + reason="arrow conversion breaks", + ), ], ), ], @@ -1691,6 +1705,7 @@ def test_integer_cast_to_timestamp_scalar(alltypes, df): @pytest.mark.notyet( ["pyspark"], reason="PySpark doesn't handle big timestamps", + condition=not IS_SPARK_REMOTE, raises=pd.errors.OutOfBoundsDatetime, ) @pytest.mark.notimpl(["flink"], raises=ArrowInvalid) @@ -1753,7 +1768,9 @@ def test_timestamp_date_comparison(backend, alltypes, df, left_fn, right_fn): @pytest.mark.notimpl( ["clickhouse"], reason="returns incorrect results", raises=AssertionError ) -@pytest.mark.notimpl(["pyspark"], raises=pd.errors.OutOfBoundsDatetime) +@pytest.mark.notimpl( + ["pyspark"], condition=not IS_SPARK_REMOTE, raises=pd.errors.OutOfBoundsDatetime +) @pytest.mark.notimpl(["polars"], raises=AssertionError, reason="returns NaT") @pytest.mark.notyet( ["flink"], diff --git a/ibis/backends/tests/test_udf.py b/ibis/backends/tests/test_udf.py index 3713a4cd3058..f27397dc256d 100644 --- a/ibis/backends/tests/test_udf.py +++ b/ibis/backends/tests/test_udf.py @@ -6,7 +6,8 @@ import ibis.common.exceptions as com from ibis import _, udf -from ibis.backends.tests.errors import Py4JJavaError +from ibis.backends.tests.errors import Py4JJavaError, PySparkPythonException +from ibis.conftest import IS_SPARK_REMOTE no_python_udfs = mark.notimpl( [ @@ -151,6 +152,12 @@ def add_one_pyarrow(s: int) -> int: # s is series, int is the element type raises=NotImplementedError, reason="postgres only supports Python-native UDFs", ) +@mark.notyet( + ["pyspark"], + condition=IS_SPARK_REMOTE, + raises=PySparkPythonException, + reason="remote udfs not yet tested due to environment complexities", +) @mark.parametrize( "add_one", [ diff --git a/ibis/backends/tests/test_vectorized_udf.py b/ibis/backends/tests/test_vectorized_udf.py index 9a3ffb6120f9..cbcfce905c1e 100644 --- a/ibis/backends/tests/test_vectorized_udf.py +++ b/ibis/backends/tests/test_vectorized_udf.py @@ -6,12 +6,25 @@ import ibis import ibis.common.exceptions as com import ibis.expr.datatypes as dt +from ibis.backends.tests.errors import PySparkPythonException +from ibis.conftest import IS_SPARK_REMOTE from ibis.legacy.udf.vectorized import analytic, elementwise, reduction np = pytest.importorskip("numpy") pd = pytest.importorskip("pandas") -pytestmark = pytest.mark.notimpl(["druid", "oracle", "risingwave"]) +pytestmark = [ + pytest.mark.notimpl(["druid", "oracle", "risingwave"]), + pytest.mark.notyet( + ["pyspark"], + condition=IS_SPARK_REMOTE, + raises=PySparkPythonException, + # TODO(cpcloud): this API is deprecated in 10.0.0, no use copypasting a + # bunch of markers just for two passing tests + strict=False, + reason="remote udfs not yet tested due to environment complexities", + ), +] def _format_udf_return_type(func, result_formatter): diff --git a/ibis/backends/tests/test_window.py b/ibis/backends/tests/test_window.py index 0942160ad1c0..c57bacb06698 100644 --- a/ibis/backends/tests/test_window.py +++ b/ibis/backends/tests/test_window.py @@ -18,8 +18,10 @@ Py4JJavaError, PyDruidProgrammingError, PyODBCProgrammingError, + PySparkPythonException, SnowflakeProgrammingError, ) +from ibis.conftest import IS_SPARK_REMOTE from ibis.legacy.udf.vectorized import analytic, reduction np = pytest.importorskip("numpy") @@ -368,6 +370,12 @@ def test_grouped_bounded_expanding_window( ], raises=com.OperationNotDefinedError, ), + pytest.mark.notyet( + ["pyspark"], + condition=IS_SPARK_REMOTE, + raises=PySparkPythonException, + reason="remote udfs not yet tested due to environment complexities", + ), ], ), ], @@ -550,6 +558,12 @@ def test_grouped_bounded_preceding_window( ], raises=com.OperationNotDefinedError, ), + pytest.mark.notyet( + ["pyspark"], + condition=IS_SPARK_REMOTE, + raises=PySparkPythonException, + reason="remote udfs not yet tested due to environment complexities", + ), ], ), ], @@ -708,6 +722,12 @@ def test_simple_ungrouped_window_with_scalar_order_by(alltypes): ], raises=com.OperationNotDefinedError, ), + pytest.mark.notyet( + ["pyspark"], + condition=IS_SPARK_REMOTE, + raises=PySparkPythonException, + reason="remote udfs not yet tested due to environment complexities", + ), ], ), param( @@ -736,6 +756,12 @@ def test_simple_ungrouped_window_with_scalar_order_by(alltypes): ], raises=com.OperationNotDefinedError, ), + pytest.mark.notyet( + ["pyspark"], + condition=IS_SPARK_REMOTE, + raises=PySparkPythonException, + reason="remote udfs not yet tested due to environment complexities", + ), ], ), # Analytic ops diff --git a/ibis/conftest.py b/ibis/conftest.py index 40d309c47d04..e0af794b5c03 100644 --- a/ibis/conftest.py +++ b/ibis/conftest.py @@ -17,6 +17,8 @@ WINDOWS = platform.system() == "Windows" ARM64 = platform.machine() in ("arm64", "aarch64") CI = os.environ.get("CI") is not None +SPARK_REMOTE = os.environ.get("SPARK_REMOTE") +IS_SPARK_REMOTE = bool(SPARK_REMOTE) @pytest.fixture(autouse=True) diff --git a/pyproject.toml b/pyproject.toml index ca83f20f971d..bf4eebd93637 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -295,6 +295,7 @@ filterwarnings = [ "error", # pyspark uses a deprecated pandas API "ignore:is_datetime64tz_dtype is deprecated and will be removed in a future version:DeprecationWarning", + "ignore:is_categorical_dtype is deprecated .+:DeprecationWarning", # pyspark and impala leave sockets open "ignore:Exception ignored in:", # pandas