|
|
|
|
@@ -1,9 +1,24 @@
|
|
|
|
|
"""Tox plugin for installing environments using Poetry
|
|
|
|
|
|
|
|
|
|
This plugin makes use of the ``tox_testenv_install_deps`` Tox plugin hook to replace the default
|
|
|
|
|
This plugin makes use of the ``tox_testenv_install_deps`` Tox plugin hook to augment the default
|
|
|
|
|
installation functionality to install dependencies from the Poetry lockfile for the project. It
|
|
|
|
|
does this by using ``poetry`` to read in the lockfile, identify necessary dependencies, and then
|
|
|
|
|
use Poetry's ``PipInstaller`` class to install those packages into the Tox environment.
|
|
|
|
|
|
|
|
|
|
Quick definition of terminology:
|
|
|
|
|
|
|
|
|
|
* "project package" - the package that Tox is testing, usually the one the current project is
|
|
|
|
|
is developing; definitionally, this is the package that is built by Tox in the ``.package`` env.
|
|
|
|
|
* "project package dependency" or "project dependency" - a dependency required by the project
|
|
|
|
|
package for installation; i.e. a package that would be installed when running
|
|
|
|
|
``pip install <project package>``.
|
|
|
|
|
* "environment dependency" - a dependency specified for a given testenv in the Tox configuration.
|
|
|
|
|
* "locked dependency" - a package that is present in the Poetry lockfile and will be installed
|
|
|
|
|
according to the metadata in the lockfile.
|
|
|
|
|
* "unlocked dependency" - a package that is either not present in the Poetry lockfile or is not
|
|
|
|
|
specified to be installed according to the metadata in the lockfile.
|
|
|
|
|
* "transiety dependency" - a package not explicitly specified for installation, but required by a
|
|
|
|
|
package that is explicitly specified.
|
|
|
|
|
"""
|
|
|
|
|
from pathlib import Path
|
|
|
|
|
from typing import Dict
|
|
|
|
|
@@ -30,7 +45,7 @@ from tox.venv import VirtualEnv as ToxVirtualEnv
|
|
|
|
|
|
|
|
|
|
__title__ = "tox-poetry-installer"
|
|
|
|
|
__summary__ = "Tox plugin to install Tox environment dependencies using the Poetry backend and lockfile"
|
|
|
|
|
__version__ = "0.3.0"
|
|
|
|
|
__version__ = "0.4.0"
|
|
|
|
|
__url__ = "https://github.com/enpaul/tox-poetry-installer/"
|
|
|
|
|
__license__ = "MIT"
|
|
|
|
|
__authors__ = ["Ethan Paul <24588726+enpaul@users.noreply.github.com>"]
|
|
|
|
|
@@ -52,6 +67,7 @@ _REPORTER_PREFIX = f"[{__title__}]:"
|
|
|
|
|
_MAGIC_SUFFIX_MARKER = "@poetry"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# Map of package names to the package object
|
|
|
|
|
PackageMap = Dict[str, PoetryPackage]
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@@ -118,10 +134,10 @@ def _sort_env_deps(venv: ToxVirtualEnv) -> _SortedEnvDeps:
|
|
|
|
|
unlocked_deps.append(dep)
|
|
|
|
|
|
|
|
|
|
reporter.verbosity1(
|
|
|
|
|
f"{_REPORTER_PREFIX} identified {len(locked_deps)} locked env dependencies for installation from poetry lockfile: {[item.name for item in locked_deps]}"
|
|
|
|
|
f"{_REPORTER_PREFIX} identified {len(locked_deps)} locked env dependencies: {[item.name for item in locked_deps]}"
|
|
|
|
|
)
|
|
|
|
|
reporter.verbosity1(
|
|
|
|
|
f"{_REPORTER_PREFIX} identified {len(unlocked_deps)} unlocked env dependencies for installation using default backend: {[item.name for item in unlocked_deps]}"
|
|
|
|
|
f"{_REPORTER_PREFIX} identified {len(unlocked_deps)} unlocked env dependencies: {[item.name for item in unlocked_deps]}"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
return _SortedEnvDeps(locked_deps=locked_deps, unlocked_deps=unlocked_deps)
|
|
|
|
|
@@ -136,6 +152,11 @@ def _install_to_venv(
|
|
|
|
|
:param venv: Tox virtual environment to install the packages to
|
|
|
|
|
:param packages: List of packages to install to the virtual environment
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
|
reporter.verbosity1(
|
|
|
|
|
f"{_REPORTER_PREFIX} Installing {len(packages)} packages to environment at {venv.envconfig.envdir}"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
installer = PoetryPipInstaller(
|
|
|
|
|
env=PoetryVirtualEnv(path=Path(venv.envconfig.envdir)),
|
|
|
|
|
io=PoetryNullIO(),
|
|
|
|
|
@@ -166,7 +187,7 @@ def _find_transients(packages: PackageMap, dependency_name: str) -> Set[PoetryPa
|
|
|
|
|
def find_deps_of_deps(name: str) -> List[PoetryPackage]:
|
|
|
|
|
if name in PoetryProvider.UNSAFE_PACKAGES:
|
|
|
|
|
reporter.warning(
|
|
|
|
|
f"{_REPORTER_PREFIX} installing '{name}' using Poetry is not supported; skipping"
|
|
|
|
|
f"{_REPORTER_PREFIX} installing package '{name}' using Poetry is not supported; skipping installation of package '{name}'"
|
|
|
|
|
)
|
|
|
|
|
return []
|
|
|
|
|
transients = [packages[name]]
|
|
|
|
|
@@ -191,6 +212,15 @@ def _find_transients(packages: PackageMap, dependency_name: str) -> Set[PoetryPa
|
|
|
|
|
def _install_env_dependencies(
|
|
|
|
|
venv: ToxVirtualEnv, poetry: Poetry, packages: PackageMap
|
|
|
|
|
):
|
|
|
|
|
"""Install the packages for a specified testenv
|
|
|
|
|
|
|
|
|
|
Processes the tox environment config, identifies any locked environment dependencies, pulls
|
|
|
|
|
them from the lockfile, and installs them to the virtual environment.
|
|
|
|
|
|
|
|
|
|
:param venv: Tox virtual environment to install the packages to
|
|
|
|
|
:param poetry: Poetry object the packages were sourced from
|
|
|
|
|
:param packages: Mapping of package names to the corresponding package object
|
|
|
|
|
"""
|
|
|
|
|
env_deps = _sort_env_deps(venv)
|
|
|
|
|
|
|
|
|
|
dependencies: List[PoetryPackage] = []
|
|
|
|
|
@@ -202,8 +232,25 @@ def _install_env_dependencies(
|
|
|
|
|
reporter.error(f"{_REPORTER_PREFIX} {err}")
|
|
|
|
|
raise err
|
|
|
|
|
|
|
|
|
|
if venv.envconfig.install_dev_deps:
|
|
|
|
|
reporter.verbosity1(
|
|
|
|
|
f"{_REPORTER_PREFIX} identified {len(dependencies)} actual dependencies from {len(venv.envconfig.deps)} specified env dependencies"
|
|
|
|
|
f"{_REPORTER_PREFIX} env specifies 'install_env_deps = true', including Poetry dev dependencies"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
dev_dependencies = [
|
|
|
|
|
dep
|
|
|
|
|
for dep in poetry.locker.locked_repository(True).packages
|
|
|
|
|
if dep not in poetry.locker.locked_repository(False).packages
|
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
reporter.verbosity1(
|
|
|
|
|
f"{_REPORTER_PREFIX} identified {len(dev_dependencies)} Poetry dev dependencies"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
dependencies = list(set(dev_dependencies + dependencies))
|
|
|
|
|
|
|
|
|
|
reporter.verbosity1(
|
|
|
|
|
f"{_REPORTER_PREFIX} identified {len(dependencies)} total dependencies from {len(env_deps.locked_deps)} locked env dependencies"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
reporter.verbosity1(
|
|
|
|
|
@@ -217,20 +264,31 @@ def _install_env_dependencies(
|
|
|
|
|
_install_to_venv(poetry, venv, dependencies)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def _install_package_dependencies(
|
|
|
|
|
def _install_project_dependencies(
|
|
|
|
|
venv: ToxVirtualEnv, poetry: Poetry, packages: PackageMap
|
|
|
|
|
):
|
|
|
|
|
"""Install the dependencies of the project package
|
|
|
|
|
|
|
|
|
|
Install all primary dependencies of the project package.
|
|
|
|
|
|
|
|
|
|
:param venv: Tox virtual environment to install the packages to
|
|
|
|
|
:param poetry: Poetry object the packages were sourced from
|
|
|
|
|
:param packages: Mapping of package names to the corresponding package object
|
|
|
|
|
"""
|
|
|
|
|
reporter.verbosity1(
|
|
|
|
|
f"{_REPORTER_PREFIX} performing installation of project dependencies"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
base_dependencies = [
|
|
|
|
|
packages[item.name] for item in poetry.package.requires if not item.is_optional
|
|
|
|
|
base_dependencies: List[PoetryPackage] = [
|
|
|
|
|
packages[item.name]
|
|
|
|
|
for item in poetry.package.requires
|
|
|
|
|
if not item.is_optional()
|
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
extra_dependencies: List[PoetryPackage] = []
|
|
|
|
|
for extra in venv.envconfig.extras:
|
|
|
|
|
try:
|
|
|
|
|
extra_dependencies = [
|
|
|
|
|
extra_dependencies += [
|
|
|
|
|
packages[item.name] for item in poetry.package.extras[extra]
|
|
|
|
|
]
|
|
|
|
|
except KeyError:
|
|
|
|
|
@@ -248,7 +306,7 @@ def _install_package_dependencies(
|
|
|
|
|
raise err
|
|
|
|
|
|
|
|
|
|
reporter.verbosity1(
|
|
|
|
|
f"{_REPORTER_PREFIX} identified {len(dependencies)} dependencies of project '{poetry.package.name}'"
|
|
|
|
|
f"{_REPORTER_PREFIX} identified {len(dependencies)} total dependencies from {len(poetry.package.requires)} project dependencies"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
reporter.verbosity0(
|
|
|
|
|
@@ -265,6 +323,13 @@ def tox_addoption(parser: ToxParser):
|
|
|
|
|
dependencies should be treated as locked or not.
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
|
parser.add_testenv_attribute(
|
|
|
|
|
name="install_dev_deps",
|
|
|
|
|
type="bool",
|
|
|
|
|
default=False,
|
|
|
|
|
help="Automatically install all Poetry development dependencies to the environment",
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
parser.add_testenv_attribute(
|
|
|
|
|
name="require_locked_deps",
|
|
|
|
|
type="bool",
|
|
|
|
|
@@ -333,4 +398,4 @@ def tox_testenv_install_deps(venv: ToxVirtualEnv, action: ToxAction):
|
|
|
|
|
)
|
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
_install_package_dependencies(venv, poetry, package_map)
|
|
|
|
|
_install_project_dependencies(venv, poetry, package_map)
|
|
|
|
|
|