2020-09-24 00:34:22 +00:00
|
|
|
"""Tox plugin for installing environments using Poetry
|
|
|
|
|
|
|
|
This plugin makes use of the ``tox_testenv_install_deps`` Tox plugin hook to replace 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.
|
|
|
|
"""
|
2020-09-23 06:17:39 +00:00
|
|
|
import logging
|
2020-09-24 00:34:22 +00:00
|
|
|
from pathlib import Path
|
|
|
|
from typing import Dict
|
|
|
|
from typing import List
|
|
|
|
from typing import Optional
|
|
|
|
from typing import Tuple
|
2020-09-23 06:17:39 +00:00
|
|
|
|
2020-09-24 00:34:22 +00:00
|
|
|
from poetry.factory import Factory as PoetryFactory
|
2020-09-23 06:17:39 +00:00
|
|
|
from poetry.factory import Poetry
|
2020-09-24 00:34:22 +00:00
|
|
|
from poetry.installation.pip_installer import PipInstaller as PoetryPipInstaller
|
|
|
|
from poetry.io.null_io import NullIO as PoetryNullIO
|
|
|
|
from poetry.packages import Package as PoetryPackage
|
|
|
|
from poetry.puzzle.provider import Provider as PoetryProvider
|
|
|
|
from poetry.utils.env import VirtualEnv as PoetryVirtualEnv
|
|
|
|
from tox import hookimpl
|
2020-09-23 06:17:39 +00:00
|
|
|
from tox.action import Action as ToxAction
|
|
|
|
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"
|
2020-09-25 01:09:09 +00:00
|
|
|
__version__ = "0.1.2"
|
2020-09-23 06:17:39 +00:00
|
|
|
__url__ = "https://github.com/enpaul/tox-poetry-installer/"
|
|
|
|
__license__ = "MIT"
|
2020-09-25 01:07:52 +00:00
|
|
|
__authors__ = ["Ethan Paul <24588726+enpaul@users.noreply.github.com>"]
|
2020-09-23 06:17:39 +00:00
|
|
|
|
|
|
|
|
2020-09-25 01:07:52 +00:00
|
|
|
_PEP508_VERSION_DELIMITERS: Tuple[str, ...] = ("~=", "==", "!=", ">", "<")
|
2020-09-24 00:34:22 +00:00
|
|
|
|
|
|
|
|
|
|
|
class ToxPoetryInstallerException(Exception):
|
|
|
|
"""Error while installing locked dependencies to the test environment"""
|
|
|
|
|
|
|
|
|
|
|
|
class NoLockedDependencyError(ToxPoetryInstallerException):
|
|
|
|
"""Cannot install a package that is not in the lockfile"""
|
|
|
|
|
|
|
|
|
2020-09-23 06:17:39 +00:00
|
|
|
def _make_poetry(venv: ToxVirtualEnv) -> Poetry:
|
2020-09-24 00:34:22 +00:00
|
|
|
"""Helper to make a poetry object from a toxenv"""
|
|
|
|
return PoetryFactory().create_poetry(venv.envconfig.config.toxinidir)
|
|
|
|
|
|
|
|
|
|
|
|
def _find_locked_dependencies(
|
|
|
|
poetry: Poetry, dependency_name: str
|
|
|
|
) -> List[PoetryPackage]:
|
|
|
|
"""Using a poetry object identify all dependencies of a specific dependency
|
2020-09-23 06:17:39 +00:00
|
|
|
|
2020-09-24 00:34:22 +00:00
|
|
|
:param poetry: Populated poetry object which can be used to build a populated locked
|
|
|
|
repository object.
|
|
|
|
:param dependency_name: Bare name (without version) of the dependency to fetch the transient
|
|
|
|
dependencies of.
|
|
|
|
:returns: List of packages that need to be installed for the requested dependency.
|
2020-09-23 06:17:39 +00:00
|
|
|
|
2020-09-24 00:34:22 +00:00
|
|
|
.. note:: The package corresponding to the dependency named by ``dependency_name`` is included
|
|
|
|
in the list of returned packages.
|
|
|
|
"""
|
|
|
|
packages: Dict[str, PoetryPackage] = {
|
2020-09-23 06:17:39 +00:00
|
|
|
package.name: package
|
|
|
|
for package in poetry.locker.locked_repository(True).packages
|
|
|
|
}
|
|
|
|
|
|
|
|
try:
|
|
|
|
top_level = packages[dependency_name]
|
|
|
|
|
2020-09-24 00:34:22 +00:00
|
|
|
def find_transients(name: str) -> List[PoetryPackage]:
|
|
|
|
if name in PoetryProvider.UNSAFE_PACKAGES:
|
|
|
|
return []
|
|
|
|
transients = [packages[name]]
|
|
|
|
for dep in packages[name].requires:
|
|
|
|
transients += find_transients(dep.name)
|
|
|
|
return transients
|
|
|
|
|
|
|
|
return find_transients(top_level.name)
|
2020-09-23 06:17:39 +00:00
|
|
|
|
2020-09-24 00:34:22 +00:00
|
|
|
except KeyError:
|
2020-09-25 01:07:52 +00:00
|
|
|
if any(
|
|
|
|
delimiter in dependency_name for delimiter in _PEP508_VERSION_DELIMITERS
|
|
|
|
):
|
2020-09-24 00:34:22 +00:00
|
|
|
message = "specifying a version in the tox environment definition is incompatible with installing from a lockfile"
|
|
|
|
else:
|
|
|
|
message = (
|
|
|
|
"no version of the package was found in the current project's lockfile"
|
|
|
|
)
|
|
|
|
|
|
|
|
raise NoLockedDependencyError(
|
|
|
|
f"Cannot install requirement '{dependency_name}': {message}"
|
|
|
|
) from None
|
2020-09-23 06:17:39 +00:00
|
|
|
|
|
|
|
|
|
|
|
@hookimpl
|
2020-09-24 00:34:22 +00:00
|
|
|
def tox_testenv_install_deps(
|
|
|
|
venv: ToxVirtualEnv, action: ToxAction
|
|
|
|
) -> Optional[List[PoetryPackage]]:
|
|
|
|
"""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
|
|
|
|
"""
|
2020-09-23 06:17:39 +00:00
|
|
|
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
if action.name == venv.envconfig.config.isolated_build_env:
|
2020-09-24 00:34:22 +00:00
|
|
|
logger.debug(
|
|
|
|
f"Environment {action.name} is isolated build environment; skipping Poetry-based dependency installation"
|
|
|
|
)
|
2020-09-23 06:17:39 +00:00
|
|
|
return None
|
|
|
|
|
|
|
|
poetry = _make_poetry(venv)
|
|
|
|
|
|
|
|
logger.debug(f"Loaded project pyproject.toml from {poetry.file}")
|
|
|
|
|
2020-09-24 00:34:22 +00:00
|
|
|
dependencies: List[PoetryPackage] = []
|
2020-09-23 06:17:39 +00:00
|
|
|
for env_dependency in venv.envconfig.deps:
|
|
|
|
dependencies += _find_locked_dependencies(poetry, env_dependency.name)
|
|
|
|
|
2020-09-24 00:34:22 +00:00
|
|
|
logger.debug(
|
|
|
|
f"Identified {len(dependencies)} dependencies for environment {action.name}"
|
|
|
|
)
|
2020-09-23 06:17:39 +00:00
|
|
|
|
2020-09-24 00:34:22 +00:00
|
|
|
installer = PoetryPipInstaller(
|
|
|
|
env=PoetryVirtualEnv(path=Path(venv.envconfig.envdir)),
|
|
|
|
io=PoetryNullIO(),
|
|
|
|
pool=poetry.pool,
|
2020-09-23 06:17:39 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
for dependency in dependencies:
|
|
|
|
logger.info(f"Installing environment dependency: {dependency}")
|
|
|
|
installer.install(dependency)
|
|
|
|
|
|
|
|
return dependencies
|