1
0
Fork 0
mirror of https://github.com/ansible-collections/community.general.git synced 2024-09-14 20:13:21 +02:00

cargo module install from source in a given directory (#8480)

* Fixes installed version for git/local.

* Support latest determination with local source.

* Adds docs.

* Improves error message.

* Setup for tests.

* Updates copyright.

* Align closer to #7895.

* Adds changelog.

* Check directory exists.

* Stop using format strings.

* Corrects directory arg type in docs.

* Setup test repo dynamically.

* Adds tests.

* Adds version matching tests.

* Update changelog fragment to match PR ID.

* Updates copyright.

* Import new directory tests.
This commit is contained in:
Colin Nolan 2024-06-17 06:15:31 +01:00 committed by GitHub
parent 3314d5c8db
commit 69b72e4a8e
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
4 changed files with 183 additions and 4 deletions

View file

@ -0,0 +1,2 @@
minor_changes:
- "cargo - add option ``directory``, which allows source directory to be specified (https://github.com/ansible-collections/community.general/pull/8480)."

View file

@ -1,6 +1,7 @@
#!/usr/bin/python #!/usr/bin/python
# -*- coding: utf-8 -*- # -*- coding: utf-8 -*-
# Copyright (c) 2021 Radek Sprta <mail@radeksprta.eu> # Copyright (c) 2021 Radek Sprta <mail@radeksprta.eu>
# Copyright (c) 2024 Colin Nolan <cn580@alumni.york.ac.uk>
# GNU General Public License v3.0+ (see LICENSES/GPL-3.0-or-later.txt or https://www.gnu.org/licenses/gpl-3.0.txt) # 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 # SPDX-License-Identifier: GPL-3.0-or-later
@ -65,6 +66,13 @@ options:
type: str type: str
default: present default: present
choices: [ "present", "absent", "latest" ] choices: [ "present", "absent", "latest" ]
directory:
description:
- Path to the source directory to install the Rust package from.
- This is only used when installing packages.
type: path
required: false
version_added: 9.1.0
requirements: requirements:
- cargo installed - cargo installed
""" """
@ -98,8 +106,14 @@ EXAMPLES = r"""
community.general.cargo: community.general.cargo:
name: ludusavi name: ludusavi
state: latest state: latest
- name: Install "ludusavi" Rust package from source directory
community.general.cargo:
name: ludusavi
directory: /path/to/ludusavi/source
""" """
import json
import os import os
import re import re
@ -115,6 +129,7 @@ class Cargo(object):
self.state = kwargs["state"] self.state = kwargs["state"]
self.version = kwargs["version"] self.version = kwargs["version"]
self.locked = kwargs["locked"] self.locked = kwargs["locked"]
self.directory = kwargs["directory"]
@property @property
def path(self): def path(self):
@ -143,7 +158,7 @@ class Cargo(object):
data, dummy = self._exec(cmd, True, False, False) data, dummy = self._exec(cmd, True, False, False)
package_regex = re.compile(r"^([\w\-]+) v(.+):$") package_regex = re.compile(r"^([\w\-]+) v(\S+).*:$")
installed = {} installed = {}
for line in data.splitlines(): for line in data.splitlines():
package_info = package_regex.match(line) package_info = package_regex.match(line)
@ -163,19 +178,53 @@ class Cargo(object):
if self.version: if self.version:
cmd.append("--version") cmd.append("--version")
cmd.append(self.version) cmd.append(self.version)
if self.directory:
cmd.append("--path")
cmd.append(self.directory)
return self._exec(cmd) return self._exec(cmd)
def is_outdated(self, name): def is_outdated(self, name):
installed_version = self.get_installed().get(name) installed_version = self.get_installed().get(name)
latest_version = (
self.get_latest_published_version(name)
if not self.directory
else self.get_source_directory_version(name)
)
return installed_version != latest_version
def get_latest_published_version(self, name):
cmd = ["search", name, "--limit", "1"] cmd = ["search", name, "--limit", "1"]
data, dummy = self._exec(cmd, True, False, False) data, dummy = self._exec(cmd, True, False, False)
match = re.search(r'"(.+)"', data) match = re.search(r'"(.+)"', data)
if match: if not match:
latest_version = match.group(1) self.module.fail_json(
msg="No published version for package %s found" % name
)
return match.group(1)
return installed_version != latest_version def get_source_directory_version(self, name):
cmd = [
"metadata",
"--format-version",
"1",
"--no-deps",
"--manifest-path",
os.path.join(self.directory, "Cargo.toml"),
]
data, dummy = self._exec(cmd, True, False, False)
manifest = json.loads(data)
package = next(
(package for package in manifest["packages"] if package["name"] == name),
None,
)
if not package:
self.module.fail_json(
msg="Package %s not defined in source, found: %s"
% (name, [x["name"] for x in manifest["packages"]])
)
return package["version"]
def uninstall(self, packages=None): def uninstall(self, packages=None):
cmd = ["uninstall"] cmd = ["uninstall"]
@ -191,16 +240,21 @@ def main():
state=dict(default="present", choices=["present", "absent", "latest"]), state=dict(default="present", choices=["present", "absent", "latest"]),
version=dict(default=None, type="str"), version=dict(default=None, type="str"),
locked=dict(default=False, type="bool"), locked=dict(default=False, type="bool"),
directory=dict(default=None, type="path"),
) )
module = AnsibleModule(argument_spec=arg_spec, supports_check_mode=True) module = AnsibleModule(argument_spec=arg_spec, supports_check_mode=True)
name = module.params["name"] name = module.params["name"]
state = module.params["state"] state = module.params["state"]
version = module.params["version"] version = module.params["version"]
directory = module.params["directory"]
if not name: if not name:
module.fail_json(msg="Package name must be specified") module.fail_json(msg="Package name must be specified")
if directory is not None and not os.path.isdir(directory):
module.fail_json(msg="Source directory does not exist")
# Set LANG env since we parse stdout # Set LANG env since we parse stdout
module.run_command_environ_update = dict( module.run_command_environ_update = dict(
LANG="C", LC_ALL="C", LC_MESSAGES="C", LC_CTYPE="C" LANG="C", LC_ALL="C", LC_MESSAGES="C", LC_CTYPE="C"

View file

@ -16,6 +16,7 @@
- block: - block:
- import_tasks: test_general.yml - import_tasks: test_general.yml
- import_tasks: test_version.yml - import_tasks: test_version.yml
- import_tasks: test_directory.yml
environment: "{{ cargo_environment }}" environment: "{{ cargo_environment }}"
when: has_cargo | default(false) when: has_cargo | default(false)
- import_tasks: test_rustup_cargo.yml - import_tasks: test_rustup_cargo.yml

View file

@ -0,0 +1,122 @@
---
# Copyright (c) 2024 Colin Nolan <cn580@alumni.york.ac.uk>
# 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
- name: Create temp directory
tempfile:
state: directory
register: temp_directory
- name: Test block
vars:
manifest_path: "{{ temp_directory.path }}/Cargo.toml"
package_name: hello-world-directory-test
block:
- name: Initialize package
ansible.builtin.command:
cmd: "cargo init --name {{ package_name }}"
args:
chdir: "{{ temp_directory.path }}"
- name: Set package version (1.0.0)
ansible.builtin.lineinfile:
path: "{{ manifest_path }}"
regexp: '^version = ".*"$'
line: 'version = "1.0.0"'
- name: Ensure package is uninstalled
community.general.cargo:
name: "{{ package_name }}"
state: absent
directory: "{{ temp_directory.path }}"
register: uninstall_absent
- name: Install package
community.general.cargo:
name: "{{ package_name }}"
directory: "{{ temp_directory.path }}"
register: install_absent
- name: Change package version (1.0.1)
ansible.builtin.lineinfile:
path: "{{ manifest_path }}"
regexp: '^version = ".*"$'
line: 'version = "1.0.1"'
- name: Install package again (present)
community.general.cargo:
name: "{{ package_name }}"
state: present
directory: "{{ temp_directory.path }}"
register: install_present_state
- name: Install package again (latest)
community.general.cargo:
name: "{{ package_name }}"
state: latest
directory: "{{ temp_directory.path }}"
register: install_latest_state
- name: Change package version (2.0.0)
ansible.builtin.lineinfile:
path: "{{ manifest_path }}"
regexp: '^version = ".*"$'
line: 'version = "2.0.0"'
- name: Install package with given version (matched)
community.general.cargo:
name: "{{ package_name }}"
version: "2.0.0"
directory: "{{ temp_directory.path }}"
register: install_given_version_matched
- name: Install package with given version (unmatched)
community.general.cargo:
name: "{{ package_name }}"
version: "2.0.1"
directory: "{{ temp_directory.path }}"
register: install_given_version_unmatched
ignore_errors: true
- name: Uninstall package
community.general.cargo:
name: "{{ package_name }}"
state: absent
directory: "{{ temp_directory.path }}"
register: uninstall_present
- name: Install non-existant package
community.general.cargo:
name: "{{ package_name }}-non-existant"
state: present
directory: "{{ temp_directory.path }}"
register: install_non_existant
ignore_errors: true
- name: Install non-existant source directory
community.general.cargo:
name: "{{ package_name }}"
state: present
directory: "{{ temp_directory.path }}/non-existant"
register: install_non_existant_source
ignore_errors: true
always:
- name: Remove temp directory
file:
path: "{{ temp_directory.path }}"
state: absent
- name: Check assertions
assert:
that:
- uninstall_absent is not changed
- install_absent is changed
- install_present_state is not changed
- install_latest_state is changed
- install_given_version_matched is changed
- install_given_version_unmatched is failed
- uninstall_present is changed
- install_non_existant is failed
- install_non_existant_source is failed