mirror of
				https://github.com/ansible-collections/community.general.git
				synced 2025-10-26 05:50:36 -07:00 
			
		
		
		
	[PR #9764/8425464c backport][stable-10] Add new systemd_info module (#9800)
Add new systemd_info module (#9764)
* add systemd_info module
* fix object results
* apply review changes
* apply module change and add doc_fragments
* removed use_unsafe_shell and doc_fragments/systemd
* fix unitname description doc
* fixed doc, replaced systemctl show syntax, added base prop result doc
* fix documentation
* fix RV values in description
* fix RV() description values
* add get_bin_path try/fail and remove list()
* fix doc, removed try block
* add Archlinux in integration test
(cherry picked from commit 8425464c0a)
Co-authored-by: Nocchia <133043574+NomakCooper@users.noreply.github.com>
	
	
This commit is contained in:
		
					parent
					
						
							
								ceb051851e
							
						
					
				
			
			
				commit
				
					
						5735c5a045
					
				
			
		
					 5 changed files with 506 additions and 0 deletions
				
			
		
							
								
								
									
										2
									
								
								.github/BOTMETA.yml
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/BOTMETA.yml
									
										
									
									
										vendored
									
									
								
							|  | @ -1364,6 +1364,8 @@ files: | ||||||
|     maintainers: konstruktoid |     maintainers: konstruktoid | ||||||
|   $modules/systemd_creds_encrypt.py: |   $modules/systemd_creds_encrypt.py: | ||||||
|     maintainers: konstruktoid |     maintainers: konstruktoid | ||||||
|  |   $modules/systemd_info.py: | ||||||
|  |     maintainers: NomakCooper | ||||||
|   $modules/sysupgrade.py: |   $modules/sysupgrade.py: | ||||||
|     maintainers: precurse |     maintainers: precurse | ||||||
|   $modules/taiga_issue.py: |   $modules/taiga_issue.py: | ||||||
|  |  | ||||||
							
								
								
									
										361
									
								
								plugins/modules/systemd_info.py
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										361
									
								
								plugins/modules/systemd_info.py
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,361 @@ | ||||||
|  | #!/usr/bin/python | ||||||
|  | # -*- 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 | ||||||
|  | 
 | ||||||
|  | DOCUMENTATION = r''' | ||||||
|  | --- | ||||||
|  | module: systemd_info | ||||||
|  | short_description: Gather C(systemd) unit info | ||||||
|  | description: | ||||||
|  |   - This module gathers info about systemd units (services, targets, sockets, mount). | ||||||
|  |   - It runs C(systemctl list-units) (or processes selected units) and collects properties | ||||||
|  |     for each unit using C(systemctl show). | ||||||
|  |   - Even if a unit has a RV(units.loadstate) of V(not-found) or V(masked), it is returned, | ||||||
|  |     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, | ||||||
|  |     then check if properties exist. If not, the module fails. | ||||||
|  | version_added: "10.4.0" | ||||||
|  | options: | ||||||
|  |   unitname: | ||||||
|  |     description: | ||||||
|  |       - List of unit names to process. | ||||||
|  |       - 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. | ||||||
|  |     type: list | ||||||
|  |     elements: str | ||||||
|  |     default: [] | ||||||
|  |   extra_properties: | ||||||
|  |     description: | ||||||
|  |       - Additional properties to retrieve (appended to the default ones). | ||||||
|  |       - Note that all property names are converted to lower-case. | ||||||
|  |     type: list | ||||||
|  |     elements: str | ||||||
|  |     default: [] | ||||||
|  | author: | ||||||
|  |   - Marco Noce (@NomakCooper) | ||||||
|  | extends_documentation_fragment: | ||||||
|  |   - community.general.attributes | ||||||
|  |   - community.general.attributes.info_module | ||||||
|  | ''' | ||||||
|  | 
 | ||||||
|  | EXAMPLES = r''' | ||||||
|  | --- | ||||||
|  | # Gather info for all systemd services, targets, sockets and mount | ||||||
|  | - name: Gather all systemd unit info | ||||||
|  |   community.general.systemd_info: | ||||||
|  |   register: results | ||||||
|  | 
 | ||||||
|  | # Gather info for selected units with extra properties. | ||||||
|  | - name: Gather info for selected unit(s) | ||||||
|  |   community.general.systemd_info: | ||||||
|  |     unitname: | ||||||
|  |       - systemd-journald.service | ||||||
|  |       - systemd-journald.socket | ||||||
|  |       - sshd-keygen.target | ||||||
|  |       - -.mount | ||||||
|  |     extra_properties: | ||||||
|  |       - Description | ||||||
|  |     register: results | ||||||
|  | ''' | ||||||
|  | 
 | ||||||
|  | RETURN = r''' | ||||||
|  | units: | ||||||
|  |   description: | ||||||
|  |     - Dictionary of systemd unit info keyed by unit name. | ||||||
|  |     - Additional fields will be returned depending on the value of O(extra_properties). | ||||||
|  |   returned: success | ||||||
|  |   type: dict | ||||||
|  |   elements: dict | ||||||
|  |   contains: | ||||||
|  |     name: | ||||||
|  |       description: Unit full name. | ||||||
|  |       returned: always | ||||||
|  |       type: str | ||||||
|  |       sample: systemd-journald.service | ||||||
|  |     loadstate: | ||||||
|  |       description: | ||||||
|  |         - The state of the unit's configuration load. | ||||||
|  |         - The most common values are V(loaded), V(not-found), and V(masked), but other values are possible as well. | ||||||
|  |       returned: always | ||||||
|  |       type: str | ||||||
|  |       sample: loaded | ||||||
|  |     activestate: | ||||||
|  |       description: | ||||||
|  |         - The current active state of the unit. | ||||||
|  |         - The most common values are V(active), V(inactive), and V(failed), but other values are possible as well. | ||||||
|  |       returned: always | ||||||
|  |       type: str | ||||||
|  |       sample: active | ||||||
|  |     substate: | ||||||
|  |       description: | ||||||
|  |         - The detailed sub state of the unit. | ||||||
|  |         - The most common values are V(running), V(dead), V(exited), V(failed), V(listening), V(active), and V(mounted), but other values are possible as well. | ||||||
|  |       returned: always | ||||||
|  |       type: str | ||||||
|  |       sample: running | ||||||
|  |     fragmentpath: | ||||||
|  |       description: Path to the unit's fragment file. | ||||||
|  |       returned: always except for C(.mount) units. | ||||||
|  |       type: str | ||||||
|  |       sample: /usr/lib/systemd/system/systemd-journald.service | ||||||
|  |     unitfilepreset: | ||||||
|  |       description: | ||||||
|  |         - The preset configuration state for the unit file. | ||||||
|  |         - The most common values are V(enabled), V(disabled), and V(static), but other values are possible as well. | ||||||
|  |       returned: always except for C(.mount) units. | ||||||
|  |       type: str | ||||||
|  |       sample: disabled | ||||||
|  |     unitfilestate: | ||||||
|  |       description: | ||||||
|  |         - The actual configuration state for the unit file. | ||||||
|  |         - The most common values are V(enabled), V(disabled), and V(static), but other values are possible as well. | ||||||
|  |       returned: always except for C(.mount) units. | ||||||
|  |       type: str | ||||||
|  |       sample: enabled | ||||||
|  |     mainpid: | ||||||
|  |       description: PID of the main process of the unit. | ||||||
|  |       returned: only for C(.service) units. | ||||||
|  |       type: str | ||||||
|  |       sample: 798 | ||||||
|  |     execmainpid: | ||||||
|  |       description: PID of the ExecStart process of the unit. | ||||||
|  |       returned: only for C(.service) units. | ||||||
|  |       type: str | ||||||
|  |       sample: 799 | ||||||
|  |     options: | ||||||
|  |       description: The mount options. | ||||||
|  |       returned: only for C(.mount) units. | ||||||
|  |       type: str | ||||||
|  |       sample: rw,relatime,noquota | ||||||
|  |     type: | ||||||
|  |       description: The filesystem type of the mounted device. | ||||||
|  |       returned: only for C(.mount) units. | ||||||
|  |       type: str | ||||||
|  |       sample: ext4 | ||||||
|  |     what: | ||||||
|  |       description: The device that is mounted. | ||||||
|  |       returned: only for C(.mount) units. | ||||||
|  |       type: str | ||||||
|  |       sample: /dev/sda1 | ||||||
|  |     where: | ||||||
|  |       description: The mount point where the device is mounted. | ||||||
|  |       returned: only for C(.mount) units. | ||||||
|  |       type: str | ||||||
|  |       sample: / | ||||||
|  |   sample: { | ||||||
|  |     "-.mount": { | ||||||
|  |         "activestate": "active", | ||||||
|  |         "description": "Root Mount", | ||||||
|  |         "loadstate": "loaded", | ||||||
|  |         "name": "-.mount", | ||||||
|  |         "options": "rw,relatime,seclabel,attr2,inode64,logbufs=8,logbsize=32k,noquota", | ||||||
|  |         "substate": "mounted", | ||||||
|  |         "type": "xfs", | ||||||
|  |         "what": "/dev/mapper/cs-root", | ||||||
|  |         "where": "/" | ||||||
|  |     }, | ||||||
|  |     "sshd-keygen.target": { | ||||||
|  |         "activestate": "active", | ||||||
|  |         "description": "sshd-keygen.target", | ||||||
|  |         "fragmentpath": "/usr/lib/systemd/system/sshd-keygen.target", | ||||||
|  |         "loadstate": "loaded", | ||||||
|  |         "name": "sshd-keygen.target", | ||||||
|  |         "substate": "active", | ||||||
|  |         "unitfilepreset": "disabled", | ||||||
|  |         "unitfilestate": "static" | ||||||
|  |     }, | ||||||
|  |     "systemd-journald.service": { | ||||||
|  |         "activestate": "active", | ||||||
|  |         "description": "Journal Service", | ||||||
|  |         "execmainpid": "613", | ||||||
|  |         "fragmentpath": "/usr/lib/systemd/system/systemd-journald.service", | ||||||
|  |         "loadstate": "loaded", | ||||||
|  |         "mainpid": "613", | ||||||
|  |         "name": "systemd-journald.service", | ||||||
|  |         "substate": "running", | ||||||
|  |         "unitfilepreset": "disabled", | ||||||
|  |         "unitfilestate": "static" | ||||||
|  |     }, | ||||||
|  |     "systemd-journald.socket": { | ||||||
|  |         "activestate": "active", | ||||||
|  |         "description": "Journal Socket", | ||||||
|  |         "fragmentpath": "/usr/lib/systemd/system/systemd-journald.socket", | ||||||
|  |         "loadstate": "loaded", | ||||||
|  |         "name": "systemd-journald.socket", | ||||||
|  |         "substate": "running", | ||||||
|  |         "unitfilepreset": "disabled", | ||||||
|  |         "unitfilestate": "static" | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | ''' | ||||||
|  | 
 | ||||||
|  | from ansible.module_utils.basic import AnsibleModule | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def run_command(module, cmd): | ||||||
|  |     rc, stdout, stderr = module.run_command(cmd, check_rc=True) | ||||||
|  |     return stdout.strip() | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def parse_show_output(output): | ||||||
|  |     result = {} | ||||||
|  |     for line in output.splitlines(): | ||||||
|  |         if "=" in line: | ||||||
|  |             key, val = line.split("=", 1) | ||||||
|  |             key = key.lower() | ||||||
|  |             if key not in result: | ||||||
|  |                 result[key] = val | ||||||
|  |     return result | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def get_unit_properties(module, systemctl_bin, unit, prop_list): | ||||||
|  |     cmd = [systemctl_bin, "show", "-p", ",".join(prop_list), "--", unit] | ||||||
|  |     output = run_command(module, cmd) | ||||||
|  |     return parse_show_output(output) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def determine_category(unit): | ||||||
|  |     if unit.endswith('.service'): | ||||||
|  |         return 'service' | ||||||
|  |     elif unit.endswith('.target'): | ||||||
|  |         return 'target' | ||||||
|  |     elif unit.endswith('.socket'): | ||||||
|  |         return 'socket' | ||||||
|  |     elif unit.endswith('.mount'): | ||||||
|  |         return 'mount' | ||||||
|  |     else: | ||||||
|  |         return None | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def extract_unit_properties(unit_data, prop_list): | ||||||
|  |     lowerprop = [x.lower() for x in prop_list] | ||||||
|  |     extracted = { | ||||||
|  |         prop: unit_data[prop] for prop in lowerprop if prop in unit_data | ||||||
|  |     } | ||||||
|  |     return extracted | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def unit_exists(module, systemctl_bin, unit): | ||||||
|  |     cmd = [systemctl_bin, "show", "-p", "LoadState", "--", unit] | ||||||
|  |     rc, stdout, stderr = module.run_command(cmd) | ||||||
|  |     return (rc == 0) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def validate_unit_and_properties(module, systemctl_bin, unit, extra_properties): | ||||||
|  |     cmd = [systemctl_bin, "show", "-p", "LoadState", "--", unit] | ||||||
|  | 
 | ||||||
|  |     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'], | ||||||
|  |         'target': ['FragmentPath', 'UnitFileState', 'UnitFilePreset'], | ||||||
|  |         'socket': ['FragmentPath', 'UnitFileState', 'UnitFilePreset'], | ||||||
|  |         'mount': ['Where', 'What', 'Options', 'Type'] | ||||||
|  |     } | ||||||
|  |     state_props = ['LoadState', 'ActiveState', 'SubState'] | ||||||
|  | 
 | ||||||
|  |     results = {} | ||||||
|  | 
 | ||||||
|  |     if not module.params['unitname']: | ||||||
|  |         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] | ||||||
|  |             loadstate = tokens[1] | ||||||
|  |             activestate = tokens[2] | ||||||
|  |             substate = tokens[3] | ||||||
|  | 
 | ||||||
