mirror of
https://github.com/ansible-collections/community.general.git
synced 2025-06-29 11:40:22 -07:00
pipx: accept python version specs in parameter name
(#10031)
* pipx: accept python version specs in parameter "name" * pipx_info: adjustment for backward compatibility * remove unnecessary comprehension * remove f-str * no shebang for module utils * remove f-str * fix syntax error * fix pipx_info * rollback adjustments in existing tests * docs & test update * add debugging tasks to int test * integration test checks for version of packaging * move assertion to block * fix idempotency when using version specifier * add changelog frag * fix docs * dial down the version of tox used in tests To accommodate old Pythons * Update plugins/modules/pipx.py * Apply suggestions from code review * refactor/rename package requirements code * fix filename in BOTMETA * Update plugins/modules/pipx.py Co-authored-by: Felix Fontein <felix@fontein.de> * Update plugins/modules/pipx.py Co-authored-by: Felix Fontein <felix@fontein.de> * pipx mod utils: create make_process_dict and deprecate make_process_list * pkg_req: make method private * make_process_dict is simpler and more specialized * ensure version specifiers are honored when state=install * fix insanity * pipx: reformat yaml blocks * pipx: doc wordsmithing --------- Co-authored-by: Felix Fontein <felix@fontein.de>
This commit is contained in:
parent
626ee3115d
commit
2b4cb6dabc
8 changed files with 302 additions and 64 deletions
2
.github/BOTMETA.yml
vendored
2
.github/BOTMETA.yml
vendored
|
@ -387,6 +387,8 @@ files:
|
||||||
$module_utils/pipx.py:
|
$module_utils/pipx.py:
|
||||||
labels: pipx
|
labels: pipx
|
||||||
maintainers: russoz
|
maintainers: russoz
|
||||||
|
$module_utils/pkg_req.py:
|
||||||
|
maintainers: russoz
|
||||||
$module_utils/python_runner.py:
|
$module_utils/python_runner.py:
|
||||||
maintainers: russoz
|
maintainers: russoz
|
||||||
$module_utils/puppet.py:
|
$module_utils/puppet.py:
|
||||||
|
|
8
changelogs/fragments/10031-pipx-python-version.yml
Normal file
8
changelogs/fragments/10031-pipx-python-version.yml
Normal file
|
@ -0,0 +1,8 @@
|
||||||
|
minor_changes:
|
||||||
|
- pipx module_utils - filtering application list by name now happens in the modules (https://github.com/ansible-collections/community.general/pull/10031).
|
||||||
|
- pipx_info - filtering application list by name now happens in the module (https://github.com/ansible-collections/community.general/pull/10031).
|
||||||
|
- >
|
||||||
|
pipx - parameter ``name`` now accepts Python package specifiers
|
||||||
|
(https://github.com/ansible-collections/community.general/issues/7815, https://github.com/ansible-collections/community.general/pull/10031).
|
||||||
|
deprecated_features:
|
||||||
|
- pipx module_utils - function ``make_process_list()`` is deprecated and will be removed in community.general 13.0.0 (https://github.com/ansible-collections/community.general/pull/10031).
|
|
@ -71,36 +71,51 @@ def pipx_runner(module, command, **kwargs):
|
||||||
return runner
|
return runner
|
||||||
|
|
||||||
|
|
||||||
def make_process_list(mod_helper, **kwargs):
|
def _make_entry(venv_name, venv, include_injected, include_deps):
|
||||||
def process_list(rc, out, err):
|
entry = {
|
||||||
if not out:
|
'name': venv_name,
|
||||||
return []
|
'version': venv['metadata']['main_package']['package_version'],
|
||||||
|
'pinned': venv['metadata']['main_package'].get('pinned'),
|
||||||
|
}
|
||||||
|
if include_injected:
|
||||||
|
entry['injected'] = {k: v['package_version'] for k, v in venv['metadata']['injected_packages'].items()}
|
||||||
|
if include_deps:
|
||||||
|
entry['dependencies'] = list(venv['metadata']['main_package']['app_paths_of_dependencies'])
|
||||||
|
return entry
|
||||||
|
|
||||||
results = []
|
|
||||||
|
def make_process_dict(include_injected, include_deps=False):
|
||||||
|
def process_dict(rc, out, err):
|
||||||
|
if not out:
|
||||||
|
return {}
|
||||||
|
|
||||||
|
results = {}
|
||||||
raw_data = json.loads(out)
|
raw_data = json.loads(out)
|
||||||
|
for venv_name, venv in raw_data['venvs'].items():
|
||||||
|
results[venv_name] = _make_entry(venv_name, venv, include_injected, include_deps)
|
||||||
|
|
||||||
|
return results, raw_data
|
||||||
|
|
||||||
|
return process_dict
|
||||||
|
|
||||||
|
|
||||||
|
def make_process_list(mod_helper, **kwargs):
|
||||||
|
#
|
||||||
|
# ATTENTION!
|
||||||
|
#
|
||||||
|
# The function `make_process_list()` is deprecated and will be removed in community.general 13.0.0
|
||||||
|
#
|
||||||
|
process_dict = make_process_dict(mod_helper, **kwargs)
|
||||||
|
|
||||||
|
def process_list(rc, out, err):
|
||||||
|
res_dict, raw_data = process_dict(rc, out, err)
|
||||||
|
|
||||||
if kwargs.get("include_raw"):
|
if kwargs.get("include_raw"):
|
||||||
mod_helper.vars.raw_output = raw_data
|
mod_helper.vars.raw_output = raw_data
|
||||||
|
|
||||||
if kwargs["name"]:
|
return [
|
||||||
if kwargs["name"] in raw_data['venvs']:
|
entry
|
||||||
data = {kwargs["name"]: raw_data['venvs'][kwargs["name"]]}
|
for name, entry in res_dict.items()
|
||||||
else:
|
if name == kwargs.get("name")
|
||||||
data = {}
|
]
|
||||||
else:
|
|
||||||
data = raw_data['venvs']
|
|
||||||
|
|
||||||
for venv_name, venv in data.items():
|
|
||||||
entry = {
|
|
||||||
'name': venv_name,
|
|
||||||
'version': venv['metadata']['main_package']['package_version'],
|
|
||||||
'pinned': venv['metadata']['main_package'].get('pinned'),
|
|
||||||
}
|
|
||||||
if kwargs.get("include_injected"):
|
|
||||||
entry['injected'] = {k: v['package_version'] for k, v in venv['metadata']['injected_packages'].items()}
|
|
||||||
if kwargs.get("include_deps"):
|
|
||||||
entry['dependencies'] = list(venv['metadata']['main_package']['app_paths_of_dependencies'])
|
|
||||||
results.append(entry)
|
|
||||||
|
|
||||||
return results
|
|
||||||
|
|
||||||
return process_list
|
return process_list
|
||||||
|
|
75
plugins/module_utils/pkg_req.py
Normal file
75
plugins/module_utils/pkg_req.py
Normal file
|
@ -0,0 +1,75 @@
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
# Copyright (c) 2025, Alexei Znamensky <russoz@gmail.com>
|
||||||
|
# GNU General Public License v3.0+ (see LICENSES/GPL-3.0-or-later.txt or https://www.gnu.org/licenses/gpl-3.0.txt)
|
||||||
|
# SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
|
||||||
|
from __future__ import absolute_import, division, print_function
|
||||||
|
__metaclass__ = type
|
||||||
|
|
||||||
|
from ansible.module_utils.six import raise_from
|
||||||
|
|
||||||
|
from ansible_collections.community.general.plugins.module_utils import deps
|
||||||
|
|
||||||
|
|
||||||
|
with deps.declare("packaging"):
|
||||||
|
from packaging.requirements import Requirement
|
||||||
|
from packaging.version import parse as parse_version, InvalidVersion
|
||||||
|
|
||||||
|
|
||||||
|
class PackageRequirement:
|
||||||
|
def __init__(self, module, name):
|
||||||
|
self.module = module
|
||||||
|
self.parsed_name, self.requirement = self._parse_spec(name)
|
||||||
|
|
||||||
|
def _parse_spec(self, name):
|
||||||
|
"""
|
||||||
|
Parse a package name that may include version specifiers using PEP 508.
|
||||||
|
Returns a tuple of (name, requirement) where requirement is of type packaging.requirements.Requirement and it may be None.
|
||||||
|
|
||||||
|
Example inputs:
|
||||||
|
"package"
|
||||||
|
"package>=1.0"
|
||||||
|
"package>=1.0,<2.0"
|
||||||
|
"package[extra]>=1.0"
|
||||||
|
"package[foo,bar]>=1.0,!=1.5"
|
||||||
|
|
||||||
|
:param name: Package name with optional version specifiers and extras
|
||||||
|
:return: Tuple of (name, requirement)
|
||||||
|
:raises ValueError: If the package specification is invalid
|
||||||
|
"""
|
||||||
|
if not name:
|
||||||
|
return name, None
|
||||||
|
|
||||||
|
# Quick check for simple package names
|
||||||
|
if not any(c in name for c in '>=<!~[]'):
|
||||||
|
return name.strip(), None
|
||||||
|
|
||||||
|
deps.validate(self.module, "packaging")
|
||||||
|
|
||||||
|
try:
|
||||||
|
req = Requirement(name)
|
||||||
|
return req.name, req
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
raise_from(ValueError("Invalid package specification for '{0}': {1}".format(name, e)), e)
|
||||||
|
|
||||||
|
def matches_version(self, version):
|
||||||
|
"""
|
||||||
|
Check if a version string fulfills a version specifier.
|
||||||
|
|
||||||
|
:param version: Version string to check
|
||||||
|
:return: True if version fulfills the requirement, False otherwise
|
||||||
|
:raises ValueError: If version is invalid
|
||||||
|
"""
|
||||||
|
# If no spec provided, any version is valid
|
||||||
|
if not self.requirement:
|
||||||
|
return True
|
||||||
|
|
||||||
|
try:
|
||||||
|
# Parse version string
|
||||||
|
ver = parse_version(version)
|
||||||
|
|
||||||
|
return ver in self.requirement.specifier
|
||||||
|
|
||||||
|
except InvalidVersion as e:
|
||||||
|
raise_from(ValueError("Invalid version '{0}': {1}".format(version, e)))
|
|
@ -54,11 +54,17 @@ options:
|
||||||
name:
|
name:
|
||||||
type: str
|
type: str
|
||||||
description:
|
description:
|
||||||
- The name of the application. In C(pipx) documentation it is also referred to as the name of the virtual environment
|
- The name of the application and also the name of the Python package being installed.
|
||||||
where the application will be installed.
|
- In C(pipx) documentation it is also referred to as the name of the virtual environment where the application is installed.
|
||||||
- If O(name) is a simple package name without version specifiers, then that name is used as the Python package name
|
- If O(name) is a simple package name without version specifiers, then that name is used as the Python package name
|
||||||
to be installed.
|
to be installed.
|
||||||
- Use O(source) for passing package specifications or installing from URLs or directories.
|
- Starting in community.general 10.7.0, you can use package specifiers when O(state=present) or O(state=install). For
|
||||||
|
example, O(name=tox<4.0.0) or O(name=tox>3.0.27).
|
||||||
|
- Please note that when you use O(state=present) and O(name) with version specifiers, contrary to the behavior of C(pipx),
|
||||||
|
this module honors the version specifier and installs a version of the application that satisfies it. If you want
|
||||||
|
to ensure the reinstallation of the application even when the version specifier is met, then you must use O(force=true),
|
||||||
|
or perhaps use O(state=upgrade) instead.
|
||||||
|
- Use O(source) for installing from URLs or directories.
|
||||||
source:
|
source:
|
||||||
type: str
|
type: str
|
||||||
description:
|
description:
|
||||||
|
@ -69,6 +75,7 @@ options:
|
||||||
- The value of this option is passed as-is to C(pipx).
|
- The value of this option is passed as-is to C(pipx).
|
||||||
- O(name) is still required when using O(source) to establish the application name without fetching the package from
|
- O(name) is still required when using O(source) to establish the application name without fetching the package from
|
||||||
a remote source.
|
a remote source.
|
||||||
|
- The module is not idempotent when using O(source).
|
||||||
install_apps:
|
install_apps:
|
||||||
description:
|
description:
|
||||||
- Add apps from the injected packages.
|
- Add apps from the injected packages.
|
||||||
|
@ -92,6 +99,7 @@ options:
|
||||||
description:
|
description:
|
||||||
- Force modification of the application's virtual environment. See C(pipx) for details.
|
- Force modification of the application's virtual environment. See C(pipx) for details.
|
||||||
- Only used when O(state=install), O(state=upgrade), O(state=upgrade_all), O(state=latest), or O(state=inject).
|
- Only used when O(state=install), O(state=upgrade), O(state=upgrade_all), O(state=latest), or O(state=inject).
|
||||||
|
- The module is not idempotent when O(force=true).
|
||||||
type: bool
|
type: bool
|
||||||
default: false
|
default: false
|
||||||
include_injected:
|
include_injected:
|
||||||
|
@ -144,10 +152,10 @@ options:
|
||||||
with O(community.general.pipx_info#module:include_raw=true) and obtaining the content from the RV(community.general.pipx_info#module:raw_output).
|
with O(community.general.pipx_info#module:include_raw=true) and obtaining the content from the RV(community.general.pipx_info#module:raw_output).
|
||||||
type: path
|
type: path
|
||||||
version_added: 9.4.0
|
version_added: 9.4.0
|
||||||
notes:
|
requirements:
|
||||||
- This first implementation does not verify whether a specified version constraint has been installed or not. Hence, when
|
- When using O(name) with version specifiers, the Python package C(packaging) is required.
|
||||||
using version operators, C(pipx) module will always try to execute the operation, even when the application was previously
|
- If the package C(packaging) is at a version lesser than C(22.0.0), it will fail silently when processing invalid specifiers,
|
||||||
installed. This feature will be added in the future.
|
like C(tox<<<<4.0).
|
||||||
author:
|
author:
|
||||||
- "Alexei Znamensky (@russoz)"
|
- "Alexei Znamensky (@russoz)"
|
||||||
"""
|
"""
|
||||||
|
@ -201,7 +209,8 @@ version:
|
||||||
|
|
||||||
|
|
||||||
from ansible_collections.community.general.plugins.module_utils.module_helper import StateModuleHelper
|
from ansible_collections.community.general.plugins.module_utils.module_helper import StateModuleHelper
|
||||||
from ansible_collections.community.general.plugins.module_utils.pipx import pipx_runner, pipx_common_argspec, make_process_list
|
from ansible_collections.community.general.plugins.module_utils.pipx import pipx_runner, pipx_common_argspec, make_process_dict
|
||||||
|
from ansible_collections.community.general.plugins.module_utils.pkg_req import PackageRequirement
|
||||||
|
|
||||||
from ansible.module_utils.facts.compat import ansible_facts
|
from ansible.module_utils.facts.compat import ansible_facts
|
||||||
|
|
||||||
|
@ -258,18 +267,13 @@ class PipX(StateModuleHelper):
|
||||||
use_old_vardict = False
|
use_old_vardict = False
|
||||||
|
|
||||||
def _retrieve_installed(self):
|
def _retrieve_installed(self):
|
||||||
name = _make_name(self.vars.name, self.vars.suffix)
|
output_process = make_process_dict(include_injected=True)
|
||||||
output_process = make_process_list(self, include_injected=True, name=name)
|
installed, dummy = self.runner('_list global', output_process=output_process).run()
|
||||||
installed = self.runner('_list global', output_process=output_process).run()
|
|
||||||
|
|
||||||
if name is not None:
|
if self.app_name is None:
|
||||||
app_list = [app for app in installed if app['name'] == name]
|
return installed
|
||||||
if app_list:
|
|
||||||
return {name: app_list[0]}
|
|
||||||
else:
|
|
||||||
return {}
|
|
||||||
|
|
||||||
return installed
|
return {k: v for k, v in installed.items() if k == self.app_name}
|
||||||
|
|
||||||
def __init_module__(self):
|
def __init_module__(self):
|
||||||
if self.vars.executable:
|
if self.vars.executable:
|
||||||
|
@ -279,6 +283,11 @@ class PipX(StateModuleHelper):
|
||||||
self.command = [facts['python']['executable'], '-m', 'pipx']
|
self.command = [facts['python']['executable'], '-m', 'pipx']
|
||||||
self.runner = pipx_runner(self.module, self.command)
|
self.runner = pipx_runner(self.module, self.command)
|
||||||
|
|
||||||
|
pkg_req = PackageRequirement(self.module, self.vars.name)
|
||||||
|
self.parsed_name = pkg_req.parsed_name
|
||||||
|
self.parsed_req = pkg_req.requirement
|
||||||
|
self.app_name = _make_name(self.parsed_name, self.vars.suffix)
|
||||||
|
|
||||||
self.vars.set('application', self._retrieve_installed(), change=True, diff=True)
|
self.vars.set('application', self._retrieve_installed(), change=True, diff=True)
|
||||||
|
|
||||||
with self.runner("version") as ctx:
|
with self.runner("version") as ctx:
|
||||||
|
@ -295,12 +304,27 @@ class PipX(StateModuleHelper):
|
||||||
self.vars.set('run_info', ctx.run_info, verbosity=4)
|
self.vars.set('run_info', ctx.run_info, verbosity=4)
|
||||||
|
|
||||||
def state_install(self):
|
def state_install(self):
|
||||||
if not self.vars.application or self.vars.force:
|
# If we have a version spec and no source, use the version spec as source
|
||||||
self.changed = True
|
if self.parsed_req and not self.vars.source:
|
||||||
args_order = 'state global index_url install_deps force python system_site_packages editable pip_args suffix name_source'
|
self.vars.source = self.vars.name
|
||||||
with self.runner(args_order, check_mode_skip=True) as ctx:
|
|
||||||
ctx.run(name_source=[self.vars.name, self.vars.source])
|
if self.vars.application.get(self.app_name):
|
||||||
self._capture_results(ctx)
|
is_installed = True
|
||||||
|
version_match = self.vars.application[self.app_name]['version'] in self.parsed_req.specifier if self.parsed_req else True
|
||||||
|
force = self.vars.force or (not version_match)
|
||||||
|
else:
|
||||||
|
is_installed = False
|
||||||
|
version_match = False
|
||||||
|
force = self.vars.force
|
||||||
|
|
||||||
|
if is_installed and version_match and not force:
|
||||||
|
return
|
||||||
|
|
||||||
|
self.changed = True
|
||||||
|
args_order = 'state global index_url install_deps force python system_site_packages editable pip_args suffix name_source'
|
||||||
|
with self.runner(args_order, check_mode_skip=True) as ctx:
|
||||||
|
ctx.run(name_source=[self.parsed_name, self.vars.source], force=force)
|
||||||
|
self._capture_results(ctx)
|
||||||
|
|
||||||
state_present = state_install
|
state_present = state_install
|
||||||
|
|
||||||
|
|
|
@ -126,7 +126,7 @@ version:
|
||||||
"""
|
"""
|
||||||
|
|
||||||
from ansible_collections.community.general.plugins.module_utils.module_helper import ModuleHelper
|
from ansible_collections.community.general.plugins.module_utils.module_helper import ModuleHelper
|
||||||
from ansible_collections.community.general.plugins.module_utils.pipx import pipx_runner, pipx_common_argspec, make_process_list
|
from ansible_collections.community.general.plugins.module_utils.pipx import pipx_runner, pipx_common_argspec, make_process_dict
|
||||||
|
|
||||||
from ansible.module_utils.facts.compat import ansible_facts
|
from ansible.module_utils.facts.compat import ansible_facts
|
||||||
|
|
||||||
|
@ -158,9 +158,20 @@ class PipXInfo(ModuleHelper):
|
||||||
self.vars.version = out.strip()
|
self.vars.version = out.strip()
|
||||||
|
|
||||||
def __run__(self):
|
def __run__(self):
|
||||||
output_process = make_process_list(self, **self.vars.as_dict())
|
output_process = make_process_dict(self.vars.include_injected, self.vars.include_deps)
|
||||||
with self.runner('_list global', output_process=output_process) as ctx:
|
with self.runner('_list global', output_process=output_process) as ctx:
|
||||||
self.vars.application = ctx.run()
|
applications, raw_data = ctx.run()
|
||||||
|
if self.vars.include_raw:
|
||||||
|
self.vars.raw_output = raw_data
|
||||||
|
|
||||||
|
if self.vars.name:
|
||||||
|
self.vars.application = [
|
||||||
|
v
|
||||||
|
for k, v in applications.items()
|
||||||
|
if k == self.vars.name
|
||||||
|
]
|
||||||
|
else:
|
||||||
|
self.vars.application = list(applications.values())
|
||||||
self._capture_results(ctx)
|
self._capture_results(ctx)
|
||||||
|
|
||||||
def _capture_results(self, ctx):
|
def _capture_results(self, ctx):
|
||||||
|
|
|
@ -6,7 +6,7 @@
|
||||||
- name: Determine pipx level
|
- name: Determine pipx level
|
||||||
block:
|
block:
|
||||||
- name: Install pipx>=1.7.0
|
- name: Install pipx>=1.7.0
|
||||||
pip:
|
ansible.builtin.pip:
|
||||||
name: pipx>=1.7.0
|
name: pipx>=1.7.0
|
||||||
- name: Set has_pipx170 fact true
|
- name: Set has_pipx170 fact true
|
||||||
ansible.builtin.set_fact:
|
ansible.builtin.set_fact:
|
||||||
|
@ -16,9 +16,24 @@
|
||||||
ansible.builtin.set_fact:
|
ansible.builtin.set_fact:
|
||||||
has_pipx170: false
|
has_pipx170: false
|
||||||
- name: Install pipx (no version spec)
|
- name: Install pipx (no version spec)
|
||||||
pip:
|
ansible.builtin.pip:
|
||||||
name: pipx
|
name: pipx
|
||||||
|
|
||||||
|
- name: Determine packaging level
|
||||||
|
block:
|
||||||
|
- name: Install packaging>=22.0
|
||||||
|
ansible.builtin.pip:
|
||||||
|
name: packaging>=22.0
|
||||||
|
- name: Set has_packaging22 fact true
|
||||||
|
ansible.builtin.set_fact:
|
||||||
|
has_packaging22: true
|
||||||
|
rescue:
|
||||||
|
- name: Set has_packaging22 fact false
|
||||||
|
ansible.builtin.set_fact:
|
||||||
|
has_packaging22: false
|
||||||
|
- name: Install has_packaging (no version spec)
|
||||||
|
ansible.builtin.pip:
|
||||||
|
name: packaging
|
||||||
|
|
||||||
##############################################################################
|
##############################################################################
|
||||||
- name: ensure application tox is uninstalled
|
- name: ensure application tox is uninstalled
|
||||||
|
@ -208,26 +223,31 @@
|
||||||
community.general.pipx:
|
community.general.pipx:
|
||||||
state: absent
|
state: absent
|
||||||
name: tox
|
name: tox
|
||||||
register: uninstall_tox_again
|
register: uninstall_tox_latest_yet_again
|
||||||
|
|
||||||
- name: check assertions tox latest
|
- name: check assertions tox latest
|
||||||
assert:
|
assert:
|
||||||
that:
|
that:
|
||||||
- install_tox_latest is changed
|
- install_tox_latest is changed
|
||||||
|
- "'tox' in install_tox_latest.application"
|
||||||
|
- install_tox_latest.application.tox.version != '3.24.0'
|
||||||
- uninstall_tox_latest is changed
|
- uninstall_tox_latest is changed
|
||||||
|
- "'tox' not in uninstall_tox_latest.application"
|
||||||
- install_tox_324_for_latest is changed
|
- install_tox_324_for_latest is changed
|
||||||
|
- "'tox' in install_tox_324_for_latest.application"
|
||||||
- install_tox_324_for_latest.application.tox.version == '3.24.0'
|
- install_tox_324_for_latest.application.tox.version == '3.24.0'
|
||||||
- install_tox_latest_with_preinstall is changed
|
- install_tox_latest_with_preinstall is changed
|
||||||
- install_tox_latest_with_preinstall.application.tox.version == latest_tox_version
|
- "'tox' in install_tox_latest_with_preinstall.application"
|
||||||
|
- install_tox_latest_with_preinstall.application.tox.version != '3.24.0'
|
||||||
- install_tox_latest_with_preinstall_again is not changed
|
- install_tox_latest_with_preinstall_again is not changed
|
||||||
- install_tox_latest_with_preinstall_again.application.tox.version == latest_tox_version
|
|
||||||
- install_tox_latest_with_preinstall_again_force is changed
|
- install_tox_latest_with_preinstall_again_force is changed
|
||||||
- install_tox_latest_with_preinstall_again_force.application.tox.version == latest_tox_version
|
|
||||||
- uninstall_tox_latest_again is changed
|
- uninstall_tox_latest_again is changed
|
||||||
- install_tox_with_deps is changed
|
- "'tox' not in uninstall_tox_latest_again.application"
|
||||||
- install_tox_with_deps.application.tox.version == latest_tox_version
|
|
||||||
- uninstall_tox_again is changed
|
##############################################################################
|
||||||
- "'tox' not in uninstall_tox_again.application"
|
# Test version specifiers in name parameter
|
||||||
|
- name: Run version specifier tests
|
||||||
|
ansible.builtin.include_tasks: testcase-10031-version-specs.yml
|
||||||
|
|
||||||
##############################################################################
|
##############################################################################
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,83 @@
|
||||||
|
---
|
||||||
|
# Copyright (c) Ansible Project
|
||||||
|
# GNU General Public License v3.0+ (see LICENSES/GPL-3.0-or-later.txt or https://www.gnu.org/licenses/gpl-3.0.txt)
|
||||||
|
# SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
|
||||||
|
##############################################################################
|
||||||
|
# Test version specifiers in name parameter
|
||||||
|
|
||||||
|
- name: Ensure tox is uninstalled
|
||||||
|
community.general.pipx:
|
||||||
|
state: absent
|
||||||
|
name: tox
|
||||||
|
register: uninstall_tox
|
||||||
|
|
||||||
|
- name: Install tox with version specifier in name
|
||||||
|
community.general.pipx:
|
||||||
|
name: tox>=3.22.0,<3.27.0
|
||||||
|
register: install_tox_version
|
||||||
|
|
||||||
|
- name: Install tox with same version specifier (idempotency check)
|
||||||
|
community.general.pipx:
|
||||||
|
name: tox>=3.22.0,<3.27.0
|
||||||
|
register: install_tox_version_again
|
||||||
|
|
||||||
|
- name: Ensure tox is uninstalled again
|
||||||
|
community.general.pipx:
|
||||||
|
state: absent
|
||||||
|
name: tox
|
||||||
|
|
||||||
|
- name: Install tox with extras and version
|
||||||
|
community.general.pipx:
|
||||||
|
name: "tox[testing]>=3.22.0,<3.27.0"
|
||||||
|
register: install_tox_extras
|
||||||
|
ignore_errors: true # Some versions might not have this extra
|
||||||
|
|
||||||
|
- name: Install tox with higher version specifier
|
||||||
|
community.general.pipx:
|
||||||
|
name: "tox>=3.27.0"
|
||||||
|
register: install_tox_higher_version
|
||||||
|
|
||||||
|
- name: Install tox with higher version specifier (force)
|
||||||
|
community.general.pipx:
|
||||||
|
name: "tox>=3.27.0"
|
||||||
|
force: true
|
||||||
|
register: install_tox_higher_version_force
|
||||||
|
|
||||||
|
- name: Cleanup tox
|
||||||
|
community.general.pipx:
|
||||||
|
state: absent
|
||||||
|
name: tox
|
||||||
|
register: uninstall_tox_final
|
||||||
|
|
||||||
|
- name: Check version specifier assertions
|
||||||
|
assert:
|
||||||
|
that:
|
||||||
|
- install_tox_version is changed
|
||||||
|
- "'tox' in install_tox_version.application"
|
||||||
|
- "install_tox_version.application.tox.version is version('3.22.0', '>=')"
|
||||||
|
- "install_tox_version.application.tox.version is version('3.27.0', '<')"
|
||||||
|
- install_tox_version_again is not changed
|
||||||
|
- "'tox' in install_tox_extras.application"
|
||||||
|
- "install_tox_extras.application.tox.version is version('3.22.0', '>=')"
|
||||||
|
- "install_tox_extras.application.tox.version is version('3.27.0', '<')"
|
||||||
|
- install_tox_higher_version is changed
|
||||||
|
- install_tox_higher_version_force is changed
|
||||||
|
- uninstall_tox_final is changed
|
||||||
|
- "'tox' not in uninstall_tox_final.application"
|
||||||
|
|
||||||
|
- name: If packaging is recent
|
||||||
|
when:
|
||||||
|
- has_packaging22
|
||||||
|
block:
|
||||||
|
- name: Install tox with invalid version specifier
|
||||||
|
community.general.pipx:
|
||||||
|
name: "tox>>>>>3.27.0"
|
||||||
|
register: install_tox_invalid
|
||||||
|
ignore_errors: true
|
||||||
|
|
||||||
|
- name: Check version specifier assertions
|
||||||
|
assert:
|
||||||
|
that:
|
||||||
|
- install_tox_invalid is failed
|
||||||
|
- "'Invalid package specification' in install_tox_invalid.msg"
|
Loading…
Add table
Add a link
Reference in a new issue