1
0
Fork 0
mirror of https://github.com/ansible-collections/community.general.git synced 2026-02-04 07:51:50 +00:00
community.general/plugins/modules/layman.py
patchback[bot] f49c5f79a7
[PR #11343/e8f2b135 backport][stable-12] batch 3 - update Python idiom to 3.7 using pyupgrade (#11352)
batch 3 - update Python idiom to 3.7 using pyupgrade (#11343)

* batch 3 - update Python idiom to 3.7 using pyupgrade

* add changelog frag

* bring back sanity

* adjust test

* Apply suggestions from code review

(cherry picked from commit e8f2b135ba)

Co-authored-by: Alexei Znamensky <103110+russoz@users.noreply.github.com>
2025-12-30 22:43:24 +01:00

275 lines
7.6 KiB
Python

#!/usr/bin/python
# Copyright (c) 2014, Jakub Jirutka <jakub@jirutka.cz>
#
# 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 annotations
DOCUMENTATION = r"""
module: layman
author: "Jakub Jirutka (@jirutka)"
short_description: Manage Gentoo overlays
deprecated:
removed_in: 14.0.0
why: Gentoo deprecated C(layman) in mid-2023.
alternative: None.
description:
- Uses Layman to manage an additional repositories for the Portage package manager on Gentoo Linux. Please note that Layman
must be installed on a managed node prior using this module.
requirements:
- layman python module
extends_documentation_fragment:
- community.general.attributes
attributes:
check_mode:
support: full
diff_mode:
support: none
options:
name:
description:
- The overlay ID to install, synchronize, or uninstall. Use V(ALL) to sync all of the installed overlays (can be used
only when O(state=updated)).
required: true
type: str
list_url:
description:
- An URL of the alternative overlays list that defines the overlay to install. This list is fetched and saved under
C(${overlay_defs}/${name}.xml), where C(overlay_defs) is read from the Layman's configuration.
aliases: [url]
type: str
state:
description:
- Whether to install (V(present)), sync (V(updated)), or uninstall (V(absent)) the overlay.
default: present
choices: [present, absent, updated]
type: str
validate_certs:
description:
- If V(false), SSL certificates are not validated. This should only be set to V(false) when no other option exists.
type: bool
default: true
"""
EXAMPLES = r"""
- name: Install the overlay mozilla which is on the central overlays list
community.general.layman:
name: mozilla
- name: Install the overlay cvut from the specified alternative list
community.general.layman:
name: cvut
list_url: 'http://raw.github.com/cvut/gentoo-overlay/master/overlay.xml'
- name: Update (sync) the overlay cvut or install if not installed yet
community.general.layman:
name: cvut
list_url: 'http://raw.github.com/cvut/gentoo-overlay/master/overlay.xml'
state: updated
- name: Update (sync) all of the installed overlays
community.general.layman:
name: ALL
state: updated
- name: Uninstall the overlay cvut
community.general.layman:
name: cvut
state: absent
"""
import shutil
import traceback
from os import path
LAYMAN_IMP_ERR = None
try:
from layman.api import LaymanAPI
from layman.config import BareConfig
HAS_LAYMAN_API = True
except ImportError:
LAYMAN_IMP_ERR = traceback.format_exc()
HAS_LAYMAN_API = False
from ansible.module_utils.basic import AnsibleModule, missing_required_lib
from ansible.module_utils.urls import fetch_url
USERAGENT = "ansible-httpget"
class ModuleError(Exception):
pass
def init_layman(config=None):
"""Returns the initialized ``LaymanAPI``.
:param config: the layman's configuration to use (optional)
"""
if config is None:
config = BareConfig(read_configfile=True, quietness=1)
return LaymanAPI(config)
def download_url(module, url, dest):
"""
:param url: the URL to download
:param dest: the absolute path of where to save the downloaded content to;
it must be writable and not a directory
:raises ModuleError
"""
# Hack to add params in the form that fetch_url expects
module.params["http_agent"] = USERAGENT
response, info = fetch_url(module, url)
if info["status"] != 200:
raise ModuleError(f"Failed to get {url}: {info['msg']}")
try:
with open(dest, "w") as f:
shutil.copyfileobj(response, f)
except OSError as e:
raise ModuleError(f"Failed to write: {e}") from e
def install_overlay(module, name, list_url=None):
"""Installs the overlay repository. If not on the central overlays list,
then :list_url of an alternative list must be provided. The list will be
fetched and saved under ``%(overlay_defs)/%(name.xml)`` (location of the
``overlay_defs`` is read from the Layman's configuration).
:param name: the overlay id
:param list_url: the URL of the remote repositories list to look for the overlay
definition (optional, default: None)
:returns: True if the overlay was installed, or False if already exists
(i.e. nothing has changed)
:raises ModuleError
"""
# read Layman configuration
layman_conf = BareConfig(read_configfile=True)
layman = init_layman(layman_conf)
if layman.is_installed(name):
return False
if module.check_mode:
mymsg = f"Would add layman repo '{name}'"
module.exit_json(changed=True, msg=mymsg)
if not layman.is_repo(name):
if not list_url:
raise ModuleError(
f"Overlay '{name}' is not on the list of known overlays and URL of the remote list was not provided."
)
overlay_defs = layman_conf.get_option("overlay_defs")
dest = path.join(overlay_defs, f"{name}.xml")
download_url(module, list_url, dest)
# reload config
layman = init_layman()
if not layman.add_repos(name):
raise ModuleError(layman.get_errors())
return True
def uninstall_overlay(module, name):
"""Uninstalls the given overlay repository from the system.
:param name: the overlay id to uninstall
:returns: True if the overlay was uninstalled, or False if doesn't exist
(i.e. nothing has changed)
:raises ModuleError
"""
layman = init_layman()
if not layman.is_installed(name):
return False
if module.check_mode:
mymsg = f"Would remove layman repo '{name}'"
module.exit_json(changed=True, msg=mymsg)
layman.delete_repos(name)
if layman.get_errors():
raise ModuleError(layman.get_errors())
return True
def sync_overlay(name):
"""Synchronizes the specified overlay repository.
:param name: the overlay repository id to sync
:raises ModuleError
"""
layman = init_layman()
if not layman.sync(name):
messages = [str(item[1]) for item in layman.sync_results[2]]
raise ModuleError(messages)
def sync_overlays():
"""Synchronize all of the installed overlays.
:raises ModuleError
"""
layman = init_layman()
for name in layman.get_installed():
sync_overlay(name)
def main():
# define module
module = AnsibleModule(
argument_spec=dict(
name=dict(required=True),
list_url=dict(aliases=["url"]),
state=dict(default="present", choices=["present", "absent", "updated"]),
validate_certs=dict(default=True, type="bool"),
),
supports_check_mode=True,
)
if not HAS_LAYMAN_API:
module.fail_json(msg=missing_required_lib("Layman"), exception=LAYMAN_IMP_ERR)
state, name, url = (module.params[key] for key in ["state", "name", "list_url"])
changed = False
try:
if state == "present":
changed = install_overlay(module, name, url)
elif state == "updated":
if name == "ALL":
sync_overlays()
elif install_overlay(module, name, url):
changed = True
else:
sync_overlay(name)
else:
changed = uninstall_overlay(module, name)
except ModuleError as e:
module.fail_json(msg=e.message)
else:
module.exit_json(changed=changed, name=name)
if __name__ == "__main__":
main()