From b8a5fb6e226b25aaa81d43be1679bee13ef9046c Mon Sep 17 00:00:00 2001
From: Reese Hyde <148883979+reesehyde@users.noreply.github.com>
Date: Sat, 30 Nov 2024 01:14:49 -0500
Subject: [PATCH] Pass Install Extras to Markers (#9553)
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
Adds support for conflicting dependencies in extras.
Co-authored-by: Randy Döring <30527984+radoering@users.noreply.github.com>
---
docs/dependency-specification.md | 103 +++++
poetry.lock | 189 +++++-----
src/poetry/installation/installer.py | 1 +
src/poetry/puzzle/provider.py | 101 ++++-
src/poetry/puzzle/solver.py | 9 +-
src/poetry/puzzle/transaction.py | 16 +-
...th-conflicting-dependency-extras-root.test | 28 ++
...flicting-dependency-extras-transitive.test | 52 +++
.../with-dependencies-differing-extras.test | 52 +++
.../fixtures/with-exclusive-extras.test | 38 ++
tests/installation/test_installer.py | 352 ++++++++++++++++++
tests/puzzle/test_solver.py | 130 +++++++
tests/puzzle/test_transaction.py | 52 +++
13 files changed, 1003 insertions(+), 120 deletions(-)
create mode 100644 tests/installation/fixtures/with-conflicting-dependency-extras-root.test
create mode 100644 tests/installation/fixtures/with-conflicting-dependency-extras-transitive.test
create mode 100644 tests/installation/fixtures/with-dependencies-differing-extras.test
create mode 100644 tests/installation/fixtures/with-exclusive-extras.test
diff --git a/docs/dependency-specification.md b/docs/dependency-specification.md
index 63299e91c8e..500999394e5 100644
--- a/docs/dependency-specification.md
+++ b/docs/dependency-specification.md
@@ -572,6 +572,109 @@ pathlib2 = { version = "^2.2", markers = "python_version <= '3.4' or sys_platfor
{{< /tab >}}
{{< /tabs >}}
+### `extra` environment marker
+
+Poetry populates the `extra` marker with each of the selected extras of the root package.
+For example, consider the following dependency:
+```toml
+[project.optional-dependencies]
+paths = [
+ "pathlib2 (>=2.2,<3.0) ; sys_platform == 'win32'"
+]
+```
+
+`pathlib2` will be installed when you install your package with `--extras paths` on a `win32` machine.
+
+#### Exclusive extras
+
+{{% warning %}}
+The first example will only work completely if you configure Poetry to not re-resolve for installation:
+
+```bash
+poetry config installer.re-resolve false
+```
+
+This is a new feature of Poetry 2.0 that may become the default in a future version of Poetry.
+
+{{% /warning %}}
+
+Keep in mind that all combinations of possible extras available in your project need to be compatible with each other.
+This means that in order to use differing or incompatible versions across different combinations, you need to make your
+extra markers *exclusive*. For example, the following installs PyTorch from one source repository with CPU versions
+when the `cuda` extra is *not* specified, while the other installs from another repository with a separate version set
+for GPUs when the `cuda` extra *is* specified:
+
+```toml
+[project]
+name = "torch-example"
+requires-python = ">=3.10"
+dependencies = [
+ "torch (==2.3.1+cpu) ; extra != 'cuda'",
+]
+
+[project.optional-dependencies]
+cuda = [
+ "torch (==2.3.1+cu118)",
+]
+
+[tool.poetry]
+package-mode = false
+
+[tool.poetry.dependencies]
+torch = [
+ { markers = "extra != 'cuda'", source = "pytorch-cpu"},
+ { markers = "extra == 'cuda'", source = "pytorch-cuda"},
+ ]
+
+[[tool.poetry.source]]
+name = "pytorch-cpu"
+url = "https://download.pytorch.org/whl/cpu"
+priority = "explicit"
+
+[[tool.poetry.source]]
+name = "pytorch-cuda"
+url = "https://download.pytorch.org/whl/cu118"
+priority = "explicit"
+```
+
+For the CPU case, we have to specify `"extra != 'cuda'"` because the version specified is not compatible with the
+GPU (`cuda`) version.
+
+This same logic applies when you want either-or extras:
+
+```toml
+[project]
+name = "torch-example"
+requires-python = ">=3.10"
+
+[project.optional-dependencies]
+cpu = [
+ "torch (==2.3.1+cpu)",
+]
+cuda = [
+ "torch (==2.3.1+cu118)",
+]
+
+[tool.poetry]
+package-mode = false
+
+[tool.poetry.dependencies]
+torch = [
+ { markers = "extra == 'cpu' and extra != 'cuda'", source = "pytorch-cpu"},
+ { markers = "extra == 'cuda' and extra != 'cpu'", source = "pytorch-cuda"},
+ ]
+
+[[tool.poetry.source]]
+name = "pytorch-cpu"
+url = "https://download.pytorch.org/whl/cpu"
+priority = "explicit"
+
+[[tool.poetry.source]]
+name = "pytorch-cuda"
+url = "https://download.pytorch.org/whl/cu118"
+priority = "explicit"
+```
+
## Multiple constraints dependencies
Sometimes, one of your dependency may have different version ranges depending
diff --git a/poetry.lock b/poetry.lock
index 35b8db01c61..5a6b6a44b07 100644
--- a/poetry.lock
+++ b/poetry.lock
@@ -970,7 +970,7 @@ develop = false
type = "git"
url = "https://github.com/python-poetry/poetry-core.git"
reference = "main"
-resolved_reference = "616d7bfaf018d50bd09bc24c630b697b6368d5a3"
+resolved_reference = "b95ec5321f1842286b042ac206c9f5395850c684"
[[package]]
name = "pre-commit"
@@ -1221,108 +1221,103 @@ files = [
[[package]]
name = "rapidfuzz"
-version = "3.9.4"
+version = "3.10.1"
description = "rapid fuzzy string matching"
optional = false
-python-versions = ">=3.8"
+python-versions = ">=3.9"
files = [
- {file = "rapidfuzz-3.9.4-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:c9b9793c19bdf38656c8eaefbcf4549d798572dadd70581379e666035c9df781"},
- {file = "rapidfuzz-3.9.4-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:015b5080b999404fe06ec2cb4f40b0be62f0710c926ab41e82dfbc28e80675b4"},
- {file = "rapidfuzz-3.9.4-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:acc5ceca9c1e1663f3e6c23fb89a311f69b7615a40ddd7645e3435bf3082688a"},
- {file = "rapidfuzz-3.9.4-cp310-cp310-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:1424e238bc3f20e1759db1e0afb48a988a9ece183724bef91ea2a291c0b92a95"},
- {file = "rapidfuzz-3.9.4-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:ed01378f605aa1f449bee82cd9c83772883120d6483e90aa6c5a4ce95dc5c3aa"},
- {file = "rapidfuzz-3.9.4-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:eb26d412271e5a76cdee1c2d6bf9881310665d3fe43b882d0ed24edfcb891a84"},
- {file = "rapidfuzz-3.9.4-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8f37e9e1f17be193c41a31c864ad4cd3ebd2b40780db11cd5c04abf2bcf4201b"},
- {file = "rapidfuzz-3.9.4-cp310-cp310-musllinux_1_2_aarch64.whl", hash = "sha256:d070ec5cf96b927c4dc5133c598c7ff6db3b833b363b2919b13417f1002560bc"},
- {file = "rapidfuzz-3.9.4-cp310-cp310-musllinux_1_2_i686.whl", hash = "sha256:10e61bb7bc807968cef09a0e32ce253711a2d450a4dce7841d21d45330ffdb24"},
- {file = "rapidfuzz-3.9.4-cp310-cp310-musllinux_1_2_ppc64le.whl", hash = "sha256:31a2fc60bb2c7face4140010a7aeeafed18b4f9cdfa495cc644a68a8c60d1ff7"},
- {file = "rapidfuzz-3.9.4-cp310-cp310-musllinux_1_2_s390x.whl", hash = "sha256:fbebf1791a71a2e89f5c12b78abddc018354d5859e305ec3372fdae14f80a826"},
- {file = "rapidfuzz-3.9.4-cp310-cp310-musllinux_1_2_x86_64.whl", hash = "sha256:aee9fc9e3bb488d040afc590c0a7904597bf4ccd50d1491c3f4a5e7e67e6cd2c"},
- {file = "rapidfuzz-3.9.4-cp310-cp310-win32.whl", hash = "sha256:005a02688a51c7d2451a2d41c79d737aa326ff54167211b78a383fc2aace2c2c"},
- {file = "rapidfuzz-3.9.4-cp310-cp310-win_amd64.whl", hash = "sha256:3a2e75e41ee3274754d3b2163cc6c82cd95b892a85ab031f57112e09da36455f"},
- {file = "rapidfuzz-3.9.4-cp310-cp310-win_arm64.whl", hash = "sha256:2c99d355f37f2b289e978e761f2f8efeedc2b14f4751d9ff7ee344a9a5ca98d9"},
- {file = "rapidfuzz-3.9.4-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:07141aa6099e39d48637ce72a25b893fc1e433c50b3e837c75d8edf99e0c63e1"},
- {file = "rapidfuzz-3.9.4-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:db1664eaff5d7d0f2542dd9c25d272478deaf2c8412e4ad93770e2e2d828e175"},
- {file = "rapidfuzz-3.9.4-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:bc01a223f6605737bec3202e94dcb1a449b6c76d46082cfc4aa980f2a60fd40e"},
- {file = "rapidfuzz-3.9.4-cp311-cp311-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:1869c42e73e2a8910b479be204fa736418741b63ea2325f9cc583c30f2ded41a"},
- {file = "rapidfuzz-3.9.4-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:62ea7007941fb2795fff305ac858f3521ec694c829d5126e8f52a3e92ae75526"},
- {file = "rapidfuzz-3.9.4-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:698e992436bf7f0afc750690c301215a36ff952a6dcd62882ec13b9a1ebf7a39"},
- {file = "rapidfuzz-3.9.4-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b76f611935f15a209d3730c360c56b6df8911a9e81e6a38022efbfb96e433bab"},
- {file = "rapidfuzz-3.9.4-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:129627d730db2e11f76169344a032f4e3883d34f20829419916df31d6d1338b1"},
- {file = "rapidfuzz-3.9.4-cp311-cp311-musllinux_1_2_i686.whl", hash = "sha256:90a82143c14e9a14b723a118c9ef8d1bbc0c5a16b1ac622a1e6c916caff44dd8"},
- {file = "rapidfuzz-3.9.4-cp311-cp311-musllinux_1_2_ppc64le.whl", hash = "sha256:ded58612fe3b0e0d06e935eaeaf5a9fd27da8ba9ed3e2596307f40351923bf72"},
- {file = "rapidfuzz-3.9.4-cp311-cp311-musllinux_1_2_s390x.whl", hash = "sha256:f16f5d1c4f02fab18366f2d703391fcdbd87c944ea10736ca1dc3d70d8bd2d8b"},
- {file = "rapidfuzz-3.9.4-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:26aa7eece23e0df55fb75fbc2a8fb678322e07c77d1fd0e9540496e6e2b5f03e"},
- {file = "rapidfuzz-3.9.4-cp311-cp311-win32.whl", hash = "sha256:f187a9c3b940ce1ee324710626daf72c05599946bd6748abe9e289f1daa9a077"},
- {file = "rapidfuzz-3.9.4-cp311-cp311-win_amd64.whl", hash = "sha256:d8e9130fe5d7c9182990b366ad78fd632f744097e753e08ace573877d67c32f8"},
- {file = "rapidfuzz-3.9.4-cp311-cp311-win_arm64.whl", hash = "sha256:40419e98b10cd6a00ce26e4837a67362f658fc3cd7a71bd8bd25c99f7ee8fea5"},
- {file = "rapidfuzz-3.9.4-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:b5d5072b548db1b313a07d62d88fe0b037bd2783c16607c647e01b070f6cf9e5"},
- {file = "rapidfuzz-3.9.4-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:cf5bcf22e1f0fd273354462631d443ef78d677f7d2fc292de2aec72ae1473e66"},
- {file = "rapidfuzz-3.9.4-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:0c8fc973adde8ed52810f590410e03fb6f0b541bbaeb04c38d77e63442b2df4c"},
- {file = "rapidfuzz-3.9.4-cp312-cp312-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:f2464bb120f135293e9a712e342c43695d3d83168907df05f8c4ead1612310c7"},
- {file = "rapidfuzz-3.9.4-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:8d9d58689aca22057cf1a5851677b8a3ccc9b535ca008c7ed06dc6e1899f7844"},
- {file = "rapidfuzz-3.9.4-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:167e745f98baa0f3034c13583e6302fb69249a01239f1483d68c27abb841e0a1"},
- {file = "rapidfuzz-3.9.4-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:db0bf0663b4b6da1507869722420ea9356b6195aa907228d6201303e69837af9"},
- {file = "rapidfuzz-3.9.4-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:cd6ac61b74fdb9e23f04d5f068e6cf554f47e77228ca28aa2347a6ca8903972f"},
- {file = "rapidfuzz-3.9.4-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:60ff67c690acecf381759c16cb06c878328fe2361ddf77b25d0e434ea48a29da"},
- {file = "rapidfuzz-3.9.4-cp312-cp312-musllinux_1_2_ppc64le.whl", hash = "sha256:cb934363380c60f3a57d14af94325125cd8cded9822611a9f78220444034e36e"},
- {file = "rapidfuzz-3.9.4-cp312-cp312-musllinux_1_2_s390x.whl", hash = "sha256:fe833493fb5cc5682c823ea3e2f7066b07612ee8f61ecdf03e1268f262106cdd"},
- {file = "rapidfuzz-3.9.4-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:2797fb847d89e04040d281cb1902cbeffbc4b5131a5c53fc0db490fd76b2a547"},
- {file = "rapidfuzz-3.9.4-cp312-cp312-win32.whl", hash = "sha256:52e3d89377744dae68ed7c84ad0ddd3f5e891c82d48d26423b9e066fc835cc7c"},
- {file = "rapidfuzz-3.9.4-cp312-cp312-win_amd64.whl", hash = "sha256:c76da20481c906e08400ee9be230f9e611d5931a33707d9df40337c2655c84b5"},
- {file = "rapidfuzz-3.9.4-cp312-cp312-win_arm64.whl", hash = "sha256:f2d2846f3980445864c7e8b8818a29707fcaff2f0261159ef6b7bd27ba139296"},
- {file = "rapidfuzz-3.9.4-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:355fc4a268ffa07bab88d9adee173783ec8d20136059e028d2a9135c623c44e6"},
- {file = "rapidfuzz-3.9.4-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:4d81a78f90269190b568a8353d4ea86015289c36d7e525cd4d43176c88eff429"},
- {file = "rapidfuzz-3.9.4-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:9e618625ffc4660b26dc8e56225f8b966d5842fa190e70c60db6cd393e25b86e"},
- {file = "rapidfuzz-3.9.4-cp38-cp38-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:b712336ad6f2bacdbc9f1452556e8942269ef71f60a9e6883ef1726b52d9228a"},
- {file = "rapidfuzz-3.9.4-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:84fc1ee19fdad05770c897e793836c002344524301501d71ef2e832847425707"},
- {file = "rapidfuzz-3.9.4-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:1950f8597890c0c707cb7e0416c62a1cf03dcdb0384bc0b2dbda7e05efe738ec"},
- {file = "rapidfuzz-3.9.4-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:4a6c35f272ec9c430568dc8c1c30cb873f6bc96be2c79795e0bce6db4e0e101d"},
- {file = "rapidfuzz-3.9.4-cp38-cp38-musllinux_1_2_aarch64.whl", hash = "sha256:1df0f9e9239132a231c86ae4f545ec2b55409fa44470692fcfb36b1bd00157ad"},
- {file = "rapidfuzz-3.9.4-cp38-cp38-musllinux_1_2_i686.whl", hash = "sha256:d2c51955329bfccf99ae26f63d5928bf5be9fcfcd9f458f6847fd4b7e2b8986c"},
- {file = "rapidfuzz-3.9.4-cp38-cp38-musllinux_1_2_ppc64le.whl", hash = "sha256:3c522f462d9fc504f2ea8d82e44aa580e60566acc754422c829ad75c752fbf8d"},
- {file = "rapidfuzz-3.9.4-cp38-cp38-musllinux_1_2_s390x.whl", hash = "sha256:d8a52fc50ded60d81117d7647f262c529659fb21d23e14ebfd0b35efa4f1b83d"},
- {file = "rapidfuzz-3.9.4-cp38-cp38-musllinux_1_2_x86_64.whl", hash = "sha256:04dbdfb0f0bfd3f99cf1e9e24fadc6ded2736d7933f32f1151b0f2abb38f9a25"},
- {file = "rapidfuzz-3.9.4-cp38-cp38-win32.whl", hash = "sha256:4968c8bd1df84b42f382549e6226710ad3476f976389839168db3e68fd373298"},
- {file = "rapidfuzz-3.9.4-cp38-cp38-win_amd64.whl", hash = "sha256:3fe4545f89f8d6c27b6bbbabfe40839624873c08bd6700f63ac36970a179f8f5"},
- {file = "rapidfuzz-3.9.4-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:9f256c8fb8f3125574c8c0c919ab0a1f75d7cba4d053dda2e762dcc36357969d"},
- {file = "rapidfuzz-3.9.4-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:f5fdc09cf6e9d8eac3ce48a4615b3a3ee332ea84ac9657dbbefef913b13e632f"},
- {file = "rapidfuzz-3.9.4-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:d395d46b80063d3b5d13c0af43d2c2cedf3ab48c6a0c2aeec715aa5455b0c632"},
- {file = "rapidfuzz-3.9.4-cp39-cp39-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:7fa714fb96ce9e70c37e64c83b62fe8307030081a0bfae74a76fac7ba0f91715"},
- {file = "rapidfuzz-3.9.4-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:1bc1a0f29f9119be7a8d3c720f1d2068317ae532e39e4f7f948607c3a6de8396"},
- {file = "rapidfuzz-3.9.4-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:6022674aa1747d6300f699cd7c54d7dae89bfe1f84556de699c4ac5df0838082"},
- {file = "rapidfuzz-3.9.4-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:dcb72e5f9762fd469701a7e12e94b924af9004954f8c739f925cb19c00862e38"},
- {file = "rapidfuzz-3.9.4-cp39-cp39-musllinux_1_2_aarch64.whl", hash = "sha256:ad04ae301129f0eb5b350a333accd375ce155a0c1cec85ab0ec01f770214e2e4"},
- {file = "rapidfuzz-3.9.4-cp39-cp39-musllinux_1_2_i686.whl", hash = "sha256:f46a22506f17c0433e349f2d1dc11907c393d9b3601b91d4e334fa9a439a6a4d"},
- {file = "rapidfuzz-3.9.4-cp39-cp39-musllinux_1_2_ppc64le.whl", hash = "sha256:01b42a8728c36011718da409aa86b84984396bf0ca3bfb6e62624f2014f6022c"},
- {file = "rapidfuzz-3.9.4-cp39-cp39-musllinux_1_2_s390x.whl", hash = "sha256:e590d5d5443cf56f83a51d3c4867bd1f6be8ef8cfcc44279522bcef3845b2a51"},
- {file = "rapidfuzz-3.9.4-cp39-cp39-musllinux_1_2_x86_64.whl", hash = "sha256:4c72078b5fdce34ba5753f9299ae304e282420e6455e043ad08e4488ca13a2b0"},
- {file = "rapidfuzz-3.9.4-cp39-cp39-win32.whl", hash = "sha256:f75639277304e9b75e6a7b3c07042d2264e16740a11e449645689ed28e9c2124"},
- {file = "rapidfuzz-3.9.4-cp39-cp39-win_amd64.whl", hash = "sha256:e81e27e8c32a1e1278a4bb1ce31401bfaa8c2cc697a053b985a6f8d013df83ec"},
- {file = "rapidfuzz-3.9.4-cp39-cp39-win_arm64.whl", hash = "sha256:15bc397ee9a3ed1210b629b9f5f1da809244adc51ce620c504138c6e7095b7bd"},
- {file = "rapidfuzz-3.9.4-pp310-pypy310_pp73-macosx_10_15_x86_64.whl", hash = "sha256:20488ade4e1ddba3cfad04f400da7a9c1b91eff5b7bd3d1c50b385d78b587f4f"},
- {file = "rapidfuzz-3.9.4-pp310-pypy310_pp73-macosx_11_0_arm64.whl", hash = "sha256:e61b03509b1a6eb31bc5582694f6df837d340535da7eba7bedb8ae42a2fcd0b9"},
- {file = "rapidfuzz-3.9.4-pp310-pypy310_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:098d231d4e51644d421a641f4a5f2f151f856f53c252b03516e01389b2bfef99"},
- {file = "rapidfuzz-3.9.4-pp310-pypy310_pp73-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:17ab8b7d10fde8dd763ad428aa961c0f30a1b44426e675186af8903b5d134fb0"},
- {file = "rapidfuzz-3.9.4-pp310-pypy310_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e272df61bee0a056a3daf99f9b1bd82cf73ace7d668894788139c868fdf37d6f"},
- {file = "rapidfuzz-3.9.4-pp310-pypy310_pp73-win_amd64.whl", hash = "sha256:d6481e099ff8c4edda85b8b9b5174c200540fd23c8f38120016c765a86fa01f5"},
- {file = "rapidfuzz-3.9.4-pp38-pypy38_pp73-macosx_10_9_x86_64.whl", hash = "sha256:ad61676e9bdae677d577fe80ec1c2cea1d150c86be647e652551dcfe505b1113"},
- {file = "rapidfuzz-3.9.4-pp38-pypy38_pp73-macosx_11_0_arm64.whl", hash = "sha256:af65020c0dd48d0d8ae405e7e69b9d8ae306eb9b6249ca8bf511a13f465fad85"},
- {file = "rapidfuzz-3.9.4-pp38-pypy38_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:4d38b4e026fcd580e0bda6c0ae941e0e9a52c6bc66cdce0b8b0da61e1959f5f8"},
- {file = "rapidfuzz-3.9.4-pp38-pypy38_pp73-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:f74ed072c2b9dc6743fb19994319d443a4330b0e64aeba0aa9105406c7c5b9c2"},
- {file = "rapidfuzz-3.9.4-pp38-pypy38_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:aee5f6b8321f90615c184bd8a4c676e9becda69b8e4e451a90923db719d6857c"},
- {file = "rapidfuzz-3.9.4-pp38-pypy38_pp73-win_amd64.whl", hash = "sha256:3a555e3c841d6efa350f862204bb0a3fea0c006b8acc9b152b374fa36518a1c6"},
- {file = "rapidfuzz-3.9.4-pp39-pypy39_pp73-macosx_10_15_x86_64.whl", hash = "sha256:0772150d37bf018110351c01d032bf9ab25127b966a29830faa8ad69b7e2f651"},
- {file = "rapidfuzz-3.9.4-pp39-pypy39_pp73-macosx_11_0_arm64.whl", hash = "sha256:addcdd3c3deef1bd54075bd7aba0a6ea9f1d01764a08620074b7a7b1e5447cb9"},
- {file = "rapidfuzz-3.9.4-pp39-pypy39_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:3fe86b82b776554add8f900b6af202b74eb5efe8f25acdb8680a5c977608727f"},
- {file = "rapidfuzz-3.9.4-pp39-pypy39_pp73-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:b0fc91ac59f4414d8542454dfd6287a154b8e6f1256718c898f695bdbb993467"},
- {file = "rapidfuzz-3.9.4-pp39-pypy39_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:3a944e546a296a5fdcaabb537b01459f1b14d66f74e584cb2a91448bffadc3c1"},
- {file = "rapidfuzz-3.9.4-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:4fb96ba96d58c668a17a06b5b5e8340fedc26188e87b0d229d38104556f30cd8"},
- {file = "rapidfuzz-3.9.4.tar.gz", hash = "sha256:366bf8947b84e37f2f4cf31aaf5f37c39f620d8c0eddb8b633e6ba0129ca4a0a"},
+ {file = "rapidfuzz-3.10.1-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:f17d9f21bf2f2f785d74f7b0d407805468b4c173fa3e52c86ec94436b338e74a"},
+ {file = "rapidfuzz-3.10.1-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:b31f358a70efc143909fb3d75ac6cd3c139cd41339aa8f2a3a0ead8315731f2b"},
+ {file = "rapidfuzz-3.10.1-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:7f4f43f2204b56a61448ec2dd061e26fd344c404da99fb19f3458200c5874ba2"},
+ {file = "rapidfuzz-3.10.1-cp310-cp310-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:9d81bf186a453a2757472133b24915768abc7c3964194406ed93e170e16c21cb"},
+ {file = "rapidfuzz-3.10.1-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:3611c8f45379a12063d70075c75134f2a8bd2e4e9b8a7995112ddae95ca1c982"},
+ {file = "rapidfuzz-3.10.1-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:3c3b537b97ac30da4b73930fa8a4fe2f79c6d1c10ad535c5c09726612cd6bed9"},
+ {file = "rapidfuzz-3.10.1-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:231ef1ec9cf7b59809ce3301006500b9d564ddb324635f4ea8f16b3e2a1780da"},
+ {file = "rapidfuzz-3.10.1-cp310-cp310-musllinux_1_2_aarch64.whl", hash = "sha256:ed4f3adc1294834955b7e74edd3c6bd1aad5831c007f2d91ea839e76461a5879"},
+ {file = "rapidfuzz-3.10.1-cp310-cp310-musllinux_1_2_i686.whl", hash = "sha256:7b6015da2e707bf632a71772a2dbf0703cff6525732c005ad24987fe86e8ec32"},
+ {file = "rapidfuzz-3.10.1-cp310-cp310-musllinux_1_2_ppc64le.whl", hash = "sha256:1b35a118d61d6f008e8e3fb3a77674d10806a8972c7b8be433d6598df4d60b01"},
+ {file = "rapidfuzz-3.10.1-cp310-cp310-musllinux_1_2_s390x.whl", hash = "sha256:bc308d79a7e877226f36bdf4e149e3ed398d8277c140be5c1fd892ec41739e6d"},
+ {file = "rapidfuzz-3.10.1-cp310-cp310-musllinux_1_2_x86_64.whl", hash = "sha256:f017dbfecc172e2d0c37cf9e3d519179d71a7f16094b57430dffc496a098aa17"},
+ {file = "rapidfuzz-3.10.1-cp310-cp310-win32.whl", hash = "sha256:36c0e1483e21f918d0f2f26799fe5ac91c7b0c34220b73007301c4f831a9c4c7"},
+ {file = "rapidfuzz-3.10.1-cp310-cp310-win_amd64.whl", hash = "sha256:10746c1d4c8cd8881c28a87fd7ba0c9c102346dfe7ff1b0d021cdf093e9adbff"},
+ {file = "rapidfuzz-3.10.1-cp310-cp310-win_arm64.whl", hash = "sha256:dfa64b89dcb906835e275187569e51aa9d546a444489e97aaf2cc84011565fbe"},
+ {file = "rapidfuzz-3.10.1-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:92958ae075c87fef393f835ed02d4fe8d5ee2059a0934c6c447ea3417dfbf0e8"},
+ {file = "rapidfuzz-3.10.1-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:ba7521e072c53e33c384e78615d0718e645cab3c366ecd3cc8cb732befd94967"},
+ {file = "rapidfuzz-3.10.1-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:00d02cbd75d283c287471b5b3738b3e05c9096150f93f2d2dfa10b3d700f2db9"},
+ {file = "rapidfuzz-3.10.1-cp311-cp311-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:efa1582a397da038e2f2576c9cd49b842f56fde37d84a6b0200ffebc08d82350"},
+ {file = "rapidfuzz-3.10.1-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:f12912acee1f506f974f58de9fdc2e62eea5667377a7e9156de53241c05fdba8"},
+ {file = "rapidfuzz-3.10.1-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:666d5d8b17becc3f53447bcb2b6b33ce6c2df78792495d1fa82b2924cd48701a"},
+ {file = "rapidfuzz-3.10.1-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:26f71582c0d62445067ee338ddad99b655a8f4e4ed517a90dcbfbb7d19310474"},
+ {file = "rapidfuzz-3.10.1-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:8a2ef08b27167bcff230ffbfeedd4c4fa6353563d6aaa015d725dd3632fc3de7"},
+ {file = "rapidfuzz-3.10.1-cp311-cp311-musllinux_1_2_i686.whl", hash = "sha256:365e4fc1a2b95082c890f5e98489b894e6bf8c338c6ac89bb6523c2ca6e9f086"},
+ {file = "rapidfuzz-3.10.1-cp311-cp311-musllinux_1_2_ppc64le.whl", hash = "sha256:1996feb7a61609fa842e6b5e0c549983222ffdedaf29644cc67e479902846dfe"},
+ {file = "rapidfuzz-3.10.1-cp311-cp311-musllinux_1_2_s390x.whl", hash = "sha256:cf654702f144beaa093103841a2ea6910d617d0bb3fccb1d1fd63c54dde2cd49"},
+ {file = "rapidfuzz-3.10.1-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:ec108bf25de674781d0a9a935030ba090c78d49def3d60f8724f3fc1e8e75024"},
+ {file = "rapidfuzz-3.10.1-cp311-cp311-win32.whl", hash = "sha256:031f8b367e5d92f7a1e27f7322012f3c321c3110137b43cc3bf678505583ef48"},
+ {file = "rapidfuzz-3.10.1-cp311-cp311-win_amd64.whl", hash = "sha256:f98f36c6a1bb9a6c8bbec99ad87c8c0e364f34761739b5ea9adf7b48129ae8cf"},
+ {file = "rapidfuzz-3.10.1-cp311-cp311-win_arm64.whl", hash = "sha256:f1da2028cb4e41be55ee797a82d6c1cf589442504244249dfeb32efc608edee7"},
+ {file = "rapidfuzz-3.10.1-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:1340b56340896bede246f612b6ecf685f661a56aabef3d2512481bfe23ac5835"},
+ {file = "rapidfuzz-3.10.1-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:2316515169b7b5a453f0ce3adbc46c42aa332cae9f2edb668e24d1fc92b2f2bb"},
+ {file = "rapidfuzz-3.10.1-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8e06fe6a12241ec1b72c0566c6b28cda714d61965d86569595ad24793d1ab259"},
+ {file = "rapidfuzz-3.10.1-cp312-cp312-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:d99c1cd9443b19164ec185a7d752f4b4db19c066c136f028991a480720472e23"},
+ {file = "rapidfuzz-3.10.1-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:a1d9aa156ed52d3446388ba4c2f335e312191d1ca9d1f5762ee983cf23e4ecf6"},
+ {file = "rapidfuzz-3.10.1-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:54bcf4efaaee8e015822be0c2c28214815f4f6b4f70d8362cfecbd58a71188ac"},
+ {file = "rapidfuzz-3.10.1-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:c0c955e32afdbfdf6e9ee663d24afb25210152d98c26d22d399712d29a9b976b"},
+ {file = "rapidfuzz-3.10.1-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:191633722203f5b7717efcb73a14f76f3b124877d0608c070b827c5226d0b972"},
+ {file = "rapidfuzz-3.10.1-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:195baad28057ec9609e40385991004e470af9ef87401e24ebe72c064431524ab"},
+ {file = "rapidfuzz-3.10.1-cp312-cp312-musllinux_1_2_ppc64le.whl", hash = "sha256:0fff4a6b87c07366662b62ae994ffbeadc472e72f725923f94b72a3db49f4671"},
+ {file = "rapidfuzz-3.10.1-cp312-cp312-musllinux_1_2_s390x.whl", hash = "sha256:4ffed25f9fdc0b287f30a98467493d1e1ce5b583f6317f70ec0263b3c97dbba6"},
+ {file = "rapidfuzz-3.10.1-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:d02cf8e5af89a9ac8f53c438ddff6d773f62c25c6619b29db96f4aae248177c0"},
+ {file = "rapidfuzz-3.10.1-cp312-cp312-win32.whl", hash = "sha256:f3bb81d4fe6a5d20650f8c0afcc8f6e1941f6fecdb434f11b874c42467baded0"},
+ {file = "rapidfuzz-3.10.1-cp312-cp312-win_amd64.whl", hash = "sha256:aaf83e9170cb1338922ae42d320699dccbbdca8ffed07faeb0b9257822c26e24"},
+ {file = "rapidfuzz-3.10.1-cp312-cp312-win_arm64.whl", hash = "sha256:c5da802a0d085ad81b0f62828fb55557996c497b2d0b551bbdfeafd6d447892f"},
+ {file = "rapidfuzz-3.10.1-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:fc22d69a1c9cccd560a5c434c0371b2df0f47c309c635a01a913e03bbf183710"},
+ {file = "rapidfuzz-3.10.1-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:38b0dac2c8e057562b8f0d8ae5b663d2d6a28c5ab624de5b73cef9abb6129a24"},
+ {file = "rapidfuzz-3.10.1-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6fde3bbb14e92ce8fcb5c2edfff72e474d0080cadda1c97785bf4822f037a309"},
+ {file = "rapidfuzz-3.10.1-cp313-cp313-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:9141fb0592e55f98fe9ac0f3ce883199b9c13e262e0bf40c5b18cdf926109d16"},
+ {file = "rapidfuzz-3.10.1-cp313-cp313-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:237bec5dd1bfc9b40bbd786cd27949ef0c0eb5fab5eb491904c6b5df59d39d3c"},
+ {file = "rapidfuzz-3.10.1-cp313-cp313-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:18123168cba156ab5794ea6de66db50f21bb3c66ae748d03316e71b27d907b95"},
+ {file = "rapidfuzz-3.10.1-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:0b75fe506c8e02769cc47f5ab21ce3e09b6211d3edaa8f8f27331cb6988779be"},
+ {file = "rapidfuzz-3.10.1-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:9da82aa4b46973aaf9e03bb4c3d6977004648c8638febfc0f9d237e865761270"},
+ {file = "rapidfuzz-3.10.1-cp313-cp313-musllinux_1_2_i686.whl", hash = "sha256:c34c022d5ad564f1a5a57a4a89793bd70d7bad428150fb8ff2760b223407cdcf"},
+ {file = "rapidfuzz-3.10.1-cp313-cp313-musllinux_1_2_ppc64le.whl", hash = "sha256:1e96c84d6c2a0ca94e15acb5399118fff669f4306beb98a6d8ec6f5dccab4412"},
+ {file = "rapidfuzz-3.10.1-cp313-cp313-musllinux_1_2_s390x.whl", hash = "sha256:e8e154b84a311263e1aca86818c962e1fa9eefdd643d1d5d197fcd2738f88cb9"},
+ {file = "rapidfuzz-3.10.1-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:335fee93188f8cd585552bb8057228ce0111bd227fa81bfd40b7df6b75def8ab"},
+ {file = "rapidfuzz-3.10.1-cp313-cp313-win32.whl", hash = "sha256:6729b856166a9e95c278410f73683957ea6100c8a9d0a8dbe434c49663689255"},
+ {file = "rapidfuzz-3.10.1-cp313-cp313-win_amd64.whl", hash = "sha256:0e06d99ad1ad97cb2ef7f51ec6b1fedd74a3a700e4949353871cf331d07b382a"},
+ {file = "rapidfuzz-3.10.1-cp313-cp313-win_arm64.whl", hash = "sha256:8d1b7082104d596a3eb012e0549b2634ed15015b569f48879701e9d8db959dbb"},
+ {file = "rapidfuzz-3.10.1-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:779027d3307e1a2b1dc0c03c34df87a470a368a1a0840a9d2908baf2d4067956"},
+ {file = "rapidfuzz-3.10.1-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:440b5608ab12650d0390128d6858bc839ae77ffe5edf0b33a1551f2fa9860651"},
+ {file = "rapidfuzz-3.10.1-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:82cac41a411e07a6f3dc80dfbd33f6be70ea0abd72e99c59310819d09f07d945"},
+ {file = "rapidfuzz-3.10.1-cp39-cp39-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:958473c9f0bca250590200fd520b75be0dbdbc4a7327dc87a55b6d7dc8d68552"},
+ {file = "rapidfuzz-3.10.1-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:9ef60dfa73749ef91cb6073be1a3e135f4846ec809cc115f3cbfc6fe283a5584"},
+ {file = "rapidfuzz-3.10.1-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:a7fbac18f2c19fc983838a60611e67e3262e36859994c26f2ee85bb268de2355"},
+ {file = "rapidfuzz-3.10.1-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:9a0d519ff39db887cd73f4e297922786d548f5c05d6b51f4e6754f452a7f4296"},
+ {file = "rapidfuzz-3.10.1-cp39-cp39-musllinux_1_2_aarch64.whl", hash = "sha256:bebb7bc6aeb91cc57e4881b222484c26759ca865794187217c9dcea6c33adae6"},
+ {file = "rapidfuzz-3.10.1-cp39-cp39-musllinux_1_2_i686.whl", hash = "sha256:fe07f8b9c3bb5c5ad1d2c66884253e03800f4189a60eb6acd6119ebaf3eb9894"},
+ {file = "rapidfuzz-3.10.1-cp39-cp39-musllinux_1_2_ppc64le.whl", hash = "sha256:bfa48a4a2d45a41457f0840c48e579db157a927f4e97acf6e20df8fc521c79de"},
+ {file = "rapidfuzz-3.10.1-cp39-cp39-musllinux_1_2_s390x.whl", hash = "sha256:2cf44d01bfe8ee605b7eaeecbc2b9ca64fc55765f17b304b40ed8995f69d7716"},
+ {file = "rapidfuzz-3.10.1-cp39-cp39-musllinux_1_2_x86_64.whl", hash = "sha256:1e6bbca9246d9eedaa1c84e04a7f555493ba324d52ae4d9f3d9ddd1b740dcd87"},
+ {file = "rapidfuzz-3.10.1-cp39-cp39-win32.whl", hash = "sha256:567f88180f2c1423b4fe3f3ad6e6310fc97b85bdba574801548597287fc07028"},
+ {file = "rapidfuzz-3.10.1-cp39-cp39-win_amd64.whl", hash = "sha256:6b2cd7c29d6ecdf0b780deb587198f13213ac01c430ada6913452fd0c40190fc"},
+ {file = "rapidfuzz-3.10.1-cp39-cp39-win_arm64.whl", hash = "sha256:9f912d459e46607ce276128f52bea21ebc3e9a5ccf4cccfef30dd5bddcf47be8"},
+ {file = "rapidfuzz-3.10.1-pp310-pypy310_pp73-macosx_10_15_x86_64.whl", hash = "sha256:ac4452f182243cfab30ba4668ef2de101effaedc30f9faabb06a095a8c90fd16"},
+ {file = "rapidfuzz-3.10.1-pp310-pypy310_pp73-macosx_11_0_arm64.whl", hash = "sha256:565c2bd4f7d23c32834652b27b51dd711814ab614b4e12add8476be4e20d1cf5"},
+ {file = "rapidfuzz-3.10.1-pp310-pypy310_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:187d9747149321607be4ccd6f9f366730078bed806178ec3eeb31d05545e9e8f"},
+ {file = "rapidfuzz-3.10.1-pp310-pypy310_pp73-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:616290fb9a8fa87e48cb0326d26f98d4e29f17c3b762c2d586f2b35c1fd2034b"},
+ {file = "rapidfuzz-3.10.1-pp310-pypy310_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:073a5b107e17ebd264198b78614c0206fa438cce749692af5bc5f8f484883f50"},
+ {file = "rapidfuzz-3.10.1-pp310-pypy310_pp73-win_amd64.whl", hash = "sha256:39c4983e2e2ccb9732f3ac7d81617088822f4a12291d416b09b8a1eadebb3e29"},
+ {file = "rapidfuzz-3.10.1-pp39-pypy39_pp73-macosx_10_15_x86_64.whl", hash = "sha256:ac7adee6bcf0c6fee495d877edad1540a7e0f5fc208da03ccb64734b43522d7a"},
+ {file = "rapidfuzz-3.10.1-pp39-pypy39_pp73-macosx_11_0_arm64.whl", hash = "sha256:425f4ac80b22153d391ee3f94bc854668a0c6c129f05cf2eaf5ee74474ddb69e"},
+ {file = "rapidfuzz-3.10.1-pp39-pypy39_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:65a2fa13e8a219f9b5dcb9e74abe3ced5838a7327e629f426d333dfc8c5a6e66"},
+ {file = "rapidfuzz-3.10.1-pp39-pypy39_pp73-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:75561f3df9a906aaa23787e9992b228b1ab69007932dc42070f747103e177ba8"},
+ {file = "rapidfuzz-3.10.1-pp39-pypy39_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:edd062490537e97ca125bc6c7f2b7331c2b73d21dc304615afe61ad1691e15d5"},
+ {file = "rapidfuzz-3.10.1-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:cfcc8feccf63245a22dfdd16e222f1a39771a44b870beb748117a0e09cbb4a62"},
+ {file = "rapidfuzz-3.10.1.tar.gz", hash = "sha256:5a15546d847a915b3f42dc79ef9b0c78b998b4e2c53b252e7166284066585979"},
]
[package.extras]
-full = ["numpy"]
+all = ["numpy"]
[[package]]
name = "requests"
diff --git a/src/poetry/installation/installer.py b/src/poetry/installation/installer.py
index eff9aeffdd5..8a243e15b45 100644
--- a/src/poetry/installation/installer.py
+++ b/src/poetry/installation/installer.py
@@ -305,6 +305,7 @@ def _do_install(self) -> int:
self._installed_repository.packages,
locked_repository.packages,
NullIO(),
+ active_root_extras=self._extras,
)
# Everything is resolved at this point, so we no longer need
# to load deferred dependencies (i.e. VCS, URL and path dependencies)
diff --git a/src/poetry/puzzle/provider.py b/src/poetry/puzzle/provider.py
index 9bae818bc56..b3b29e5f6c6 100644
--- a/src/poetry/puzzle/provider.py
+++ b/src/poetry/puzzle/provider.py
@@ -8,6 +8,7 @@
from collections import defaultdict
from contextlib import contextmanager
from typing import TYPE_CHECKING
+from typing import Any
from typing import ClassVar
from typing import cast
@@ -17,6 +18,7 @@
from poetry.core.constraints.version import VersionRange
from poetry.core.packages.utils.utils import get_python_constraint_from_marker
from poetry.core.version.markers import AnyMarker
+from poetry.core.version.markers import parse_marker
from poetry.core.version.markers import union as marker_union
from poetry.mixology.incompatibility import Incompatibility
@@ -115,6 +117,7 @@ def __init__(
io: IO,
*,
locked: list[Package] | None = None,
+ active_root_extras: Collection[NormalizedName] | None = None,
) -> None:
self._package = package
self._pool = pool
@@ -130,6 +133,9 @@ def __init__(
self._direct_origin_packages: dict[str, Package] = {}
self._locked: dict[NormalizedName, list[DependencyPackage]] = defaultdict(list)
self._use_latest: Collection[NormalizedName] = []
+ self._active_root_extras = (
+ frozenset(active_root_extras) if active_root_extras is not None else None
+ )
self._explicit_sources: dict[str, str] = {}
for package in locked or []:
@@ -416,21 +422,12 @@ def incompatibilities_for(
)
]
- _dependencies = [
- dep
- for dep in dependencies
- if dep.name not in self.UNSAFE_PACKAGES
- and self._python_constraint.allows_any(dep.python_constraint)
- and (not self._env or dep.marker.validate(self._env.marker_env))
- ]
- dependencies = self._get_dependencies_with_overrides(_dependencies, package)
-
return [
Incompatibility(
[Term(package.to_dependency(), True), Term(dep, False)],
DependencyCauseError(),
)
- for dep in dependencies
+ for dep in self._get_dependencies_with_overrides(dependencies, package)
]
def complete_package(
@@ -480,7 +477,7 @@ def complete_package(
package = dependency_package.package
dependency = dependency_package.dependency
new_dependency = package.without_features().to_dependency()
- new_dependency.marker = AnyMarker()
+ new_dependency.marker = dependency.marker
# When adding dependency foo[extra] -> foo, preserve foo's source, if it's
# specified. This prevents us from trying to get foo from PyPI
@@ -497,8 +494,14 @@ def complete_package(
if dep.name in self.UNSAFE_PACKAGES:
continue
- if self._env and not dep.marker.validate(self._env.marker_env):
- continue
+ if self._env:
+ marker_values = (
+ self._marker_values(self._active_root_extras)
+ if package.is_root()
+ else self._env.marker_env
+ )
+ if not dep.marker.validate(marker_values):
+ continue
if not package.is_root() and (
(dep.is_optional() and dep.name not in optional_dependencies)
@@ -509,6 +512,24 @@ def complete_package(
):
continue
+ # For normal dependency resolution, we have to make sure that root extras
+ # are represented in the markers. This is required to identify mutually
+ # exclusive markers in cases like 'extra == "foo"' and 'extra != "foo"'.
+ # However, for installation with re-resolving (installer.re-resolve=true,
+ # which results in self._env being not None), this spoils the result
+ # because we have to keep extras so that they are uninstalled
+ # when calculating the operations of the transaction.
+ if self._env is None and package.is_root() and dep.in_extras:
+ # The clone is required for installation with re-resolving
+ # without an existing lock file because the root package is used
+ # once for solving and a second time for re-resolving for installation.
+ dep = dep.clone()
+ dep.marker = dep.marker.intersect(
+ parse_marker(
+ " or ".join(f'extra == "{extra}"' for extra in dep.in_extras)
+ )
+ )
+
_dependencies.append(dep)
if self._load_deferred:
@@ -545,7 +566,7 @@ def complete_package(
# • pypiwin32 (219); sys_platform == "win32" and python_version < "3.6"
duplicates: dict[str, list[Dependency]] = defaultdict(list)
for dep in dependencies:
- duplicates[dep.complete_name].append(dep)
+ duplicates[dep.name].append(dep)
dependencies = []
for dep_name, deps in duplicates.items():
@@ -556,9 +577,39 @@ def complete_package(
self.debug(f"Duplicate dependencies for {dep_name}")
# For dependency resolution, markers of duplicate dependencies must be
- # mutually exclusive.
- active_extras = None if package.is_root() else dependency.extras
- deps = self._resolve_overlapping_markers(package, deps, active_extras)
+ # mutually exclusive. However, we have to take care about duplicates
+ # with differing extras.
+ duplicates_by_extras: dict[str, list[Dependency]] = defaultdict(list)
+ for dep in deps:
+ duplicates_by_extras[dep.complete_name].append(dep)
+
+ if len(duplicates_by_extras) == 1:
+ active_extras = (
+ self._active_root_extras if package.is_root() else dependency.extras
+ )
+ deps = self._resolve_overlapping_markers(package, deps, active_extras)
+ else:
+ # There are duplicates with different extras.
+ for complete_dep_name, deps_by_extra in duplicates_by_extras.items():
+ if len(deps_by_extra) > 1:
+ duplicates_by_extras[complete_dep_name] = (
+ self._resolve_overlapping_markers(package, deps, None)
+ )
+ if all(len(d) == 1 for d in duplicates_by_extras.values()) and all(
+ d1[0].marker.intersect(d2[0].marker).is_empty()
+ for d1, d2 in itertools.combinations(
+ duplicates_by_extras.values(), 2
+ )
+ ):
+ # Since all markers are mutually exclusive,
+ # we can trigger overrides.
+ deps = list(itertools.chain(*duplicates_by_extras.values()))
+ else:
+ # Too complicated to handle with overrides,
+ # fallback to basic handling without overrides.
+ for d in duplicates_by_extras.values():
+ dependencies.extend(d)
+ continue
if len(deps) == 1:
self.debug(f"Merging requirements for {dep_name}")
@@ -909,3 +960,19 @@ def _resolve_overlapping_markers(
# dependencies by constraint again. After overlapping markers were
# resolved, there might be new dependencies with the same constraint.
return self._merge_dependencies_by_constraint(new_dependencies)
+
+ def _marker_values(
+ self, extras: Collection[NormalizedName] | None = None
+ ) -> dict[str, Any]:
+ """
+ Marker values, from `self._env` if present plus the supplied extras
+
+ :param extras: the values to add to the 'extra' marker value
+ """
+ result = self._env.marker_env.copy() if self._env is not None else {}
+ if extras is not None:
+ assert (
+ "extra" not in result
+ ), "'extra' marker key is already present in environment"
+ result["extra"] = set(extras)
+ return result
diff --git a/src/poetry/puzzle/solver.py b/src/poetry/puzzle/solver.py
index e32d000df98..234518fc52e 100644
--- a/src/poetry/puzzle/solver.py
+++ b/src/poetry/puzzle/solver.py
@@ -49,6 +49,7 @@ def __init__(
installed: list[Package],
locked: list[Package],
io: IO,
+ active_root_extras: Collection[NormalizedName] | None = None,
) -> None:
self._package = package
self._pool = pool
@@ -56,7 +57,13 @@ def __init__(
self._locked_packages = locked
self._io = io
- self._provider = Provider(self._package, self._pool, self._io, locked=locked)
+ self._provider = Provider(
+ self._package,
+ self._pool,
+ self._io,
+ locked=locked,
+ active_root_extras=active_root_extras,
+ )
self._overrides: list[dict[Package, dict[str, Dependency]]] = []
@property
diff --git a/src/poetry/puzzle/transaction.py b/src/poetry/puzzle/transaction.py
index 05a9432818f..338e43d8b5c 100644
--- a/src/poetry/puzzle/transaction.py
+++ b/src/poetry/puzzle/transaction.py
@@ -78,7 +78,7 @@ def calculate_operations(
else:
priorities = defaultdict(int)
relevant_result_packages: set[NormalizedName] = set()
- uninstalls: set[NormalizedName] = set()
+ pending_extra_uninstalls: list[Package] = [] # list for deterministic order
for result_package in self._result_packages:
is_unsolicited_extra = False
if self._marker_env:
@@ -95,11 +95,12 @@ def calculate_operations(
else:
continue
else:
- relevant_result_packages.add(result_package.name)
is_unsolicited_extra = extras is not None and (
result_package.optional
and result_package.name not in extra_packages
)
+ if not is_unsolicited_extra:
+ relevant_result_packages.add(result_package.name)
installed = False
for installed_package in self._installed_packages:
@@ -108,9 +109,7 @@ def calculate_operations(
# Extras that were not requested are always uninstalled.
if is_unsolicited_extra:
- uninstalls.add(installed_package.name)
- if installed_package.name not in system_site_packages:
- operations.append(Uninstall(installed_package))
+ pending_extra_uninstalls.append(installed_package)
# We have to perform an update if the version or another
# attribute of the package has changed (source type, url, ref, ...).
@@ -153,6 +152,13 @@ def calculate_operations(
op.skip("Not required")
operations.append(op)
+ uninstalls: set[NormalizedName] = set()
+ for package in pending_extra_uninstalls:
+ if package.name not in (relevant_result_packages | uninstalls):
+ uninstalls.add(package.name)
+ if package.name not in system_site_packages:
+ operations.append(Uninstall(package))
+
if with_uninstalls:
for current_package in self._current_packages:
found = current_package.name in (relevant_result_packages | uninstalls)
diff --git a/tests/installation/fixtures/with-conflicting-dependency-extras-root.test b/tests/installation/fixtures/with-conflicting-dependency-extras-root.test
new file mode 100644
index 00000000000..c99c37a5694
--- /dev/null
+++ b/tests/installation/fixtures/with-conflicting-dependency-extras-root.test
@@ -0,0 +1,28 @@
+[[package]]
+name = "conflicting-dep"
+version = "1.1.0"
+description = ""
+optional = true
+python-versions = "*"
+files = [ ]
+groups = [ "main" ]
+markers = "extra == \"extra-one\" and extra != \"extra-two\""
+
+[[package]]
+name = "conflicting-dep"
+version = "1.2.0"
+description = ""
+optional = true
+python-versions = "*"
+files = [ ]
+groups = [ "main" ]
+markers = "extra != \"extra-one\" and extra == \"extra-two\""
+
+[extras]
+extra-one = [ "conflicting-dep", "conflicting-dep" ]
+extra-two = [ "conflicting-dep", "conflicting-dep" ]
+
+[metadata]
+lock-version = "2.1"
+python-versions = "*"
+content-hash = "123456789"
diff --git a/tests/installation/fixtures/with-conflicting-dependency-extras-transitive.test b/tests/installation/fixtures/with-conflicting-dependency-extras-transitive.test
new file mode 100644
index 00000000000..5e6f11ab7f9
--- /dev/null
+++ b/tests/installation/fixtures/with-conflicting-dependency-extras-transitive.test
@@ -0,0 +1,52 @@
+[[package]]
+name = "conflicting-dep"
+version = "1.1.0"
+description = ""
+optional = true
+python-versions = "*"
+files = [ ]
+groups = [ "main" ]
+markers = "extra == \"root-extra-one\" and extra != \"root-extra-two\""
+
+[[package]]
+name = "conflicting-dep"
+version = "1.2.0"
+description = ""
+optional = true
+python-versions = "*"
+files = [ ]
+groups = [ "main" ]
+markers = "extra != \"root-extra-one\" and extra == \"root-extra-two\""
+
+[[package]]
+name = "intermediate-dep"
+version = "1.0.0"
+description = ""
+optional = true
+python-versions = "*"
+files = [ ]
+groups = [ "main" ]
+markers = "extra == \"root-extra-one\" and extra != \"root-extra-two\" or extra == \"root-extra-two\" and extra != \"root-extra-one\""
+
+[[package.dependencies.conflicting-dep]]
+version = "1.1.0"
+optional = true
+markers = 'extra == "extra-one" and extra != "extra-two"'
+
+[[package.dependencies.conflicting-dep]]
+version = "1.2.0"
+optional = true
+markers = 'extra != "extra-one" and extra == "extra-two"'
+
+ [package.extras]
+ extra-one = [ "conflicting-dep (==1.1.0)", "conflicting-dep (==1.2.0)" ]
+ extra-two = [ "conflicting-dep (==1.1.0)", "conflicting-dep (==1.2.0)" ]
+
+[extras]
+root-extra-one = [ "intermediate-dep", "intermediate-dep" ]
+root-extra-two = [ "intermediate-dep", "intermediate-dep" ]
+
+[metadata]
+lock-version = "2.1"
+python-versions = "*"
+content-hash = "123456789"
diff --git a/tests/installation/fixtures/with-dependencies-differing-extras.test b/tests/installation/fixtures/with-dependencies-differing-extras.test
new file mode 100644
index 00000000000..7dcaacf26ae
--- /dev/null
+++ b/tests/installation/fixtures/with-dependencies-differing-extras.test
@@ -0,0 +1,52 @@
+[[package]]
+name = "demo"
+version = "1.0.0"
+description = ""
+optional = true
+python-versions = "*"
+files = [ ]
+groups = [ "main" ]
+markers = "extra == \"extra-one\" and extra != \"extra-two\" or extra == \"extra-two\" and extra != \"extra-one\""
+
+[package.dependencies.transitive-dep-one]
+version = "1.1.0"
+optional = true
+markers = 'extra == "demo-extra-one" and extra != "demo-extra-two"'
+
+[package.dependencies.transitive-dep-two]
+version = "1.2.0"
+optional = true
+markers = 'extra != "demo-extra-one" and extra == "demo-extra-two"'
+
+ [package.extras]
+ demo-extra-one = [ "transitive-dep-one", "transitive-dep-two" ]
+ demo-extra-two = [ "transitive-dep-one", "transitive-dep-two" ]
+
+[[package]]
+name = "transitive-dep-one"
+version = "1.1.0"
+description = ""
+optional = true
+python-versions = "*"
+files = [ ]
+groups = [ "main" ]
+markers = "extra == \"extra-one\" and extra != \"extra-two\""
+
+[[package]]
+name = "transitive-dep-two"
+version = "1.2.0"
+description = ""
+optional = true
+python-versions = "*"
+files = [ ]
+groups = [ "main" ]
+markers = "extra != \"extra-one\" and extra == \"extra-two\""
+
+[extras]
+extra-one = [ "demo", "demo" ]
+extra-two = [ "demo", "demo" ]
+
+[metadata]
+lock-version = "2.1"
+python-versions = "*"
+content-hash = "123456789"
diff --git a/tests/installation/fixtures/with-exclusive-extras.test b/tests/installation/fixtures/with-exclusive-extras.test
new file mode 100644
index 00000000000..c4764e4a8af
--- /dev/null
+++ b/tests/installation/fixtures/with-exclusive-extras.test
@@ -0,0 +1,38 @@
+[[package]]
+name = "torch"
+version = "1.11.0+cpu"
+description = ""
+optional = true
+python-versions = "*"
+files = []
+groups = [ "main" ]
+markers = "extra == \"cpu\" and extra != \"cuda\""
+
+[package.source]
+reference = "pytorch-cpu"
+type = "legacy"
+url = "https://download.pytorch.org/whl/cpu"
+
+[[package]]
+name = "torch"
+version = "1.11.0+cuda"
+description = ""
+optional = true
+python-versions = "*"
+files = []
+groups = [ "main" ]
+markers = "extra != \"cpu\" and extra == \"cuda\""
+
+[package.source]
+reference = "pytorch-cuda"
+type = "legacy"
+url = "https://download.pytorch.org/whl/cuda"
+
+[extras]
+cpu = ["torch", "torch"]
+cuda = ["torch", "torch"]
+
+[metadata]
+python-versions = "*"
+lock-version = "2.1"
+content-hash = "123456789"
diff --git a/tests/installation/test_installer.py b/tests/installation/test_installer.py
index f4b81757c82..a8e3b726c36 100644
--- a/tests/installation/test_installer.py
+++ b/tests/installation/test_installer.py
@@ -1077,6 +1077,358 @@ def test_run_with_dependencies_nested_extras(
assert locker.written_data == expected
+@pytest.mark.parametrize("root", [True, False])
+@pytest.mark.parametrize("locked", [False, True])
+@pytest.mark.parametrize("extra", [None, "extra-one", "extra-two"])
+def test_run_with_conflicting_dependency_extras(
+ installer: Installer,
+ pool: RepositoryPool,
+ locker: Locker,
+ installed: CustomInstalledRepository,
+ repo: Repository,
+ config: Config,
+ package: ProjectPackage,
+ extra: str | None,
+ locked: bool,
+ root: bool,
+) -> None:
+ """
+ - https://github.com/python-poetry/poetry/issues/6419
+
+ Tests resolution of extras with conflicting dependencies. Tests in both as direct dependencies of
+ root package and as transitive dependencies.
+ """
+ # A package with two optional dependencies, one for each extra
+ # If root, this is the root package, otherwise an intermediate package
+ main_package = package if root else get_package("intermediate-dep", "1.0.0")
+
+ # Two conflicting versions of a dependency, one in each extra
+ conflicting_dep_one_pkg = get_package("conflicting-dep", "1.1.0")
+ conflicting_dep_two_pkg = get_package("conflicting-dep", "1.2.0")
+
+ conflicting_dep_one = Factory.create_dependency(
+ "conflicting-dep",
+ {
+ "version": "1.1.0",
+ "markers": "extra == 'extra-one' and extra != 'extra-two'",
+ "optional": True,
+ },
+ )
+ conflicting_dep_two = Factory.create_dependency(
+ "conflicting-dep",
+ {
+ "version": "1.2.0",
+ "markers": "extra != 'extra-one' and extra == 'extra-two'",
+ "optional": True,
+ },
+ )
+
+ # Include both just for extra validation that our marker validation works as expected
+ main_package.extras = {
+ canonicalize_name("extra-one"): [conflicting_dep_one, conflicting_dep_two],
+ canonicalize_name("extra-two"): [conflicting_dep_one, conflicting_dep_two],
+ }
+ main_package.add_dependency(conflicting_dep_one)
+ main_package.add_dependency(conflicting_dep_two)
+
+ repo.add_package(conflicting_dep_one_pkg)
+ repo.add_package(conflicting_dep_two_pkg)
+ if not root:
+ repo.add_package(main_package)
+
+ # If we have an intermediate package, add extras to our root package
+ if not root:
+ extra_one_dep = Factory.create_dependency(
+ "intermediate-dep",
+ {
+ "version": "1.0.0",
+ "markers": "extra == 'root-extra-one' and extra != 'root-extra-two'",
+ "extras": ["extra-one"],
+ "optional": True,
+ },
+ )
+ extra_two_dep = Factory.create_dependency(
+ "intermediate-dep",
+ {
+ "version": "1.0.0",
+ "markers": "extra != 'root-extra-one' and extra == 'root-extra-two'",
+ "extras": ["extra-two"],
+ "optional": True,
+ },
+ )
+ package.add_dependency(extra_one_dep)
+ package.add_dependency(extra_two_dep)
+ # Include both just for extra validation that our marker validation works as expected
+ package.extras = {
+ canonicalize_name("root-extra-one"): [extra_one_dep, extra_two_dep],
+ canonicalize_name("root-extra-two"): [extra_one_dep, extra_two_dep],
+ }
+
+ fixture_name = "with-conflicting-dependency-extras-" + (
+ "root" if root else "transitive"
+ )
+ locker.locked(locked)
+ if locked:
+ locker.mock_lock_data(dict(fixture(fixture_name)))
+
+ if extra is not None:
+ extras = [f"root-{extra}"] if not root else [extra]
+ installer.extras(extras)
+ result = installer.run()
+ assert result == 0
+
+ if not locked:
+ expected = fixture(fixture_name)
+ assert locker.written_data == expected
+
+ # Results of installation are consistent with the 'extra' input
+ assert isinstance(installer.executor, Executor)
+
+ expected_installations = []
+ if extra == "extra-one":
+ expected_installations.append(conflicting_dep_one_pkg)
+ elif extra == "extra-two":
+ expected_installations.append(conflicting_dep_two_pkg)
+ if not root and extra is not None:
+ expected_installations.append(get_package("intermediate-dep", "1.0.0"))
+
+ assert len(installer.executor.installations) == len(expected_installations)
+ assert set(installer.executor.installations) == set(expected_installations)
+
+
+@pytest.mark.parametrize("locked", [True, False])
+@pytest.mark.parametrize("extra", [None, "cpu", "cuda"])
+def test_run_with_exclusive_extras_different_sources(
+ installer: Installer,
+ locker: Locker,
+ installed: CustomInstalledRepository,
+ config: Config,
+ package: ProjectPackage,
+ extra: str | None,
+ locked: bool,
+) -> None:
+ """
+ - https://github.com/python-poetry/poetry/issues/6409
+ - https://github.com/python-poetry/poetry/issues/6419
+ - https://github.com/python-poetry/poetry/issues/7748
+ - https://github.com/python-poetry/poetry/issues/9537
+ """
+ # Setup repo for each of our sources
+ cpu_repo = Repository("pytorch-cpu")
+ cuda_repo = Repository("pytorch-cuda")
+ pool = RepositoryPool()
+ pool.add_repository(cpu_repo)
+ pool.add_repository(cuda_repo)
+ config.config["repositories"] = {
+ "pytorch-cpu": {"url": "https://download.pytorch.org/whl/cpu"},
+ "pytorch-cuda": {"url": "https://download.pytorch.org/whl/cuda"},
+ }
+
+ # Configure packages that read from each of the different sources
+ torch_cpu_pkg = get_package("torch", "1.11.0+cpu")
+ torch_cpu_pkg._source_reference = "pytorch-cpu"
+ torch_cpu_pkg._source_type = "legacy"
+ torch_cpu_pkg._source_url = "https://download.pytorch.org/whl/cpu"
+ torch_cuda_pkg = get_package("torch", "1.11.0+cuda")
+ torch_cuda_pkg._source_reference = "pytorch-cuda"
+ torch_cuda_pkg._source_type = "legacy"
+ torch_cuda_pkg._source_url = "https://download.pytorch.org/whl/cuda"
+ cpu_repo.add_package(torch_cpu_pkg)
+ cuda_repo.add_package(torch_cuda_pkg)
+
+ # Depend on each package based on exclusive extras
+ torch_cpu_dep = Factory.create_dependency(
+ "torch",
+ {
+ "version": "1.11.0+cpu",
+ "markers": "extra == 'cpu' and extra != 'cuda'",
+ "source": "pytorch-cpu",
+ },
+ )
+ torch_cuda_dep = Factory.create_dependency(
+ "torch",
+ {
+ "version": "1.11.0+cuda",
+ "markers": "extra != 'cpu' and extra == 'cuda'",
+ "source": "pytorch-cuda",
+ },
+ )
+ package.add_dependency(torch_cpu_dep)
+ package.add_dependency(torch_cuda_dep)
+ # We don't want to cheat by only including the correct dependency in the 'extra' mapping
+ package.extras = {
+ canonicalize_name("cpu"): [torch_cpu_dep, torch_cuda_dep],
+ canonicalize_name("cuda"): [torch_cpu_dep, torch_cuda_dep],
+ }
+
+ # Set locker state
+ locker.locked(locked)
+ if locked:
+ locker.mock_lock_data(dict(fixture("with-exclusive-extras")))
+
+ # Perform install
+ installer = Installer(
+ NullIO(),
+ MockEnv(),
+ package,
+ locker,
+ pool,
+ config,
+ installed=installed,
+ executor=Executor(
+ MockEnv(),
+ pool,
+ config,
+ NullIO(),
+ ),
+ )
+ if extra is not None:
+ installer.extras([extra])
+ result = installer.run()
+ assert result == 0
+
+ # Results of locking are expected and installation are consistent with the 'extra' input
+ if not locked:
+ expected = fixture("with-exclusive-extras")
+ assert locker.written_data == expected
+ assert isinstance(installer.executor, Executor)
+ if extra is None:
+ assert len(installer.executor.installations) == 0
+ else:
+ assert len(installer.executor.installations) == 1
+ version = f"1.11.0+{extra}"
+ source_url = f"https://download.pytorch.org/whl/{extra}"
+ source_reference = f"pytorch-{extra}"
+ assert installer.executor.installations[0] == Package(
+ "torch",
+ version,
+ source_type="legacy",
+ source_url=source_url,
+ source_reference=source_reference,
+ )
+
+
+@pytest.mark.parametrize("locked", [True, False])
+@pytest.mark.parametrize("extra", [None, "extra-one", "extra-two"])
+def test_run_with_different_dependency_extras(
+ installer: Installer,
+ pool: RepositoryPool,
+ locker: Locker,
+ installed: CustomInstalledRepository,
+ repo: Repository,
+ config: Config,
+ package: ProjectPackage,
+ extra: str | None,
+ locked: bool,
+) -> None:
+ """
+ - https://github.com/python-poetry/poetry/issues/834
+ - https://github.com/python-poetry/poetry/issues/7748
+
+ This tests different sets of extras in a dependency of the root project. These different dependency extras are
+ themselves conditioned on extras in the root project.
+ """
+ # Three packages in addition to root: demo (direct dependency) and two transitive dep packages
+ demo_pkg = get_package("demo", "1.0.0")
+ transitive_one_pkg = get_package("transitive-dep-one", "1.1.0")
+ transitive_two_pkg = get_package("transitive-dep-two", "1.2.0")
+
+ # Switch each transitive dependency based on extra markers in the 'demo' package
+ transitive_dep_one = Factory.create_dependency(
+ "transitive-dep-one",
+ {
+ "version": "1.1.0",
+ "markers": "extra == 'demo-extra-one' and extra != 'demo-extra-two'",
+ "optional": True,
+ },
+ )
+ transitive_dep_two = Factory.create_dependency(
+ "transitive-dep-two",
+ {
+ "version": "1.2.0",
+ "markers": "extra != 'demo-extra-one' and extra == 'demo-extra-two'",
+ "optional": True,
+ },
+ )
+ # Include both packages in both demo extras, to validate that they're filtered out based on extra markers alone
+ demo_pkg.extras = {
+ canonicalize_name("demo-extra-one"): [
+ get_dependency("transitive-dep-one"),
+ get_dependency("transitive-dep-two"),
+ ],
+ canonicalize_name("demo-extra-two"): [
+ get_dependency("transitive-dep-one"),
+ get_dependency("transitive-dep-two"),
+ ],
+ }
+ demo_pkg.add_dependency(transitive_dep_one)
+ demo_pkg.add_dependency(transitive_dep_two)
+
+ # Now define the demo dependency, similarly switched on extra markers in the root package
+ extra_one_dep = Factory.create_dependency(
+ "demo",
+ {
+ "version": "1.0.0",
+ "markers": "extra == 'extra-one' and extra != 'extra-two'",
+ "extras": ["demo-extra-one"],
+ },
+ )
+ extra_two_dep = Factory.create_dependency(
+ "demo",
+ {
+ "version": "1.0.0",
+ "markers": "extra != 'extra-one' and extra == 'extra-two'",
+ "extras": ["demo-extra-two"],
+ },
+ )
+ package.add_dependency(extra_one_dep)
+ package.add_dependency(extra_two_dep)
+ # Again we don't want to cheat by only including the correct dependency in the 'extra' mapping
+ package.extras = {
+ canonicalize_name("extra-one"): [extra_one_dep, extra_two_dep],
+ canonicalize_name("extra-two"): [extra_one_dep, extra_two_dep],
+ }
+
+ repo.add_package(demo_pkg)
+ repo.add_package(transitive_one_pkg)
+ repo.add_package(transitive_two_pkg)
+
+ locker.locked(locked)
+ if locked:
+ locker.mock_lock_data(dict(fixture("with-dependencies-differing-extras")))
+
+ installer = Installer(
+ NullIO(),
+ MockEnv(),
+ package,
+ locker,
+ pool,
+ config,
+ installed=installed,
+ executor=Executor(
+ MockEnv(),
+ pool,
+ config,
+ NullIO(),
+ ),
+ )
+ if extra is not None:
+ installer.extras([extra])
+ result = installer.run()
+ assert result == 0
+
+ if not locked:
+ expected = fixture("with-dependencies-differing-extras")
+ assert locker.written_data == expected
+
+ # Results of installation are consistent with the 'extra' input
+ assert isinstance(installer.executor, Executor)
+ if extra is None:
+ assert len(installer.executor.installations) == 0
+ else:
+ assert len(installer.executor.installations) == 2
+
+
@pytest.mark.parametrize("is_locked", [False, True])
@pytest.mark.parametrize("is_installed", [False, True])
@pytest.mark.parametrize("with_extras", [False, True])
diff --git a/tests/puzzle/test_solver.py b/tests/puzzle/test_solver.py
index a51a520dc33..a566e5c588e 100644
--- a/tests/puzzle/test_solver.py
+++ b/tests/puzzle/test_solver.py
@@ -4755,6 +4755,136 @@ def test_solver_resolves_duplicate_dependency_in_extra(
)
+def test_solver_resolves_conflicting_dependency_in_root_extra(
+ package: ProjectPackage,
+ pool: RepositoryPool,
+ repo: Repository,
+ io: NullIO,
+) -> None:
+ package_a1 = get_package("A", "1.0")
+ package_a2 = get_package("A", "2.0")
+
+ dep = get_dependency("A", {"version": "1.0", "markers": "extra != 'foo'"})
+ package.add_dependency(dep)
+
+ dep_extra = get_dependency("A", "2.0", optional=True)
+ dep_extra._in_extras = [canonicalize_name("foo")]
+ package.extras = {canonicalize_name("foo"): [dep_extra]}
+ package.add_dependency(dep_extra)
+
+ repo.add_package(package_a1)
+ repo.add_package(package_a2)
+
+ solver = Solver(package, pool, [], [], io)
+ transaction = solver.solve()
+
+ check_solver_result(
+ transaction,
+ (
+ [
+ {"job": "install", "package": package_a1},
+ {"job": "install", "package": package_a2},
+ ]
+ ),
+ )
+ solved_packages = transaction.get_solved_packages()
+ assert solved_packages[package_a1].markers["main"] == parse_marker("extra != 'foo'")
+ assert solved_packages[package_a2].markers["main"] == parse_marker("extra == 'foo'")
+
+
+def test_solver_resolves_conflicting_dependency_in_root_extras(
+ package: ProjectPackage,
+ pool: RepositoryPool,
+ repo: Repository,
+ io: NullIO,
+) -> None:
+ package_a1 = get_package("A", "1.0")
+ package_a2 = get_package("A", "2.0")
+
+ dep_extra1 = get_dependency( # extra == 'foo' is implicit via _in_extras!
+ "A", {"version": "1.0", "markers": "extra != 'bar'"}, optional=True
+ )
+ dep_extra1._in_extras = [canonicalize_name("foo")]
+ package.add_dependency(dep_extra1)
+
+ dep_extra2 = get_dependency( # extra == 'bar' is implicit via _in_extras!
+ "A", {"version": "2.0", "markers": "extra != 'foo'"}, optional=True
+ )
+ dep_extra2._in_extras = [canonicalize_name("bar")]
+ package.extras = {
+ canonicalize_name("foo"): [dep_extra1],
+ canonicalize_name("bar"): [dep_extra2],
+ }
+ package.add_dependency(dep_extra1)
+ package.add_dependency(dep_extra2)
+
+ repo.add_package(package_a1)
+ repo.add_package(package_a2)
+
+ solver = Solver(package, pool, [], [], io)
+ transaction = solver.solve()
+
+ check_solver_result(
+ transaction,
+ (
+ [
+ {"job": "install", "package": package_a1},
+ {"job": "install", "package": package_a2},
+ ]
+ ),
+ )
+ solved_packages = transaction.get_solved_packages()
+ assert solved_packages[package_a1].markers["main"] == parse_marker(
+ "extra != 'bar' and extra == 'foo'"
+ )
+ assert solved_packages[package_a2].markers["main"] == parse_marker(
+ "extra != 'foo' and extra == 'bar'"
+ )
+
+
+@pytest.mark.parametrize("with_extra", [False, True])
+def test_solver_resolves_duplicate_dependency_in_root_extra_for_installation(
+ package: ProjectPackage,
+ pool: RepositoryPool,
+ repo: Repository,
+ io: NullIO,
+ with_extra: bool,
+) -> None:
+ """
+ Without extras, a newer version of A can be chosen than with root extras.
+ """
+ extra = [canonicalize_name("foo")] if with_extra else []
+
+ package_a1 = get_package("A", "1.0")
+ package_a2 = get_package("A", "2.0")
+
+ dep = get_dependency("A", ">=1.0")
+ package.add_dependency(dep)
+
+ dep_extra = get_dependency("A", "^1.0", optional=True)
+ dep_extra.marker = parse_marker("extra == 'foo'")
+ package.extras = {canonicalize_name("foo"): [dep_extra]}
+ package.add_dependency(dep_extra)
+
+ repo.add_package(package_a1)
+ repo.add_package(package_a2)
+
+ solver = Solver(
+ package, pool, [], [package_a1, package_a2], io, active_root_extras=extra
+ )
+ with solver.use_environment(MockEnv()):
+ transaction = solver.solve()
+
+ check_solver_result(
+ transaction,
+ (
+ [
+ {"job": "install", "package": package_a1 if with_extra else package_a2},
+ ]
+ ),
+ )
+
+
def test_solver_resolves_duplicate_dependencies_with_restricted_extras(
package: ProjectPackage,
pool: RepositoryPool,
diff --git a/tests/puzzle/test_transaction.py b/tests/puzzle/test_transaction.py
index a68c981d159..2dcea971e79 100644
--- a/tests/puzzle/test_transaction.py
+++ b/tests/puzzle/test_transaction.py
@@ -429,3 +429,55 @@ def test_calculate_operations_extras(
),
ops,
)
+
+
+@pytest.mark.parametrize("extra", ["", "foo", "bar"])
+def test_calculate_operations_extras_no_redundant_uninstall(extra: str) -> None:
+ extra1 = canonicalize_name("foo")
+ extra2 = canonicalize_name("bar")
+ package = ProjectPackage("root", "1.0")
+ dep_a1 = Dependency("a", "1", optional=True)
+ dep_a1._in_extras = [canonicalize_name("foo")]
+ dep_a1.marker = parse_marker("extra != 'bar'")
+ dep_a2 = Dependency("a", "2", optional=True)
+ dep_a2._in_extras = [canonicalize_name("bar")]
+ dep_a2.marker = parse_marker("extra != 'foo'")
+ package.add_dependency(dep_a1)
+ package.add_dependency(dep_a2)
+ package.extras = {extra1: [dep_a1], extra2: [dep_a2]}
+ opt_a1 = Package("a", "1")
+ opt_a1.optional = True
+ opt_a2 = Package("a", "2")
+ opt_a2.optional = True
+
+ transaction = Transaction(
+ [Package("a", "1")],
+ {
+ opt_a1: TransitivePackageInfo(
+ 0, {"main"}, {"main": parse_marker("extra == 'foo' and extra != 'bar'")}
+ ),
+ opt_a2: TransitivePackageInfo(
+ 0, {"main"}, {"main": parse_marker("extra == 'bar' and extra != 'foo'")}
+ ),
+ },
+ [Package("a", "1")],
+ package,
+ {"python_version": "3.9"},
+ {"main"},
+ )
+
+ if not extra:
+ ops = [{"job": "remove", "package": Package("a", "1")}]
+ elif extra == "foo":
+ ops = [{"job": "install", "package": Package("a", "1"), "skipped": True}]
+ elif extra == "bar":
+ ops = [{"job": "update", "from": Package("a", "1"), "to": Package("a", "2")}]
+ else:
+ raise NotImplementedError
+
+ check_operations(
+ transaction.calculate_operations(
+ extras=set() if not extra else {canonicalize_name(extra)},
+ ),
+ ops,
+ )