2019-07-17 15:42:23 +02:00
|
|
|
import os
|
|
|
|
import shutil
|
2020-08-10 11:58:17 +02:00
|
|
|
import subprocess
|
2019-07-17 15:42:23 +02:00
|
|
|
import sys
|
2019-07-17 13:54:09 +02:00
|
|
|
import tox
|
|
|
|
|
2020-08-27 17:35:19 +02:00
|
|
|
try:
|
|
|
|
import importlib.metadata as importlib_metadata
|
|
|
|
except ImportError:
|
|
|
|
import importlib_metadata
|
|
|
|
|
2019-07-17 13:54:09 +02:00
|
|
|
|
2019-07-17 15:42:23 +02:00
|
|
|
@tox.hookimpl
|
|
|
|
def tox_addoption(parser):
|
|
|
|
parser.add_argument(
|
|
|
|
"--current-env",
|
|
|
|
action="store_true",
|
|
|
|
dest="current_env",
|
|
|
|
default=False,
|
|
|
|
help="Run tests in current environment, not creating any virtual environment",
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
"--print-deps-only",
|
|
|
|
action="store_true",
|
|
|
|
dest="print_deps_only",
|
|
|
|
default=False,
|
2019-08-12 16:17:41 +02:00
|
|
|
help="Don't run tests, only print the dependencies to stdout",
|
2019-07-17 15:42:23 +02:00
|
|
|
)
|
2019-08-01 12:39:12 +02:00
|
|
|
parser.add_argument(
|
|
|
|
"--print-deps-to-file",
|
|
|
|
action="store",
|
|
|
|
dest="print_deps_path",
|
|
|
|
metavar="PATH",
|
|
|
|
default=None,
|
2019-08-12 16:05:18 +02:00
|
|
|
help="Like --print-deps-only, but to a file. Overwrites the file if it exists.",
|
2019-08-01 12:39:12 +02:00
|
|
|
)
|
2019-07-17 15:42:23 +02:00
|
|
|
|
|
|
|
|
|
|
|
@tox.hookimpl
|
|
|
|
def tox_configure(config):
|
2019-07-18 11:33:42 +02:00
|
|
|
"""Stores options in the config. Makes all commands external and skips sdist"""
|
2019-08-12 16:17:41 +02:00
|
|
|
if config.option.print_deps_only and config.option.print_deps_path:
|
|
|
|
raise tox.exception.ConfigError(
|
|
|
|
"--print-deps-only cannot be used together with --print-deps-to-file"
|
|
|
|
)
|
2019-08-01 12:39:12 +02:00
|
|
|
if config.option.print_deps_path is not None:
|
|
|
|
config.option.print_deps_only = True
|
|
|
|
with open(config.option.print_deps_path, "w", encoding="utf-8") as f:
|
|
|
|
f.write("")
|
2019-07-18 11:33:42 +02:00
|
|
|
if config.option.current_env or config.option.print_deps_only:
|
2019-07-17 15:42:23 +02:00
|
|
|
config.skipsdist = True
|
|
|
|
for testenv in config.envconfigs:
|
|
|
|
config.envconfigs[testenv].whitelist_externals = "*"
|
|
|
|
|
|
|
|
return config
|
|
|
|
|
2019-07-17 18:59:50 +02:00
|
|
|
|
2019-07-17 16:48:12 +02:00
|
|
|
class InterpreterMismatch(tox.exception.InterpreterNotFound):
|
|
|
|
"""Interpreter version in current env does not match requested version"""
|
2019-07-17 15:42:23 +02:00
|
|
|
|
2019-07-17 18:59:50 +02:00
|
|
|
|
2019-07-26 13:35:26 +02:00
|
|
|
def _python_activate_exists(venv):
|
|
|
|
python = venv.envconfig.get_envpython()
|
|
|
|
bindir = os.path.dirname(python)
|
|
|
|
activate = os.path.join(bindir, "activate")
|
|
|
|
return os.path.exists(python), os.path.exists(activate)
|
|
|
|
|
|
|
|
|
|
|
|
def is_current_env_link(venv):
|
|
|
|
python, activate = _python_activate_exists(venv)
|
|
|
|
return python and not activate
|
|
|
|
|
|
|
|
|
|
|
|
def is_proper_venv(venv):
|
|
|
|
python, activate = _python_activate_exists(venv)
|
|
|
|
return python and activate
|
|
|
|
|
|
|
|
|
2019-07-26 13:49:54 +02:00
|
|
|
def is_any_env(venv):
|
|
|
|
python, activate = _python_activate_exists(venv)
|
|
|
|
return python
|
|
|
|
|
|
|
|
|
2019-08-01 13:24:41 +02:00
|
|
|
def rm_venv(venv):
|
|
|
|
link = venv.envconfig.get_envpython()
|
|
|
|
shutil.rmtree(os.path.dirname(os.path.dirname(link)), ignore_errors=True)
|
|
|
|
|
|
|
|
|
2019-07-26 13:35:26 +02:00
|
|
|
def unsupported_raise(config, venv):
|
|
|
|
if config.option.recreate:
|
|
|
|
return
|
|
|
|
regular = not (config.option.current_env or config.option.print_deps_only)
|
|
|
|
if regular and is_current_env_link(venv):
|
2019-08-01 14:46:01 +02:00
|
|
|
if hasattr(tox.hookspecs, "tox_cleanup"):
|
|
|
|
raise tox.exception.ConfigError(
|
|
|
|
"Looks like previous --current-env or --print-deps-only tox run didn't finish the cleanup. "
|
|
|
|
"Run tox run with --recreate (-r) or manually remove the environment in .tox."
|
|
|
|
)
|
|
|
|
else:
|
|
|
|
raise tox.exception.ConfigError(
|
|
|
|
"Regular tox run after --current-env or --print-deps-only tox run is not supported without --recreate (-r)."
|
|
|
|
)
|
2019-07-26 13:35:26 +02:00
|
|
|
elif config.option.current_env and is_proper_venv(venv):
|
|
|
|
raise tox.exception.ConfigError(
|
|
|
|
"--current-env after regular tox run is not supported without --recreate (-r)."
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2019-07-17 13:54:09 +02:00
|
|
|
@tox.hookimpl
|
|
|
|
def tox_testenv_create(venv, action):
|
2019-07-17 18:59:50 +02:00
|
|
|
"""We create a fake virtualenv with just the symbolic link"""
|
2019-07-17 15:42:23 +02:00
|
|
|
config = venv.envconfig.config
|
2019-07-26 13:49:54 +02:00
|
|
|
create_fake_env = check_version = config.option.current_env
|
2019-07-18 11:33:42 +02:00
|
|
|
if config.option.print_deps_only:
|
2019-07-26 13:49:54 +02:00
|
|
|
if is_any_env(venv):
|
|
|
|
# We don't need anything
|
|
|
|
return True
|
|
|
|
else:
|
|
|
|
# We need at least some kind of environment,
|
|
|
|
# or tox fails without a python command
|
|
|
|
# We fallback to --current-env behavior,
|
|
|
|
# because it's cheaper, faster and won't install stuff
|
|
|
|
create_fake_env = True
|
|
|
|
if check_version:
|
|
|
|
# With real --current-env, we check this, but not with --print-deps-only only
|
2019-07-17 16:48:12 +02:00
|
|
|
version_info = venv.envconfig.python_info.version_info
|
2019-07-23 14:37:06 +02:00
|
|
|
if version_info is None:
|
|
|
|
raise tox.exception.InterpreterNotFound(venv.envconfig.basepython)
|
2019-07-17 16:48:12 +02:00
|
|
|
if version_info[:2] != sys.version_info[:2]:
|
|
|
|
raise InterpreterMismatch(
|
2019-07-17 18:59:50 +02:00
|
|
|
f"tox_current_env: interpreter versions do not match:\n"
|
|
|
|
+ f" in current env: {tuple(sys.version_info)}\n"
|
|
|
|
+ f" requested: {version_info}"
|
2019-07-17 16:48:12 +02:00
|
|
|
)
|
2019-07-26 13:49:54 +02:00
|
|
|
if create_fake_env:
|
2019-07-17 16:04:55 +02:00
|
|
|
# Make sure the `python` command on path is sys.executable.
|
|
|
|
# (We might have e.g. /usr/bin/python3, not `python`.)
|
|
|
|
# Remove the rest of the virtualenv.
|
2019-07-17 15:42:23 +02:00
|
|
|
link = venv.envconfig.get_envpython()
|
|
|
|
target = sys.executable
|
2020-01-13 00:33:21 +01:00
|
|
|
rm_venv(venv)
|
2019-07-17 15:42:23 +02:00
|
|
|
os.makedirs(os.path.dirname(link))
|
2020-08-10 11:58:17 +02:00
|
|
|
if sys.platform == "win32":
|
|
|
|
# Avoid requiring admin rights on Windows
|
|
|
|
subprocess.check_call(f'mklink /J "{link}" "{target}"', shell=True)
|
|
|
|
else:
|
|
|
|
os.symlink(target, link)
|
2020-01-13 00:33:21 +01:00
|
|
|
# prevent tox from creating the venv
|
2019-07-17 15:42:23 +02:00
|
|
|
return True
|
2020-01-13 00:33:21 +01:00
|
|
|
if not is_proper_venv(venv):
|
2019-08-01 13:24:41 +02:00
|
|
|
rm_venv(venv)
|
2020-01-13 00:33:21 +01:00
|
|
|
return None # let tox handle the rest
|
2019-07-17 15:42:23 +02:00
|
|
|
|
|
|
|
|
2019-07-29 14:37:17 +02:00
|
|
|
@tox.hookimpl
|
|
|
|
def tox_package(session, venv):
|
|
|
|
"""Fail early when unsupported"""
|
|
|
|
config = venv.envconfig.config
|
|
|
|
unsupported_raise(config, venv)
|
|
|
|
|
|
|
|
|
2019-07-17 15:42:23 +02:00
|
|
|
@tox.hookimpl
|
|
|
|
def tox_testenv_install_deps(venv, action):
|
|
|
|
"""We don't install anything"""
|
|
|
|
config = venv.envconfig.config
|
2019-07-17 18:59:50 +02:00
|
|
|
unsupported_raise(config, venv)
|
2019-07-17 15:42:23 +02:00
|
|
|
if config.option.current_env or config.option.print_deps_only:
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
|
|
@tox.hookimpl
|
|
|
|
def tox_runtest(venv, redirect):
|
2019-07-18 11:33:42 +02:00
|
|
|
"""If --print-deps-only, prints deps instead of running tests"""
|
2019-07-17 18:59:50 +02:00
|
|
|
config = venv.envconfig.config
|
|
|
|
unsupported_raise(config, venv)
|
2019-08-01 12:39:12 +02:00
|
|
|
if config.option.print_deps_path is not None:
|
|
|
|
with open(config.option.print_deps_path, "a", encoding="utf-8") as f:
|
|
|
|
print(*venv.get_resolved_dependencies(), sep="\n", file=f)
|
|
|
|
return True
|
2019-07-17 18:59:50 +02:00
|
|
|
if config.option.print_deps_only:
|
2019-08-01 12:39:12 +02:00
|
|
|
print(*venv.get_resolved_dependencies(), sep="\n")
|
2019-07-17 15:42:23 +02:00
|
|
|
return True
|
2019-08-01 13:24:41 +02:00
|
|
|
|
|
|
|
|
2020-08-12 17:04:58 +02:00
|
|
|
@tox.hookimpl
|
|
|
|
def tox_cleanup(session):
|
|
|
|
"""Remove the fake virtualenv not to collide with regular tox
|
|
|
|
Collisions can happen anyway (when tox is killed forcefully before this happens)
|
|
|
|
Note that we don't remove real venvs, as recreating them is expensive"""
|
|
|
|
for venv in session.venv_dict.values():
|
|
|
|
if is_current_env_link(venv):
|
|
|
|
rm_venv(venv)
|
2020-08-27 17:35:19 +02:00
|
|
|
|
|
|
|
|
|
|
|
@tox.hookimpl
|
|
|
|
def tox_runenvreport(venv, action):
|
|
|
|
if not venv.envconfig.config.option.current_env:
|
|
|
|
return None
|
|
|
|
return (
|
|
|
|
"{}=={}".format(d.metadata.get("name"), d.version)
|
|
|
|
for d in importlib_metadata.distributions()
|
|
|
|
)
|