mirror of
https://github.com/enpaul/tox-poetry-installer.git
synced 2025-10-28 07:00:43 +00:00
Compare commits
20 Commits
079aae0556
...
devel
| Author | SHA1 | Date | |
|---|---|---|---|
| 536fd6536b | |||
|
bbb075a1de
|
|||
|
198287a633
|
|||
|
4b38b00f81
|
|||
|
863f88d63c
|
|||
|
d816678975
|
|||
|
552f2080f5
|
|||
|
ddbf442a30
|
|||
|
13cfb8616c
|
|||
|
df343396a4
|
|||
|
f66e59ab85
|
|||
|
f37463d172
|
|||
|
6837a64121
|
|||
|
506aae0ccd
|
|||
|
5c4d861230
|
|||
|
f3ae242cf7
|
|||
|
57787c6206
|
|||
|
661072a69f
|
|||
|
0a46b2d876
|
|||
|
c06cdfe8c2
|
1
.github/scripts/setup-env.sh
vendored
1
.github/scripts/setup-env.sh
vendored
@@ -26,7 +26,6 @@ poetry --version --no-ansi;
|
|||||||
poetry run pip --version;
|
poetry run pip --version;
|
||||||
|
|
||||||
poetry install \
|
poetry install \
|
||||||
--extras poetry \
|
|
||||||
--quiet \
|
--quiet \
|
||||||
--remove-untracked \
|
--remove-untracked \
|
||||||
--no-ansi;
|
--no-ansi;
|
||||||
|
|||||||
4
.github/workflows/ci.yaml
vendored
4
.github/workflows/ci.yaml
vendored
@@ -12,8 +12,6 @@ jobs:
|
|||||||
strategy:
|
strategy:
|
||||||
matrix:
|
matrix:
|
||||||
python:
|
python:
|
||||||
- version: "3.7"
|
|
||||||
toxenv: py37
|
|
||||||
- version: "3.8"
|
- version: "3.8"
|
||||||
toxenv: py38
|
toxenv: py38
|
||||||
- version: "3.9"
|
- version: "3.9"
|
||||||
@@ -22,6 +20,8 @@ jobs:
|
|||||||
toxenv: py310
|
toxenv: py310
|
||||||
- version: "3.11"
|
- version: "3.11"
|
||||||
toxenv: py311
|
toxenv: py311
|
||||||
|
- version: "3.12"
|
||||||
|
toxenv: py312
|
||||||
fail-fast: true
|
fail-fast: true
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout
|
- name: Checkout
|
||||||
|
|||||||
@@ -47,12 +47,13 @@ repos:
|
|||||||
|
|
||||||
- id: reorder-python-imports
|
- id: reorder-python-imports
|
||||||
name: reorder-python-imports
|
name: reorder-python-imports
|
||||||
entry: reorder-python-imports
|
entry: isort
|
||||||
language: system
|
language: system
|
||||||
args:
|
require_serial: true
|
||||||
- "--unclassifiable-application-module=tox_poetry_installer"
|
|
||||||
types:
|
types:
|
||||||
- python
|
- python
|
||||||
|
args:
|
||||||
|
- "--filter-files"
|
||||||
|
|
||||||
- id: black
|
- id: black
|
||||||
name: black
|
name: black
|
||||||
|
|||||||
@@ -11,7 +11,6 @@
|
|||||||
# --disable=W"
|
# --disable=W"
|
||||||
disable=logging-fstring-interpolation
|
disable=logging-fstring-interpolation
|
||||||
,logging-format-interpolation
|
,logging-format-interpolation
|
||||||
,bad-continuation
|
|
||||||
,line-too-long
|
,line-too-long
|
||||||
,ungrouped-imports
|
,ungrouped-imports
|
||||||
,typecheck
|
,typecheck
|
||||||
|
|||||||
@@ -227,7 +227,6 @@ error will be set to one of the "Status" values below to indicate what the error
|
|||||||
| `LockedDepNotFoundError` | Indicates that an item specified in the `locked_deps` config option does not match the name of a package in the Poetry lockfile. |
|
| `LockedDepNotFoundError` | Indicates that an item specified in the `locked_deps` config option does not match the name of a package in the Poetry lockfile. |
|
||||||
| `LockedDepsRequiredError` | Indicates that a test environment with the `require_locked_deps` config option set to `true` also specified unlocked dependencies using the [`deps`](https://tox.readthedocs.io/en/latest/config.html#conf-deps) config option. |
|
| `LockedDepsRequiredError` | Indicates that a test environment with the `require_locked_deps` config option set to `true` also specified unlocked dependencies using the [`deps`](https://tox.readthedocs.io/en/latest/config.html#conf-deps) config option. |
|
||||||
| `PoetryNotInstalledError` | Indicates that the `poetry` module could not be imported under the current runtime environment, and `require_poetry = true` was specified. |
|
| `PoetryNotInstalledError` | Indicates that the `poetry` module could not be imported under the current runtime environment, and `require_poetry = true` was specified. |
|
||||||
| `RequiresUnsafeDepError` | Indicates that the package-under-test depends on a package that Poetry has classified as unsafe and cannot be installed. |
|
|
||||||
|
|
||||||
> ℹ️ **Note:** One or more of these errors can be caused by the `pyproject.toml` being out
|
> ℹ️ **Note:** One or more of these errors can be caused by the `pyproject.toml` being out
|
||||||
> of sync with the Poetry lockfile. If this is the case, than a warning will be logged
|
> of sync with the Poetry lockfile. If this is the case, than a warning will be logged
|
||||||
|
|||||||
2793
poetry.lock
generated
2793
poetry.lock
generated
File diff suppressed because it is too large
Load Diff
@@ -23,45 +23,46 @@ classifiers = [
|
|||||||
"Natural Language :: English",
|
"Natural Language :: English",
|
||||||
"Operating System :: OS Independent",
|
"Operating System :: OS Independent",
|
||||||
"Programming Language :: Python :: 3",
|
"Programming Language :: Python :: 3",
|
||||||
"Programming Language :: Python :: 3.7",
|
|
||||||
"Programming Language :: Python :: 3.8",
|
"Programming Language :: Python :: 3.8",
|
||||||
"Programming Language :: Python :: 3.9",
|
"Programming Language :: Python :: 3.9",
|
||||||
"Programming Language :: Python :: 3.10",
|
"Programming Language :: Python :: 3.10",
|
||||||
"Programming Language :: Python :: 3.11",
|
"Programming Language :: Python :: 3.11",
|
||||||
|
"Programming Language :: Python :: 3.12",
|
||||||
"Programming Language :: Python :: Implementation :: CPython",
|
"Programming Language :: Python :: Implementation :: CPython",
|
||||||
]
|
]
|
||||||
|
|
||||||
[tool.poetry.plugins.tox]
|
[tool.poetry.plugins.tox]
|
||||||
poetry_installer = "tox_poetry_installer"
|
poetry_installer = "tox_poetry_installer"
|
||||||
|
|
||||||
[tool.poetry.extras]
|
|
||||||
poetry = ["poetry", "cleo"]
|
|
||||||
|
|
||||||
[tool.poetry.dependencies]
|
[tool.poetry.dependencies]
|
||||||
python = "^3.7"
|
python = "^3.8"
|
||||||
cleo = {version = ">=1.0,<3.0", optional = true}
|
cleo = ">=1.0,<3.0"
|
||||||
poetry = {version = "^1.5.0", optional = true}
|
poetry = "^1.5.0"
|
||||||
poetry-core = "^1.1.0"
|
poetry-core = "^1.1.0"
|
||||||
tox = "^4"
|
tox = "^4.1"
|
||||||
|
|
||||||
[tool.poetry.group.dev.dependencies]
|
[tool.poetry.group.dev.dependencies]
|
||||||
bandit = "^1.6.2"
|
bandit = {version = "^1.7.7", python = "^3.10"}
|
||||||
black = "^22.3.0"
|
black = {version = "^24.3.0", python = "^3.10"}
|
||||||
blacken-docs = "^1.8.0"
|
blacken-docs = {version = "^1.18.0", python = "^3.10"}
|
||||||
ipython = {version = "^8.10.1", python = "^3.8"}
|
ipython = {version = "^8.10.1", python = "^3.10"}
|
||||||
mdformat = "^0.7"
|
isort = {version = "^5.13.2", python = "^3.10"}
|
||||||
mdformat-gfm = "^0.3"
|
mdformat = {version = "^0.7", python = "^3.10"}
|
||||||
mypy = "^0.930"
|
mdformat-gfm = {version = "^0.3", python = "^3.10"}
|
||||||
pre-commit = "^2.7.1"
|
mypy = {version = "^1.11.1", python = "^3.10"}
|
||||||
pre-commit-hooks = "^3.3.0"
|
pre-commit = {version = "^3.8.0", python = "^3.10"}
|
||||||
pylint = "^2.13.0"
|
pre-commit-hooks = {version = "^4.6.0", python = "^3.10"}
|
||||||
pytest = "^6.0.2"
|
pylint = {version = "^3.2.6", python = "^3.10"}
|
||||||
pytest-cov = "^2.10.1"
|
pytest = {version = "^8.3.2", python = "^3.10"}
|
||||||
reorder-python-imports = "^2.3.5"
|
pytest-cov = {version = "^5.0.0", python = "^3.10"}
|
||||||
safety = "^2.2.0"
|
toml = {version = "^0.10.1", python = "^3.10"}
|
||||||
toml = "^0.10.1"
|
tox = "^4.1"
|
||||||
tox = "^4"
|
types-toml = {version = "^0.10.1", python = "^3.10"}
|
||||||
types-toml = "^0.10.1"
|
|
||||||
|
[tool.isort]
|
||||||
|
profile = "black"
|
||||||
|
force_single_line = "true"
|
||||||
|
lines_after_imports = 2
|
||||||
|
|
||||||
[build-system]
|
[build-system]
|
||||||
requires = ["poetry-core>=1.1.0"]
|
requires = ["poetry-core>=1.1.0"]
|
||||||
|
|||||||
@@ -1,16 +1,16 @@
|
|||||||
# pylint: disable=missing-module-docstring, missing-function-docstring, unused-argument, too-few-public-methods
|
# pylint: disable=missing-module-docstring,missing-function-docstring,unused-argument,too-few-public-methods,protected-access
|
||||||
import time
|
import time
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
from typing import List
|
from typing import List
|
||||||
|
|
||||||
import poetry.factory
|
import poetry.factory
|
||||||
import poetry.installation.executor
|
import poetry.installation.executor
|
||||||
|
import poetry.installation.operations.operation
|
||||||
import poetry.utils.env
|
import poetry.utils.env
|
||||||
import pytest
|
import pytest
|
||||||
import tox.tox_env.python.virtual_env.runner
|
import tox.tox_env.python.virtual_env.runner
|
||||||
from poetry.installation.operations.operation import Operation
|
|
||||||
|
|
||||||
from tox_poetry_installer import utilities
|
import tox_poetry_installer.hooks._tox_on_install_helpers
|
||||||
|
|
||||||
|
|
||||||
TEST_PROJECT_PATH = Path(__file__).parent.resolve() / "test-project"
|
TEST_PROJECT_PATH = Path(__file__).parent.resolve() / "test-project"
|
||||||
@@ -40,14 +40,20 @@ class MockExecutor:
|
|||||||
def __init__(self, env: MockVirtualEnv, **kwargs):
|
def __init__(self, env: MockVirtualEnv, **kwargs):
|
||||||
self.env = env
|
self.env = env
|
||||||
|
|
||||||
def execute(self, operations: List[Operation]):
|
def execute(
|
||||||
|
self, operations: List[poetry.installation.operations.operation.Operation]
|
||||||
|
):
|
||||||
self.env.installed.extend([operation.package for operation in operations])
|
self.env.installed.extend([operation.package for operation in operations])
|
||||||
time.sleep(1)
|
time.sleep(1)
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture
|
@pytest.fixture
|
||||||
def mock_venv(monkeypatch):
|
def mock_venv(monkeypatch):
|
||||||
monkeypatch.setattr(utilities, "convert_virtualenv", lambda venv: venv)
|
monkeypatch.setattr(
|
||||||
|
tox_poetry_installer.hooks._tox_on_install_helpers,
|
||||||
|
"convert_virtualenv",
|
||||||
|
lambda venv: venv,
|
||||||
|
)
|
||||||
monkeypatch.setattr(poetry.installation.executor, "Executor", MockExecutor)
|
monkeypatch.setattr(poetry.installation.executor, "Executor", MockExecutor)
|
||||||
monkeypatch.setattr(
|
monkeypatch.setattr(
|
||||||
tox.tox_env.python.virtual_env.runner, "VirtualEnvRunner", MockVirtualEnv
|
tox.tox_env.python.virtual_env.runner, "VirtualEnvRunner", MockVirtualEnv
|
||||||
@@ -57,9 +63,9 @@ def mock_venv(monkeypatch):
|
|||||||
|
|
||||||
@pytest.fixture(scope="function")
|
@pytest.fixture(scope="function")
|
||||||
def mock_poetry_factory(monkeypatch):
|
def mock_poetry_factory(monkeypatch):
|
||||||
pypoetry = poetry.factory.Factory().create_poetry(cwd=TEST_PROJECT_PATH)
|
project = poetry.factory.Factory().create_poetry(cwd=TEST_PROJECT_PATH)
|
||||||
|
|
||||||
def mock_factory(*args, **kwargs):
|
def mock_factory(*args, **kwargs):
|
||||||
return pypoetry
|
return project
|
||||||
|
|
||||||
monkeypatch.setattr(poetry.factory.Factory, "create_poetry", mock_factory)
|
monkeypatch.setattr(poetry.factory.Factory, "create_poetry", mock_factory)
|
||||||
|
|||||||
525
tests/test-project/poetry.lock
generated
525
tests/test-project/poetry.lock
generated
@@ -1,109 +1,80 @@
|
|||||||
# This file is automatically @generated by Poetry 1.8.3 and should not be changed by hand.
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "appdirs"
|
name = "appdirs"
|
||||||
version = "1.4.4"
|
version = "1.4.4"
|
||||||
description = "A small Python module for determining appropriate platform-specific dirs, e.g. a \"user data dir\"."
|
description = "A small Python module for determining appropriate platform-specific dirs, e.g. a \"user data dir\"."
|
||||||
|
category = "main"
|
||||||
optional = false
|
optional = false
|
||||||
python-versions = "*"
|
python-versions = "*"
|
||||||
files = [
|
|
||||||
{file = "appdirs-1.4.4-py2.py3-none-any.whl", hash = "sha256:a841dacd6b99318a741b166adb07e19ee71a274450e68237b4650ca1055ab128"},
|
|
||||||
{file = "appdirs-1.4.4.tar.gz", hash = "sha256:7d5d0167b2b1ba821647616af46a749d1c653740dd0d2415100fe26e27afdf41"},
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "attrs"
|
name = "attrs"
|
||||||
version = "20.3.0"
|
version = "20.3.0"
|
||||||
description = "Classes Without Boilerplate"
|
description = "Classes Without Boilerplate"
|
||||||
|
category = "main"
|
||||||
optional = false
|
optional = false
|
||||||
python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*"
|
python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*"
|
||||||
files = [
|
|
||||||
{file = "attrs-20.3.0-py2.py3-none-any.whl", hash = "sha256:31b2eced602aa8423c2aea9c76a724617ed67cf9513173fd3a4f03e3a929c7e6"},
|
|
||||||
{file = "attrs-20.3.0.tar.gz", hash = "sha256:832aa3cde19744e49938b91fea06d69ecb9e649c93ba974535d08ad92164f700"},
|
|
||||||
]
|
|
||||||
|
|
||||||
[package.extras]
|
[package.extras]
|
||||||
dev = ["coverage[toml] (>=5.0.2)", "furo", "hypothesis", "pre-commit", "pympler", "pytest (>=4.3.0)", "six", "sphinx", "zope.interface"]
|
dev = ["coverage[toml] (>=5.0.2)", "hypothesis", "pympler", "pytest (>=4.3.0)", "six", "zope.interface", "furo", "sphinx", "pre-commit"]
|
||||||
docs = ["furo", "sphinx", "zope.interface"]
|
docs = ["furo", "sphinx", "zope.interface"]
|
||||||
tests = ["coverage[toml] (>=5.0.2)", "hypothesis", "pympler", "pytest (>=4.3.0)", "six", "zope.interface"]
|
tests = ["coverage[toml] (>=5.0.2)", "hypothesis", "pympler", "pytest (>=4.3.0)", "six", "zope.interface"]
|
||||||
tests-no-zope = ["coverage[toml] (>=5.0.2)", "hypothesis", "pympler", "pytest (>=4.3.0)", "six"]
|
tests_no_zope = ["coverage[toml] (>=5.0.2)", "hypothesis", "pympler", "pytest (>=4.3.0)", "six"]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "certifi"
|
name = "certifi"
|
||||||
version = "2020.12.5"
|
version = "2020.12.5"
|
||||||
description = "Python package for providing Mozilla's CA Bundle."
|
description = "Python package for providing Mozilla's CA Bundle."
|
||||||
|
category = "main"
|
||||||
optional = false
|
optional = false
|
||||||
python-versions = "*"
|
python-versions = "*"
|
||||||
files = [
|
|
||||||
{file = "certifi-2020.12.5-py2.py3-none-any.whl", hash = "sha256:719a74fb9e33b9bd44cc7f3a8d94bc35e4049deebe19ba7d8e108280cfd59830"},
|
|
||||||
{file = "certifi-2020.12.5.tar.gz", hash = "sha256:1a4995114262bffbc2413b159f2a1a480c969de6e6eb13ee966d470af86af59c"},
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "chardet"
|
name = "chardet"
|
||||||
version = "4.0.0"
|
version = "4.0.0"
|
||||||
description = "Universal encoding detector for Python 2 and 3"
|
description = "Universal encoding detector for Python 2 and 3"
|
||||||
|
category = "main"
|
||||||
optional = false
|
optional = false
|
||||||
python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*"
|
python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*"
|
||||||
files = [
|
|
||||||
{file = "chardet-4.0.0-py2.py3-none-any.whl", hash = "sha256:f864054d66fd9118f2e67044ac8981a54775ec5b67aed0441892edb553d21da5"},
|
|
||||||
{file = "chardet-4.0.0.tar.gz", hash = "sha256:0d6f53a15db4120f2b08c94f11e7d93d2c911ee118b6b30a04ec3ee8310179fa"},
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "click"
|
name = "click"
|
||||||
version = "7.1.2"
|
version = "7.1.2"
|
||||||
description = "Composable command line interface toolkit"
|
description = "Composable command line interface toolkit"
|
||||||
|
category = "main"
|
||||||
optional = false
|
optional = false
|
||||||
python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*"
|
python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*"
|
||||||
files = [
|
|
||||||
{file = "click-7.1.2-py2.py3-none-any.whl", hash = "sha256:dacca89f4bfadd5de3d7489b7c8a566eee0d3676333fbb50030263894c38c0dc"},
|
|
||||||
{file = "click-7.1.2.tar.gz", hash = "sha256:d2b5255c7c6349bc1bd1e59e08cd12acbbd63ce649f2588755783aa94dfb6b1a"},
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "colorama"
|
name = "colorama"
|
||||||
version = "0.4.4"
|
version = "0.4.4"
|
||||||
description = "Cross-platform colored terminal text."
|
description = "Cross-platform colored terminal text."
|
||||||
|
category = "main"
|
||||||
optional = false
|
optional = false
|
||||||
python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*"
|
python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*"
|
||||||
files = [
|
|
||||||
{file = "colorama-0.4.4-py2.py3-none-any.whl", hash = "sha256:9f47eda37229f68eee03b24b9748937c7dc3868f906e8ba69fbcbdd3bc5dc3e2"},
|
|
||||||
{file = "colorama-0.4.4.tar.gz", hash = "sha256:5941b2b48a20143d2267e95b1c2a7603ce057ee39fd88e7329b0c292aa16869b"},
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "distlib"
|
name = "distlib"
|
||||||
version = "0.3.1"
|
version = "0.3.1"
|
||||||
description = "Distribution utilities"
|
description = "Distribution utilities"
|
||||||
|
category = "main"
|
||||||
optional = false
|
optional = false
|
||||||
python-versions = "*"
|
python-versions = "*"
|
||||||
files = [
|
|
||||||
{file = "distlib-0.3.1-py2.py3-none-any.whl", hash = "sha256:8c09de2c67b3e7deef7184574fc060ab8a793e7adbb183d942c389c8b13c52fb"},
|
|
||||||
{file = "distlib-0.3.1.zip", hash = "sha256:edf6116872c863e1aa9d5bb7cb5e05a022c519a4594dc703843343a9ddd9bff1"},
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "filelock"
|
name = "filelock"
|
||||||
version = "3.0.12"
|
version = "3.0.12"
|
||||||
description = "A platform independent file lock."
|
description = "A platform independent file lock."
|
||||||
|
category = "main"
|
||||||
optional = false
|
optional = false
|
||||||
python-versions = "*"
|
python-versions = "*"
|
||||||
files = [
|
|
||||||
{file = "filelock-3.0.12-py3-none-any.whl", hash = "sha256:929b7d63ec5b7d6b71b0fa5ac14e030b3f70b75747cef1b10da9b879fef15836"},
|
|
||||||
{file = "filelock-3.0.12.tar.gz", hash = "sha256:18d82244ee114f543149c66a6e0c14e9c4f8a1044b5cdaadd0f82159d6a6ff59"},
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "flask"
|
name = "flask"
|
||||||
version = "1.1.2"
|
version = "1.1.2"
|
||||||
description = "A simple framework for building complex web applications."
|
description = "A simple framework for building complex web applications."
|
||||||
|
category = "main"
|
||||||
optional = false
|
optional = false
|
||||||
python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*"
|
python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*"
|
||||||
files = [
|
|
||||||
{file = "Flask-1.1.2-py2.py3-none-any.whl", hash = "sha256:8a4fdd8936eba2512e9c85df320a37e694c93945b33ef33c89946a340a238557"},
|
|
||||||
{file = "Flask-1.1.2.tar.gz", hash = "sha256:4efa1ae2d7c9865af48986de8aeb8504bf32c7f3d6fdc9353d34b21f4b127060"},
|
|
||||||
]
|
|
||||||
|
|
||||||
[package.dependencies]
|
[package.dependencies]
|
||||||
click = ">=5.1"
|
click = ">=5.1"
|
||||||
@@ -112,79 +83,64 @@ Jinja2 = ">=2.10.1"
|
|||||||
Werkzeug = ">=0.15"
|
Werkzeug = ">=0.15"
|
||||||
|
|
||||||
[package.extras]
|
[package.extras]
|
||||||
dev = ["coverage", "pallets-sphinx-themes", "pytest", "sphinx", "sphinx-issues", "sphinxcontrib-log-cabinet", "tox"]
|
dev = ["pytest", "coverage", "tox", "sphinx", "pallets-sphinx-themes", "sphinxcontrib-log-cabinet", "sphinx-issues"]
|
||||||
docs = ["pallets-sphinx-themes", "sphinx", "sphinx-issues", "sphinxcontrib-log-cabinet"]
|
docs = ["sphinx", "pallets-sphinx-themes", "sphinxcontrib-log-cabinet", "sphinx-issues"]
|
||||||
dotenv = ["python-dotenv"]
|
dotenv = ["python-dotenv"]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "idna"
|
name = "idna"
|
||||||
version = "2.10"
|
version = "2.10"
|
||||||
description = "Internationalized Domain Names in Applications (IDNA)"
|
description = "Internationalized Domain Names in Applications (IDNA)"
|
||||||
|
category = "main"
|
||||||
optional = false
|
optional = false
|
||||||
python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*"
|
python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*"
|
||||||
files = [
|
|
||||||
{file = "idna-2.10-py2.py3-none-any.whl", hash = "sha256:b97d804b1e9b523befed77c48dacec60e6dcb0b5391d57af6a65a312a90648c0"},
|
|
||||||
{file = "idna-2.10.tar.gz", hash = "sha256:b307872f855b18632ce0c21c5e45be78c0ea7ae4c15c828c20788b26921eb3f6"},
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "importlib-metadata"
|
name = "importlib-metadata"
|
||||||
version = "3.10.1"
|
version = "3.10.1"
|
||||||
description = "Read metadata from Python packages"
|
description = "Read metadata from Python packages"
|
||||||
|
category = "main"
|
||||||
optional = false
|
optional = false
|
||||||
python-versions = ">=3.6"
|
python-versions = ">=3.6"
|
||||||
files = [
|
|
||||||
{file = "importlib_metadata-3.10.1-py3-none-any.whl", hash = "sha256:2ec0faae539743ae6aaa84b49a169670a465f7f5d64e6add98388cc29fd1f2f6"},
|
|
||||||
{file = "importlib_metadata-3.10.1.tar.gz", hash = "sha256:c9356b657de65c53744046fa8f7358afe0714a1af7d570c00c3835c2d724a7c1"},
|
|
||||||
]
|
|
||||||
|
|
||||||
[package.dependencies]
|
[package.dependencies]
|
||||||
typing-extensions = {version = ">=3.6.4", markers = "python_version < \"3.8\""}
|
typing-extensions = {version = ">=3.6.4", markers = "python_version < \"3.8\""}
|
||||||
zipp = ">=0.5"
|
zipp = ">=0.5"
|
||||||
|
|
||||||
[package.extras]
|
[package.extras]
|
||||||
docs = ["jaraco.packaging (>=8.2)", "rst.linker (>=1.9)", "sphinx"]
|
docs = ["sphinx", "jaraco.packaging (>=8.2)", "rst.linker (>=1.9)"]
|
||||||
testing = ["flufl.flake8", "importlib-resources (>=1.3)", "packaging", "pep517", "pyfakefs", "pytest (>=4.6)", "pytest-black (>=0.3.7)", "pytest-checkdocs (>=2.4)", "pytest-cov", "pytest-enabler (>=1.0.1)", "pytest-flake8", "pytest-mypy"]
|
testing = ["pytest (>=4.6)", "pytest-checkdocs (>=2.4)", "pytest-flake8", "pytest-cov", "pytest-enabler (>=1.0.1)", "packaging", "pep517", "pyfakefs", "flufl.flake8", "pytest-black (>=0.3.7)", "pytest-mypy", "importlib-resources (>=1.3)"]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "importlib-resources"
|
name = "importlib-resources"
|
||||||
version = "5.1.2"
|
version = "5.1.2"
|
||||||
description = "Read resources from Python packages"
|
description = "Read resources from Python packages"
|
||||||
|
category = "main"
|
||||||
optional = false
|
optional = false
|
||||||
python-versions = ">=3.6"
|
python-versions = ">=3.6"
|
||||||
files = [
|
|
||||||
{file = "importlib_resources-5.1.2-py3-none-any.whl", hash = "sha256:ebab3efe74d83b04d6bf5cd9a17f0c5c93e60fb60f30c90f56265fce4682a469"},
|
|
||||||
{file = "importlib_resources-5.1.2.tar.gz", hash = "sha256:642586fc4740bd1cad7690f836b3321309402b20b332529f25617ff18e8e1370"},
|
|
||||||
]
|
|
||||||
|
|
||||||
[package.dependencies]
|
[package.dependencies]
|
||||||
zipp = {version = ">=0.4", markers = "python_version < \"3.8\""}
|
zipp = {version = ">=0.4", markers = "python_version < \"3.8\""}
|
||||||
|
|
||||||
[package.extras]
|
[package.extras]
|
||||||
docs = ["jaraco.packaging (>=8.2)", "rst.linker (>=1.9)", "sphinx"]
|
docs = ["sphinx", "jaraco.packaging (>=8.2)", "rst.linker (>=1.9)"]
|
||||||
testing = ["pytest (>=4.6)", "pytest-black (>=0.3.7)", "pytest-checkdocs (>=1.2.3)", "pytest-cov", "pytest-enabler", "pytest-flake8", "pytest-mypy"]
|
testing = ["pytest (>=4.6)", "pytest-checkdocs (>=1.2.3)", "pytest-flake8", "pytest-cov", "pytest-enabler", "pytest-black (>=0.3.7)", "pytest-mypy"]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "itsdangerous"
|
name = "itsdangerous"
|
||||||
version = "1.1.0"
|
version = "1.1.0"
|
||||||
description = "Various helpers to pass data to untrusted environments and back."
|
description = "Various helpers to pass data to untrusted environments and back."
|
||||||
|
category = "main"
|
||||||
optional = false
|
optional = false
|
||||||
python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*"
|
python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*"
|
||||||
files = [
|
|
||||||
{file = "itsdangerous-1.1.0-py2.py3-none-any.whl", hash = "sha256:b12271b2047cb23eeb98c8b5622e2e5c5e9abd9784a153e9d8ef9cb4dd09d749"},
|
|
||||||
{file = "itsdangerous-1.1.0.tar.gz", hash = "sha256:321b033d07f2a4136d3ec762eac9f16a10ccd60f53c0c91af90217ace7ba1f19"},
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "jinja2"
|
name = "jinja2"
|
||||||
version = "2.11.3"
|
version = "2.11.3"
|
||||||
description = "A very fast and expressive template engine."
|
description = "A very fast and expressive template engine."
|
||||||
|
category = "main"
|
||||||
optional = false
|
optional = false
|
||||||
python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*"
|
python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*"
|
||||||
files = [
|
|
||||||
{file = "Jinja2-2.11.3-py2.py3-none-any.whl", hash = "sha256:03e47ad063331dd6a3f04a43eddca8a966a26ba0c5b7207a9a9e4e08f1b29419"},
|
|
||||||
{file = "Jinja2-2.11.3.tar.gz", hash = "sha256:a6d58433de0ae800347cab1fa3043cebbabe8baa9d29e668f1c768cb87a333c6"},
|
|
||||||
]
|
|
||||||
|
|
||||||
[package.dependencies]
|
[package.dependencies]
|
||||||
MarkupSafe = ">=0.23"
|
MarkupSafe = ">=0.23"
|
||||||
@@ -196,9 +152,247 @@ i18n = ["Babel (>=0.8)"]
|
|||||||
name = "markupsafe"
|
name = "markupsafe"
|
||||||
version = "1.1.1"
|
version = "1.1.1"
|
||||||
description = "Safely add untrusted strings to HTML/XML markup."
|
description = "Safely add untrusted strings to HTML/XML markup."
|
||||||
|
category = "main"
|
||||||
optional = false
|
optional = false
|
||||||
python-versions = ">=2.7,!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*"
|
python-versions = ">=2.7,!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*"
|
||||||
files = [
|
|
||||||
|
[[package]]
|
||||||
|
name = "packaging"
|
||||||
|
version = "20.9"
|
||||||
|
description = "Core utilities for Python packages"
|
||||||
|
category = "main"
|
||||||
|
optional = false
|
||||||
|
python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*"
|
||||||
|
|
||||||
|
[package.dependencies]
|
||||||
|
pyparsing = ">=2.0.2"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "pluggy"
|
||||||
|
version = "0.13.1"
|
||||||
|
description = "plugin and hook calling mechanisms for python"
|
||||||
|
category = "main"
|
||||||
|
optional = false
|
||||||
|
python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*"
|
||||||
|
|
||||||
|
[package.dependencies]
|
||||||
|
importlib-metadata = {version = ">=0.12", markers = "python_version < \"3.8\""}
|
||||||
|
|
||||||
|
[package.extras]
|
||||||
|
dev = ["pre-commit", "tox"]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "py"
|
||||||
|
version = "1.10.0"
|
||||||
|
description = "library with cross-python path, ini-parsing, io, code, log facilities"
|
||||||
|
category = "main"
|
||||||
|
optional = false
|
||||||
|
python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "pyparsing"
|
||||||
|
version = "2.4.7"
|
||||||
|
description = "Python parsing module"
|
||||||
|
category = "main"
|
||||||
|
optional = false
|
||||||
|
python-versions = ">=2.6, !=3.0.*, !=3.1.*, !=3.2.*"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "python-dateutil"
|
||||||
|
version = "2.8.1"
|
||||||
|
description = "Extensions to the standard Python datetime module"
|
||||||
|
category = "main"
|
||||||
|
optional = false
|
||||||
|
python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,>=2.7"
|
||||||
|
|
||||||
|
[package.dependencies]
|
||||||
|
six = ">=1.5"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "requests"
|
||||||
|
version = "2.25.1"
|
||||||
|
description = "Python HTTP for Humans."
|
||||||
|
category = "main"
|
||||||
|
optional = false
|
||||||
|
python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*"
|
||||||
|
|
||||||
|
[package.dependencies]
|
||||||
|
certifi = ">=2017.4.17"
|
||||||
|
chardet = ">=3.0.2,<5"
|
||||||
|
idna = ">=2.5,<3"
|
||||||
|
urllib3 = ">=1.21.1,<1.27"
|
||||||
|
|
||||||
|
[package.extras]
|
||||||
|
security = ["pyOpenSSL (>=0.14)", "cryptography (>=1.3.4)"]
|
||||||
|
socks = ["PySocks (>=1.5.6,!=1.5.7)", "win-inet-pton"]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "six"
|
||||||
|
version = "1.15.0"
|
||||||
|
description = "Python 2 and 3 compatibility utilities"
|
||||||
|
category = "main"
|
||||||
|
optional = false
|
||||||
|
python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "toml"
|
||||||
|
version = "0.10.2"
|
||||||
|
description = "Python Library for Tom's Obvious, Minimal Language"
|
||||||
|
category = "main"
|
||||||
|
optional = false
|
||||||
|
python-versions = ">=2.6, !=3.0.*, !=3.1.*, !=3.2.*"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "tox"
|
||||||
|
version = "3.23.0"
|
||||||
|
description = "tox is a generic virtualenv management and test command line tool"
|
||||||
|
category = "main"
|
||||||
|
optional = false
|
||||||
|
python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,>=2.7"
|
||||||
|
|
||||||
|
[package.dependencies]
|
||||||
|
colorama = {version = ">=0.4.1", markers = "platform_system == \"Windows\""}
|
||||||
|
filelock = ">=3.0.0"
|
||||||
|
importlib-metadata = {version = ">=0.12", markers = "python_version < \"3.8\""}
|
||||||
|
packaging = ">=14"
|
||||||
|
pluggy = ">=0.12.0"
|
||||||
|
py = ">=1.4.17"
|
||||||
|
six = ">=1.14.0"
|
||||||
|
toml = ">=0.9.4"
|
||||||
|
virtualenv = ">=16.0.0,<20.0.0 || >20.0.0,<20.0.1 || >20.0.1,<20.0.2 || >20.0.2,<20.0.3 || >20.0.3,<20.0.4 || >20.0.4,<20.0.5 || >20.0.5,<20.0.6 || >20.0.6,<20.0.7 || >20.0.7"
|
||||||
|
|
||||||
|
[package.extras]
|
||||||
|
docs = ["pygments-github-lexers (>=0.0.5)", "sphinx (>=2.0.0)", "sphinxcontrib-autoprogram (>=0.1.5)", "towncrier (>=18.5.0)"]
|
||||||
|
testing = ["flaky (>=3.4.0)", "freezegun (>=0.3.11)", "psutil (>=5.6.1)", "pytest (>=4.0.0)", "pytest-cov (>=2.5.1)", "pytest-mock (>=1.10.0)", "pytest-randomly (>=1.0.0)", "pytest-xdist (>=1.22.2)", "pathlib2 (>=2.3.3)"]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "typing-extensions"
|
||||||
|
version = "3.7.4.3"
|
||||||
|
description = "Backported and Experimental Type Hints for Python 3.5+"
|
||||||
|
category = "main"
|
||||||
|
optional = false
|
||||||
|
python-versions = "*"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "urllib3"
|
||||||
|
version = "1.26.4"
|
||||||
|
description = "HTTP library with thread-safe connection pooling, file post, and more."
|
||||||
|
category = "main"
|
||||||
|
optional = false
|
||||||
|
python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*, <4"
|
||||||
|
|
||||||
|
[package.extras]
|
||||||
|
secure = ["pyOpenSSL (>=0.14)", "cryptography (>=1.3.4)", "idna (>=2.0.0)", "certifi", "ipaddress"]
|
||||||
|
socks = ["PySocks (>=1.5.6,!=1.5.7,<2.0)"]
|
||||||
|
brotli = ["brotlipy (>=0.6.0)"]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "virtualenv"
|
||||||
|
version = "20.4.3"
|
||||||
|
description = "Virtual Python Environment builder"
|
||||||
|
category = "main"
|
||||||
|
optional = false
|
||||||
|
python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,>=2.7"
|
||||||
|
|
||||||
|
[package.dependencies]
|
||||||
|
appdirs = ">=1.4.3,<2"
|
||||||
|
distlib = ">=0.3.1,<1"
|
||||||
|
filelock = ">=3.0.0,<4"
|
||||||
|
importlib-metadata = {version = ">=0.12", markers = "python_version < \"3.8\""}
|
||||||
|
importlib-resources = {version = ">=1.0", markers = "python_version < \"3.7\""}
|
||||||
|
six = ">=1.9.0,<2"
|
||||||
|
|
||||||
|
[package.extras]
|
||||||
|
docs = ["proselint (>=0.10.2)", "sphinx (>=3)", "sphinx-argparse (>=0.2.5)", "sphinx-rtd-theme (>=0.4.3)", "towncrier (>=19.9.0rc1)"]
|
||||||
|
testing = ["coverage (>=4)", "coverage-enable-subprocess (>=1)", "flaky (>=3)", "pytest (>=4)", "pytest-env (>=0.6.2)", "pytest-freezegun (>=0.4.1)", "pytest-mock (>=2)", "pytest-randomly (>=1)", "pytest-timeout (>=1)", "packaging (>=20.0)", "xonsh (>=0.9.16)"]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "werkzeug"
|
||||||
|
version = "1.0.1"
|
||||||
|
description = "The comprehensive WSGI web application library."
|
||||||
|
category = "main"
|
||||||
|
optional = false
|
||||||
|
python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*"
|
||||||
|
|
||||||
|
[package.extras]
|
||||||
|
dev = ["pytest", "pytest-timeout", "coverage", "tox", "sphinx", "pallets-sphinx-themes", "sphinx-issues"]
|
||||||
|
watchdog = ["watchdog"]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "zipp"
|
||||||
|
version = "3.4.1"
|
||||||
|
description = "Backport of pathlib-compatible object wrapper for zip files"
|
||||||
|
category = "main"
|
||||||
|
optional = false
|
||||||
|
python-versions = ">=3.6"
|
||||||
|
|
||||||
|
[package.extras]
|
||||||
|
docs = ["sphinx", "jaraco.packaging (>=8.2)", "rst.linker (>=1.9)"]
|
||||||
|
testing = ["pytest (>=4.6)", "pytest-checkdocs (>=1.2.3)", "pytest-flake8", "pytest-cov", "pytest-enabler", "jaraco.itertools", "func-timeout", "pytest-black (>=0.3.7)", "pytest-mypy"]
|
||||||
|
|
||||||
|
[metadata]
|
||||||
|
lock-version = "1.1"
|
||||||
|
python-versions = "^3.6.1"
|
||||||
|
content-hash = "af9db950cd722e7dc52b691fb58abc1e22ab48b34ddfe4c5258b3c755a3892fa"
|
||||||
|
|
||||||
|
[metadata.files]
|
||||||
|
appdirs = [
|
||||||
|
{file = "appdirs-1.4.4-py2.py3-none-any.whl", hash = "sha256:a841dacd6b99318a741b166adb07e19ee71a274450e68237b4650ca1055ab128"},
|
||||||
|
{file = "appdirs-1.4.4.tar.gz", hash = "sha256:7d5d0167b2b1ba821647616af46a749d1c653740dd0d2415100fe26e27afdf41"},
|
||||||
|
]
|
||||||
|
attrs = [
|
||||||
|
{file = "attrs-20.3.0-py2.py3-none-any.whl", hash = "sha256:31b2eced602aa8423c2aea9c76a724617ed67cf9513173fd3a4f03e3a929c7e6"},
|
||||||
|
{file = "attrs-20.3.0.tar.gz", hash = "sha256:832aa3cde19744e49938b91fea06d69ecb9e649c93ba974535d08ad92164f700"},
|
||||||
|
]
|
||||||
|
certifi = [
|
||||||
|
{file = "certifi-2020.12.5-py2.py3-none-any.whl", hash = "sha256:719a74fb9e33b9bd44cc7f3a8d94bc35e4049deebe19ba7d8e108280cfd59830"},
|
||||||
|
{file = "certifi-2020.12.5.tar.gz", hash = "sha256:1a4995114262bffbc2413b159f2a1a480c969de6e6eb13ee966d470af86af59c"},
|
||||||
|
]
|
||||||
|
chardet = [
|
||||||
|
{file = "chardet-4.0.0-py2.py3-none-any.whl", hash = "sha256:f864054d66fd9118f2e67044ac8981a54775ec5b67aed0441892edb553d21da5"},
|
||||||
|
{file = "chardet-4.0.0.tar.gz", hash = "sha256:0d6f53a15db4120f2b08c94f11e7d93d2c911ee118b6b30a04ec3ee8310179fa"},
|
||||||
|
]
|
||||||
|
click = [
|
||||||
|
{file = "click-7.1.2-py2.py3-none-any.whl", hash = "sha256:dacca89f4bfadd5de3d7489b7c8a566eee0d3676333fbb50030263894c38c0dc"},
|
||||||
|
{file = "click-7.1.2.tar.gz", hash = "sha256:d2b5255c7c6349bc1bd1e59e08cd12acbbd63ce649f2588755783aa94dfb6b1a"},
|
||||||
|
]
|
||||||
|
colorama = [
|
||||||
|
{file = "colorama-0.4.4-py2.py3-none-any.whl", hash = "sha256:9f47eda37229f68eee03b24b9748937c7dc3868f906e8ba69fbcbdd3bc5dc3e2"},
|
||||||
|
{file = "colorama-0.4.4.tar.gz", hash = "sha256:5941b2b48a20143d2267e95b1c2a7603ce057ee39fd88e7329b0c292aa16869b"},
|
||||||
|
]
|
||||||
|
distlib = [
|
||||||
|
{file = "distlib-0.3.1-py2.py3-none-any.whl", hash = "sha256:8c09de2c67b3e7deef7184574fc060ab8a793e7adbb183d942c389c8b13c52fb"},
|
||||||
|
{file = "distlib-0.3.1.zip", hash = "sha256:edf6116872c863e1aa9d5bb7cb5e05a022c519a4594dc703843343a9ddd9bff1"},
|
||||||
|
]
|
||||||
|
filelock = [
|
||||||
|
{file = "filelock-3.0.12-py3-none-any.whl", hash = "sha256:929b7d63ec5b7d6b71b0fa5ac14e030b3f70b75747cef1b10da9b879fef15836"},
|
||||||
|
{file = "filelock-3.0.12.tar.gz", hash = "sha256:18d82244ee114f543149c66a6e0c14e9c4f8a1044b5cdaadd0f82159d6a6ff59"},
|
||||||
|
]
|
||||||
|
flask = [
|
||||||
|
{file = "Flask-1.1.2-py2.py3-none-any.whl", hash = "sha256:8a4fdd8936eba2512e9c85df320a37e694c93945b33ef33c89946a340a238557"},
|
||||||
|
{file = "Flask-1.1.2.tar.gz", hash = "sha256:4efa1ae2d7c9865af48986de8aeb8504bf32c7f3d6fdc9353d34b21f4b127060"},
|
||||||
|
]
|
||||||
|
idna = [
|
||||||
|
{file = "idna-2.10-py2.py3-none-any.whl", hash = "sha256:b97d804b1e9b523befed77c48dacec60e6dcb0b5391d57af6a65a312a90648c0"},
|
||||||
|
{file = "idna-2.10.tar.gz", hash = "sha256:b307872f855b18632ce0c21c5e45be78c0ea7ae4c15c828c20788b26921eb3f6"},
|
||||||
|
]
|
||||||
|
importlib-metadata = [
|
||||||
|
{file = "importlib_metadata-3.10.1-py3-none-any.whl", hash = "sha256:2ec0faae539743ae6aaa84b49a169670a465f7f5d64e6add98388cc29fd1f2f6"},
|
||||||
|
{file = "importlib_metadata-3.10.1.tar.gz", hash = "sha256:c9356b657de65c53744046fa8f7358afe0714a1af7d570c00c3835c2d724a7c1"},
|
||||||
|
]
|
||||||
|
importlib-resources = [
|
||||||
|
{file = "importlib_resources-5.1.2-py3-none-any.whl", hash = "sha256:ebab3efe74d83b04d6bf5cd9a17f0c5c93e60fb60f30c90f56265fce4682a469"},
|
||||||
|
{file = "importlib_resources-5.1.2.tar.gz", hash = "sha256:642586fc4740bd1cad7690f836b3321309402b20b332529f25617ff18e8e1370"},
|
||||||
|
]
|
||||||
|
itsdangerous = [
|
||||||
|
{file = "itsdangerous-1.1.0-py2.py3-none-any.whl", hash = "sha256:b12271b2047cb23eeb98c8b5622e2e5c5e9abd9784a153e9d8ef9cb4dd09d749"},
|
||||||
|
{file = "itsdangerous-1.1.0.tar.gz", hash = "sha256:321b033d07f2a4136d3ec762eac9f16a10ccd60f53c0c91af90217ace7ba1f19"},
|
||||||
|
]
|
||||||
|
jinja2 = [
|
||||||
|
{file = "Jinja2-2.11.3-py2.py3-none-any.whl", hash = "sha256:03e47ad063331dd6a3f04a43eddca8a966a26ba0c5b7207a9a9e4e08f1b29419"},
|
||||||
|
{file = "Jinja2-2.11.3.tar.gz", hash = "sha256:a6d58433de0ae800347cab1fa3043cebbabe8baa9d29e668f1c768cb87a333c6"},
|
||||||
|
]
|
||||||
|
markupsafe = [
|
||||||
{file = "MarkupSafe-1.1.1-cp27-cp27m-macosx_10_6_intel.whl", hash = "sha256:09027a7803a62ca78792ad89403b1b7a73a01c8cb65909cd876f7fcebd79b161"},
|
{file = "MarkupSafe-1.1.1-cp27-cp27m-macosx_10_6_intel.whl", hash = "sha256:09027a7803a62ca78792ad89403b1b7a73a01c8cb65909cd876f7fcebd79b161"},
|
||||||
{file = "MarkupSafe-1.1.1-cp27-cp27m-manylinux1_i686.whl", hash = "sha256:e249096428b3ae81b08327a63a485ad0878de3fb939049038579ac0ef61e17e7"},
|
{file = "MarkupSafe-1.1.1-cp27-cp27m-manylinux1_i686.whl", hash = "sha256:e249096428b3ae81b08327a63a485ad0878de3fb939049038579ac0ef61e17e7"},
|
||||||
{file = "MarkupSafe-1.1.1-cp27-cp27m-manylinux1_x86_64.whl", hash = "sha256:500d4957e52ddc3351cabf489e79c91c17f6e0899158447047588650b5e69183"},
|
{file = "MarkupSafe-1.1.1-cp27-cp27m-manylinux1_x86_64.whl", hash = "sha256:500d4957e52ddc3351cabf489e79c91c17f6e0899158447047588650b5e69183"},
|
||||||
@@ -252,225 +446,60 @@ files = [
|
|||||||
{file = "MarkupSafe-1.1.1-cp39-cp39-win_amd64.whl", hash = "sha256:b7d644ddb4dbd407d31ffb699f1d140bc35478da613b441c582aeb7c43838dd8"},
|
{file = "MarkupSafe-1.1.1-cp39-cp39-win_amd64.whl", hash = "sha256:b7d644ddb4dbd407d31ffb699f1d140bc35478da613b441c582aeb7c43838dd8"},
|
||||||
{file = "MarkupSafe-1.1.1.tar.gz", hash = "sha256:29872e92839765e546828bb7754a68c418d927cd064fd4708fab9fe9c8bb116b"},
|
{file = "MarkupSafe-1.1.1.tar.gz", hash = "sha256:29872e92839765e546828bb7754a68c418d927cd064fd4708fab9fe9c8bb116b"},
|
||||||
]
|
]
|
||||||
|
packaging = [
|
||||||
[[package]]
|
|
||||||
name = "packaging"
|
|
||||||
version = "20.9"
|
|
||||||
description = "Core utilities for Python packages"
|
|
||||||
optional = false
|
|
||||||
python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*"
|
|
||||||
files = [
|
|
||||||
{file = "packaging-20.9-py2.py3-none-any.whl", hash = "sha256:67714da7f7bc052e064859c05c595155bd1ee9f69f76557e21f051443c20947a"},
|
{file = "packaging-20.9-py2.py3-none-any.whl", hash = "sha256:67714da7f7bc052e064859c05c595155bd1ee9f69f76557e21f051443c20947a"},
|
||||||
{file = "packaging-20.9.tar.gz", hash = "sha256:5b327ac1320dc863dca72f4514ecc086f31186744b84a230374cc1fd776feae5"},
|
{file = "packaging-20.9.tar.gz", hash = "sha256:5b327ac1320dc863dca72f4514ecc086f31186744b84a230374cc1fd776feae5"},
|
||||||
]
|
]
|
||||||
|
pluggy = [
|
||||||
[package.dependencies]
|
|
||||||
pyparsing = ">=2.0.2"
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "pluggy"
|
|
||||||
version = "0.13.1"
|
|
||||||
description = "plugin and hook calling mechanisms for python"
|
|
||||||
optional = false
|
|
||||||
python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*"
|
|
||||||
files = [
|
|
||||||
{file = "pluggy-0.13.1-py2.py3-none-any.whl", hash = "sha256:966c145cd83c96502c3c3868f50408687b38434af77734af1e9ca461a4081d2d"},
|
{file = "pluggy-0.13.1-py2.py3-none-any.whl", hash = "sha256:966c145cd83c96502c3c3868f50408687b38434af77734af1e9ca461a4081d2d"},
|
||||||
{file = "pluggy-0.13.1.tar.gz", hash = "sha256:15b2acde666561e1298d71b523007ed7364de07029219b604cf808bfa1c765b0"},
|
{file = "pluggy-0.13.1.tar.gz", hash = "sha256:15b2acde666561e1298d71b523007ed7364de07029219b604cf808bfa1c765b0"},
|
||||||
]
|
]
|
||||||
|
py = [
|
||||||
[package.dependencies]
|
|
||||||
importlib-metadata = {version = ">=0.12", markers = "python_version < \"3.8\""}
|
|
||||||
|
|
||||||
[package.extras]
|
|
||||||
dev = ["pre-commit", "tox"]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "py"
|
|
||||||
version = "1.10.0"
|
|
||||||
description = "library with cross-python path, ini-parsing, io, code, log facilities"
|
|
||||||
optional = false
|
|
||||||
python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*"
|
|
||||||
files = [
|
|
||||||
{file = "py-1.10.0-py2.py3-none-any.whl", hash = "sha256:3b80836aa6d1feeaa108e046da6423ab8f6ceda6468545ae8d02d9d58d18818a"},
|
{file = "py-1.10.0-py2.py3-none-any.whl", hash = "sha256:3b80836aa6d1feeaa108e046da6423ab8f6ceda6468545ae8d02d9d58d18818a"},
|
||||||
{file = "py-1.10.0.tar.gz", hash = "sha256:21b81bda15b66ef5e1a777a21c4dcd9c20ad3efd0b3f817e7a809035269e1bd3"},
|
{file = "py-1.10.0.tar.gz", hash = "sha256:21b81bda15b66ef5e1a777a21c4dcd9c20ad3efd0b3f817e7a809035269e1bd3"},
|
||||||
]
|
]
|
||||||
|
pyparsing = [
|
||||||
[[package]]
|
|
||||||
name = "pyparsing"
|
|
||||||
version = "2.4.7"
|
|
||||||
description = "Python parsing module"
|
|
||||||
optional = false
|
|
||||||
python-versions = ">=2.6, !=3.0.*, !=3.1.*, !=3.2.*"
|
|
||||||
files = [
|
|
||||||
{file = "pyparsing-2.4.7-py2.py3-none-any.whl", hash = "sha256:ef9d7589ef3c200abe66653d3f1ab1033c3c419ae9b9bdb1240a85b024efc88b"},
|
{file = "pyparsing-2.4.7-py2.py3-none-any.whl", hash = "sha256:ef9d7589ef3c200abe66653d3f1ab1033c3c419ae9b9bdb1240a85b024efc88b"},
|
||||||
{file = "pyparsing-2.4.7.tar.gz", hash = "sha256:c203ec8783bf771a155b207279b9bccb8dea02d8f0c9e5f8ead507bc3246ecc1"},
|
{file = "pyparsing-2.4.7.tar.gz", hash = "sha256:c203ec8783bf771a155b207279b9bccb8dea02d8f0c9e5f8ead507bc3246ecc1"},
|
||||||
]
|
]
|
||||||
|
python-dateutil = [
|
||||||
[[package]]
|
|
||||||
name = "python-dateutil"
|
|
||||||
version = "2.8.1"
|
|
||||||
description = "Extensions to the standard Python datetime module"
|
|
||||||
optional = false
|
|
||||||
python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,>=2.7"
|
|
||||||
files = [
|
|
||||||
{file = "python-dateutil-2.8.1.tar.gz", hash = "sha256:73ebfe9dbf22e832286dafa60473e4cd239f8592f699aa5adaf10050e6e1823c"},
|
{file = "python-dateutil-2.8.1.tar.gz", hash = "sha256:73ebfe9dbf22e832286dafa60473e4cd239f8592f699aa5adaf10050e6e1823c"},
|
||||||
{file = "python_dateutil-2.8.1-py2.py3-none-any.whl", hash = "sha256:75bb3f31ea686f1197762692a9ee6a7550b59fc6ca3a1f4b5d7e32fb98e2da2a"},
|
{file = "python_dateutil-2.8.1-py2.py3-none-any.whl", hash = "sha256:75bb3f31ea686f1197762692a9ee6a7550b59fc6ca3a1f4b5d7e32fb98e2da2a"},
|
||||||
]
|
]
|
||||||
|
requests = [
|
||||||
[package.dependencies]
|
|
||||||
six = ">=1.5"
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "requests"
|
|
||||||
version = "2.25.1"
|
|
||||||
description = "Python HTTP for Humans."
|
|
||||||
optional = false
|
|
||||||
python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*"
|
|
||||||
files = [
|
|
||||||
{file = "requests-2.25.1-py2.py3-none-any.whl", hash = "sha256:c210084e36a42ae6b9219e00e48287def368a26d03a048ddad7bfee44f75871e"},
|
{file = "requests-2.25.1-py2.py3-none-any.whl", hash = "sha256:c210084e36a42ae6b9219e00e48287def368a26d03a048ddad7bfee44f75871e"},
|
||||||
{file = "requests-2.25.1.tar.gz", hash = "sha256:27973dd4a904a4f13b263a19c866c13b92a39ed1c964655f025f3f8d3d75b804"},
|
{file = "requests-2.25.1.tar.gz", hash = "sha256:27973dd4a904a4f13b263a19c866c13b92a39ed1c964655f025f3f8d3d75b804"},
|
||||||
]
|
]
|
||||||
|
six = [
|
||||||
[package.dependencies]
|
|
||||||
certifi = ">=2017.4.17"
|
|
||||||
chardet = ">=3.0.2,<5"
|
|
||||||
idna = ">=2.5,<3"
|
|
||||||
urllib3 = ">=1.21.1,<1.27"
|
|
||||||
|
|
||||||
[package.extras]
|
|
||||||
security = ["cryptography (>=1.3.4)", "pyOpenSSL (>=0.14)"]
|
|
||||||
socks = ["PySocks (>=1.5.6,!=1.5.7)", "win-inet-pton"]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "six"
|
|
||||||
version = "1.15.0"
|
|
||||||
description = "Python 2 and 3 compatibility utilities"
|
|
||||||
optional = false
|
|
||||||
python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*"
|
|
||||||
files = [
|
|
||||||
{file = "six-1.15.0-py2.py3-none-any.whl", hash = "sha256:8b74bedcbbbaca38ff6d7491d76f2b06b3592611af620f8426e82dddb04a5ced"},
|
{file = "six-1.15.0-py2.py3-none-any.whl", hash = "sha256:8b74bedcbbbaca38ff6d7491d76f2b06b3592611af620f8426e82dddb04a5ced"},
|
||||||
{file = "six-1.15.0.tar.gz", hash = "sha256:30639c035cdb23534cd4aa2dd52c3bf48f06e5f4a941509c8bafd8ce11080259"},
|
{file = "six-1.15.0.tar.gz", hash = "sha256:30639c035cdb23534cd4aa2dd52c3bf48f06e5f4a941509c8bafd8ce11080259"},
|
||||||
]
|
]
|
||||||
|
toml = [
|
||||||
[[package]]
|
|
||||||
name = "toml"
|
|
||||||
version = "0.10.2"
|
|
||||||
description = "Python Library for Tom's Obvious, Minimal Language"
|
|
||||||
optional = false
|
|
||||||
python-versions = ">=2.6, !=3.0.*, !=3.1.*, !=3.2.*"
|
|
||||||
files = [
|
|
||||||
{file = "toml-0.10.2-py2.py3-none-any.whl", hash = "sha256:806143ae5bfb6a3c6e736a764057db0e6a0e05e338b5630894a5f779cabb4f9b"},
|
{file = "toml-0.10.2-py2.py3-none-any.whl", hash = "sha256:806143ae5bfb6a3c6e736a764057db0e6a0e05e338b5630894a5f779cabb4f9b"},
|
||||||
{file = "toml-0.10.2.tar.gz", hash = "sha256:b3bda1d108d5dd99f4a20d24d9c348e91c4db7ab1b749200bded2f839ccbe68f"},
|
{file = "toml-0.10.2.tar.gz", hash = "sha256:b3bda1d108d5dd99f4a20d24d9c348e91c4db7ab1b749200bded2f839ccbe68f"},
|
||||||
]
|
]
|
||||||
|
tox = [
|
||||||
[[package]]
|
|
||||||
name = "tox"
|
|
||||||
version = "3.23.0"
|
|
||||||
description = "tox is a generic virtualenv management and test command line tool"
|
|
||||||
optional = false
|
|
||||||
python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,>=2.7"
|
|
||||||
files = [
|
|
||||||
{file = "tox-3.23.0-py2.py3-none-any.whl", hash = "sha256:e007673f3595cede9b17a7c4962389e4305d4a3682a6c5a4159a1453b4f326aa"},
|
{file = "tox-3.23.0-py2.py3-none-any.whl", hash = "sha256:e007673f3595cede9b17a7c4962389e4305d4a3682a6c5a4159a1453b4f326aa"},
|
||||||
{file = "tox-3.23.0.tar.gz", hash = "sha256:05a4dbd5e4d3d8269b72b55600f0b0303e2eb47ad5c6fe76d3576f4c58d93661"},
|
{file = "tox-3.23.0.tar.gz", hash = "sha256:05a4dbd5e4d3d8269b72b55600f0b0303e2eb47ad5c6fe76d3576f4c58d93661"},
|
||||||
]
|
]
|
||||||
|
typing-extensions = [
|
||||||
[package.dependencies]
|
|
||||||
colorama = {version = ">=0.4.1", markers = "platform_system == \"Windows\""}
|
|
||||||
filelock = ">=3.0.0"
|
|
||||||
importlib-metadata = {version = ">=0.12", markers = "python_version < \"3.8\""}
|
|
||||||
packaging = ">=14"
|
|
||||||
pluggy = ">=0.12.0"
|
|
||||||
py = ">=1.4.17"
|
|
||||||
six = ">=1.14.0"
|
|
||||||
toml = ">=0.9.4"
|
|
||||||
virtualenv = ">=16.0.0,<20.0.0 || >20.0.0,<20.0.1 || >20.0.1,<20.0.2 || >20.0.2,<20.0.3 || >20.0.3,<20.0.4 || >20.0.4,<20.0.5 || >20.0.5,<20.0.6 || >20.0.6,<20.0.7 || >20.0.7"
|
|
||||||
|
|
||||||
[package.extras]
|
|
||||||
docs = ["pygments-github-lexers (>=0.0.5)", "sphinx (>=2.0.0)", "sphinxcontrib-autoprogram (>=0.1.5)", "towncrier (>=18.5.0)"]
|
|
||||||
testing = ["flaky (>=3.4.0)", "freezegun (>=0.3.11)", "pathlib2 (>=2.3.3)", "psutil (>=5.6.1)", "pytest (>=4.0.0)", "pytest-cov (>=2.5.1)", "pytest-mock (>=1.10.0)", "pytest-randomly (>=1.0.0)", "pytest-xdist (>=1.22.2)"]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "typing-extensions"
|
|
||||||
version = "3.7.4.3"
|
|
||||||
description = "Backported and Experimental Type Hints for Python 3.5+"
|
|
||||||
optional = false
|
|
||||||
python-versions = "*"
|
|
||||||
files = [
|
|
||||||
{file = "typing_extensions-3.7.4.3-py2-none-any.whl", hash = "sha256:dafc7639cde7f1b6e1acc0f457842a83e722ccca8eef5270af2d74792619a89f"},
|
{file = "typing_extensions-3.7.4.3-py2-none-any.whl", hash = "sha256:dafc7639cde7f1b6e1acc0f457842a83e722ccca8eef5270af2d74792619a89f"},
|
||||||
{file = "typing_extensions-3.7.4.3-py3-none-any.whl", hash = "sha256:7cb407020f00f7bfc3cb3e7881628838e69d8f3fcab2f64742a5e76b2f841918"},
|
{file = "typing_extensions-3.7.4.3-py3-none-any.whl", hash = "sha256:7cb407020f00f7bfc3cb3e7881628838e69d8f3fcab2f64742a5e76b2f841918"},
|
||||||
{file = "typing_extensions-3.7.4.3.tar.gz", hash = "sha256:99d4073b617d30288f569d3f13d2bd7548c3a7e4c8de87db09a9d29bb3a4a60c"},
|
{file = "typing_extensions-3.7.4.3.tar.gz", hash = "sha256:99d4073b617d30288f569d3f13d2bd7548c3a7e4c8de87db09a9d29bb3a4a60c"},
|
||||||
]
|
]
|
||||||
|
urllib3 = [
|
||||||
[[package]]
|
{file = "urllib3-1.26.4-py2.py3-none-any.whl", hash = "sha256:2f4da4594db7e1e110a944bb1b551fdf4e6c136ad42e4234131391e21eb5b0df"},
|
||||||
name = "urllib3"
|
{file = "urllib3-1.26.4.tar.gz", hash = "sha256:e7b021f7241115872f92f43c6508082facffbd1c048e3c6e2bb9c2a157e28937"},
|
||||||
version = "1.26.19"
|
|
||||||
description = "HTTP library with thread-safe connection pooling, file post, and more."
|
|
||||||
optional = false
|
|
||||||
python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,!=3.5.*,>=2.7"
|
|
||||||
files = [
|
|
||||||
{file = "urllib3-1.26.19-py2.py3-none-any.whl", hash = "sha256:37a0344459b199fce0e80b0d3569837ec6b6937435c5244e7fd73fa6006830f3"},
|
|
||||||
{file = "urllib3-1.26.19.tar.gz", hash = "sha256:3e3d753a8618b86d7de333b4223005f68720bcd6a7d2bcb9fbd2229ec7c1e429"},
|
|
||||||
]
|
]
|
||||||
|
virtualenv = [
|
||||||
[package.extras]
|
|
||||||
brotli = ["brotli (==1.0.9)", "brotli (>=1.0.9)", "brotlicffi (>=0.8.0)", "brotlipy (>=0.6.0)"]
|
|
||||||
secure = ["certifi", "cryptography (>=1.3.4)", "idna (>=2.0.0)", "ipaddress", "pyOpenSSL (>=0.14)", "urllib3-secure-extra"]
|
|
||||||
socks = ["PySocks (>=1.5.6,!=1.5.7,<2.0)"]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "virtualenv"
|
|
||||||
version = "20.4.3"
|
|
||||||
description = "Virtual Python Environment builder"
|
|
||||||
optional = false
|
|
||||||
python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,>=2.7"
|
|
||||||
files = [
|
|
||||||
{file = "virtualenv-20.4.3-py2.py3-none-any.whl", hash = "sha256:83f95875d382c7abafe06bd2a4cdd1b363e1bb77e02f155ebe8ac082a916b37c"},
|
{file = "virtualenv-20.4.3-py2.py3-none-any.whl", hash = "sha256:83f95875d382c7abafe06bd2a4cdd1b363e1bb77e02f155ebe8ac082a916b37c"},
|
||||||
{file = "virtualenv-20.4.3.tar.gz", hash = "sha256:49ec4eb4c224c6f7dd81bb6d0a28a09ecae5894f4e593c89b0db0885f565a107"},
|
{file = "virtualenv-20.4.3.tar.gz", hash = "sha256:49ec4eb4c224c6f7dd81bb6d0a28a09ecae5894f4e593c89b0db0885f565a107"},
|
||||||
]
|
]
|
||||||
|
werkzeug = [
|
||||||
[package.dependencies]
|
|
||||||
appdirs = ">=1.4.3,<2"
|
|
||||||
distlib = ">=0.3.1,<1"
|
|
||||||
filelock = ">=3.0.0,<4"
|
|
||||||
importlib-metadata = {version = ">=0.12", markers = "python_version < \"3.8\""}
|
|
||||||
importlib-resources = {version = ">=1.0", markers = "python_version < \"3.7\""}
|
|
||||||
six = ">=1.9.0,<2"
|
|
||||||
|
|
||||||
[package.extras]
|
|
||||||
docs = ["proselint (>=0.10.2)", "sphinx (>=3)", "sphinx-argparse (>=0.2.5)", "sphinx-rtd-theme (>=0.4.3)", "towncrier (>=19.9.0rc1)"]
|
|
||||||
testing = ["coverage (>=4)", "coverage-enable-subprocess (>=1)", "flaky (>=3)", "packaging (>=20.0)", "pytest (>=4)", "pytest-env (>=0.6.2)", "pytest-freezegun (>=0.4.1)", "pytest-mock (>=2)", "pytest-randomly (>=1)", "pytest-timeout (>=1)", "xonsh (>=0.9.16)"]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "werkzeug"
|
|
||||||
version = "1.0.1"
|
|
||||||
description = "The comprehensive WSGI web application library."
|
|
||||||
optional = false
|
|
||||||
python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*"
|
|
||||||
files = [
|
|
||||||
{file = "Werkzeug-1.0.1-py2.py3-none-any.whl", hash = "sha256:2de2a5db0baeae7b2d2664949077c2ac63fbd16d98da0ff71837f7d1dea3fd43"},
|
{file = "Werkzeug-1.0.1-py2.py3-none-any.whl", hash = "sha256:2de2a5db0baeae7b2d2664949077c2ac63fbd16d98da0ff71837f7d1dea3fd43"},
|
||||||
{file = "Werkzeug-1.0.1.tar.gz", hash = "sha256:6c80b1e5ad3665290ea39320b91e1be1e0d5f60652b964a3070216de83d2e47c"},
|
{file = "Werkzeug-1.0.1.tar.gz", hash = "sha256:6c80b1e5ad3665290ea39320b91e1be1e0d5f60652b964a3070216de83d2e47c"},
|
||||||
]
|
]
|
||||||
|
zipp = [
|
||||||
[package.extras]
|
|
||||||
dev = ["coverage", "pallets-sphinx-themes", "pytest", "pytest-timeout", "sphinx", "sphinx-issues", "tox"]
|
|
||||||
watchdog = ["watchdog"]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "zipp"
|
|
||||||
version = "3.4.1"
|
|
||||||
description = "Backport of pathlib-compatible object wrapper for zip files"
|
|
||||||
optional = false
|
|
||||||
python-versions = ">=3.6"
|
|
||||||
files = [
|
|
||||||
{file = "zipp-3.4.1-py3-none-any.whl", hash = "sha256:51cb66cc54621609dd593d1787f286ee42a5c0adbb4b29abea5a63edc3e03098"},
|
{file = "zipp-3.4.1-py3-none-any.whl", hash = "sha256:51cb66cc54621609dd593d1787f286ee42a5c0adbb4b29abea5a63edc3e03098"},
|
||||||
{file = "zipp-3.4.1.tar.gz", hash = "sha256:3607921face881ba3e026887d8150cca609d517579abe052ac81fc5aeffdbd76"},
|
{file = "zipp-3.4.1.tar.gz", hash = "sha256:3607921face881ba3e026887d8150cca609d517579abe052ac81fc5aeffdbd76"},
|
||||||
]
|
]
|
||||||
|
|
||||||
[package.extras]
|
|
||||||
docs = ["jaraco.packaging (>=8.2)", "rst.linker (>=1.9)", "sphinx"]
|
|
||||||
testing = ["func-timeout", "jaraco.itertools", "pytest (>=4.6)", "pytest-black (>=0.3.7)", "pytest-checkdocs (>=1.2.3)", "pytest-cov", "pytest-enabler", "pytest-flake8", "pytest-mypy"]
|
|
||||||
|
|
||||||
[metadata]
|
|
||||||
lock-version = "2.0"
|
|
||||||
python-versions = "^3.6.1"
|
|
||||||
content-hash = "af9db950cd722e7dc52b691fb58abc1e22ab48b34ddfe4c5258b3c755a3892fa"
|
|
||||||
|
|||||||
@@ -1,37 +1,40 @@
|
|||||||
# pylint: disable=missing-module-docstring, redefined-outer-name, unused-argument, wrong-import-order, unused-import
|
# pylint: disable=missing-module-docstring,redefined-outer-name,unused-argument,unused-import,protected-access
|
||||||
import time
|
import time
|
||||||
from unittest import mock
|
from unittest import mock
|
||||||
|
|
||||||
|
import poetry.factory
|
||||||
|
import poetry.installation.executor
|
||||||
import pytest
|
import pytest
|
||||||
import tox.tox_env.python.virtual_env.runner
|
import tox.tox_env.python.virtual_env.runner
|
||||||
from poetry.factory import Factory
|
|
||||||
|
import tox_poetry_installer.hooks._tox_on_install_helpers
|
||||||
|
|
||||||
from .fixtures import mock_poetry_factory
|
from .fixtures import mock_poetry_factory
|
||||||
from .fixtures import mock_venv
|
from .fixtures import mock_venv
|
||||||
from tox_poetry_installer import installer
|
|
||||||
from tox_poetry_installer import utilities
|
|
||||||
|
|
||||||
|
|
||||||
def test_deduplication(mock_venv, mock_poetry_factory):
|
def test_deduplication(mock_venv, mock_poetry_factory):
|
||||||
"""Test that the installer does not install duplicate dependencies"""
|
"""Test that the installer does not install duplicate dependencies"""
|
||||||
poetry = Factory().create_poetry(None)
|
project = poetry.factory.Factory().create_poetry(None)
|
||||||
packages: utilities.PackageMap = {
|
packages: tox_poetry_installer.hooks._tox_on_install_helpers.PackageMap = {
|
||||||
item.name: item for item in poetry.locker.locked_repository().packages
|
item.name: item for item in project.locker.locked_repository().packages
|
||||||
}
|
}
|
||||||
|
|
||||||
venv = tox.tox_env.python.virtual_env.runner.VirtualEnvRunner()
|
venv = tox.tox_env.python.virtual_env.runner.VirtualEnvRunner()
|
||||||
to_install = [packages["toml"], packages["toml"]]
|
to_install = [packages["toml"], packages["toml"]]
|
||||||
|
|
||||||
installer.install(poetry, venv, to_install)
|
tox_poetry_installer.hooks._tox_on_install_helpers.install_package(
|
||||||
|
project, venv, to_install
|
||||||
|
)
|
||||||
|
|
||||||
assert len(set(to_install)) == len(venv.installed) # pylint: disable=no-member
|
assert len(set(to_install)) == len(venv.installed) # pylint: disable=no-member
|
||||||
|
|
||||||
|
|
||||||
def test_parallelization(mock_venv, mock_poetry_factory):
|
def test_parallelization(mock_venv, mock_poetry_factory):
|
||||||
"""Test that behavior is consistent between parallel and non-parallel usage"""
|
"""Test that behavior is consistent between parallel and non-parallel usage"""
|
||||||
poetry = Factory().create_poetry(None)
|
project = poetry.factory.Factory().create_poetry(None)
|
||||||
packages: utilities.PackageMap = {
|
packages: tox_poetry_installer.hooks._tox_on_install_helpers.PackageMap = {
|
||||||
item.name: item for item in poetry.locker.locked_repository().packages
|
item.name: item for item in project.locker.locked_repository().packages
|
||||||
}
|
}
|
||||||
|
|
||||||
to_install = [
|
to_install = [
|
||||||
@@ -45,12 +48,16 @@ def test_parallelization(mock_venv, mock_poetry_factory):
|
|||||||
|
|
||||||
venv_sequential = tox.tox_env.python.virtual_env.runner.VirtualEnvRunner()
|
venv_sequential = tox.tox_env.python.virtual_env.runner.VirtualEnvRunner()
|
||||||
start_sequential = time.time()
|
start_sequential = time.time()
|
||||||
installer.install(poetry, venv_sequential, to_install, 0)
|
tox_poetry_installer.hooks._tox_on_install_helpers.install_package(
|
||||||
|
project, venv_sequential, to_install, 0
|
||||||
|
)
|
||||||
sequential = time.time() - start_sequential
|
sequential = time.time() - start_sequential
|
||||||
|
|
||||||
venv_parallel = tox.tox_env.python.virtual_env.runner.VirtualEnvRunner()
|
venv_parallel = tox.tox_env.python.virtual_env.runner.VirtualEnvRunner()
|
||||||
start_parallel = time.time()
|
start_parallel = time.time()
|
||||||
installer.install(poetry, venv_parallel, to_install, 5)
|
tox_poetry_installer.hooks._tox_on_install_helpers.install_package(
|
||||||
|
project, venv_parallel, to_install, 5
|
||||||
|
)
|
||||||
parallel = time.time() - start_parallel
|
parallel = time.time() - start_parallel
|
||||||
|
|
||||||
# The mock delay during package install is static (one second) so these values should all
|
# The mock delay during package install is static (one second) so these values should all
|
||||||
@@ -69,22 +76,22 @@ def test_propagates_exceptions_during_installation(
|
|||||||
|
|
||||||
Regression test for https://github.com/enpaul/tox-poetry-installer/issues/86
|
Regression test for https://github.com/enpaul/tox-poetry-installer/issues/86
|
||||||
"""
|
"""
|
||||||
from tox_poetry_installer import _poetry # pylint: disable=import-outside-toplevel
|
project = poetry.factory.Factory().create_poetry(None)
|
||||||
|
packages: tox_poetry_installer.hooks._tox_on_install_helpers.PackageMap = {
|
||||||
poetry = Factory().create_poetry(None)
|
item.name: item for item in project.locker.locked_repository().packages
|
||||||
packages: utilities.PackageMap = {
|
|
||||||
item.name: item for item in poetry.locker.locked_repository().packages
|
|
||||||
}
|
}
|
||||||
to_install = [packages["toml"]]
|
to_install = [packages["toml"]]
|
||||||
venv = tox.tox_env.python.virtual_env.runner.VirtualEnvRunner()
|
venv = tox.tox_env.python.virtual_env.runner.VirtualEnvRunner()
|
||||||
fake_exception = ValueError("my testing exception")
|
fake_exception = ValueError("my testing exception")
|
||||||
|
|
||||||
with mock.patch.object(
|
with mock.patch.object(
|
||||||
_poetry,
|
poetry.installation.executor,
|
||||||
"Executor",
|
"Executor",
|
||||||
**{"return_value.execute.side_effect": fake_exception},
|
**{"return_value.execute.side_effect": fake_exception},
|
||||||
):
|
):
|
||||||
with pytest.raises(ValueError) as exc_info:
|
with pytest.raises(ValueError) as exc_info:
|
||||||
installer.install(poetry, venv, to_install, num_threads)
|
tox_poetry_installer.hooks._tox_on_install_helpers.install_package(
|
||||||
|
project, venv, to_install, num_threads
|
||||||
|
)
|
||||||
|
|
||||||
assert exc_info.value is fake_exception
|
assert exc_info.value is fake_exception
|
||||||
|
|||||||
@@ -3,6 +3,7 @@
|
|||||||
The next best thing to having one source of truth is having a way to ensure all of your
|
The next best thing to having one source of truth is having a way to ensure all of your
|
||||||
sources of truth agree with each other.
|
sources of truth agree with each other.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
|
|
||||||
import toml
|
import toml
|
||||||
|
|||||||
@@ -1,33 +1,23 @@
|
|||||||
# pylint: disable=missing-module-docstring, redefined-outer-name, unused-argument, wrong-import-order, unused-import
|
# pylint: disable=missing-module-docstring,redefined-outer-name,unused-argument,unused-import,protected-access
|
||||||
import poetry.factory
|
import poetry.factory
|
||||||
import poetry.utils.env
|
import poetry.utils.env
|
||||||
import pytest
|
import pytest
|
||||||
from poetry.puzzle.provider import Provider
|
|
||||||
|
import tox_poetry_installer.hooks._tox_on_install_helpers
|
||||||
|
from tox_poetry_installer import exceptions
|
||||||
|
|
||||||
from .fixtures import mock_poetry_factory
|
from .fixtures import mock_poetry_factory
|
||||||
from .fixtures import mock_venv
|
from .fixtures import mock_venv
|
||||||
from tox_poetry_installer import constants
|
|
||||||
from tox_poetry_installer import exceptions
|
|
||||||
from tox_poetry_installer import utilities
|
|
||||||
|
|
||||||
|
|
||||||
def test_exclude_unsafe():
|
|
||||||
"""Test that the unsafe packages are properly excluded
|
|
||||||
|
|
||||||
Also ensure that the internal constant matches the value from Poetry
|
|
||||||
"""
|
|
||||||
assert Provider.UNSAFE_PACKAGES == constants.UNSAFE_PACKAGES
|
|
||||||
|
|
||||||
for dep in constants.UNSAFE_PACKAGES:
|
|
||||||
assert not utilities.identify_transients(dep, {}, None)
|
|
||||||
|
|
||||||
|
|
||||||
def test_allow_missing():
|
def test_allow_missing():
|
||||||
"""Test that the ``allow_missing`` parameter works as expected"""
|
"""Test that the ``allow_missing`` parameter works as expected"""
|
||||||
with pytest.raises(exceptions.LockedDepNotFoundError):
|
with pytest.raises(exceptions.LockedDepNotFoundError):
|
||||||
utilities.identify_transients("luke-skywalker", {}, None)
|
tox_poetry_installer.hooks._tox_on_install_helpers.identify_transients(
|
||||||
|
"luke-skywalker", {}, None
|
||||||
|
)
|
||||||
|
|
||||||
assert not utilities.identify_transients(
|
assert not tox_poetry_installer.hooks._tox_on_install_helpers.identify_transients(
|
||||||
"darth-vader", {}, None, allow_missing=["darth-vader"]
|
"darth-vader", {}, None, allow_missing=["darth-vader"]
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -47,7 +37,9 @@ def test_exclude_pep508():
|
|||||||
"=>foo",
|
"=>foo",
|
||||||
]:
|
]:
|
||||||
with pytest.raises(exceptions.LockedDepVersionConflictError):
|
with pytest.raises(exceptions.LockedDepVersionConflictError):
|
||||||
utilities.identify_transients(version, {}, None)
|
tox_poetry_installer.hooks._tox_on_install_helpers.identify_transients(
|
||||||
|
version, {}, None
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
def test_functional(mock_poetry_factory, mock_venv):
|
def test_functional(mock_poetry_factory, mock_venv):
|
||||||
@@ -56,8 +48,10 @@ def test_functional(mock_poetry_factory, mock_venv):
|
|||||||
Trivially test that it resolves dependencies properly and that the parent package
|
Trivially test that it resolves dependencies properly and that the parent package
|
||||||
is always the last in the returned list.
|
is always the last in the returned list.
|
||||||
"""
|
"""
|
||||||
pypoetry = poetry.factory.Factory().create_poetry(None)
|
project = poetry.factory.Factory().create_poetry(None)
|
||||||
packages = utilities.build_package_map(pypoetry)
|
packages = tox_poetry_installer.hooks._tox_on_install_helpers.build_package_map(
|
||||||
|
project
|
||||||
|
)
|
||||||
venv = poetry.utils.env.VirtualEnv() # pylint: disable=no-value-for-parameter
|
venv = poetry.utils.env.VirtualEnv() # pylint: disable=no-value-for-parameter
|
||||||
|
|
||||||
requests_requires = [
|
requests_requires = [
|
||||||
@@ -68,12 +62,18 @@ def test_functional(mock_poetry_factory, mock_venv):
|
|||||||
packages["requests"][0],
|
packages["requests"][0],
|
||||||
]
|
]
|
||||||
|
|
||||||
transients = utilities.identify_transients("requests", packages, venv)
|
transients = tox_poetry_installer.hooks._tox_on_install_helpers.identify_transients(
|
||||||
|
"requests", packages, venv
|
||||||
|
)
|
||||||
|
|
||||||
assert all((item in requests_requires) for item in transients)
|
assert all((item in requests_requires) for item in transients)
|
||||||
assert all((item in transients) for item in requests_requires)
|
assert all((item in transients) for item in requests_requires)
|
||||||
|
|
||||||
for package in [packages["requests"][0], packages["tox"][0], packages["flask"][0]]:
|
for package in [packages["requests"][0], packages["tox"][0], packages["flask"][0]]:
|
||||||
transients = utilities.identify_transients(package.name, packages, venv)
|
transients = (
|
||||||
|
tox_poetry_installer.hooks._tox_on_install_helpers.identify_transients(
|
||||||
|
package.name, packages, venv
|
||||||
|
)
|
||||||
|
)
|
||||||
assert transients[-1] == package
|
assert transients[-1] == package
|
||||||
assert len(transients) == len(set(transients))
|
assert len(transients) == len(set(transients))
|
||||||
|
|||||||
18
tox.ini
18
tox.ini
@@ -1,13 +1,11 @@
|
|||||||
[tox]
|
[tox]
|
||||||
envlist = py37, py38, py39, py310, py311, static, static-tests, security
|
envlist = py3{8,9,10,11,12} static, static-tests, security
|
||||||
skip_missing_interpreters = true
|
skip_missing_interpreters = true
|
||||||
|
|
||||||
[testenv]
|
[testenv]
|
||||||
description = Run the tests
|
description = Run the tests
|
||||||
require_locked_deps = true
|
require_locked_deps = true
|
||||||
require_poetry = true
|
require_poetry = true
|
||||||
extras =
|
|
||||||
poetry
|
|
||||||
locked_deps =
|
locked_deps =
|
||||||
pytest
|
pytest
|
||||||
pytest-cov
|
pytest-cov
|
||||||
@@ -26,10 +24,10 @@ ignore_errors = true
|
|||||||
locked_deps =
|
locked_deps =
|
||||||
black
|
black
|
||||||
blacken-docs
|
blacken-docs
|
||||||
|
isort
|
||||||
mdformat
|
mdformat
|
||||||
mdformat-gfm
|
mdformat-gfm
|
||||||
mypy
|
mypy
|
||||||
reorder-python-imports
|
|
||||||
pre-commit
|
pre-commit
|
||||||
pre-commit-hooks
|
pre-commit-hooks
|
||||||
pylint
|
pylint
|
||||||
@@ -52,6 +50,7 @@ locked_deps =
|
|||||||
pylint
|
pylint
|
||||||
pytest
|
pytest
|
||||||
mypy
|
mypy
|
||||||
|
toml
|
||||||
types-toml
|
types-toml
|
||||||
commands =
|
commands =
|
||||||
pylint {toxinidir}/tests/ \
|
pylint {toxinidir}/tests/ \
|
||||||
@@ -78,14 +77,3 @@ commands =
|
|||||||
--recursive \
|
--recursive \
|
||||||
--quiet \
|
--quiet \
|
||||||
--skip B101
|
--skip B101
|
||||||
poetry export \
|
|
||||||
--format requirements.txt \
|
|
||||||
--output {envtmpdir}/requirements.txt \
|
|
||||||
--without-hashes \
|
|
||||||
--with dev \
|
|
||||||
--extras poetry
|
|
||||||
safety check \
|
|
||||||
--file {envtmpdir}/requirements.txt \
|
|
||||||
--output text \
|
|
||||||
# https://github.com/pytest-dev/py/issues/287
|
|
||||||
--ignore 51457
|
|
||||||
|
|||||||
@@ -1,43 +0,0 @@
|
|||||||
"""You've heard of vendoirization, now get ready for internal namespace shadowing
|
|
||||||
|
|
||||||
Poetry is an optional dependency of this package explicitly to support the use case of having the
|
|
||||||
plugin and the `poetry` package installed to the same python environment; this is most common in
|
|
||||||
containers and/or CI. In this case there are two potential problems that can arise in this case:
|
|
||||||
|
|
||||||
* The installation of the plugin overwrites the installed version of Poetry resulting in
|
|
||||||
compatibility issues.
|
|
||||||
* Running `poetry install --no-dev`, when this plugin is in the dev-deps, results in poetry being
|
|
||||||
uninstalled from the environment.
|
|
||||||
|
|
||||||
To support these edge cases, and more broadly to support not messing with a system package manager,
|
|
||||||
the `poetry` package dependency is listed as optional dependency. This allows the plugin to be
|
|
||||||
installed to the same environment as Poetry and import that same Poetry installation here.
|
|
||||||
|
|
||||||
However, simply importing Poetry on the assumption that it is installed breaks another valid use
|
|
||||||
case: having this plugin installed alongside Tox when not using a Poetry-based project. To account
|
|
||||||
for this the imports in this module are isolated and the resultant import error that would result
|
|
||||||
is converted to an internal error that can be caught by callers. Rather than importing this module
|
|
||||||
at the module scope it is imported into function scope wherever Poetry components are needed. This
|
|
||||||
moves import errors from load time to runtime which allows the plugin to be skipped if Poetry isn't
|
|
||||||
installed and/or a more helpful error be raised within the Tox framework.
|
|
||||||
"""
|
|
||||||
# pylint: disable=unused-import
|
|
||||||
import sys
|
|
||||||
|
|
||||||
from tox_poetry_installer import exceptions
|
|
||||||
|
|
||||||
|
|
||||||
try:
|
|
||||||
from cleo.io.null_io import NullIO
|
|
||||||
from poetry.config.config import Config
|
|
||||||
from poetry.core.packages.dependency import Dependency as PoetryDependency
|
|
||||||
from poetry.core.packages.package import Package as PoetryPackage
|
|
||||||
from poetry.factory import Factory
|
|
||||||
from poetry.installation.executor import Executor
|
|
||||||
from poetry.installation.operations.install import Install
|
|
||||||
from poetry.poetry import Poetry
|
|
||||||
from poetry.utils.env import VirtualEnv
|
|
||||||
except ImportError:
|
|
||||||
raise exceptions.PoetryNotInstalledError(
|
|
||||||
f"No version of Poetry could be imported under the current environment for '{sys.executable}'"
|
|
||||||
) from None
|
|
||||||
@@ -5,7 +5,7 @@ in this module.
|
|||||||
|
|
||||||
All constants should be type hinted.
|
All constants should be type hinted.
|
||||||
"""
|
"""
|
||||||
from typing import Set
|
|
||||||
from typing import Tuple
|
from typing import Tuple
|
||||||
|
|
||||||
from tox_poetry_installer import __about__
|
from tox_poetry_installer import __about__
|
||||||
@@ -19,9 +19,5 @@ PEP508_VERSION_DELIMITERS: Tuple[str, ...] = ("~=", "==", "!=", ">", "<")
|
|||||||
# console output.
|
# console output.
|
||||||
REPORTER_PREFIX: str = f"{__about__.__title__}:"
|
REPORTER_PREFIX: str = f"{__about__.__title__}:"
|
||||||
|
|
||||||
# Internal list of packages that poetry has deemed unsafe and are excluded from the lockfile
|
|
||||||
# TODO: This functionality is no longer needed, should remove in a future update.
|
|
||||||
UNSAFE_PACKAGES: Set[str] = set()
|
|
||||||
|
|
||||||
# Number of threads to use for installing dependencies by default
|
# Number of threads to use for installing dependencies by default
|
||||||
DEFAULT_INSTALL_THREADS: int = 10
|
DEFAULT_INSTALL_THREADS: int = 10
|
||||||
|
|||||||
@@ -11,7 +11,7 @@ All exceptions should inherit from the common base exception :exc:`ToxPoetryInst
|
|||||||
+-- LockedDepNotFoundError
|
+-- LockedDepNotFoundError
|
||||||
+-- ExtraNotFoundError
|
+-- ExtraNotFoundError
|
||||||
+-- LockedDepsRequiredError
|
+-- LockedDepsRequiredError
|
||||||
+-- RequiresUnsafeDepError
|
+-- LockfileParsingError
|
||||||
|
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@@ -44,5 +44,5 @@ class LockedDepsRequiredError(ToxPoetryInstallerException):
|
|||||||
"""Environment cannot specify unlocked dependencies when locked dependencies are required"""
|
"""Environment cannot specify unlocked dependencies when locked dependencies are required"""
|
||||||
|
|
||||||
|
|
||||||
class RequiresUnsafeDepError(ToxPoetryInstallerException):
|
class LockfileParsingError(ToxPoetryInstallerException):
|
||||||
"""Package under test depends on an unsafe dependency and cannot be installed"""
|
"""Failed to load or parse the Poetry lockfile"""
|
||||||
|
|||||||
@@ -1,175 +0,0 @@
|
|||||||
"""Main hook definition module
|
|
||||||
|
|
||||||
All implementations of tox hooks are defined here, as well as any single-use helper functions
|
|
||||||
specifically related to implementing the hooks (to keep the size/readability of the hook functions
|
|
||||||
themselves manageable).
|
|
||||||
"""
|
|
||||||
from itertools import chain
|
|
||||||
from typing import List
|
|
||||||
|
|
||||||
from tox.config.cli.parser import ToxParser
|
|
||||||
from tox.config.sets import EnvConfigSet
|
|
||||||
from tox.plugin import impl
|
|
||||||
from tox.tox_env.api import ToxEnv as ToxVirtualEnv
|
|
||||||
|
|
||||||
from tox_poetry_installer import constants
|
|
||||||
from tox_poetry_installer import exceptions
|
|
||||||
from tox_poetry_installer import installer
|
|
||||||
from tox_poetry_installer import logger
|
|
||||||
from tox_poetry_installer import utilities
|
|
||||||
|
|
||||||
|
|
||||||
@impl
|
|
||||||
def tox_add_option(parser: ToxParser):
|
|
||||||
"""Add additional command line arguments to tox to configure plugin behavior"""
|
|
||||||
parser.add_argument(
|
|
||||||
"--require-poetry",
|
|
||||||
action="store_true",
|
|
||||||
dest="require_poetry",
|
|
||||||
help="(deprecated) Trigger a failure if Poetry is not available to Tox",
|
|
||||||
)
|
|
||||||
|
|
||||||
parser.add_argument(
|
|
||||||
"--parallel-install-threads",
|
|
||||||
type=int,
|
|
||||||
dest="parallel_install_threads",
|
|
||||||
default=constants.DEFAULT_INSTALL_THREADS,
|
|
||||||
help="Number of locked dependencies to install simultaneously; set to 0 to disable parallel installation",
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
@impl
|
|
||||||
def tox_add_env_config(env_conf: EnvConfigSet):
|
|
||||||
"""Add required env configuration options to the tox INI file"""
|
|
||||||
env_conf.add_config(
|
|
||||||
"poetry_dep_groups",
|
|
||||||
of_type=List[str],
|
|
||||||
default=[],
|
|
||||||
desc="List of Poetry dependency groups to install to the environment",
|
|
||||||
)
|
|
||||||
|
|
||||||
env_conf.add_config(
|
|
||||||
"install_project_deps",
|
|
||||||
of_type=bool,
|
|
||||||
default=True,
|
|
||||||
desc="Automatically install all Poetry primary dependencies to the environment",
|
|
||||||
)
|
|
||||||
|
|
||||||
env_conf.add_config(
|
|
||||||
"require_locked_deps",
|
|
||||||
of_type=bool,
|
|
||||||
default=False,
|
|
||||||
desc="Require all dependencies in the environment be installed using the Poetry lockfile",
|
|
||||||
)
|
|
||||||
|
|
||||||
env_conf.add_config(
|
|
||||||
"require_poetry",
|
|
||||||
of_type=bool,
|
|
||||||
default=False,
|
|
||||||
desc="Trigger a failure if Poetry is not available to Tox",
|
|
||||||
)
|
|
||||||
|
|
||||||
env_conf.add_config(
|
|
||||||
"locked_deps",
|
|
||||||
of_type=List[str],
|
|
||||||
default=[],
|
|
||||||
desc="List of locked dependencies to install to the environment using the Poetry lockfile",
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
@impl
|
|
||||||
def tox_on_install(
|
|
||||||
tox_env: ToxVirtualEnv, section: str # pylint: disable=unused-argument
|
|
||||||
) -> None:
|
|
||||||
"""Install the dependencies for the current environment
|
|
||||||
|
|
||||||
Loads the local Poetry environment and the corresponding lockfile then pulls the dependencies
|
|
||||||
specified by the Tox environment. Finally these dependencies are installed into the Tox
|
|
||||||
environment using the Poetry ``PipInstaller`` backend.
|
|
||||||
|
|
||||||
:param venv: Tox virtual environment object with configuration for the local Tox environment.
|
|
||||||
:param action: Tox action object
|
|
||||||
"""
|
|
||||||
try:
|
|
||||||
poetry = utilities.check_preconditions(tox_env)
|
|
||||||
except exceptions.SkipEnvironment as err:
|
|
||||||
if (
|
|
||||||
isinstance(err, exceptions.PoetryNotInstalledError)
|
|
||||||
and tox_env.conf["require_poetry"]
|
|
||||||
):
|
|
||||||
logger.error(str(err))
|
|
||||||
raise err
|
|
||||||
logger.info(str(err))
|
|
||||||
return
|
|
||||||
|
|
||||||
logger.info(f"Loaded project pyproject.toml from {poetry.file}")
|
|
||||||
|
|
||||||
virtualenv = utilities.convert_virtualenv(tox_env)
|
|
||||||
|
|
||||||
if not poetry.locker.is_fresh():
|
|
||||||
logger.warning(
|
|
||||||
f"The Poetry lock file is not up to date with the latest changes in {poetry.file}"
|
|
||||||
)
|
|
||||||
|
|
||||||
try:
|
|
||||||
if tox_env.conf["require_locked_deps"] and tox_env.conf["deps"].lines():
|
|
||||||
raise exceptions.LockedDepsRequiredError(
|
|
||||||
f"Unlocked dependencies '{tox_env.conf['deps']}' specified for environment '{tox_env.name}' which requires locked dependencies"
|
|
||||||
)
|
|
||||||
|
|
||||||
packages = utilities.build_package_map(poetry)
|
|
||||||
|
|
||||||
group_deps = utilities.dedupe_packages(
|
|
||||||
list(
|
|
||||||
chain(
|
|
||||||
*[
|
|
||||||
utilities.find_group_deps(group, packages, virtualenv, poetry)
|
|
||||||
for group in tox_env.conf["poetry_dep_groups"]
|
|
||||||
]
|
|
||||||
)
|
|
||||||
)
|
|
||||||
)
|
|
||||||
logger.info(
|
|
||||||
f"Identified {len(group_deps)} group dependencies to install to env"
|
|
||||||
)
|
|
||||||
|
|
||||||
env_deps = utilities.find_additional_deps(
|
|
||||||
packages, virtualenv, poetry, tox_env.conf["locked_deps"]
|
|
||||||
)
|
|
||||||
|
|
||||||
logger.info(
|
|
||||||
f"Identified {len(env_deps)} environment dependencies to install to env"
|
|
||||||
)
|
|
||||||
|
|
||||||
# extras are not set in a testenv if skip_install=true
|
|
||||||
try:
|
|
||||||
extras = tox_env.conf["extras"]
|
|
||||||
except KeyError:
|
|
||||||
extras = []
|
|
||||||
|
|
||||||
if tox_env.conf["install_project_deps"]:
|
|
||||||
project_deps = utilities.find_project_deps(
|
|
||||||
packages, virtualenv, poetry, extras
|
|
||||||
)
|
|
||||||
logger.info(
|
|
||||||
f"Identified {len(project_deps)} project dependencies to install to env"
|
|
||||||
)
|
|
||||||
else:
|
|
||||||
project_deps = []
|
|
||||||
logger.info("Env does not install project package dependencies, skipping")
|
|
||||||
except exceptions.ToxPoetryInstallerException as err:
|
|
||||||
logger.error(str(err))
|
|
||||||
raise err
|
|
||||||
except Exception as err:
|
|
||||||
logger.error(f"Internal plugin error: {err}")
|
|
||||||
raise err
|
|
||||||
|
|
||||||
dependencies = utilities.dedupe_packages(group_deps + env_deps + project_deps)
|
|
||||||
|
|
||||||
logger.info(f"Installing {len(dependencies)} dependencies from Poetry lock file")
|
|
||||||
installer.install(
|
|
||||||
poetry,
|
|
||||||
tox_env,
|
|
||||||
dependencies,
|
|
||||||
tox_env.options.parallel_install_threads,
|
|
||||||
)
|
|
||||||
4
tox_poetry_installer/hooks/__init__.py
Normal file
4
tox_poetry_installer/hooks/__init__.py
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
# pylint: disable=missing-module-docstring
|
||||||
|
from tox_poetry_installer.hooks.tox_add_env_config import tox_add_env_config
|
||||||
|
from tox_poetry_installer.hooks.tox_add_option import tox_add_option
|
||||||
|
from tox_poetry_installer.hooks.tox_on_install import tox_on_install
|
||||||
@@ -1,32 +1,38 @@
|
|||||||
"""Helper utility functions, usually bridging Tox and Poetry functionality"""
|
"""Helper functions for the :func:`tox_on_install` hook"""
|
||||||
# Silence this one globally to support the internal function imports for the proxied poetry module.
|
|
||||||
# See the docstring in 'tox_poetry_installer._poetry' for more context.
|
|
||||||
# pylint: disable=import-outside-toplevel
|
|
||||||
import collections
|
import collections
|
||||||
import typing
|
import concurrent.futures
|
||||||
from pathlib import Path
|
import contextlib
|
||||||
|
import datetime
|
||||||
|
import pathlib
|
||||||
|
from typing import Collection
|
||||||
from typing import Dict
|
from typing import Dict
|
||||||
from typing import List
|
from typing import List
|
||||||
from typing import Sequence
|
from typing import Sequence
|
||||||
from typing import Set
|
from typing import Set
|
||||||
|
|
||||||
from poetry.core.packages.dependency import Dependency as PoetryDependency
|
import cleo.io.null_io
|
||||||
from poetry.core.packages.package import Package as PoetryPackage
|
import packaging.utils
|
||||||
from tox.tox_env.api import ToxEnv as ToxVirtualEnv
|
import poetry.config.config
|
||||||
from tox.tox_env.package import PackageToxEnv
|
import poetry.core.packages.dependency
|
||||||
|
import poetry.core.packages.package
|
||||||
|
import poetry.factory
|
||||||
|
import poetry.installation.executor
|
||||||
|
import poetry.installation.operations.install
|
||||||
|
import poetry.poetry
|
||||||
|
import poetry.utils.env
|
||||||
|
import tox.tox_env.api
|
||||||
|
import tox.tox_env.package
|
||||||
|
|
||||||
from tox_poetry_installer import constants
|
from tox_poetry_installer import constants
|
||||||
from tox_poetry_installer import exceptions
|
from tox_poetry_installer import exceptions
|
||||||
from tox_poetry_installer import logger
|
from tox_poetry_installer import logger
|
||||||
|
|
||||||
if typing.TYPE_CHECKING:
|
|
||||||
from tox_poetry_installer import _poetry
|
PackageMap = Dict[str, List[poetry.core.packages.package.Package]]
|
||||||
|
|
||||||
|
|
||||||
PackageMap = Dict[str, List[PoetryPackage]]
|
def check_preconditions(venv: tox.tox_env.api.ToxEnv) -> poetry.poetry.Poetry:
|
||||||
|
|
||||||
|
|
||||||
def check_preconditions(venv: ToxVirtualEnv) -> "_poetry.Poetry":
|
|
||||||
"""Check that the local project environment meets expectations"""
|
"""Check that the local project environment meets expectations"""
|
||||||
|
|
||||||
# Skip running the plugin for the provisioning environment. The provisioned environment,
|
# Skip running the plugin for the provisioning environment. The provisioned environment,
|
||||||
@@ -34,61 +40,28 @@ def check_preconditions(venv: ToxVirtualEnv) -> "_poetry.Poetry":
|
|||||||
# handled by Tox and is out of scope for this plugin. Since one of the ways to install this
|
# handled by Tox and is out of scope for this plugin. Since one of the ways to install this
|
||||||
# plugin in the first place is via the Tox provisioning environment, it quickly becomes a
|
# plugin in the first place is via the Tox provisioning environment, it quickly becomes a
|
||||||
# chicken-and-egg problem.
|
# chicken-and-egg problem.
|
||||||
if isinstance(venv, PackageToxEnv):
|
if isinstance(venv, tox.tox_env.package.PackageToxEnv):
|
||||||
raise exceptions.SkipEnvironment(f"Skipping Tox provisioning env '{venv.name}'")
|
raise exceptions.SkipEnvironment(f"Skipping Tox provisioning env '{venv.name}'")
|
||||||
|
|
||||||
if venv.options.require_poetry:
|
|
||||||
logger.warning(
|
|
||||||
"DEPRECATION: The '--require-poetry' runtime option is deprecated and will be "
|
|
||||||
"removed in version 1.0.0. Please update test environments that require Poetry to "
|
|
||||||
"set the 'require_poetry = true' option in tox.ini"
|
|
||||||
)
|
|
||||||
|
|
||||||
from tox_poetry_installer import _poetry
|
|
||||||
|
|
||||||
try:
|
try:
|
||||||
return _poetry.Factory().create_poetry(venv.core["tox_root"])
|
return poetry.factory.Factory().create_poetry(venv.core["tox_root"])
|
||||||
# Support running the plugin when the current tox project does not use Poetry for its
|
# Support running the plugin when the current tox project does not use Poetry for its
|
||||||
# environment/dependency management.
|
# environment/dependency management.
|
||||||
#
|
#
|
||||||
# ``RuntimeError`` is dangerous to blindly catch because it can be (and in Poetry's case,
|
# ``RuntimeError`` is dangerous to blindly catch because it can be (and in Poetry's case,
|
||||||
# is) raised in many different places for different purposes.
|
# is) raised in many different places for different purposes.
|
||||||
except RuntimeError:
|
except RuntimeError as err:
|
||||||
raise exceptions.SkipEnvironment(
|
raise exceptions.SkipEnvironment(
|
||||||
"Project does not use Poetry for env management, skipping installation of locked dependencies"
|
f"Skipping installation of locked dependencies due to a Poetry error: {err}"
|
||||||
) from None
|
) from None
|
||||||
|
|
||||||
|
|
||||||
def convert_virtualenv(venv: ToxVirtualEnv) -> "_poetry.VirtualEnv":
|
|
||||||
"""Convert a Tox venv to a Poetry venv
|
|
||||||
|
|
||||||
:param venv: Tox ``VirtualEnv`` object representing a tox virtual environment
|
|
||||||
:returns: Poetry ``VirtualEnv`` object representing a poetry virtual environment
|
|
||||||
"""
|
|
||||||
from tox_poetry_installer import _poetry
|
|
||||||
|
|
||||||
return _poetry.VirtualEnv(path=Path(venv.env_dir))
|
|
||||||
|
|
||||||
|
|
||||||
def build_package_map(poetry: "_poetry.Poetry") -> PackageMap:
|
|
||||||
"""Build the mapping of package names to objects
|
|
||||||
|
|
||||||
:param poetry: Populated poetry object to load locked packages from
|
|
||||||
:returns: Mapping of package names to Poetry package objects
|
|
||||||
"""
|
|
||||||
packages = collections.defaultdict(list)
|
|
||||||
for package in poetry.locker.locked_repository().packages:
|
|
||||||
packages[package.name].append(package)
|
|
||||||
|
|
||||||
return packages
|
|
||||||
|
|
||||||
|
|
||||||
def identify_transients(
|
def identify_transients(
|
||||||
dep_name: str,
|
dep_name: str,
|
||||||
packages: PackageMap,
|
packages: PackageMap,
|
||||||
venv: "_poetry.VirtualEnv",
|
venv: poetry.utils.env.VirtualEnv,
|
||||||
allow_missing: Sequence[str] = (),
|
allow_missing: Sequence[str] = (),
|
||||||
) -> List[PoetryPackage]:
|
) -> List[poetry.core.packages.package.Package]:
|
||||||
"""Using a pool of packages, identify all transient dependencies of a given package name
|
"""Using a pool of packages, identify all transient dependencies of a given package name
|
||||||
|
|
||||||
:param dep_name: Either the Poetry dependency or the dependency's bare package name to recursively
|
:param dep_name: Either the Poetry dependency or the dependency's bare package name to recursively
|
||||||
@@ -105,10 +78,12 @@ def identify_transients(
|
|||||||
"""
|
"""
|
||||||
searched: Set[str] = set()
|
searched: Set[str] = set()
|
||||||
|
|
||||||
def _transients(transient: PoetryDependency) -> List[PoetryPackage]:
|
def _transients(
|
||||||
|
transient: poetry.core.packages.dependency.Dependency,
|
||||||
|
) -> List[poetry.core.packages.package.Package]:
|
||||||
searched.add(transient.name)
|
searched.add(transient.name)
|
||||||
|
|
||||||
results: List[PoetryPackage] = []
|
results: List[poetry.core.packages.package.Package] = []
|
||||||
for option in packages[transient.name]:
|
for option in packages[transient.name]:
|
||||||
if venv.is_valid_for_marker(option.to_dependency().marker):
|
if venv.is_valid_for_marker(option.to_dependency().marker):
|
||||||
for requirement in option.requires:
|
for requirement in option.requires:
|
||||||
@@ -139,13 +114,6 @@ def identify_transients(
|
|||||||
except KeyError as err:
|
except KeyError as err:
|
||||||
missing = err.args[0]
|
missing = err.args[0]
|
||||||
|
|
||||||
if missing in constants.UNSAFE_PACKAGES:
|
|
||||||
logger.warning(
|
|
||||||
f"Installing package '{missing}' using Poetry is not supported and will be skipped"
|
|
||||||
)
|
|
||||||
logger.debug(f"Skipping {missing}: designated unsafe by Poetry")
|
|
||||||
return []
|
|
||||||
|
|
||||||
if missing in allow_missing:
|
if missing in allow_missing:
|
||||||
logger.debug(f"Skipping {missing}: package is allowed to be unlocked")
|
logger.debug(f"Skipping {missing}: package is allowed to be unlocked")
|
||||||
return []
|
return []
|
||||||
@@ -164,43 +132,43 @@ def identify_transients(
|
|||||||
|
|
||||||
def find_project_deps(
|
def find_project_deps(
|
||||||
packages: PackageMap,
|
packages: PackageMap,
|
||||||
venv: "_poetry.VirtualEnv",
|
venv: poetry.utils.env.VirtualEnv,
|
||||||
poetry: "_poetry.Poetry",
|
project: poetry.poetry.Poetry,
|
||||||
extras: Sequence[str] = (),
|
extras: Sequence[str] = (),
|
||||||
) -> List[PoetryPackage]:
|
) -> List[poetry.core.packages.package.Package]:
|
||||||
"""Find the root project dependencies
|
"""Find the root project dependencies
|
||||||
|
|
||||||
Recursively identify the dependencies of the root project package
|
Recursively identify the dependencies of the root project package
|
||||||
|
|
||||||
:param packages: Mapping of all locked package names to their corresponding package object
|
:param packages: Mapping of all locked package names to their corresponding package object
|
||||||
:param venv: Poetry virtual environment to use for package compatibility checks
|
:param venv: Poetry virtual environment to use for package compatibility checks
|
||||||
:param poetry: Poetry object for the current project
|
:param project: Poetry object for the current project
|
||||||
:param extras: Sequence of extra names to include the dependencies of
|
:param extras: Sequence of extra names to include the dependencies of
|
||||||
"""
|
"""
|
||||||
|
|
||||||
if any(dep.name in constants.UNSAFE_PACKAGES for dep in poetry.package.requires):
|
|
||||||
raise exceptions.RequiresUnsafeDepError(
|
|
||||||
f"Project package requires one or more unsafe dependencies ({', '.join(constants.UNSAFE_PACKAGES)}) which cannot be installed with Poetry"
|
|
||||||
)
|
|
||||||
|
|
||||||
required_dep_names = [
|
required_dep_names = [
|
||||||
item.name for item in poetry.package.requires if not item.is_optional()
|
item.name for item in project.package.requires if not item.is_optional()
|
||||||
]
|
]
|
||||||
|
|
||||||
extra_dep_names: List[str] = []
|
extra_dep_names: List[str] = []
|
||||||
for extra in extras:
|
for extra in extras:
|
||||||
logger.info(f"Processing project extra '{extra}'")
|
logger.info(f"Processing project extra '{extra}'")
|
||||||
try:
|
try:
|
||||||
extra_dep_names += [item.name for item in poetry.package.extras[extra]]
|
extra_dep_names += [
|
||||||
|
item.name
|
||||||
|
for item in project.package.extras[
|
||||||
|
packaging.utils.NormalizedName(extra)
|
||||||
|
]
|
||||||
|
]
|
||||||
except KeyError:
|
except KeyError:
|
||||||
raise exceptions.ExtraNotFoundError(
|
raise exceptions.ExtraNotFoundError(
|
||||||
f"Environment specifies project extra '{extra}' which was not found in the lockfile"
|
f"Environment specifies project extra '{extra}' which was not found in the lockfile"
|
||||||
) from None
|
) from None
|
||||||
|
|
||||||
dependencies: List[PoetryPackage] = []
|
dependencies: List[poetry.core.packages.package.Package] = []
|
||||||
for dep_name in required_dep_names + extra_dep_names:
|
for dep_name in required_dep_names + extra_dep_names:
|
||||||
dependencies += identify_transients(
|
dependencies += identify_transients(
|
||||||
dep_name.lower(), packages, venv, allow_missing=[poetry.package.name]
|
dep_name.lower(), packages, venv, allow_missing=[project.package.name]
|
||||||
)
|
)
|
||||||
|
|
||||||
return dedupe_packages(dependencies)
|
return dedupe_packages(dependencies)
|
||||||
@@ -208,24 +176,24 @@ def find_project_deps(
|
|||||||
|
|
||||||
def find_additional_deps(
|
def find_additional_deps(
|
||||||
packages: PackageMap,
|
packages: PackageMap,
|
||||||
venv: "_poetry.VirtualEnv",
|
venv: poetry.utils.env.VirtualEnv,
|
||||||
poetry: "_poetry.Poetry",
|
project: poetry.poetry.Poetry,
|
||||||
dep_names: Sequence[str],
|
dep_names: Sequence[str],
|
||||||
) -> List[PoetryPackage]:
|
) -> List[poetry.core.packages.package.Package]:
|
||||||
"""Find additional dependencies
|
"""Find additional dependencies
|
||||||
|
|
||||||
Recursively identify the dependencies of an arbitrary list of package names
|
Recursively identify the dependencies of an arbitrary list of package names
|
||||||
|
|
||||||
:param packages: Mapping of all locked package names to their corresponding package object
|
:param packages: Mapping of all locked package names to their corresponding package object
|
||||||
:param venv: Poetry virtual environment to use for package compatibility checks
|
:param venv: Poetry virtual environment to use for package compatibility checks
|
||||||
:param poetry: Poetry object for the current project
|
:param project: Poetry object for the current project
|
||||||
:param dep_names: Sequence of additional dependency names to recursively find the transient
|
:param dep_names: Sequence of additional dependency names to recursively find the transient
|
||||||
dependencies for
|
dependencies for
|
||||||
"""
|
"""
|
||||||
dependencies: List[PoetryPackage] = []
|
dependencies: List[poetry.core.packages.package.Package] = []
|
||||||
for dep_name in dep_names:
|
for dep_name in dep_names:
|
||||||
dependencies += identify_transients(
|
dependencies += identify_transients(
|
||||||
dep_name.lower(), packages, venv, allow_missing=[poetry.package.name]
|
dep_name.lower(), packages, venv, allow_missing=[project.package.name]
|
||||||
)
|
)
|
||||||
|
|
||||||
return dedupe_packages(dependencies)
|
return dedupe_packages(dependencies)
|
||||||
@@ -234,9 +202,9 @@ def find_additional_deps(
|
|||||||
def find_group_deps(
|
def find_group_deps(
|
||||||
group: str,
|
group: str,
|
||||||
packages: PackageMap,
|
packages: PackageMap,
|
||||||
venv: "_poetry.VirtualEnv",
|
venv: poetry.utils.env.VirtualEnv,
|
||||||
poetry: "_poetry.Poetry",
|
project: poetry.poetry.Poetry,
|
||||||
) -> List[PoetryPackage]:
|
) -> List[poetry.core.packages.package.Package]:
|
||||||
"""Find the dependencies belonging to a dependency group
|
"""Find the dependencies belonging to a dependency group
|
||||||
|
|
||||||
Recursively identify the Poetry dev dependencies
|
Recursively identify the Poetry dev dependencies
|
||||||
@@ -244,13 +212,17 @@ def find_group_deps(
|
|||||||
:param group: Name of the dependency group from the project's ``pyproject.toml``
|
:param group: Name of the dependency group from the project's ``pyproject.toml``
|
||||||
:param packages: Mapping of all locked package names to their corresponding package object
|
:param packages: Mapping of all locked package names to their corresponding package object
|
||||||
:param venv: Poetry virtual environment to use for package compatibility checks
|
:param venv: Poetry virtual environment to use for package compatibility checks
|
||||||
:param poetry: Poetry object for the current project
|
:param project: Poetry object for the current project
|
||||||
"""
|
"""
|
||||||
return find_additional_deps(
|
return find_additional_deps(
|
||||||
packages,
|
packages,
|
||||||
venv,
|
venv,
|
||||||
poetry,
|
project,
|
||||||
poetry.pyproject.data["tool"]["poetry"]
|
# the type ignore here is due to the difficulties around getting nested data
|
||||||
|
# from the inherrently unstructured toml structure (which necessarily is flexibly
|
||||||
|
# typed) but in a situation where there is a meta-structure applied to it (i.e. a
|
||||||
|
# pyproject.toml structure).
|
||||||
|
project.pyproject.data["tool"]["poetry"] # type: ignore
|
||||||
.get("group", {})
|
.get("group", {})
|
||||||
.get(group, {})
|
.get(group, {})
|
||||||
.get("dependencies", {})
|
.get("dependencies", {})
|
||||||
@@ -259,36 +231,136 @@ def find_group_deps(
|
|||||||
|
|
||||||
|
|
||||||
def find_dev_deps(
|
def find_dev_deps(
|
||||||
packages: PackageMap, venv: "_poetry.VirtualEnv", poetry: "_poetry.Poetry"
|
packages: PackageMap,
|
||||||
) -> List[PoetryPackage]:
|
venv: poetry.utils.env.VirtualEnv,
|
||||||
|
project: poetry.poetry.Poetry,
|
||||||
|
) -> List[poetry.core.packages.package.Package]:
|
||||||
"""Find the dev dependencies
|
"""Find the dev dependencies
|
||||||
|
|
||||||
Recursively identify the Poetry dev dependencies
|
Recursively identify the Poetry dev dependencies
|
||||||
|
|
||||||
:param packages: Mapping of all locked package names to their corresponding package object
|
:param packages: Mapping of all locked package names to their corresponding package object
|
||||||
:param venv: Poetry virtual environment to use for package compatibility checks
|
:param venv: Poetry virtual environment to use for package compatibility checks
|
||||||
:param poetry: Poetry object for the current project
|
:param project: Poetry object for the current project
|
||||||
"""
|
"""
|
||||||
dev_group_deps = find_group_deps("dev", packages, venv, poetry)
|
dev_group_deps = find_group_deps("dev", packages, venv, project)
|
||||||
|
|
||||||
# Legacy pyproject.toml poetry format:
|
# Legacy pyproject.toml poetry format:
|
||||||
legacy_dev_group_deps = find_additional_deps(
|
legacy_dev_group_deps = find_additional_deps(
|
||||||
packages,
|
packages,
|
||||||
venv,
|
venv,
|
||||||
poetry,
|
project,
|
||||||
poetry.pyproject.data["tool"]["poetry"].get("dev-dependencies", {}).keys(),
|
# the type ignore here is due to the difficulties around getting nested data
|
||||||
|
# from the inherrently unstructured toml structure (which necessarily is flexibly
|
||||||
|
# typed) but in a situation where there is a meta-structure applied to it (i.e. a
|
||||||
|
# pyproject.toml structure).
|
||||||
|
project.pyproject.data["tool"]["poetry"].get("dev-dependencies", {}).keys(), # type: ignore
|
||||||
)
|
)
|
||||||
|
|
||||||
# Poetry 1.2 unions these two toml sections.
|
# Poetry 1.2 unions these two toml sections.
|
||||||
return dedupe_packages(dev_group_deps + legacy_dev_group_deps)
|
return dedupe_packages(dev_group_deps + legacy_dev_group_deps)
|
||||||
|
|
||||||
|
|
||||||
def dedupe_packages(packages: Sequence[PoetryPackage]) -> List[PoetryPackage]:
|
@contextlib.contextmanager
|
||||||
"""Deduplicates a sequence of PoetryPackages while preserving ordering
|
def _optional_parallelize(parallels: int):
|
||||||
|
"""A bit of cheat, really
|
||||||
|
|
||||||
|
A context manager that exposes a common interface for the caller that optionally
|
||||||
|
enables/disables the usage of the parallel thread pooler depending on the value of
|
||||||
|
the ``parallels`` parameter.
|
||||||
|
"""
|
||||||
|
if parallels > 0:
|
||||||
|
with concurrent.futures.ThreadPoolExecutor(max_workers=parallels) as executor:
|
||||||
|
yield executor.submit
|
||||||
|
else:
|
||||||
|
yield lambda func, arg: func(arg)
|
||||||
|
|
||||||
|
|
||||||
|
def install_package(
|
||||||
|
project: poetry.poetry.Poetry,
|
||||||
|
venv: tox.tox_env.api.ToxEnv,
|
||||||
|
packages: Collection[poetry.core.packages.package.Package],
|
||||||
|
parallels: int = 0,
|
||||||
|
):
|
||||||
|
"""Install a bunch of packages to a virtualenv
|
||||||
|
|
||||||
|
:param project: Poetry object the packages were sourced from
|
||||||
|
:param venv: Tox virtual environment to install the packages to
|
||||||
|
:param packages: List of packages to install to the virtual environment
|
||||||
|
:param parallels: Number of parallel processes to use for installing dependency packages, or
|
||||||
|
``None`` to disable parallelization.
|
||||||
|
"""
|
||||||
|
|
||||||
|
logger.info(f"Installing {len(packages)} packages to environment at {venv.env_dir}")
|
||||||
|
|
||||||
|
install_executor = poetry.installation.executor.Executor(
|
||||||
|
env=convert_virtualenv(venv),
|
||||||
|
io=cleo.io.null_io.NullIO(),
|
||||||
|
pool=project.pool,
|
||||||
|
config=poetry.config.config.Config(),
|
||||||
|
)
|
||||||
|
|
||||||
|
installed: Set[poetry.core.packages.package.Package] = set()
|
||||||
|
|
||||||
|
def logged_install(dependency: poetry.core.packages.package.Package) -> None:
|
||||||
|
start = datetime.datetime.now()
|
||||||
|
logger.debug(f"Installing {dependency}")
|
||||||
|
install_executor.execute(
|
||||||
|
[poetry.installation.operations.install.Install(package=dependency)]
|
||||||
|
)
|
||||||
|
end = datetime.datetime.now()
|
||||||
|
logger.debug(f"Finished installing {dependency} in {end - start}")
|
||||||
|
|
||||||
|
with _optional_parallelize(parallels) as executor:
|
||||||
|
futures = []
|
||||||
|
for dependency in packages:
|
||||||
|
if dependency not in installed:
|
||||||
|
installed.add(dependency)
|
||||||
|
logger.debug(f"Queuing {dependency}")
|
||||||
|
future = executor(logged_install, dependency)
|
||||||
|
if future is not None:
|
||||||
|
futures.append(future)
|
||||||
|
else:
|
||||||
|
logger.debug(f"Skipping {dependency}, already installed")
|
||||||
|
logger.debug("Waiting for installs to finish...")
|
||||||
|
|
||||||
|
for future in concurrent.futures.as_completed(futures):
|
||||||
|
# Don't actually care about the return value, just waiting on the
|
||||||
|
# future to ensure any exceptions that were raised in the called
|
||||||
|
# function are propagated.
|
||||||
|
future.result()
|
||||||
|
|
||||||
|
|
||||||
|
def dedupe_packages(
|
||||||
|
packages: Sequence[poetry.core.packages.package.Package],
|
||||||
|
) -> List[poetry.core.packages.package.Package]:
|
||||||
|
"""Deduplicates a sequence of Packages while preserving ordering
|
||||||
|
|
||||||
Adapted from StackOverflow: https://stackoverflow.com/a/480227
|
Adapted from StackOverflow: https://stackoverflow.com/a/480227
|
||||||
"""
|
"""
|
||||||
seen: Set[PoetryPackage] = set()
|
seen: Set[poetry.core.packages.package.Package] = set()
|
||||||
# Make this faster, avoid method lookup below
|
# Make this faster, avoid method lookup below
|
||||||
seen_add = seen.add
|
seen_add = seen.add
|
||||||
return [p for p in packages if not (p in seen or seen_add(p))]
|
return [item for item in packages if not (item in seen or seen_add(item))]
|
||||||
|
|
||||||
|
|
||||||
|
def convert_virtualenv(venv: tox.tox_env.api.ToxEnv) -> poetry.utils.env.VirtualEnv:
|
||||||
|
"""Convert a Tox venv to a Poetry venv
|
||||||
|
|
||||||
|
:param venv: Tox ``VirtualEnv`` object representing a tox virtual environment
|
||||||
|
:returns: Poetry ``VirtualEnv`` object representing a poetry virtual environment
|
||||||
|
"""
|
||||||
|
return poetry.utils.env.VirtualEnv(path=pathlib.Path(venv.env_dir))
|
||||||
|
|
||||||
|
|
||||||
|
def build_package_map(project: poetry.poetry.Poetry) -> PackageMap:
|
||||||
|
"""Build the mapping of package names to objects
|
||||||
|
|
||||||
|
:param project: Populated poetry object to load locked packages from
|
||||||
|
:returns: Mapping of package names to Poetry package objects
|
||||||
|
"""
|
||||||
|
packages = collections.defaultdict(list)
|
||||||
|
for package in project.locker.locked_repository().packages:
|
||||||
|
packages[str(package.name)].append(package)
|
||||||
|
|
||||||
|
return packages
|
||||||
47
tox_poetry_installer/hooks/tox_add_env_config.py
Normal file
47
tox_poetry_installer/hooks/tox_add_env_config.py
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
"""Add required env configuration options to the tox INI file"""
|
||||||
|
|
||||||
|
from typing import List
|
||||||
|
|
||||||
|
import tox.config.sets
|
||||||
|
import tox.plugin
|
||||||
|
|
||||||
|
|
||||||
|
# pylint: disable=missing-function-docstring
|
||||||
|
@tox.plugin.impl
|
||||||
|
def tox_add_env_config(
|
||||||
|
env_conf: tox.config.sets.EnvConfigSet,
|
||||||
|
):
|
||||||
|
env_conf.add_config(
|
||||||
|
"poetry_dep_groups",
|
||||||
|
of_type=List[str],
|
||||||
|
default=[],
|
||||||
|
desc="List of Poetry dependency groups to install to the environment",
|
||||||
|
)
|
||||||
|
|
||||||
|
env_conf.add_config(
|
||||||
|
"install_project_deps",
|
||||||
|
of_type=bool,
|
||||||
|
default=True,
|
||||||
|
desc="Automatically install all Poetry primary dependencies to the environment",
|
||||||
|
)
|
||||||
|
|
||||||
|
env_conf.add_config(
|
||||||
|
"require_locked_deps",
|
||||||
|
of_type=bool,
|
||||||
|
default=False,
|
||||||
|
desc="Require all dependencies in the environment be installed using the Poetry lockfile",
|
||||||
|
)
|
||||||
|
|
||||||
|
env_conf.add_config(
|
||||||
|
"require_poetry",
|
||||||
|
of_type=bool,
|
||||||
|
default=False,
|
||||||
|
desc="Trigger a failure if Poetry is not available to Tox",
|
||||||
|
)
|
||||||
|
|
||||||
|
env_conf.add_config(
|
||||||
|
"locked_deps",
|
||||||
|
of_type=List[str],
|
||||||
|
default=[],
|
||||||
|
desc="List of locked dependencies to install to the environment using the Poetry lockfile",
|
||||||
|
)
|
||||||
18
tox_poetry_installer/hooks/tox_add_option.py
Normal file
18
tox_poetry_installer/hooks/tox_add_option.py
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
"""Add additional command line arguments to tox to configure plugin behavior"""
|
||||||
|
|
||||||
|
import tox.config.cli.parser
|
||||||
|
import tox.plugin
|
||||||
|
|
||||||
|
from tox_poetry_installer import constants
|
||||||
|
|
||||||
|
|
||||||
|
# pylint: disable=missing-function-docstring
|
||||||
|
@tox.plugin.impl
|
||||||
|
def tox_add_option(parser: tox.config.cli.parser.ToxParser):
|
||||||
|
parser.add_argument(
|
||||||
|
"--parallel-install-threads",
|
||||||
|
type=int,
|
||||||
|
dest="parallel_install_threads",
|
||||||
|
default=constants.DEFAULT_INSTALL_THREADS,
|
||||||
|
help="Number of locked dependencies to install simultaneously; set to 0 to disable parallel installation",
|
||||||
|
)
|
||||||
114
tox_poetry_installer/hooks/tox_on_install.py
Normal file
114
tox_poetry_installer/hooks/tox_on_install.py
Normal file
@@ -0,0 +1,114 @@
|
|||||||
|
"""Install the dependencies for the current environment
|
||||||
|
|
||||||
|
Loads the local Poetry environment and the corresponding lockfile then pulls the dependencies
|
||||||
|
specified by the Tox environment. Finally these dependencies are installed into the Tox
|
||||||
|
environment using the Poetry ``PipInstaller`` backend.
|
||||||
|
"""
|
||||||
|
|
||||||
|
import itertools
|
||||||
|
|
||||||
|
import tox.plugin
|
||||||
|
import tox.tox_env.api
|
||||||
|
|
||||||
|
from tox_poetry_installer import exceptions
|
||||||
|
from tox_poetry_installer import logger
|
||||||
|
from tox_poetry_installer.hooks._tox_on_install_helpers import build_package_map
|
||||||
|
from tox_poetry_installer.hooks._tox_on_install_helpers import check_preconditions
|
||||||
|
from tox_poetry_installer.hooks._tox_on_install_helpers import convert_virtualenv
|
||||||
|
from tox_poetry_installer.hooks._tox_on_install_helpers import dedupe_packages
|
||||||
|
from tox_poetry_installer.hooks._tox_on_install_helpers import find_additional_deps
|
||||||
|
from tox_poetry_installer.hooks._tox_on_install_helpers import find_group_deps
|
||||||
|
from tox_poetry_installer.hooks._tox_on_install_helpers import find_project_deps
|
||||||
|
from tox_poetry_installer.hooks._tox_on_install_helpers import install_package
|
||||||
|
|
||||||
|
|
||||||
|
# pylint: disable=missing-function-docstring,unused-argument
|
||||||
|
@tox.plugin.impl
|
||||||
|
def tox_on_install(tox_env: tox.tox_env.api.ToxEnv, *args) -> None:
|
||||||
|
try:
|
||||||
|
poetry = check_preconditions(tox_env)
|
||||||
|
except exceptions.SkipEnvironment as err:
|
||||||
|
if (
|
||||||
|
isinstance(err, exceptions.PoetryNotInstalledError)
|
||||||
|
and tox_env.conf["require_poetry"]
|
||||||
|
):
|
||||||
|
logger.error(str(err))
|
||||||
|
raise err
|
||||||
|
logger.info(str(err))
|
||||||
|
return
|
||||||
|
|
||||||
|
logger.info(f"Loaded project pyproject.toml from {poetry.file}")
|
||||||
|
|
||||||
|
virtualenv = convert_virtualenv(tox_env)
|
||||||
|
|
||||||
|
try:
|
||||||
|
if not poetry.locker.is_fresh():
|
||||||
|
logger.warning(
|
||||||
|
f"The Poetry lock file is not up to date with the latest changes in {poetry.file}"
|
||||||
|
)
|
||||||
|
except FileNotFoundError as err:
|
||||||
|
logger.error(f"Could not parse lockfile: {err}")
|
||||||
|
raise exceptions.LockfileParsingError(
|
||||||
|
f"Could not parse lockfile: {err}"
|
||||||
|
) from err
|
||||||
|
|
||||||
|
try:
|
||||||
|
if tox_env.conf["require_locked_deps"] and tox_env.conf["deps"].lines():
|
||||||
|
raise exceptions.LockedDepsRequiredError(
|
||||||
|
f"Unlocked dependencies '{tox_env.conf['deps']}' specified for environment '{tox_env.name}' which requires locked dependencies"
|
||||||
|
)
|
||||||
|
|
||||||
|
packages = build_package_map(poetry)
|
||||||
|
|
||||||
|
group_deps = dedupe_packages(
|
||||||
|
list(
|
||||||
|
itertools.chain(
|
||||||
|
*[
|
||||||
|
find_group_deps(group, packages, virtualenv, poetry)
|
||||||
|
for group in tox_env.conf["poetry_dep_groups"]
|
||||||
|
]
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
logger.info(
|
||||||
|
f"Identified {len(group_deps)} group dependencies to install to env"
|
||||||
|
)
|
||||||
|
|
||||||
|
env_deps = find_additional_deps(
|
||||||
|
packages, virtualenv, poetry, tox_env.conf["locked_deps"]
|
||||||
|
)
|
||||||
|
|
||||||
|
logger.info(
|
||||||
|
f"Identified {len(env_deps)} environment dependencies to install to env"
|
||||||
|
)
|
||||||
|
|
||||||
|
# extras are not set in a testenv if skip_install=true
|
||||||
|
try:
|
||||||
|
extras = tox_env.conf["extras"]
|
||||||
|
except KeyError:
|
||||||
|
extras = []
|
||||||
|
|
||||||
|
if tox_env.conf["install_project_deps"]:
|
||||||
|
project_deps = find_project_deps(packages, virtualenv, poetry, extras)
|
||||||
|
logger.info(
|
||||||
|
f"Identified {len(project_deps)} project dependencies to install to env"
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
project_deps = []
|
||||||
|
logger.info("Env does not install project package dependencies, skipping")
|
||||||
|
except exceptions.ToxPoetryInstallerException as err:
|
||||||
|
logger.error(str(err))
|
||||||
|
raise err
|
||||||
|
except Exception as err:
|
||||||
|
logger.error(f"Internal plugin error: {err}")
|
||||||
|
raise err
|
||||||
|
|
||||||
|
dependencies = dedupe_packages(group_deps + env_deps + project_deps)
|
||||||
|
|
||||||
|
logger.info(f"Installing {len(dependencies)} dependencies from Poetry lock file")
|
||||||
|
install_package(
|
||||||
|
poetry,
|
||||||
|
tox_env,
|
||||||
|
dependencies,
|
||||||
|
tox_env.options.parallel_install_threads,
|
||||||
|
)
|
||||||
@@ -1,88 +0,0 @@
|
|||||||
"""Funcationality for performing virtualenv installation"""
|
|
||||||
# Silence this one globally to support the internal function imports for the proxied poetry module.
|
|
||||||
# See the docstring in 'tox_poetry_installer._poetry' for more context.
|
|
||||||
# pylint: disable=import-outside-toplevel
|
|
||||||
import concurrent.futures
|
|
||||||
import contextlib
|
|
||||||
import typing
|
|
||||||
from datetime import datetime
|
|
||||||
from typing import Collection
|
|
||||||
from typing import Set
|
|
||||||
|
|
||||||
from tox.tox_env.api import ToxEnv as ToxVirtualEnv
|
|
||||||
|
|
||||||
from tox_poetry_installer import logger
|
|
||||||
from tox_poetry_installer import utilities
|
|
||||||
|
|
||||||
if typing.TYPE_CHECKING:
|
|
||||||
from tox_poetry_installer import _poetry
|
|
||||||
|
|
||||||
|
|
||||||
def install(
|
|
||||||
poetry: "_poetry.Poetry",
|
|
||||||
venv: ToxVirtualEnv,
|
|
||||||
packages: Collection["_poetry.PoetryPackage"],
|
|
||||||
parallels: int = 0,
|
|
||||||
):
|
|
||||||
"""Install a bunch of packages to a virtualenv
|
|
||||||
|
|
||||||
:param poetry: Poetry object the packages were sourced from
|
|
||||||
:param venv: Tox virtual environment to install the packages to
|
|
||||||
:param packages: List of packages to install to the virtual environment
|
|
||||||
:param parallels: Number of parallel processes to use for installing dependency packages, or
|
|
||||||
``None`` to disable parallelization.
|
|
||||||
"""
|
|
||||||
from tox_poetry_installer import _poetry
|
|
||||||
|
|
||||||
logger.info(f"Installing {len(packages)} packages to environment at {venv.env_dir}")
|
|
||||||
|
|
||||||
install_executor = _poetry.Executor(
|
|
||||||
env=utilities.convert_virtualenv(venv),
|
|
||||||
io=_poetry.NullIO(),
|
|
||||||
pool=poetry.pool,
|
|
||||||
config=_poetry.Config(),
|
|
||||||
)
|
|
||||||
|
|
||||||
installed: Set[_poetry.PoetryPackage] = set()
|
|
||||||
|
|
||||||
def logged_install(dependency: _poetry.PoetryPackage) -> None:
|
|
||||||
start = datetime.now()
|
|
||||||
logger.debug(f"Installing {dependency}")
|
|
||||||
install_executor.execute([_poetry.Install(package=dependency)])
|
|
||||||
end = datetime.now()
|
|
||||||
logger.debug(f"Finished installing {dependency} in {end - start}")
|
|
||||||
|
|
||||||
@contextlib.contextmanager
|
|
||||||
def _optional_parallelize():
|
|
||||||
"""A bit of cheat, really
|
|
||||||
|
|
||||||
A context manager that exposes a common interface for the caller that optionally
|
|
||||||
enables/disables the usage of the parallel thread pooler depending on the value of
|
|
||||||
the ``parallels`` parameter.
|
|
||||||
"""
|
|
||||||
if parallels > 0:
|
|
||||||
with concurrent.futures.ThreadPoolExecutor(
|
|
||||||
max_workers=parallels
|
|
||||||
) as executor:
|
|
||||||
yield executor.submit
|
|
||||||
else:
|
|
||||||
yield lambda func, arg: func(arg)
|
|
||||||
|
|
||||||
with _optional_parallelize() as executor:
|
|
||||||
futures = []
|
|
||||||
for dependency in packages:
|
|
||||||
if dependency not in installed:
|
|
||||||
installed.add(dependency)
|
|
||||||
logger.debug(f"Queuing {dependency}")
|
|
||||||
future = executor(logged_install, dependency)
|
|
||||||
if future is not None:
|
|
||||||
futures.append(future)
|
|
||||||
else:
|
|
||||||
logger.debug(f"Skipping {dependency}, already installed")
|
|
||||||
logger.debug("Waiting for installs to finish...")
|
|
||||||
|
|
||||||
for future in concurrent.futures.as_completed(futures):
|
|
||||||
# Don't actually care about the return value, just waiting on the
|
|
||||||
# future to ensure any exceptions that were raised in the called
|
|
||||||
# function are propagated.
|
|
||||||
future.result()
|
|
||||||
@@ -4,6 +4,7 @@ Calling ``tox.reporter.something()`` and having to format a string with the pref
|
|||||||
gets really old fast, but more importantly it also makes the flow of the main code
|
gets really old fast, but more importantly it also makes the flow of the main code
|
||||||
more difficult to follow because of the added complexity.
|
more difficult to follow because of the added complexity.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
import logging
|
import logging
|
||||||
|
|
||||||
from tox_poetry_installer import constants
|
from tox_poetry_installer import constants
|
||||||
|
|||||||
Reference in New Issue
Block a user