mirror of
https://github.com/ansible-collections/community.general.git
synced 2025-04-05 10:10:31 -07:00
systemd_info - add wildcards support (#9821)
* systemd_info - add wildcards support * systemd_info - add wildcards fragments * systemd_info - improved dedicated functions * systemd_info - improved code and functions for better maintenance and timing * fix unitname description * removed redundancies and keys() in lists, replaced fnmatch with filter and run_command with cmdrunner * systemd_info - add new cmdrunner * systemd_info - fix runner * systemd_info - fix env in runner * systemd_info - rename runner and get_version * systemd_info - change args runner, fix fragment, add botmeta * systemd_info - merge type args
This commit is contained in:
parent
abe4e5ce95
commit
3bd0ab4a49
5 changed files with 221 additions and 109 deletions
2
.github/BOTMETA.yml
vendored
2
.github/BOTMETA.yml
vendored
|
@ -404,6 +404,8 @@ files:
|
||||||
maintainers: russoz
|
maintainers: russoz
|
||||||
$module_utils/ssh.py:
|
$module_utils/ssh.py:
|
||||||
maintainers: russoz
|
maintainers: russoz
|
||||||
|
$module_utils/systemd.py:
|
||||||
|
maintainers: NomakCooper
|
||||||
$module_utils/storage/hpe3par/hpe3par.py:
|
$module_utils/storage/hpe3par/hpe3par.py:
|
||||||
maintainers: farhan7500 gautamphegde
|
maintainers: farhan7500 gautamphegde
|
||||||
$module_utils/utm_utils.py:
|
$module_utils/utm_utils.py:
|
||||||
|
|
2
changelogs/fragments/9821-systemd_info-add-wildcards.yml
Normal file
2
changelogs/fragments/9821-systemd_info-add-wildcards.yml
Normal file
|
@ -0,0 +1,2 @@
|
||||||
|
minor_changes:
|
||||||
|
- systemd_info - add wildcard expression support in ``unitname`` option (https://github.com/ansible-collections/community.general/pull/9821).
|
34
plugins/module_utils/systemd.py
Normal file
34
plugins/module_utils/systemd.py
Normal file
|
@ -0,0 +1,34 @@
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
# Copyright (c) 2025, Marco Noce <nce.marco@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_collections.community.general.plugins.module_utils.cmd_runner import CmdRunner, cmd_runner_fmt
|
||||||
|
|
||||||
|
|
||||||
|
def systemd_runner(module, command, **kwargs):
|
||||||
|
arg_formats = dict(
|
||||||
|
version=cmd_runner_fmt.as_fixed("--version"),
|
||||||
|
list_units=cmd_runner_fmt.as_fixed(["list-units", "--no-pager"]),
|
||||||
|
types=cmd_runner_fmt.as_func(lambda v: [] if not v else ["--type", ",".join(v)]),
|
||||||
|
all=cmd_runner_fmt.as_fixed("--all"),
|
||||||
|
plain=cmd_runner_fmt.as_fixed("--plain"),
|
||||||
|
no_legend=cmd_runner_fmt.as_fixed("--no-legend"),
|
||||||
|
show=cmd_runner_fmt.as_fixed("show"),
|
||||||
|
props=cmd_runner_fmt.as_func(lambda v: [] if not v else ["-p", ",".join(v)]),
|
||||||
|
dashdash=cmd_runner_fmt.as_fixed("--"),
|
||||||
|
unit=cmd_runner_fmt.as_list(),
|
||||||
|
)
|
||||||
|
|
||||||
|
runner = CmdRunner(
|
||||||
|
module,
|
||||||
|
command=command,
|
||||||
|
arg_formats=arg_formats,
|
||||||
|
check_rc=True,
|
||||||
|
**kwargs
|
||||||
|
)
|
||||||
|
return runner
|
|
@ -20,13 +20,16 @@ description:
|
||||||
but only with the minimal properties (RV(units.name), RV(units.loadstate), RV(units.activestate), RV(units.substate)).
|
but only with the minimal properties (RV(units.name), RV(units.loadstate), RV(units.activestate), RV(units.substate)).
|
||||||
- When O(unitname) and O(extra_properties) are used, the module first checks if the unit exists,
|
- When O(unitname) and O(extra_properties) are used, the module first checks if the unit exists,
|
||||||
then check if properties exist. If not, the module fails.
|
then check if properties exist. If not, the module fails.
|
||||||
|
- When O(unitname) is used with wildcard expressions, the module checks for units that match the indicated expressions,
|
||||||
|
if units are not present for all the indicated expressions, the module fails.
|
||||||
version_added: "10.4.0"
|
version_added: "10.4.0"
|
||||||
options:
|
options:
|
||||||
unitname:
|
unitname:
|
||||||
description:
|
description:
|
||||||
- List of unit names to process.
|
- List of unit names to process.
|
||||||
- It supports C(.service), C(.target), C(.socket), and C(.mount) units type.
|
- It supports C(.service), C(.target), C(.socket), and C(.mount) units type.
|
||||||
- Each name must correspond to the full name of the C(systemd) unit.
|
- Each name must correspond to the full name of the C(systemd) unit or to a wildcard expression like V('ssh*') and V('*.service').
|
||||||
|
- Wildcard expressions in O(unitname) are supported since community.general 10.5.0.
|
||||||
type: list
|
type: list
|
||||||
elements: str
|
elements: str
|
||||||
default: []
|
default: []
|
||||||
|
@ -62,6 +65,13 @@ EXAMPLES = r'''
|
||||||
extra_properties:
|
extra_properties:
|
||||||
- Description
|
- Description
|
||||||
register: results
|
register: results
|
||||||
|
|
||||||
|
# Gather info using wildcards/expression
|
||||||
|
- name: Gather info of units that start with 'systemd-'
|
||||||
|
community.general.systemd_info:
|
||||||
|
unitname:
|
||||||
|
- 'systemd-*'
|
||||||
|
register: results
|
||||||
'''
|
'''
|
||||||
|
|
||||||
RETURN = r'''
|
RETURN = r'''
|
||||||
|
@ -195,11 +205,28 @@ units:
|
||||||
}
|
}
|
||||||
'''
|
'''
|
||||||
|
|
||||||
|
import fnmatch
|
||||||
from ansible.module_utils.basic import AnsibleModule
|
from ansible.module_utils.basic import AnsibleModule
|
||||||
|
from ansible_collections.community.general.plugins.module_utils.systemd import systemd_runner
|
||||||
|
|
||||||
|
|
||||||
def run_command(module, cmd):
|
def get_version(runner):
|
||||||
rc, stdout, stderr = module.run_command(cmd, check_rc=True)
|
with runner("version") as ctx:
|
||||||
|
rc, stdout, stderr = ctx.run()
|
||||||
|
return stdout.strip()
|
||||||
|
|
||||||
|
|
||||||
|
def list_units(runner, types_value):
|
||||||
|
context = "list_units types all plain no_legend"
|
||||||
|
with runner(context) as ctx:
|
||||||
|
rc, stdout, stderr = ctx.run(types=types_value)
|
||||||
|
return stdout.strip()
|
||||||
|
|
||||||
|
|
||||||
|
def show_unit_properties(runner, prop_list, unit):
|
||||||
|
context = "show props dashdash unit"
|
||||||
|
with runner(context) as ctx:
|
||||||
|
rc, stdout, stderr = ctx.run(props=prop_list, unit=unit)
|
||||||
return stdout.strip()
|
return stdout.strip()
|
||||||
|
|
||||||
|
|
||||||
|
@ -214,9 +241,8 @@ def parse_show_output(output):
|
||||||
return result
|
return result
|
||||||
|
|
||||||
|
|
||||||
def get_unit_properties(module, systemctl_bin, unit, prop_list):
|
def get_unit_properties(runner, prop_list, unit):
|
||||||
cmd = [systemctl_bin, "show", "-p", ",".join(prop_list), "--", unit]
|
output = show_unit_properties(runner, prop_list, unit)
|
||||||
output = run_command(module, cmd)
|
|
||||||
return parse_show_output(output)
|
return parse_show_output(output)
|
||||||
|
|
||||||
|
|
||||||
|
@ -235,125 +261,141 @@ def determine_category(unit):
|
||||||
|
|
||||||
def extract_unit_properties(unit_data, prop_list):
|
def extract_unit_properties(unit_data, prop_list):
|
||||||
lowerprop = [x.lower() for x in prop_list]
|
lowerprop = [x.lower() for x in prop_list]
|
||||||
extracted = {
|
return {prop: unit_data[prop] for prop in lowerprop if prop in unit_data}
|
||||||
prop: unit_data[prop] for prop in lowerprop if prop in unit_data
|
|
||||||
}
|
|
||||||
return extracted
|
|
||||||
|
|
||||||
|
|
||||||
def unit_exists(module, systemctl_bin, unit):
|
def unit_exists(unit, units_info):
|
||||||
cmd = [systemctl_bin, "show", "-p", "LoadState", "--", unit]
|
info = units_info.get(unit, {})
|
||||||
rc, stdout, stderr = module.run_command(cmd)
|
loadstate = info.get("loadstate", "").lower()
|
||||||
return (rc == 0)
|
return loadstate not in ("not-found", "masked")
|
||||||
|
|
||||||
|
|
||||||
def validate_unit_and_properties(module, systemctl_bin, unit, extra_properties):
|
def get_category_base_props(category):
|
||||||
cmd = [systemctl_bin, "show", "-p", "LoadState", "--", unit]
|
base_props = {
|
||||||
|
|
||||||
output = run_command(module, cmd)
|
|
||||||
if "loadstate=not-found" in output.lower():
|
|
||||||
module.fail_json(msg="Unit '{0}' does not exist or is inaccessible.".format(unit))
|
|
||||||
|
|
||||||
if extra_properties:
|
|
||||||
unit_data = get_unit_properties(module, systemctl_bin, unit, extra_properties)
|
|
||||||
missing_props = [prop for prop in extra_properties if prop.lower() not in unit_data]
|
|
||||||
if missing_props:
|
|
||||||
module.fail_json(msg="The following properties do not exist for unit '{0}': {1}".format(unit, ", ".join(missing_props)))
|
|
||||||
|
|
||||||
|
|
||||||
def main():
|
|
||||||
module_args = dict(
|
|
||||||
unitname=dict(type='list', elements='str', default=[]),
|
|
||||||
extra_properties=dict(type='list', elements='str', default=[])
|
|
||||||
)
|
|
||||||
|
|
||||||
module = AnsibleModule(
|
|
||||||
argument_spec=module_args,
|
|
||||||
supports_check_mode=True
|
|
||||||
)
|
|
||||||
|
|
||||||
systemctl_bin = module.get_bin_path('systemctl', required=True)
|
|
||||||
|
|
||||||
run_command(module, [systemctl_bin, '--version'])
|
|
||||||
|
|
||||||
base_properties = {
|
|
||||||
'service': ['FragmentPath', 'UnitFileState', 'UnitFilePreset', 'MainPID', 'ExecMainPID'],
|
'service': ['FragmentPath', 'UnitFileState', 'UnitFilePreset', 'MainPID', 'ExecMainPID'],
|
||||||
'target': ['FragmentPath', 'UnitFileState', 'UnitFilePreset'],
|
'target': ['FragmentPath', 'UnitFileState', 'UnitFilePreset'],
|
||||||
'socket': ['FragmentPath', 'UnitFileState', 'UnitFilePreset'],
|
'socket': ['FragmentPath', 'UnitFileState', 'UnitFilePreset'],
|
||||||
'mount': ['Where', 'What', 'Options', 'Type']
|
'mount': ['Where', 'What', 'Options', 'Type']
|
||||||
}
|
}
|
||||||
state_props = ['LoadState', 'ActiveState', 'SubState']
|
return base_props.get(category, [])
|
||||||
|
|
||||||
|
|
||||||
|
def validate_unit_and_properties(runner, unit, extra_properties, units_info, property_cache):
|
||||||
|
if not unit_exists(unit, units_info):
|
||||||
|
module.fail_json(msg="Unit '{0}' does not exist or is inaccessible.".format(unit))
|
||||||
|
|
||||||
|
category = determine_category(unit)
|
||||||
|
|
||||||
|
if not category:
|
||||||
|
module.fail_json(msg="Could not determine the category for unit '{0}'.".format(unit))
|
||||||
|
|
||||||
|
state_props = ['LoadState', 'ActiveState', 'SubState']
|
||||||
|
props = get_category_base_props(category)
|
||||||
|
full_props = set(props + state_props + extra_properties)
|
||||||
|
|
||||||
|
if unit not in property_cache:
|
||||||
|
unit_data = get_unit_properties(runner, full_props, unit)
|
||||||
|
property_cache[unit] = unit_data
|
||||||
|
else:
|
||||||
|
unit_data = property_cache[unit]
|
||||||
|
if extra_properties:
|
||||||
|
missing_props = [prop for prop in extra_properties if prop.lower() not in unit_data]
|
||||||
|
if missing_props:
|
||||||
|
module.fail_json(msg="The following properties do not exist for unit '{0}': {1}".format(unit, ", ".join(missing_props)))
|
||||||
|
|
||||||
|
return True
|
||||||
|
|
||||||
|
|
||||||
|
def process_wildcards(selected_units, all_units, module):
|
||||||
|
resolved_units = {}
|
||||||
|
non_matching_patterns = []
|
||||||
|
|
||||||
|
for pattern in selected_units:
|
||||||
|
matches = fnmatch.filter(all_units, pattern)
|
||||||
|
if not matches:
|
||||||
|
non_matching_patterns.append(pattern)
|
||||||
|
else:
|
||||||
|
for match in matches:
|
||||||
|
resolved_units[match] = True
|
||||||
|
|
||||||
|
if not resolved_units:
|
||||||
|
module.fail_json(msg="No units match any of the provided patterns: {}".format(", ".join(non_matching_patterns)))
|
||||||
|
|
||||||
|
return resolved_units, non_matching_patterns
|
||||||
|
|
||||||
|
|
||||||
|
def process_unit(runner, unit, extra_properties, units_info, property_cache, state_props):
|
||||||
|
if not unit_exists(unit, units_info):
|
||||||
|
return units_info.get(unit, {"name": unit, "loadstate": "not-found"})
|
||||||
|
|
||||||
|
validate_unit_and_properties(runner, unit, extra_properties, units_info, property_cache)
|
||||||
|
category = determine_category(unit)
|
||||||
|
|
||||||
|
if not category:
|
||||||
|
module.fail_json(msg="Could not determine the category for unit '{0}'.".format(unit))
|
||||||
|
|
||||||
|
props = get_category_base_props(category)
|
||||||
|
full_props = set(props + state_props + extra_properties)
|
||||||
|
unit_data = property_cache[unit]
|
||||||
|
fact = {"name": unit}
|
||||||
|
minimal_keys = ["LoadState", "ActiveState", "SubState"]
|
||||||
|
fact.update(extract_unit_properties(unit_data, minimal_keys))
|
||||||
|
ls = unit_data.get("loadstate", "").lower()
|
||||||
|
|
||||||
|
if ls not in ("not-found", "masked"):
|
||||||
|
fact.update(extract_unit_properties(unit_data, full_props))
|
||||||
|
|
||||||
|
return fact
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
global module
|
||||||
|
module_args = dict(
|
||||||
|
unitname=dict(type='list', elements='str', default=[]),
|
||||||
|
extra_properties=dict(type='list', elements='str', default=[])
|
||||||
|
)
|
||||||
|
module = AnsibleModule(argument_spec=module_args, supports_check_mode=True)
|
||||||
|
systemctl_bin = module.get_bin_path('systemctl', required=True)
|
||||||
|
|
||||||
|
base_runner = systemd_runner(module, systemctl_bin)
|
||||||
|
|
||||||
|
get_version(base_runner)
|
||||||
|
|
||||||
|
state_props = ['LoadState', 'ActiveState', 'SubState']
|
||||||
results = {}
|
results = {}
|
||||||
|
|
||||||
if not module.params['unitname']:
|
unit_types = ["service", "target", "socket", "mount"]
|
||||||
list_cmd = [
|
|
||||||
systemctl_bin, "list-units",
|
|
||||||
"--no-pager",
|
|
||||||
"--type", "service,target,socket,mount",
|
|
||||||
"--all",
|
|
||||||
"--plain",
|
|
||||||
"--no-legend"
|
|
||||||
]
|
|
||||||
list_output = run_command(module, list_cmd)
|
|
||||||
for line in list_output.splitlines():
|
|
||||||
tokens = line.split()
|
|
||||||
if len(tokens) < 4:
|
|
||||||
continue
|
|
||||||
|
|
||||||
unit_name = tokens[0]
|
list_output = list_units(base_runner, unit_types)
|
||||||
loadstate = tokens[1]
|
units_info = {}
|
||||||
activestate = tokens[2]
|
for line in list_output.splitlines():
|
||||||
substate = tokens[3]
|
tokens = line.split()
|
||||||
|
if len(tokens) < 4:
|
||||||
|
continue
|
||||||
|
unit_name = tokens[0]
|
||||||
|
loadstate = tokens[1]
|
||||||
|
activestate = tokens[2]
|
||||||
|
substate = tokens[3]
|
||||||
|
units_info[unit_name] = {
|
||||||
|
"name": unit_name,
|
||||||
|
"loadstate": loadstate,
|
||||||
|
"activestate": activestate,
|
||||||
|
"substate": substate,
|
||||||
|
}
|
||||||
|
|
||||||
fact = {
|
property_cache = {}
|
||||||
"name": unit_name,
|
extra_properties = module.params['extra_properties']
|
||||||
"loadstate": loadstate,
|
|
||||||
"activestate": activestate,
|
|
||||||
"substate": substate
|
|
||||||
}
|
|
||||||
|
|
||||||
if loadstate in ("not-found", "masked"):
|
if module.params['unitname']:
|
||||||
results[unit_name] = fact
|
|
||||||
continue
|
|
||||||
|
|
||||||
category = determine_category(unit_name)
|
|
||||||
if not category:
|
|
||||||
results[unit_name] = fact
|
|
||||||
continue
|
|
||||||
|
|
||||||
props = base_properties.get(category, [])
|
|
||||||
full_props = set(props + state_props)
|
|
||||||
unit_data = get_unit_properties(module, systemctl_bin, unit_name, full_props)
|
|
||||||
|
|
||||||
fact.update(extract_unit_properties(unit_data, full_props))
|
|
||||||
results[unit_name] = fact
|
|
||||||
|
|
||||||
else:
|
|
||||||
selected_units = module.params['unitname']
|
selected_units = module.params['unitname']
|
||||||
extra_properties = module.params['extra_properties']
|
all_units = list(units_info)
|
||||||
|
resolved_units, non_matching = process_wildcards(selected_units, all_units, module)
|
||||||
for unit in selected_units:
|
units_to_process = sorted(resolved_units)
|
||||||
validate_unit_and_properties(module, systemctl_bin, unit, extra_properties)
|
else:
|
||||||
category = determine_category(unit)
|
units_to_process = list(units_info)
|
||||||
|
|
||||||
if not category:
|
|
||||||
module.fail_json(msg="Could not determine the category for unit '{0}'.".format(unit))
|
|
||||||
|
|
||||||
props = base_properties.get(category, [])
|
|
||||||
full_props = set(props + state_props + extra_properties)
|
|
||||||
unit_data = get_unit_properties(module, systemctl_bin, unit, full_props)
|
|
||||||
fact = {"name": unit}
|
|
||||||
minimal_keys = ["LoadState", "ActiveState", "SubState"]
|
|
||||||
|
|
||||||
fact.update(extract_unit_properties(unit_data, minimal_keys))
|
|
||||||
|
|
||||||
ls = unit_data.get("loadstate", "").lower()
|
|
||||||
if ls not in ("not-found", "masked"):
|
|
||||||
fact.update(extract_unit_properties(unit_data, full_props))
|
|
||||||
|
|
||||||
results[unit] = fact
|
|
||||||
|
|
||||||
|
for unit in units_to_process:
|
||||||
|
results[unit] = process_unit(base_runner, unit, extra_properties, units_info, property_cache, state_props)
|
||||||
module.exit_json(changed=False, units=results)
|
module.exit_json(changed=False, units=results)
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -105,3 +105,35 @@
|
||||||
- journal_extra.units['systemd-journald.service'].description == journald_shell.Description
|
- journal_extra.units['systemd-journald.service'].description == journald_shell.Description
|
||||||
- journal_extra.units['systemd-journald.service'].restart == journald_shell.Restart
|
- journal_extra.units['systemd-journald.service'].restart == journald_shell.Restart
|
||||||
success_msg: "Success: Extra property values are correct."
|
success_msg: "Success: Extra property values are correct."
|
||||||
|
|
||||||
|
- name: Gather info using wildcard pattern for services
|
||||||
|
community.general.systemd_info:
|
||||||
|
unitname:
|
||||||
|
- '*.service'
|
||||||
|
extra_properties:
|
||||||
|
- Description
|
||||||
|
register: result_wildcards
|
||||||
|
|
||||||
|
- name: Assert that at least one service unit was returned
|
||||||
|
ansible.builtin.assert:
|
||||||
|
that:
|
||||||
|
- result_wildcards.units | length > 0
|
||||||
|
|
||||||
|
- name: Gather info using multiple wildcard patterns
|
||||||
|
community.general.systemd_info:
|
||||||
|
unitname:
|
||||||
|
- '*.service'
|
||||||
|
- 'ssh*'
|
||||||
|
register: result_multi
|
||||||
|
|
||||||
|
- name: Debug multi-wildcard results
|
||||||
|
ansible.builtin.debug:
|
||||||
|
var: result_multi.units
|
||||||
|
|
||||||
|
- name: Assert deduplication of units
|
||||||
|
ansible.builtin.assert:
|
||||||
|
that:
|
||||||
|
- unique_keys | length == all_keys | length
|
||||||
|
vars:
|
||||||
|
all_keys: "{{ result_multi.units | dict2items | map(attribute='key') | list }}"
|
||||||
|
unique_keys: "{{ all_keys | unique }}"
|
Loading…
Add table
Reference in a new issue