14 Commits
0.2.0 ... 0.2.3

Author SHA1 Message Date
707a73c922 Bump patch version 2020-09-29 19:01:59 -04:00
640dbfe102 Fix implicit case-sensitive locked dependency name handling
Fixes #7
2020-09-29 19:01:52 -04:00
05c5a26cc4 Update non-poetry error handling to be more permissive
Fixes #1
2020-09-29 19:01:43 -04:00
ee6f939b6a Fix links to roadmap issues 2020-09-29 00:58:04 -04:00
82678e81e8 Add issue links for relevant roadmap items 2020-09-29 00:56:22 -04:00
5411025612 Proofreading and editing fixes
Check spelling more thoroughly
Improve clarity in a few places
Fix grammar mistakes
Fix docs that didn't get updated for the 0.2 overhaul
2020-09-29 00:51:08 -04:00
2e1d5fc922 Bump patch version 2020-09-28 23:26:19 -04:00
b7961bec58 Add support for running in a non-poetry project
Handle the error poetry raises when not in a poetry managed project
  and use that as a sign to skip further usage of the plugin in that
  environment
2020-09-28 23:24:42 -04:00
e28159060d Add badge links
Add black code format badge
2020-09-27 20:59:41 -04:00
edcef918b3 Rename 'usage' section to 'usage examples' 2020-09-27 19:41:29 -04:00
beba9416be Add note about root env setting inheritance breaking child envs 2020-09-27 19:38:42 -04:00
18a74fab63 Bump patch version 2020-09-27 15:57:00 -04:00
516515b347 Fix duplicate installation of env dependencies
Fix always logging post-sorted unlocked env dependencies
2020-09-27 15:56:29 -04:00
c9f1f41163 Fix installing package deps when skipdist is true 2020-09-27 15:56:29 -04:00
3 changed files with 126 additions and 105 deletions

182
README.md
View File

@@ -1,19 +1,20 @@
# tox-poetry-installer # tox-poetry-installer
A plugin for [Tox](https://tox.readthedocs.io/en/latest/) that allows test environment A plugin for [Tox](https://tox.readthedocs.io/en/latest/) that allows test environment
dependencies to be installed using [Poetry](https://python-poetry.org/) using its lockfile. dependencies to be installed using [Poetry](https://python-poetry.org/) from its lockfile.
⚠️ **This project is alpha software and should not be used in a production capacity** ⚠️ ⚠️ **This project is alpha software and should not be used in production environments** ⚠️
![image](https://img.shields.io/pypi/l/tox-poetry-installer) [![image](https://img.shields.io/pypi/l/tox-poetry-installer)](https://opensource.org/licenses/MIT)
![image](https://img.shields.io/pypi/v/tox-poetry-installer) [![image](https://img.shields.io/pypi/v/tox-poetry-installer)](https://pypi.org/project/tox-poetry-installer/)
![image](https://img.shields.io/pypi/pyversions/tox-poetry-installer) [![image](https://img.shields.io/pypi/pyversions/tox-poetry-installer)](https://www.python.org)
[![Code style: black](https://img.shields.io/badge/code%20style-black-000000.svg)](https://github.com/psf/black)
**Documentation** **Documentation**
* [Installation](#installation) * [Installation](#installation)
* [Quick Start](#quick-start) * [Quick Start](#quick-start)
* [Usage](#usage) * [Usage Examples](#usage-examples)
* [Known Drawbacks and Problems](#known-drawbacks-and-problems) * [Known Drawbacks and Problems](#known-drawbacks-and-problems)
* [Why would I use this?](#why-would-i-use-this) (What problems does this solve?) * [Why would I use this?](#why-would-i-use-this) (What problems does this solve?)
* [Developing](#developing) * [Developing](#developing)
@@ -30,7 +31,7 @@ Related resources:
## Installation ## Installation
Add the plugin as a development dependency a project using Poetry: Add the plugin as a development dependency of a Poetry project:
``` ```
~ $: poetry add tox-poetry-installer --dev ~ $: poetry add tox-poetry-installer --dev
@@ -42,11 +43,10 @@ Confirm that the plugin is installed, and Tox recognizes it, by checking the Tox
~ $: poetry run tox --version ~ $: poetry run tox --version
3.20.0 imported from .venv/lib64/python3.8/site-packages/tox/__init__.py 3.20.0 imported from .venv/lib64/python3.8/site-packages/tox/__init__.py
registered plugins: registered plugins:
tox-poetry-installer-0.2.0 at .venv/lib64/python3.8/site-packages/tox_poetry_installer.py tox-poetry-installer-0.2.2 at .venv/lib64/python3.8/site-packages/tox_poetry_installer.py
``` ```
If using in a CI/automation environment using Pip, ensure that the plugin is installed to the If using Pip, ensure that the plugin is installed to the same environment as Tox:
same environment as Tox:
``` ```
# Calling the virtualenv's 'pip' binary directly will cause pip to install to that virtualenv # Calling the virtualenv's 'pip' binary directly will cause pip to install to that virtualenv
@@ -76,8 +76,8 @@ commands = ...
``` ```
To require specific dependencies be installed from the Poetry lockfile, and let the rest be To require specific dependencies be installed from the Poetry lockfile, and let the rest be
installed using the default Tox installation method, add the suffix `@poetry` to the dependencies. installed using the default Tox installation backend, add the suffix `@poetry` to the dependencies.
In the example below the `pytest`, `pytest-cov`, and `black` dependencies will be installed using In the example below the `pytest`, `pytest-cov`, and `black` dependencies will be installed from
the lockfile while `pylint` and `mypy` will be installed using the versions specified here: the lockfile while `pylint` and `mypy` will be installed using the versions specified here:
```ini ```ini
@@ -97,9 +97,9 @@ commands = ...
one Tox is testing) will always be installed from the lockfile. one Tox is testing) will always be installed from the lockfile.
## Usage ## Usage Examples
After installing the plugin to a project, your Tox automation is already benefiting from the After installing the plugin to a project your Tox automation is already benefiting from the
lockfile: when Tox installs your project package to one of your environments, all the dependencies lockfile: when Tox installs your project package to one of your environments, all the dependencies
of your project package will be installed using the versions specified in the lockfile. This of your project package will be installed using the versions specified in the lockfile. This
happens automatically and requires no configuration changes. happens automatically and requires no configuration changes.
@@ -107,7 +107,7 @@ happens automatically and requires no configuration changes.
But what about the rest of your Tox environment dependencies? But what about the rest of your Tox environment dependencies?
Let's use an example `tox.ini` file, below, that defines two environments: the main `testenv` for Let's use an example `tox.ini` file, below, that defines two environments: the main `testenv` for
running the project tests and `testenv:check` for running some other helpful checks: running the project tests and `testenv:check` for running some other helpful tools:
```ini ```ini
[tox] [tox]
@@ -151,9 +151,9 @@ Running Tox using this config gives us this error:
tox_poetry_installer.LockedDepVersionConflictError: Locked dependency 'pylint >=2.4.4,<2.6.0' cannot include version specifier tox_poetry_installer.LockedDepVersionConflictError: Locked dependency 'pylint >=2.4.4,<2.6.0' cannot include version specifier
``` ```
This is because we told the Tox environment to require all dependencies to be locked, but then also This is because we told the Tox environment to require all dependencies be locked, but then also
specified a specific version constraint for Pylint. With the `require_locked_deps = true` setting specified a specific version constraint for Pylint. With the `require_locked_deps = true` setting
Tox expects all dependencies to take their version from the lockfile, so when it got conflicting Tox expects all dependencies to take their version from the lockfile, so when it gets conflicting
information it errors. We can fix this by simply removing all version specifiers from the information it errors. We can fix this by simply removing all version specifiers from the
environment dependency list: environment dependency list:
@@ -174,9 +174,9 @@ recreated.
Now let's look at the `testenv` environment. Let's make the same changes to the `testenv` Now let's look at the `testenv` environment. Let's make the same changes to the `testenv`
environment that we made to `testenv:check` above; remove the PyTest version and add environment that we made to `testenv:check` above; remove the PyTest version and add
`require_locked_deps = true`. Then imagine that we want to add a new (made up) tool the test `require_locked_deps = true`. Then imagine that we want to add the
environment called `crash_override` to the environment: we can add `crash-override` as a dependency [Requests](https://requests.readthedocs.io/en/master/) library to the test environment: we
of the test environment, but this will cause an error: can add `requests` as a dependency of the test environment, but this will cause an error:
```ini ```ini
[testenv] [testenv]
@@ -184,30 +184,29 @@ description = Run the tests
require_locked_deps = true require_locked_deps = true
deps = deps =
pytest pytest
crash-override requests
commands = ... commands = ...
``` ```
Running Tox with this config gives us this error: Running Tox with this config gives us this error:
``` ```
tox_poetry_installer.LockedDepNotFoundError: No version of locked dependency 'crash-override' found in the project lockfile tox_poetry_installer.LockedDepNotFoundError: No version of locked dependency 'requests' found in the project lockfile
``` ```
This is because `crash-override` is not in our lockfile. Tox will refuse to install a dependency This is because `requests` is not in our lockfile yet. Tox will refuse to install a dependency
that isn't in the lockfile to an an environment that specifies `require_locked_deps = true`. We that isn't in the lockfile to an an environment that specifies `require_locked_deps = true`. We
could fix this (if `crash-override` was a real package) by running can fix this by running `poetry add requests --dev` to add it to the lockfile.
`poetry add crash-override --dev` to add it to the lockfile.
Now let's combine dependencies from the lockfile ("locked dependencies") with dependencies that are Now let's combine dependencies from the lockfile with dependencies that are
specified inline in the environment configuration ("unlocked dependencies"). specified in-line in the Tox environment configuration.
[This isn't generally recommended of course](#why-would-i-use-this), but it's a valid use case and [This isn't generally recommended](#why-would-i-use-this), but it is a valid use case and
fully supported by this plugin. Let's modify the `testenv` configuration to install PyTest from the fully supported by this plugin. Let's modify the `testenv` configuration to install PyTest
lockfile but then install an older version of the from the lockfile but then install an older version of the Requests library.
[Requests](https://requests.readthedocs.io/en/master/) library.
The first thing to do is remove the `require_locked_deps = true` setting so that we can install The first thing to do is remove the `require_locked_deps = true` setting so that we can install
Requests as an unlocked dependency. Then we can add our version of requests to the dependency list: Requests as an unlocked dependency. Then we can add our version specifier to the `requests`
entry in the dependency list:
```ini ```ini
[testenv] [testenv]
@@ -219,7 +218,8 @@ commands = ...
``` ```
However we still want `pytest` to be installed from the lockfile, so the final step is to tell Tox However we still want `pytest` to be installed from the lockfile, so the final step is to tell Tox
to install it from the lockfile by adding the suffix `@poetry` to it: to install it from the lockfile by adding the suffix `@poetry` to the `pytest` entry in the
dependency list:
```ini ```ini
[testenv] [testenv]
@@ -232,7 +232,7 @@ commands = ...
Now when the `testenv` environment is created it will install PyTest (and all of its dependencies) Now when the `testenv` environment is created it will install PyTest (and all of its dependencies)
from the lockfile while it will install Requests (and all of its dependencies) using the default from the lockfile while it will install Requests (and all of its dependencies) using the default
Tox installation backend using Pip. Tox installation backend.
## Known Drawbacks and Problems ## Known Drawbacks and Problems
@@ -253,6 +253,12 @@ Tox installation backend using Pip.
* The plugin currently depends on `poetry<1.1.0`. This can be a different version than Poetry being * The plugin currently depends on `poetry<1.1.0`. This can be a different version than Poetry being
used for actual project development. (See the [road map](#roadmap)) used for actual project development. (See the [road map](#roadmap))
* Tox environments automatically inherit their settings from the main `testenv` environment. This
means that if the `require_locked_deps = true` is specified for the `testenv` environment then
all environments will also require locked dependencies. This can be overwritten by explicitly
specifying `require_locked_deps = false` on child environments where unlocked dependencies are
needed.
* There are a handful of packages that cannot be installed from the lockfile, whether as specific * There are a handful of packages that cannot be installed from the lockfile, whether as specific
dependencies or as transient dependencies (dependencies of dependencies). This is due to dependencies or as transient dependencies (dependencies of dependencies). This is due to
[an ongoing discussion in the Poetry project](https://github.com/python-poetry/poetry/issues/1584); [an ongoing discussion in the Poetry project](https://github.com/python-poetry/poetry/issues/1584);
@@ -265,20 +271,20 @@ Tox installation backend using Pip.
**Introduction** **Introduction**
The lockfile is a file generated by a package manager for a project that lists what The lockfile is a file generated by a package manager for a project that records what
dependencies are installed, the versions of those dependencies, and additional metadata that dependencies are installed, the versions of those dependencies, and any additional metadata that
the package manager can use to recreate the local project environment. This allows developers the package manager needs to recreate the local project environment. This allows developers
to have confidence that a bug they are encountering that may be caused by one of their to have confidence that a bug they are encountering that may be caused by one of their
dependencies will be reproducible on another device. In addition, installing a project dependencies will be reproducible on another device. In addition, installing a project
environment from a lockfile gives confidence that automated systems running tests or performing environment from a lockfile gives confidence that automated systems running tests or performing
builds are using the same environment that a developer is. builds are using the same environment as a developer.
[Poetry](https://python-poetry.org/) is a project dependency manager for Python projects, and [Poetry](https://python-poetry.org/) is a project dependency manager for Python projects, and
as such it creates and manages a lockfile so that its users can benefit from all the features so it creates and manages a lockfile so that its users can benefit from all the features
described above. [Tox](https://tox.readthedocs.io/en/latest/#what-is-tox) is an automation tool described above. [Tox](https://tox.readthedocs.io/en/latest/#what-is-tox) is an automation tool
that allows Python developers to run tests suites, perform builds, and automate tasks within that allows Python developers to run tests suites, perform builds, and automate tasks within
self contained [Python virtual environments](https://docs.python.org/3/tutorial/venv.html). self-contained [Python virtual environments](https://docs.python.org/3/tutorial/venv.html).
To make these environments useful, Tox supports installing per-environment dependencies. To make these environments useful Tox supports installing dependencies in each environment.
However, since these environments are created on the fly and Tox does not maintain a lockfile, However, since these environments are created on the fly and Tox does not maintain a lockfile,
there can be subtle differences between the dependencies a developer is using and the there can be subtle differences between the dependencies a developer is using and the
dependencies Tox uses. dependencies Tox uses.
@@ -286,19 +292,17 @@ dependencies Tox uses.
This is where this plugin comes into play. This is where this plugin comes into play.
By default Tox uses [Pip](https://docs.python.org/3/tutorial/venv.html) to install the By default Tox uses [Pip](https://docs.python.org/3/tutorial/venv.html) to install the
PEP-508 compliant dependencies to a test environment. A more robust way to do this is to PEP-508 compliant dependencies to a test environment. This plugin extends the default
install dependencies directly from the lockfile so that the version installed to the Tox Tox dependency installation behavior to support installing dependencies using a Poetry-based
environment always matches the version Poetry specifies. This plugin overwrites the default installation method that makes use of the dependency metadata from Poetry's lockfile.
Tox dependency installation behavior and replaces it with a Poetry-based installation using
the dependency metadata from the lockfile.
**The Problem** **The Problem**
Environment dependencies for a Tox environment are usually done in PEP-508 format like the Environment dependencies for a Tox environment are usually specified in PEP-508 format, like
below example the below example:
```ini ```ini
# tox.ini # from tox.ini
... ...
[testenv] [testenv]
@@ -311,31 +315,35 @@ deps =
... ...
``` ```
Perhaps these dependencies are also useful during development, so they can be added to the Let's assume these dependencies are also useful during development, so they can be added to the
Poetry environment using this command: Poetry environment using this command:
``` ```
poetry add foo==1.2.3 bar>=1.3,<2.0 baz --dev poetry add --dev \
foo==1.2.3 \
bar>=1.3,<2.0 \
baz
``` ```
However there are three potential problems that could arise from each of these environment However there is a potential problem that could arise from each of these environment
dependencies that would _only_ appear in the Tox environment and not in the Poetry dependencies that would _only_ appear in the Tox environment and not in the Poetry
environment: environment in use by a developer:
* **The `foo` dependency is pinned to a specific version:** let's imagine a security * **The `foo` dependency is pinned to a specific version:** let's imagine a security
vulnerability is discovered in `foo` and the maintainers release version `1.2.4` to fix vulnerability is discovered in `foo` and the maintainers release version `1.2.4` to fix
it. A developer can run `poetry remove foo && poetry add foo^1.2` to get the new version, it. A developer can run `poetry remove foo` and then `poetry add foo^1.2` to get the new
but the Tox environment is left unchanged. The developer environment specified by the version, but the Tox environment is left unchanged. The development environment, as defined by
lockfile is now patched against the vulnerability, but the Tox environment is not. the lockfile, is now patched against the vulnerability but the Tox environment is not.
* **The `bar` dependency specifies a dynamic range:** a dynamic range allows a range of * **The `bar` dependency specifies a dynamic range:** a dynamic range allows a range of
versions to be installed, but the lockfile will have an exact version specified so that versions to be installed, but the lockfile will have an exact version specified so that
the Poetry environment is reproducible; this allows versions to be updated with the Poetry environment is reproducible; this allows versions to be updated with
`poetry update` rather than with the `remove` and `add` used above. If the maintainers of `poetry update` rather than with the `remove` and `add` commands used above. If the
`bar` release version `1.6.0` then the Tox environment will install it because it is valid maintainers of `bar` release version `1.6.0` then the Tox environment will install it
for the specified version range, meanwhile the Poetry environment will continue to install because it is valid for the specified version range. Meanwhile the Poetry environment will
the version from the lockfile until `poetry update bar` explicitly updates it. The continue to install the version from the lockfile until `poetry update bar` explicitly
development environment is now has a different version of `bar` than the Tox environment. updates it. The development environment is now has a different version of `bar` than the Tox
environment.
* **The `baz` dependency is unpinned:** unpinned dependencies are * **The `baz` dependency is unpinned:** unpinned dependencies are
[generally a bad idea](https://python-poetry.org/docs/faq/#why-are-unbound-version-constraints-a-bad-idea), [generally a bad idea](https://python-poetry.org/docs/faq/#why-are-unbound-version-constraints-a-bad-idea),
@@ -344,27 +352,26 @@ Poetry environment using this command:
but Pip (via Tox) will interpret it as a wildcard. If the latest version of `baz` is `1.0.0` but Pip (via Tox) will interpret it as a wildcard. If the latest version of `baz` is `1.0.0`
then `poetry add baz` will result in a constraint of `baz>=1.0.0,<2.0.0` while the Tox then `poetry add baz` will result in a constraint of `baz>=1.0.0,<2.0.0` while the Tox
environment will have a constraint of `baz==*`. The Tox environment can now install an environment will have a constraint of `baz==*`. The Tox environment can now install an
incompatible version of `baz` that cannot be easily caught using `poetry update`. incompatible version of `baz` and any errors that causes cannot be replicated using `poetry update`.
All of these problems can apply not only to the dependencies specified for a Tox environment, All of these problems can apply not only to the dependencies specified for a Tox environment,
but also to the dependencies of those dependencies, and so on. but also to the dependencies of those dependencies, those dependencies' dependencies, and so on.
**The Solution** **The Solution**
This plugin requires that all dependencies specified for all Tox environments be unbound This plugin allows dependencies specified in Tox environment take their version directly from
with no version constraint specified at all. This seems counter-intuitive given the problems the Poetry lockfile without needing an independent version to be specified in the Tox
outlined above, but what it allows the plugin to do is offload all version management to environment configuration. The modified version of the example environment given below appears
Poetry. less stable than the one presented above because it does not specify any versions for its
dependencies:
On initial inspection, the environment below appears less stable than the one presented above
because it does not specify any versions for its dependencies:
```ini ```ini
# tox.ini # from tox.ini
... ...
[testenv] [testenv]
description = Some very cool tests description = Some very cool tests
require_locked_deps = true
deps = deps =
foo foo
bar bar
@@ -373,19 +380,16 @@ deps =
... ...
``` ```
However with the `tox-poetry-installer` plugin installed this instructs Tox to install these However with the `tox-poetry-installer` plugin installed the `require_locked_deps = true`
dependencies using the Poetry lockfile so that the version installed to the Tox environment setting means that Tox will install these dependencies from the Poetry lockfile so that the
exactly matches the version Poetry is managing. When `poetry update` updates the lockfile version installed to the Tox environment exactly matches the version Poetry is managing. When
with new dependency versions, Tox will automatically install these new versions without needing `poetry update` updates the lockfile with new versions of these dependencies, Tox will
any changes to the configuration. automatically install these new versions without needing any changes to the configuration.
All dependencies are specified in one place (the lockfile) and dependency version management is
handled by a tool dedicated to that task (Poetry).
## Developing ## Developing
This project requires Poetry-1.0+, see the [installation instructions here](https://python-poetry.org/docs/#installation). This project requires Poetry version 1.0+, see the [installation instructions here](https://python-poetry.org/docs/#installation).
```bash ```bash
# Clone the repository... # Clone the repository...
@@ -428,7 +432,7 @@ releases on PyPI.
## Roadmap ## Roadmap
This project is under active development and is classified as alpha software, not yet ready This project is under active development and is classified as alpha software, not yet ready
usage in production systems. for usage in production environments.
* Beta classification will be assigned when the initial feature set is finalized * Beta classification will be assigned when the initial feature set is finalized
* Stable classification will be assigned when the test suite covers an acceptable number of * Stable classification will be assigned when the test suite covers an acceptable number of
@@ -439,19 +443,19 @@ usage in production systems.
- [X] Verify that primary package dependencies (from the `.package` env) are installed - [X] Verify that primary package dependencies (from the `.package` env) are installed
correctly using the Poetry backend. correctly using the Poetry backend.
- [ ] Support the [`extras`](https://tox.readthedocs.io/en/latest/config.html#conf-extras) - [ ] Support the [`extras`](https://tox.readthedocs.io/en/latest/config.html#conf-extras)
Tox configuration option Tox configuration option ([#4](https://github.com/enpaul/tox-poetry-installer/issues/4))
- [X] Add per-environment Tox configuration option to fall back to default installation - [X] Add per-environment Tox configuration option to fall back to default installation
backend. backend.
- [ ] Add detection of a changed lockfile to automatically trigger a rebuild of Tox
environments when necessary.
- [ ] Add warnings when an unsupported Tox configuration option is detected while using the - [ ] Add warnings when an unsupported Tox configuration option is detected while using the
Poetry backend. Poetry backend. ([#5](https://github.com/enpaul/tox-poetry-installer/issues/5))
- [X] Add trivial tests to ensure the project metadata is consistent between the pyproject.toml - [X] Add trivial tests to ensure the project metadata is consistent between the pyproject.toml
and the module constants. and the module constants.
- [ ] Update to use [poetry-core](https://github.com/python-poetry/poetry-core) - [ ] Update to use [poetry-core](https://github.com/python-poetry/poetry-core) and
Tox configuration option) and improve robustness of the Tox and Poetry module imports improve robustness of the Tox and Poetry module imports
to avoid potentially breaking API changes in upstream packages. to avoid potentially breaking API changes in upstream packages. ([#2](https://github.com/enpaul/tox-poetry-installer/issues/2))
- [ ] Find and implement a way to mitigate the [Poetry UNSAFE_DEPENDENCIES bug](https://github.com/python-poetry/poetry/issues/1584). - [ ] Find and implement a way to mitigate the [UNSAFE_DEPENDENCIES issue](https://github.com/python-poetry/poetry/issues/1584) in Poetry.
([#6](https://github.com/enpaul/tox-poetry-installer/issues/6))
- [ ] Fix logging to make proper use of Tox's logging reporter infrastructure
### Path to Stable ### Path to Stable

View File

@@ -1,6 +1,6 @@
[tool.poetry] [tool.poetry]
name = "tox-poetry-installer" name = "tox-poetry-installer"
version = "0.2.0" version = "0.2.3"
license = "MIT" license = "MIT"
authors = ["Ethan Paul <24588726+enpaul@users.noreply.github.com>"] authors = ["Ethan Paul <24588726+enpaul@users.noreply.github.com>"]
description = "Tox plugin to install Tox environment dependencies using the Poetry backend and lockfile" description = "Tox plugin to install Tox environment dependencies using the Poetry backend and lockfile"

View File

@@ -10,6 +10,7 @@ from typing import Dict
from typing import List from typing import List
from typing import NamedTuple from typing import NamedTuple
from typing import Sequence from typing import Sequence
from typing import Set
from typing import Tuple from typing import Tuple
from poetry.factory import Factory as PoetryFactory from poetry.factory import Factory as PoetryFactory
@@ -29,7 +30,7 @@ from tox.venv import VirtualEnv as ToxVirtualEnv
__title__ = "tox-poetry-installer" __title__ = "tox-poetry-installer"
__summary__ = "Tox plugin to install Tox environment dependencies using the Poetry backend and lockfile" __summary__ = "Tox plugin to install Tox environment dependencies using the Poetry backend and lockfile"
__version__ = "0.2.0" __version__ = "0.2.3"
__url__ = "https://github.com/enpaul/tox-poetry-installer/" __url__ = "https://github.com/enpaul/tox-poetry-installer/"
__license__ = "MIT" __license__ = "MIT"
__authors__ = ["Ethan Paul <24588726+enpaul@users.noreply.github.com>"] __authors__ = ["Ethan Paul <24588726+enpaul@users.noreply.github.com>"]
@@ -139,7 +140,7 @@ def _install_to_venv(
installer.install(dependency) installer.install(dependency)
def _find_transients(poetry: Poetry, dependency_name: str) -> List[PoetryPackage]: def _find_transients(poetry: Poetry, dependency_name: str) -> Set[PoetryPackage]:
"""Using a poetry object identify all dependencies of a specific dependency """Using a poetry object identify all dependencies of a specific dependency
:param poetry: Populated poetry object which can be used to build a populated locked :param poetry: Populated poetry object which can be used to build a populated locked
@@ -170,7 +171,7 @@ def _find_transients(poetry: Poetry, dependency_name: str) -> List[PoetryPackage
transients += find_deps_of_deps(dep.name) transients += find_deps_of_deps(dep.name)
return transients return transients
return find_deps_of_deps(top_level.name) return set(find_deps_of_deps(top_level.name))
except KeyError: except KeyError:
if any( if any(
@@ -187,17 +188,12 @@ def _find_transients(poetry: Poetry, dependency_name: str) -> List[PoetryPackage
def _install_env_dependencies(venv: ToxVirtualEnv, poetry: Poetry): def _install_env_dependencies(venv: ToxVirtualEnv, poetry: Poetry):
env_deps = _sort_env_deps(venv) env_deps = _sort_env_deps(venv)
reporter.verbosity1(
f"{_REPORTER_PREFIX} updating env config with {len(env_deps.unlocked_deps)} unlocked env dependencies for installation using the default backend"
)
venv.envconfig.deps = env_deps.unlocked_deps
dependencies: List[PoetryPackage] = [] dependencies: List[PoetryPackage] = []
for dep in env_deps.locked_deps: for dep in env_deps.locked_deps:
try: try:
dependencies += _find_transients(poetry, dep.name) dependencies += _find_transients(poetry, dep.name.lower())
except ToxPoetryInstallerException as err: except ToxPoetryInstallerException as err:
venv.status = "install-locked-deps-failed" venv.status = "lockfile installation failed"
reporter.error(f"{_REPORTER_PREFIX} {err}") reporter.error(f"{_REPORTER_PREFIX} {err}")
raise err raise err
@@ -205,6 +201,11 @@ def _install_env_dependencies(venv: ToxVirtualEnv, poetry: Poetry):
f"{_REPORTER_PREFIX} identified {len(dependencies)} actual dependencies from {len(venv.envconfig.deps)} specified env dependencies" f"{_REPORTER_PREFIX} identified {len(dependencies)} actual dependencies from {len(venv.envconfig.deps)} specified env dependencies"
) )
reporter.verbosity1(
f"{_REPORTER_PREFIX} updating env config with {len(env_deps.unlocked_deps)} unlocked env dependencies for installation using the default backend"
)
venv.envconfig.deps = env_deps.unlocked_deps
reporter.verbosity0( reporter.verbosity0(
f"{_REPORTER_PREFIX} ({venv.name}) installing {len(dependencies)} env dependencies from lockfile" f"{_REPORTER_PREFIX} ({venv.name}) installing {len(dependencies)} env dependencies from lockfile"
) )
@@ -265,7 +266,18 @@ def tox_testenv_install_deps(venv: ToxVirtualEnv, action: ToxAction):
) )
return return
poetry = PoetryFactory().create_poetry(venv.envconfig.config.toxinidir) try:
poetry = PoetryFactory().create_poetry(venv.envconfig.config.toxinidir)
except RuntimeError:
# Support running the plugin when the current tox project does not use Poetry for its
# environment/dependency management.
#
# ``RuntimeError`` is dangerous to blindly catch because it can be (and in Poetry's case,
# is) raised in many different places for different purposes.
reporter.verbosity1(
f"{_REPORTER_PREFIX} project does not use Poetry for env management, skipping installation of locked dependencies"
)
return
reporter.verbosity1( reporter.verbosity1(
f"{_REPORTER_PREFIX} loaded project pyproject.toml from {poetry.file}" f"{_REPORTER_PREFIX} loaded project pyproject.toml from {poetry.file}"
@@ -276,9 +288,14 @@ def tox_testenv_install_deps(venv: ToxVirtualEnv, action: ToxAction):
# Handle the installation of the package dependencies from the lockfile if the package is # Handle the installation of the package dependencies from the lockfile if the package is
# being installed to this venv; otherwise skip installing the package dependencies # being installed to this venv; otherwise skip installing the package dependencies
if not venv.envconfig.skip_install: if not venv.envconfig.skip_install and not venv.envconfig.config.skipsdist:
_install_package_dependencies(venv, poetry) _install_package_dependencies(venv, poetry)
else: else:
reporter.verbosity1( if venv.envconfig.skip_install:
f"{_REPORTER_PREFIX} env specifies 'skip_install = true', skipping installation of project package" reporter.verbosity1(
) f"{_REPORTER_PREFIX} env specifies 'skip_install = true', skipping installation of project package"
)
elif venv.envconfig.config.skipsdist:
reporter.verbosity1(
f"{_REPORTER_PREFIX} config specifies 'skipsdist = true', skipping installation of project package"
)