|  |             fact = { | ||||||
|  |                 "name": unit_name, | ||||||
|  |                 "loadstate": loadstate, | ||||||
|  |                 "activestate": activestate, | ||||||
|  |                 "substate": substate | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if loadstate in ("not-found", "masked"): | ||||||
|  |                 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'] | ||||||
|  |         extra_properties = module.params['extra_properties'] | ||||||
|  | 
 | ||||||
|  |         for unit in selected_units: | ||||||
|  |             validate_unit_and_properties(module, systemctl_bin, unit, extra_properties) | ||||||
|  |             category = determine_category(unit) | ||||||
|  | 
 | ||||||
|  |             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 | ||||||
|  | 
 | ||||||
|  |     module.exit_json(changed=False, units=results) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | if __name__ == '__main__': | ||||||
|  |     main() | ||||||
							
								
								
									
										10
									
								
								tests/integration/targets/systemd_info/aliases
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								tests/integration/targets/systemd_info/aliases
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,10 @@ | ||||||
|  | # 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 | ||||||
|  | 
 | ||||||
|  | needs/root | ||||||
|  | azp/posix/1 | ||||||
|  | skip/aix | ||||||
|  | skip/freebsd | ||||||
|  | skip/osx | ||||||
|  | skip/macos | ||||||
							
								
								
									
										26
									
								
								tests/integration/targets/systemd_info/tasks/main.yml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								tests/integration/targets/systemd_info/tasks/main.yml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,26 @@ | ||||||
|  | --- | ||||||
|  | # 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 | ||||||
|  | 
 | ||||||
|  | - name: skip Alpine | ||||||
|  |   meta: end_host | ||||||
|  |   when: ansible_distribution == 'Alpine' | ||||||
|  | 
 | ||||||
|  | - name: check ansible_service_mgr | ||||||
|  |   ansible.builtin.assert: | ||||||
|  |     that: ansible_service_mgr == 'systemd' | ||||||
|  | 
 | ||||||
|  | - name: Test systemd_facts | ||||||
|  |   block: | ||||||
|  | 
 | ||||||
|  |   - name: Run tests | ||||||
|  |     import_tasks: tests.yml | ||||||
|  | 
 | ||||||
|  |   when: > | ||||||
|  |     (ansible_distribution in ['RedHat', 'CentOS', 'ScientificLinux'] and ansible_distribution_major_version is version('7', '>=')) or  | ||||||
|  |     ansible_distribution == 'Fedora' or  | ||||||
|  |     (ansible_distribution == 'Ubuntu' and ansible_distribution_version is version('15.04', '>=')) or  | ||||||
|  |     (ansible_distribution == 'Debian' and ansible_distribution_version is version('8', '>=')) or  | ||||||
|  |     ansible_os_family == 'Suse' or | ||||||
|  |     ansible_distribution == 'Archlinux' | ||||||
							
								
								
									
										107
									
								
								tests/integration/targets/systemd_info/tasks/tests.yml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										107
									
								
								tests/integration/targets/systemd_info/tasks/tests.yml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,107 @@ | ||||||
|  | # 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 | ||||||
|  | 
 | ||||||
|  | - name: Gather all units from shell | ||||||
|  |   ansible.builtin.command: systemctl list-units --no-pager --type service,target,socket,mount --all --plain --no-legend | ||||||
|  |   register: all_units | ||||||
|  | 
 | ||||||
|  | - name: Assert command run successfully | ||||||
|  |   ansible.builtin.assert: | ||||||
|  |     that: | ||||||
|  |       - all_units.rc == 0 | ||||||
|  | 
 | ||||||
|  | - name: Gather all units | ||||||
|  |   community.general.systemd_info: | ||||||
|  |   register: units_all | ||||||
|  | 
 | ||||||
|  | - name: Check all units exists | ||||||
|  |   ansible.builtin.assert: | ||||||
|  |     that: | ||||||
|  |       - units_all is defined | ||||||
|  |       - units_all.units | length == all_units.stdout_lines | length | ||||||
|  |     success_msg: "Success: All units collected." | ||||||
|  | 
 | ||||||
|  | - name: Build all units list | ||||||
|  |   set_fact: | ||||||
|  |     shell_units: "{{ all_units.stdout_lines | map('split') | list }}" | ||||||
|  | 
 | ||||||
|  | - name: Check all units properties | ||||||
|  |   ansible.builtin.assert: | ||||||
|  |     that: | ||||||
|  |       - units_all.units[item[0]].name == item[0] | ||||||
|  |       - units_all.units[item[0]].loadstate == item[1] | ||||||
|  |       - units_all.units[item[0]].activestate == item[2] | ||||||
|  |       - units_all.units[item[0]].substate == item[3] | ||||||
|  |   loop: "{{ shell_units }}" | ||||||
|  |   loop_control: | ||||||
|  |     label: "{{ item[0] }}" | ||||||
|  | 
 | ||||||
|  | - name: Gather systemd-journald.service properties from shell | ||||||
|  |   ansible.builtin.command: systemctl show systemd-journald.service -p Id,LoadState,ActiveState,SubState,FragmentPath,MainPID,ExecMainPID,UnitFileState,UnitFilePreset,Description,Restart | ||||||
|  |   register: journald_prop | ||||||
|  | 
 | ||||||
|  | - name: Assert command run successfully | ||||||
|  |   ansible.builtin.assert: | ||||||
|  |     that: | ||||||
|  |       - journald_prop.rc == 0 | ||||||
|  | 
 | ||||||
|  | - name: Gather systemd-journald.service | ||||||
|  |   community.general.systemd_info: | ||||||
|  |     unitname: | ||||||
|  |       - systemd-journald.service | ||||||
|  |   register: journal_unit | ||||||
|  | 
 | ||||||
|  | - name: Check unit facts and all properties | ||||||
|  |   ansible.builtin.assert: | ||||||
|  |     that: | ||||||
|  |       - journal_unit.units is defined | ||||||
|  |       - journal_unit.units['systemd-journald.service'] is defined | ||||||
|  |       - journal_unit.units['systemd-journald.service'].name is defined | ||||||
|  |       - journal_unit.units['systemd-journald.service'].loadstate is defined | ||||||
|  |       - journal_unit.units['systemd-journald.service'].activestate is defined | ||||||
|  |       - journal_unit.units['systemd-journald.service'].substate is defined | ||||||
|  |       - journal_unit.units['systemd-journald.service'].fragmentpath is defined | ||||||
|  |       - journal_unit.units['systemd-journald.service'].mainpid is defined | ||||||
|  |       - journal_unit.units['systemd-journald.service'].execmainpid is defined | ||||||
|  |       - journal_unit.units['systemd-journald.service'].unitfilestate is defined | ||||||
|  |       - journal_unit.units['systemd-journald.service'].unitfilepreset is defined | ||||||
|  |     success_msg: "Success: All properties collected." | ||||||
|  | 
 | ||||||
|  | - name: Create dict of properties from shell | ||||||
|  |   ansible.builtin.set_fact: | ||||||
|  |     journald_shell: "{{ dict(journald_prop.stdout_lines | map('split', '=', 1) | list) }}" | ||||||
|  | 
 | ||||||
|  | - name: Check properties content | ||||||
|  |   ansible.builtin.assert: | ||||||
|  |     that: | ||||||
|  |       - journal_unit.units['systemd-journald.service'].name == journald_shell.Id | ||||||
|  |       - journal_unit.units['systemd-journald.service'].loadstate == journald_shell.LoadState | ||||||
|  |       - journal_unit.units['systemd-journald.service'].activestate == journald_shell.ActiveState | ||||||
|  |       - journal_unit.units['systemd-journald.service'].substate == journald_shell.SubState | ||||||
|  |       - journal_unit.units['systemd-journald.service'].fragmentpath == journald_shell.FragmentPath | ||||||
|  |       - journal_unit.units['systemd-journald.service'].mainpid == journald_shell.MainPID | ||||||
|  |       - journal_unit.units['systemd-journald.service'].execmainpid == journald_shell.ExecMainPID | ||||||
|  |       - journal_unit.units['systemd-journald.service'].unitfilestate == journald_shell.UnitFileState | ||||||
|  |       - journal_unit.units['systemd-journald.service'].unitfilepreset == journald_shell.UnitFilePreset | ||||||
|  |     success_msg: "Success: Property values are correct." | ||||||
|  | 
 | ||||||
|  | - name: Gather systemd-journald.service extra properties | ||||||
|  |   community.general.systemd_info: | ||||||
|  |     unitname: | ||||||
|  |       - systemd-journald.service | ||||||
|  |     extra_properties: | ||||||
|  |       - Description | ||||||
|  |       - Restart | ||||||
|  |   register: journal_extra | ||||||
|  | 
 | ||||||
|  | - name: Check new properties | ||||||
|  |   ansible.builtin.assert: | ||||||
|  |     that: | ||||||
|  |       - journal_extra.units is defined | ||||||
|  |       - journal_extra.units['systemd-journald.service'] is defined | ||||||
|  |       - journal_extra.units['systemd-journald.service'].description is defined | ||||||
|  |       - journal_extra.units['systemd-journald.service'].restart is defined | ||||||
|  |       - journal_extra.units['systemd-journald.service'].description == journald_shell.Description | ||||||
|  |       - journal_extra.units['systemd-journald.service'].restart == journald_shell.Restart | ||||||
|  |     success_msg: "Success: Extra property values are correct." | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue