From 71d21b8266ddaa8f4ac9b8bcf01978c0fca22297 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Miro=20Hron=C4=8Dok?= Date: Thu, 1 Aug 2019 12:39:12 +0200 Subject: [PATCH 1/3] Add --print-deps-to-file option for easier parsable dependencies Fixes https://github.com/fedora-python/tox-current-env/issues/13 --- README.rst | 5 +++-- src/tox_current_env/hooks.py | 21 +++++++++++++++--- tests/test_integration.py | 41 ++++++++++++++++++++++++++++++++++++ 3 files changed, 62 insertions(+), 5 deletions(-) diff --git a/README.rst b/README.rst index 21d4050..5ea96d1 100644 --- a/README.rst +++ b/README.rst @@ -14,10 +14,11 @@ The ``tox-current-env`` plugin adds two options: An attempt to run this with a Python version that doesn't match will fail (if ``tox`` is invoked from an Python 3.7 environment, any non 3.7 testenv will fail). -``tox --print-deps-only`` +``tox --print-deps-only`` / ``--print-deps-to-file`` Instead of running any ``commands``, - simply prints the declared dependencies in ``deps`` to the standard output. + simply prints the declared dependencies in ``deps`` to the standard output or specified file. This is useful for preparing the current environment for the above. + ``--print-deps-to-file`` will override the file, if it already exists. Invoking ``tox`` without any of the above options should behave as regular ``tox`` invocation without this plugin. Any deviation from this behavior is considered a bug. diff --git a/src/tox_current_env/hooks.py b/src/tox_current_env/hooks.py index 03dc706..82a2275 100644 --- a/src/tox_current_env/hooks.py +++ b/src/tox_current_env/hooks.py @@ -20,11 +20,23 @@ def tox_addoption(parser): default=False, help="Don't run tests, only print the dependencies", ) + parser.add_argument( + "--print-deps-to-file", + action="store", + dest="print_deps_path", + metavar="PATH", + default=None, + help="Like --print-deps-only, but to a file (overrides the file)", + ) @tox.hookimpl def tox_configure(config): """Stores options in the config. Makes all commands external and skips sdist""" + 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("") if config.option.current_env or config.option.print_deps_only: config.skipsdist = True for testenv in config.envconfigs: @@ -136,7 +148,10 @@ def tox_runtest(venv, redirect): """If --print-deps-only, prints deps instead of running tests""" config = venv.envconfig.config unsupported_raise(config, venv) - if config.option.print_deps_only: - for dependency in venv.get_resolved_dependencies(): - print(dependency) + 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 + if config.option.print_deps_only: + print(*venv.get_resolved_dependencies(), sep="\n") return True diff --git a/tests/test_integration.py b/tests/test_integration.py index 89cc83c..41a8c6c 100644 --- a/tests/test_integration.py +++ b/tests/test_integration.py @@ -113,6 +113,47 @@ def test_allenvs_print_deps_only(): assert result.stdout == expected +@pytest.mark.parametrize("toxenv", ["py36", "py37", "py38"]) +def test_print_deps_to_file(toxenv, tmp_path): + depspath = tmp_path / "deps" + result = tox("-e", toxenv, "--print-deps-to-file", str(depspath)) + assert depspath.read_text().splitlines() == ["six", "py"] + expected = textwrap.dedent( + f""" + ___________________________________ summary ____________________________________ + {toxenv}: commands succeeded + congratulations :) + """ + ).lstrip() + assert result.stdout == expected + + +def test_allenvs_print_deps_to_file(tmp_path): + depspath = tmp_path / "deps" + result = tox("--print-deps-to-file", str(depspath)) + assert depspath.read_text().splitlines() == ["six", "py"] * 3 + expected = textwrap.dedent( + """ + ___________________________________ summary ____________________________________ + py36: commands succeeded + py37: commands succeeded + py38: commands succeeded + congratulations :) + """ + ).lstrip() + assert result.stdout == expected + + +def test_allenvs_print_deps_to_existing_file(tmp_path): + depspath = tmp_path / "deps" + depspath.write_text("nada") + result = tox("--print-deps-to-file", str(depspath)) + lines = depspath.read_text().splitlines() + assert "nada" not in lines + assert "six" in lines + assert "py" in lines + + @needs_py3678 def test_regular_run(): result = tox() From 7b6ccc79cb36e8fea1a1ea98afbb764b6a80590d Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Miro=20Hron=C4=8Dok?= Date: Mon, 12 Aug 2019 16:05:18 +0200 Subject: [PATCH 2/3] Apply suggestions from code review Co-Authored-By: Petr Viktorin --- README.rst | 2 +- src/tox_current_env/hooks.py | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/README.rst b/README.rst index 5ea96d1..e99640c 100644 --- a/README.rst +++ b/README.rst @@ -18,7 +18,7 @@ The ``tox-current-env`` plugin adds two options: Instead of running any ``commands``, simply prints the declared dependencies in ``deps`` to the standard output or specified file. This is useful for preparing the current environment for the above. - ``--print-deps-to-file`` will override the file, if it already exists. + ``--print-deps-to-file`` will overwrite the file if it already exists. Invoking ``tox`` without any of the above options should behave as regular ``tox`` invocation without this plugin. Any deviation from this behavior is considered a bug. diff --git a/src/tox_current_env/hooks.py b/src/tox_current_env/hooks.py index 82a2275..e0fd48b 100644 --- a/src/tox_current_env/hooks.py +++ b/src/tox_current_env/hooks.py @@ -26,7 +26,7 @@ def tox_addoption(parser): dest="print_deps_path", metavar="PATH", default=None, - help="Like --print-deps-only, but to a file (overrides the file)", + help="Like --print-deps-only, but to a file. Overwrites the file if it exists.", ) From 0863536aff7226990f4e7164b433fc330ef1b1a2 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Miro=20Hron=C4=8Dok?= Date: Mon, 12 Aug 2019 16:17:41 +0200 Subject: [PATCH 3/3] Make --print-deps-only and --print-deps-to-file mutually exclusive --- src/tox_current_env/hooks.py | 6 +++++- tests/test_integration.py | 13 +++++++++++++ 2 files changed, 18 insertions(+), 1 deletion(-) diff --git a/src/tox_current_env/hooks.py b/src/tox_current_env/hooks.py index e0fd48b..14607fe 100644 --- a/src/tox_current_env/hooks.py +++ b/src/tox_current_env/hooks.py @@ -18,7 +18,7 @@ def tox_addoption(parser): action="store_true", dest="print_deps_only", default=False, - help="Don't run tests, only print the dependencies", + help="Don't run tests, only print the dependencies to stdout", ) parser.add_argument( "--print-deps-to-file", @@ -33,6 +33,10 @@ def tox_addoption(parser): @tox.hookimpl def tox_configure(config): """Stores options in the config. Makes all commands external and skips sdist""" + 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" + ) 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: diff --git a/tests/test_integration.py b/tests/test_integration.py index 41a8c6c..3f74ca4 100644 --- a/tests/test_integration.py +++ b/tests/test_integration.py @@ -154,6 +154,19 @@ def test_allenvs_print_deps_to_existing_file(tmp_path): assert "py" in lines +def test_print_deps_only_print_deps_to_file_are_mutually_exclusive(): + result = tox( + "-e", + NATIVE_TOXENV, + "--print-deps-only", + "--print-deps-to-file", + "foobar", + check=False, + ) + assert result.returncode > 0 + assert "cannot be used together" in result.stderr + + @needs_py3678 def test_regular_run(): result = tox()