mirror of
https://github.com/ansible-collections/community.general.git
synced 2025-04-06 10:40:32 -07:00
launchd: new module to control services on macOS hosts (#305)
This commit is contained in:
parent
669b7bf090
commit
80cd8329e0
21 changed files with 947 additions and 0 deletions
1
plugins/modules/launchd.py
Symbolic link
1
plugins/modules/launchd.py
Symbolic link
|
@ -0,0 +1 @@
|
||||||
|
./system/launchd.py
|
480
plugins/modules/system/launchd.py
Normal file
480
plugins/modules/system/launchd.py
Normal file
|
@ -0,0 +1,480 @@
|
||||||
|
#!/usr/bin/python
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
|
# Copyright: (c) 2018, Martin Migasiewicz <migasiew.nk@gmail.com>
|
||||||
|
# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
|
||||||
|
|
||||||
|
from __future__ import absolute_import, division, print_function
|
||||||
|
__metaclass__ = type
|
||||||
|
|
||||||
|
DOCUMENTATION = r'''
|
||||||
|
---
|
||||||
|
module: launchd
|
||||||
|
author:
|
||||||
|
- Martin Migasiewicz (@martinm82)
|
||||||
|
short_description: Manage macOS services
|
||||||
|
version_added: 1.0.0
|
||||||
|
description:
|
||||||
|
- Manage launchd services on target macOS hosts.
|
||||||
|
options:
|
||||||
|
name:
|
||||||
|
description:
|
||||||
|
- Name of the service.
|
||||||
|
type: str
|
||||||
|
required: true
|
||||||
|
state:
|
||||||
|
description:
|
||||||
|
- C(started)/C(stopped) are idempotent actions that will not run
|
||||||
|
commands unless necessary.
|
||||||
|
- Launchd does not support C(restarted) nor C(reloaded) natively.
|
||||||
|
These will trigger a stop/start (restarted) or an unload/load
|
||||||
|
(reloaded).
|
||||||
|
- C(restarted) unloads and loads the service before start to ensure
|
||||||
|
that the latest job definition (plist) is used.
|
||||||
|
- C(reloaded) unloads and loads the service to ensure that the latest
|
||||||
|
job definition (plist) is used. Whether a service is started or
|
||||||
|
stopped depends on the content of the definition file.
|
||||||
|
type: str
|
||||||
|
choices: [ reloaded, restarted, started, stopped, unloaded ]
|
||||||
|
enabled:
|
||||||
|
description:
|
||||||
|
- Whether the service should start on boot.
|
||||||
|
- B(At least one of state and enabled are required.)
|
||||||
|
type: bool
|
||||||
|
force_stop:
|
||||||
|
description:
|
||||||
|
- Whether the service should not be restarted automatically by launchd.
|
||||||
|
- Services might have the 'KeepAlive' attribute set to true in a launchd configuration.
|
||||||
|
In case this is set to true, stopping a service will cause that launchd starts the service again.
|
||||||
|
- Set this option to C(yes) to let this module change the 'KeepAlive' attribute to false.
|
||||||
|
type: bool
|
||||||
|
default: no
|
||||||
|
notes:
|
||||||
|
- A user must privileged to manage services using this module.
|
||||||
|
requirements:
|
||||||
|
- A system managed by launchd
|
||||||
|
- The plistlib python library
|
||||||
|
'''
|
||||||
|
|
||||||
|
EXAMPLES = r'''
|
||||||
|
- name: Make sure spotify webhelper is started
|
||||||
|
community.general.launchd:
|
||||||
|
name: com.spotify.webhelper
|
||||||
|
state: started
|
||||||
|
|
||||||
|
- name: Deploy custom memcached job definition
|
||||||
|
template:
|
||||||
|
src: org.memcached.plist.j2
|
||||||
|
dest: /Library/LaunchDaemons/org.memcached.plist
|
||||||
|
|
||||||
|
- name: Run memcached
|
||||||
|
community.general.launchd:
|
||||||
|
name: org.memcached
|
||||||
|
state: started
|
||||||
|
|
||||||
|
- name: Stop memcached
|
||||||
|
community.general.launchd:
|
||||||
|
name: org.memcached
|
||||||
|
state: stopped
|
||||||
|
|
||||||
|
- name: Stop memcached
|
||||||
|
community.general.launchd:
|
||||||
|
name: org.memcached
|
||||||
|
state: stopped
|
||||||
|
force_stop: yes
|
||||||
|
|
||||||
|
- name: Restart memcached
|
||||||
|
community.general.launchd:
|
||||||
|
name: org.memcached
|
||||||
|
state: restarted
|
||||||
|
|
||||||
|
- name: Unload memcached
|
||||||
|
community.general.launchd:
|
||||||
|
name: org.memcached
|
||||||
|
state: unloaded
|
||||||
|
'''
|
||||||
|
|
||||||
|
RETURN = r'''
|
||||||
|
status:
|
||||||
|
description: Metadata about service status
|
||||||
|
returned: always
|
||||||
|
type: dict
|
||||||
|
sample:
|
||||||
|
{
|
||||||
|
"current_pid": "-",
|
||||||
|
"current_state": "stopped",
|
||||||
|
"previous_pid": "82636",
|
||||||
|
"previous_state": "running"
|
||||||
|
}
|
||||||
|
'''
|
||||||
|
|
||||||
|
import os
|
||||||
|
import plistlib
|
||||||
|
from abc import ABCMeta, abstractmethod
|
||||||
|
from time import sleep
|
||||||
|
|
||||||
|
from ansible.module_utils.basic import AnsibleModule
|
||||||
|
|
||||||
|
|
||||||
|
class ServiceState:
|
||||||
|
UNKNOWN = 0
|
||||||
|
LOADED = 1
|
||||||
|
STOPPED = 2
|
||||||
|
STARTED = 3
|
||||||
|
UNLOADED = 4
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def to_string(state):
|
||||||
|
strings = {
|
||||||
|
ServiceState.UNKNOWN: 'unknown',
|
||||||
|
ServiceState.LOADED: 'loaded',
|
||||||
|
ServiceState.STOPPED: 'stopped',
|
||||||
|
ServiceState.STARTED: 'started',
|
||||||
|
ServiceState.UNLOADED: 'unloaded'
|
||||||
|
}
|
||||||
|
return strings[state]
|
||||||
|
|
||||||
|
|
||||||
|
class Plist:
|
||||||
|
def __init__(self, module, service):
|
||||||
|
self.__changed = False
|
||||||
|
self.__service = service
|
||||||
|
|
||||||
|
state, pid, dummy, dummy = LaunchCtlList(module, service).run()
|
||||||
|
|
||||||
|
self.__file = self.__find_service_plist(service)
|
||||||
|
if self.__file is None:
|
||||||
|
msg = 'Unable to infer the path of %s service plist file' % service
|
||||||
|
if pid is None and state == ServiceState.UNLOADED:
|
||||||
|
msg += ' and it was not found among active services'
|
||||||
|
module.fail_json(msg=msg)
|
||||||
|
self.__update(module)
|
||||||
|
|
||||||
|
def __find_service_plist(self, service_name):
|
||||||
|
"""Finds the plist file associated with a service"""
|
||||||
|
|
||||||
|
launchd_paths = [
|
||||||
|
'~/Library/LaunchAgents',
|
||||||
|
'/Library/LaunchAgents',
|
||||||
|
'/Library/LaunchDaemons',
|
||||||
|
'/System/Library/LaunchAgents',
|
||||||
|
'/System/Library/LaunchDaemons'
|
||||||
|
]
|
||||||
|
|
||||||
|
for path in launchd_paths:
|
||||||
|
try:
|
||||||
|
files = os.listdir(os.path.expanduser(path))
|
||||||
|
except OSError:
|
||||||
|
continue
|
||||||
|
|
||||||
|
filename = '%s.plist' % service_name
|
||||||
|
if filename in files:
|
||||||
|
return os.path.join(path, filename)
|
||||||
|
return None
|
||||||
|
|
||||||
|
def __update(self, module):
|
||||||
|
self.__handle_param_enabled(module)
|
||||||
|
self.__handle_param_force_stop(module)
|
||||||
|
|
||||||
|
def __handle_param_enabled(self, module):
|
||||||
|
if module.params['enabled'] is not None:
|
||||||
|
service_plist = plistlib.readPlist(self.__file)
|
||||||
|
|
||||||
|
# Enable/disable service startup at boot if requested
|
||||||
|
# Launchctl does not expose functionality to set the RunAtLoad
|
||||||
|
# attribute of a job definition. So we parse and modify the job
|
||||||
|
# definition plist file directly for this purpose.
|
||||||
|
if module.params['enabled'] is not None:
|
||||||
|
enabled = service_plist.get('RunAtLoad', False)
|
||||||
|
if module.params['enabled'] != enabled:
|
||||||
|
service_plist['RunAtLoad'] = module.params['enabled']
|
||||||
|
|
||||||
|
# Update the plist with one of the changes done.
|
||||||
|
if not module.check_mode:
|
||||||
|
plistlib.writePlist(service_plist, self.__file)
|
||||||
|
self.__changed = True
|
||||||
|
|
||||||
|
def __handle_param_force_stop(self, module):
|
||||||
|
if module.params['force_stop'] is not None:
|
||||||
|
service_plist = plistlib.readPlist(self.__file)
|
||||||
|
|
||||||
|
# Set KeepAlive to false in case force_stop is defined to avoid
|
||||||
|
# that the service gets restarted when stopping was requested.
|
||||||
|
if module.params['force_stop'] is not None:
|
||||||
|
keep_alive = service_plist.get('KeepAlive', False)
|
||||||
|
if module.params['force_stop'] and keep_alive:
|
||||||
|
service_plist['KeepAlive'] = not module.params['force_stop']
|
||||||
|
|
||||||
|
# Update the plist with one of the changes done.
|
||||||
|
if not module.check_mode:
|
||||||
|
plistlib.writePlist(service_plist, self.__file)
|
||||||
|
self.__changed = True
|
||||||
|
|
||||||
|
def is_changed(self):
|
||||||
|
return self.__changed
|
||||||
|
|
||||||
|
def get_file(self):
|
||||||
|
return self.__file
|
||||||
|
|
||||||
|
|
||||||
|
class LaunchCtlTask(object):
|
||||||
|
__metaclass__ = ABCMeta
|
||||||
|
WAITING_TIME = 5 # seconds
|
||||||
|
|
||||||
|
def __init__(self, module, service, plist):
|
||||||
|
self._module = module
|
||||||
|
self._service = service
|
||||||
|
self._plist = plist
|
||||||
|
self._launch = self._module.get_bin_path('launchctl', True)
|
||||||
|
|
||||||
|
def run(self):
|
||||||
|
"""Runs a launchd command like 'load', 'unload', 'start', 'stop', etc.
|
||||||
|
and returns the new state and pid.
|
||||||
|
"""
|
||||||
|
self.runCommand()
|
||||||
|
return self.get_state()
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def runCommand(self):
|
||||||
|
pass
|
||||||
|
|
||||||
|
def get_state(self):
|
||||||
|
rc, out, err = self._launchctl("list")
|
||||||
|
if rc != 0:
|
||||||
|
self._module.fail_json(
|
||||||
|
msg='Failed to get status of %s' % (self._launch))
|
||||||
|
|
||||||
|
state = ServiceState.UNLOADED
|
||||||
|
service_pid = "-"
|
||||||
|
status_code = None
|
||||||
|
for line in out.splitlines():
|
||||||
|
if line.strip():
|
||||||
|
pid, last_exit_code, label = line.split('\t')
|
||||||
|
if label.strip() == self._service:
|
||||||
|
service_pid = pid
|
||||||
|
status_code = last_exit_code
|
||||||
|
|
||||||
|
# From launchctl man page:
|
||||||
|
# If the number [...] is negative, it represents the
|
||||||
|
# negative of the signal which killed the job. Thus,
|
||||||
|
# "-15" would indicate that the job was terminated with
|
||||||
|
# SIGTERM.
|
||||||
|
if last_exit_code not in ['0', '-2', '-3', '-9', '-15']:
|
||||||
|
# Something strange happened and we have no clue in
|
||||||
|
# which state the service is now. Therefore we mark
|
||||||
|
# the service state as UNKNOWN.
|
||||||
|
state = ServiceState.UNKNOWN
|
||||||
|
elif pid != '-':
|
||||||
|
# PID seems to be an integer so we assume the service
|
||||||
|
# is started.
|
||||||
|
state = ServiceState.STARTED
|
||||||
|
else:
|
||||||
|
# Exit code is 0 and PID is not available so we assume
|
||||||
|
# the service is stopped.
|
||||||
|
state = ServiceState.STOPPED
|
||||||
|
break
|
||||||
|
return (state, service_pid, status_code, err)
|
||||||
|
|
||||||
|
def start(self):
|
||||||
|
rc, out, err = self._launchctl("start")
|
||||||
|
# Unfortunately launchd does not wait until the process really started.
|
||||||
|
sleep(self.WAITING_TIME)
|
||||||
|
return (rc, out, err)
|
||||||
|
|
||||||
|
def stop(self):
|
||||||
|
rc, out, err = self._launchctl("stop")
|
||||||
|
# Unfortunately launchd does not wait until the process really stopped.
|
||||||
|
sleep(self.WAITING_TIME)
|
||||||
|
return (rc, out, err)
|
||||||
|
|
||||||
|
def restart(self):
|
||||||
|
# TODO: check for rc, out, err
|
||||||
|
self.stop()
|
||||||
|
return self.start()
|
||||||
|
|
||||||
|
def reload(self):
|
||||||
|
# TODO: check for rc, out, err
|
||||||
|
self.unload()
|
||||||
|
return self.load()
|
||||||
|
|
||||||
|
def load(self):
|
||||||
|
return self._launchctl("load")
|
||||||
|
|
||||||
|
def unload(self):
|
||||||
|
return self._launchctl("unload")
|
||||||
|
|
||||||
|
def _launchctl(self, command):
|
||||||
|
service_or_plist = self._plist.get_file() if command in [
|
||||||
|
'load', 'unload'] else self._service if command in ['start', 'stop'] else ""
|
||||||
|
|
||||||
|
rc, out, err = self._module.run_command(
|
||||||
|
'%s %s %s' % (self._launch, command, service_or_plist))
|
||||||
|
|
||||||
|
if rc != 0:
|
||||||
|
msg = "Unable to %s '%s' (%s): '%s'" % (
|
||||||
|
command, self._service, self._plist.get_file(), err)
|
||||||
|
self._module.fail_json(msg=msg)
|
||||||
|
|
||||||
|
return (rc, out, err)
|
||||||
|
|
||||||
|
|
||||||
|
class LaunchCtlStart(LaunchCtlTask):
|
||||||
|
def __init__(self, module, service, plist):
|
||||||
|
super(LaunchCtlStart, self).__init__(module, service, plist)
|
||||||
|
|
||||||
|
def runCommand(self):
|
||||||
|
state, dummy, dummy, dummy = self.get_state()
|
||||||
|
|
||||||
|
if state == ServiceState.STOPPED or state == ServiceState.LOADED:
|
||||||
|
self.reload()
|
||||||
|
self.start()
|
||||||
|
elif state == ServiceState.STARTED:
|
||||||
|
# In case the service is already in started state but the
|
||||||
|
# job definition was changed we need to unload/load the
|
||||||
|
# service and start the service again.
|
||||||
|
if self._plist.is_changed():
|
||||||
|
self.reload()
|
||||||
|
self.start()
|
||||||
|
elif state == ServiceState.UNLOADED:
|
||||||
|
self.load()
|
||||||
|
self.start()
|
||||||
|
elif state == ServiceState.UNKNOWN:
|
||||||
|
# We are in an unknown state, let's try to reload the config
|
||||||
|
# and start the service again.
|
||||||
|
self.reload()
|
||||||
|
self.start()
|
||||||
|
|
||||||
|
|
||||||
|
class LaunchCtlStop(LaunchCtlTask):
|
||||||
|
def __init__(self, module, service, plist):
|
||||||
|
super(LaunchCtlStop, self).__init__(module, service, plist)
|
||||||
|
|
||||||
|
def runCommand(self):
|
||||||
|
state, dummy, dummy, dummy = self.get_state()
|
||||||
|
|
||||||
|
if state == ServiceState.STOPPED:
|
||||||
|
# In case the service is stopped and we might later decide
|
||||||
|
# to start it, we need to reload the job definition by
|
||||||
|
# forcing an unload and load first.
|
||||||
|
# Afterwards we need to stop it as it might have been
|
||||||
|
# started again (KeepAlive or RunAtLoad).
|
||||||
|
if self._plist.is_changed():
|
||||||
|
self.reload()
|
||||||
|
self.stop()
|
||||||
|
elif state == ServiceState.STARTED or state == ServiceState.LOADED:
|
||||||
|
if self._plist.is_changed():
|
||||||
|
self.reload()
|
||||||
|
self.stop()
|
||||||
|
elif state == ServiceState.UNKNOWN:
|
||||||
|
# We are in an unknown state, let's try to reload the config
|
||||||
|
# and stop the service gracefully.
|
||||||
|
self.reload()
|
||||||
|
self.stop()
|
||||||
|
|
||||||
|
|
||||||
|
class LaunchCtlReload(LaunchCtlTask):
|
||||||
|
def __init__(self, module, service, plist):
|
||||||
|
super(LaunchCtlReload, self).__init__(module, service, plist)
|
||||||
|
|
||||||
|
def runCommand(self):
|
||||||
|
state, dummy, dummy, dummy = self.get_state()
|
||||||
|
|
||||||
|
if state == ServiceState.UNLOADED:
|
||||||
|
# launchd throws an error if we do an unload on an already
|
||||||
|
# unloaded service.
|
||||||
|
self.load()
|
||||||
|
else:
|
||||||
|
self.reload()
|
||||||
|
|
||||||
|
|
||||||
|
class LaunchCtlUnload(LaunchCtlTask):
|
||||||
|
def __init__(self, module, service, plist):
|
||||||
|
super(LaunchCtlUnload, self).__init__(module, service, plist)
|
||||||
|
|
||||||
|
def runCommand(self):
|
||||||
|
state, dummy, dummy, dummy = self.get_state()
|
||||||
|
self.unload()
|
||||||
|
|
||||||
|
|
||||||
|
class LaunchCtlRestart(LaunchCtlReload):
|
||||||
|
def __init__(self, module, service, plist):
|
||||||
|
super(LaunchCtlRestart, self).__init__(module, service, plist)
|
||||||
|
|
||||||
|
def runCommand(self):
|
||||||
|
super(LaunchCtlRestart, self).runCommand()
|
||||||
|
self.start()
|
||||||
|
|
||||||
|
|
||||||
|
class LaunchCtlList(LaunchCtlTask):
|
||||||
|
def __init__(self, module, service):
|
||||||
|
super(LaunchCtlList, self).__init__(module, service, None)
|
||||||
|
|
||||||
|
def runCommand(self):
|
||||||
|
# Do nothing, the list functionality is done by the
|
||||||
|
# base class run method.
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
module = AnsibleModule(
|
||||||
|
argument_spec=dict(
|
||||||
|
name=dict(type='str', required=True),
|
||||||
|
state=dict(type='str', choices=['reloaded', 'restarted', 'started', 'stopped', 'unloaded']),
|
||||||
|
enabled=dict(type='bool'),
|
||||||
|
force_stop=dict(type='bool', default=False),
|
||||||
|
),
|
||||||
|
supports_check_mode=True,
|
||||||
|
required_one_of=[
|
||||||
|
['state', 'enabled'],
|
||||||
|
],
|
||||||
|
)
|
||||||
|
|
||||||
|
service = module.params['name']
|
||||||
|
action = module.params['state']
|
||||||
|
rc = 0
|
||||||
|
out = err = ''
|
||||||
|
result = {
|
||||||
|
'name': service,
|
||||||
|
'changed': False,
|
||||||
|
'status': {},
|
||||||
|
}
|
||||||
|
|
||||||
|
# We will tailor the plist file in case one of the options
|
||||||
|
# (enabled, force_stop) was specified.
|
||||||
|
plist = Plist(module, service)
|
||||||
|
result['changed'] = plist.is_changed()
|
||||||
|
|
||||||
|
# Gather information about the service to be controlled.
|
||||||
|
state, pid, dummy, dummy = LaunchCtlList(module, service).run()
|
||||||
|
result['status']['previous_state'] = ServiceState.to_string(state)
|
||||||
|
result['status']['previous_pid'] = pid
|
||||||
|
|
||||||
|
# Map the actions to specific tasks
|
||||||
|
tasks = {
|
||||||
|
'started': LaunchCtlStart(module, service, plist),
|
||||||
|
'stopped': LaunchCtlStop(module, service, plist),
|
||||||
|
'restarted': LaunchCtlRestart(module, service, plist),
|
||||||
|
'reloaded': LaunchCtlReload(module, service, plist),
|
||||||
|
'unloaded': LaunchCtlUnload(module, service, plist)
|
||||||
|
}
|
||||||
|
|
||||||
|
status_code = '0'
|
||||||
|
# Run the requested task
|
||||||
|
if not module.check_mode:
|
||||||
|
state, pid, status_code, err = tasks[action].run()
|
||||||
|
|
||||||
|
result['status']['current_state'] = ServiceState.to_string(state)
|
||||||
|
result['status']['current_pid'] = pid
|
||||||
|
result['status']['status_code'] = status_code
|
||||||
|
result['status']['error'] = err
|
||||||
|
|
||||||
|
if (result['status']['current_state'] != result['status']['previous_state'] or
|
||||||
|
result['status']['current_pid'] != result['status']['previous_pid']):
|
||||||
|
result['changed'] = True
|
||||||
|
if module.check_mode:
|
||||||
|
result['changed'] = True
|
||||||
|
module.exit_json(**result)
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
main()
|
3
tests/integration/targets/launchd/aliases
Normal file
3
tests/integration/targets/launchd/aliases
Normal file
|
@ -0,0 +1,3 @@
|
||||||
|
shippable/posix/group1
|
||||||
|
skip/freebsd
|
||||||
|
skip/rhel
|
|
@ -0,0 +1,21 @@
|
||||||
|
#!/usr/bin/env python
|
||||||
|
|
||||||
|
from __future__ import absolute_import, division, print_function
|
||||||
|
__metaclass__ = type
|
||||||
|
|
||||||
|
import sys
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
if sys.version_info[0] >= 3:
|
||||||
|
import http.server
|
||||||
|
import socketserver
|
||||||
|
PORT = int(sys.argv[1])
|
||||||
|
Handler = http.server.SimpleHTTPRequestHandler
|
||||||
|
httpd = socketserver.TCPServer(("", PORT), Handler)
|
||||||
|
httpd.serve_forever()
|
||||||
|
else:
|
||||||
|
import mimetypes
|
||||||
|
mimetypes.init()
|
||||||
|
mimetypes.add_type('application/json', '.json')
|
||||||
|
import SimpleHTTPServer
|
||||||
|
SimpleHTTPServer.test()
|
4
tests/integration/targets/launchd/meta/main.yml
Normal file
4
tests/integration/targets/launchd/meta/main.yml
Normal file
|
@ -0,0 +1,4 @@
|
||||||
|
---
|
||||||
|
|
||||||
|
dependencies:
|
||||||
|
- prepare_tests
|
22
tests/integration/targets/launchd/tasks/main.yml
Normal file
22
tests/integration/targets/launchd/tasks/main.yml
Normal file
|
@ -0,0 +1,22 @@
|
||||||
|
---
|
||||||
|
|
||||||
|
- name: Test launchd module
|
||||||
|
block:
|
||||||
|
- name: Expect that launchctl exists
|
||||||
|
stat:
|
||||||
|
path: /bin/launchctl
|
||||||
|
register: launchctl_check
|
||||||
|
failed_when:
|
||||||
|
- not launchctl_check.stat.exists
|
||||||
|
|
||||||
|
- name: Run tests
|
||||||
|
include_tasks: test.yml
|
||||||
|
with_items:
|
||||||
|
- test_unknown
|
||||||
|
- test_start_stop
|
||||||
|
- test_restart
|
||||||
|
- test_unload
|
||||||
|
- test_reload
|
||||||
|
- test_runatload
|
||||||
|
|
||||||
|
when: ansible_os_family == 'Darwin'
|
20
tests/integration/targets/launchd/tasks/setup.yml
Normal file
20
tests/integration/targets/launchd/tasks/setup.yml
Normal file
|
@ -0,0 +1,20 @@
|
||||||
|
---
|
||||||
|
|
||||||
|
- name: "[{{ item }}] Deploy test service configuration"
|
||||||
|
template:
|
||||||
|
src: "{{ launchd_service_name }}.plist.j2"
|
||||||
|
dest: "{{ launchd_plist_location }}"
|
||||||
|
become: yes
|
||||||
|
|
||||||
|
- name: install the test daemon script
|
||||||
|
copy:
|
||||||
|
src: ansible_test_service.py
|
||||||
|
dest: /usr/local/sbin/ansible_test_service
|
||||||
|
mode: '755'
|
||||||
|
|
||||||
|
- name: rewrite shebang in the test daemon script
|
||||||
|
lineinfile:
|
||||||
|
path: /usr/local/sbin/ansible_test_service
|
||||||
|
line: "#!{{ ansible_python_interpreter | realpath }}"
|
||||||
|
insertbefore: BOF
|
||||||
|
firstmatch: yes
|
27
tests/integration/targets/launchd/tasks/teardown.yml
Normal file
27
tests/integration/targets/launchd/tasks/teardown.yml
Normal file
|
@ -0,0 +1,27 @@
|
||||||
|
---
|
||||||
|
|
||||||
|
- name: "[{{ item }}] Unload service"
|
||||||
|
launchd:
|
||||||
|
name: "{{ launchd_service_name }}"
|
||||||
|
state: unloaded
|
||||||
|
become: yes
|
||||||
|
register: launchd_unloaded_result
|
||||||
|
|
||||||
|
- name: "[{{ item }}] Validation"
|
||||||
|
assert:
|
||||||
|
that:
|
||||||
|
- launchd_unloaded_result is success
|
||||||
|
- launchd_unloaded_result.status.current_state == 'unloaded'
|
||||||
|
- launchd_unloaded_result.status.current_pid == '-'
|
||||||
|
|
||||||
|
- name: "[{{ item }}] Remove test service configuration"
|
||||||
|
file:
|
||||||
|
path: "{{ launchd_plist_location }}"
|
||||||
|
state: absent
|
||||||
|
become: yes
|
||||||
|
|
||||||
|
- name: "[{{ item }}] Remove test service server"
|
||||||
|
file:
|
||||||
|
path: "/usr/local/sbin/ansible_test_service"
|
||||||
|
state: absent
|
||||||
|
become: yes
|
8
tests/integration/targets/launchd/tasks/test.yml
Normal file
8
tests/integration/targets/launchd/tasks/test.yml
Normal file
|
@ -0,0 +1,8 @@
|
||||||
|
---
|
||||||
|
|
||||||
|
- name: "Running {{ item }}"
|
||||||
|
block:
|
||||||
|
- include_tasks: setup.yml
|
||||||
|
- include_tasks: "tests/{{ item }}.yml"
|
||||||
|
always:
|
||||||
|
- include_tasks: teardown.yml
|
|
@ -0,0 +1,68 @@
|
||||||
|
---
|
||||||
|
|
||||||
|
# -----------------------------------------------------------
|
||||||
|
|
||||||
|
- name: "[{{ item }}] Given a started service in check_mode"
|
||||||
|
launchd:
|
||||||
|
name: "{{ launchd_service_name }}"
|
||||||
|
state: started
|
||||||
|
become: yes
|
||||||
|
register: "test_1_launchd_start_result_check_mode"
|
||||||
|
check_mode: yes
|
||||||
|
|
||||||
|
- name: "[{{ item }}] Assert that everything work in check mode"
|
||||||
|
assert:
|
||||||
|
that:
|
||||||
|
- test_1_launchd_start_result_check_mode is success
|
||||||
|
- test_1_launchd_start_result_check_mode is changed
|
||||||
|
|
||||||
|
- name: "[{{ item }}] Given a started service..."
|
||||||
|
launchd:
|
||||||
|
name: "{{ launchd_service_name }}"
|
||||||
|
state: started
|
||||||
|
become: yes
|
||||||
|
register: "test_1_launchd_start_result"
|
||||||
|
|
||||||
|
|
||||||
|
- name: "[{{ item }}] The started service should run on port 21212"
|
||||||
|
wait_for:
|
||||||
|
port: 21212
|
||||||
|
delay: 5
|
||||||
|
timeout: 10
|
||||||
|
|
||||||
|
- name: "[{{ item }}] Deploy a new test service configuration with a new port 21213"
|
||||||
|
template:
|
||||||
|
src: "modified.{{ launchd_service_name }}.plist.j2"
|
||||||
|
dest: "{{ launchd_plist_location }}"
|
||||||
|
become: yes
|
||||||
|
|
||||||
|
- name: "[{{ item }}] When reloading the service..."
|
||||||
|
launchd:
|
||||||
|
name: "{{ launchd_service_name }}"
|
||||||
|
state: reloaded
|
||||||
|
become: yes
|
||||||
|
register: "test_1_launchd_reload_result"
|
||||||
|
|
||||||
|
- name: "[{{ item }}] Validate that service was reloaded"
|
||||||
|
assert:
|
||||||
|
that:
|
||||||
|
- test_1_launchd_reload_result is success
|
||||||
|
- test_1_launchd_reload_result is changed
|
||||||
|
- test_1_launchd_reload_result.status.previous_pid == test_1_launchd_start_result.status.current_pid
|
||||||
|
- test_1_launchd_reload_result.status.previous_state == test_1_launchd_start_result.status.current_state
|
||||||
|
- test_1_launchd_reload_result.status.current_state == 'stopped'
|
||||||
|
- test_1_launchd_reload_result.status.current_pid == '-'
|
||||||
|
|
||||||
|
- name: "[{{ item }}] Start the service with the new configuration..."
|
||||||
|
launchd:
|
||||||
|
name: "{{ launchd_service_name }}"
|
||||||
|
state: started
|
||||||
|
become: yes
|
||||||
|
register: "test_1_launchd_start_result"
|
||||||
|
|
||||||
|
|
||||||
|
- name: "[{{ item }}] The started service should run on port 21213"
|
||||||
|
wait_for:
|
||||||
|
port: 21213
|
||||||
|
delay: 5
|
||||||
|
timeout: 10
|
|
@ -0,0 +1,43 @@
|
||||||
|
---
|
||||||
|
|
||||||
|
# -----------------------------------------------------------
|
||||||
|
|
||||||
|
- name: "[{{ item }}] Given a started service..."
|
||||||
|
launchd:
|
||||||
|
name: "{{ launchd_service_name }}"
|
||||||
|
state: started
|
||||||
|
become: yes
|
||||||
|
register: "test_1_launchd_start_result"
|
||||||
|
|
||||||
|
|
||||||
|
- name: "[{{ item }}] When restarting the service in check mode"
|
||||||
|
launchd:
|
||||||
|
name: "{{ launchd_service_name }}"
|
||||||
|
state: restarted
|
||||||
|
become: yes
|
||||||
|
register: "test_1_launchd_restart_result_check_mode"
|
||||||
|
check_mode: yes
|
||||||
|
|
||||||
|
- name: "[{{ item }}] Validate that service was restarted in check mode"
|
||||||
|
assert:
|
||||||
|
that:
|
||||||
|
- test_1_launchd_restart_result_check_mode is success
|
||||||
|
- test_1_launchd_restart_result_check_mode is changed
|
||||||
|
|
||||||
|
- name: "[{{ item }}] When restarting the service..."
|
||||||
|
launchd:
|
||||||
|
name: "{{ launchd_service_name }}"
|
||||||
|
state: restarted
|
||||||
|
become: yes
|
||||||
|
register: "test_1_launchd_restart_result"
|
||||||
|
|
||||||
|
- name: "[{{ item }}] Validate that service was restarted"
|
||||||
|
assert:
|
||||||
|
that:
|
||||||
|
- test_1_launchd_restart_result is success
|
||||||
|
- test_1_launchd_restart_result is changed
|
||||||
|
- test_1_launchd_restart_result.status.previous_pid == test_1_launchd_start_result.status.current_pid
|
||||||
|
- test_1_launchd_restart_result.status.previous_state == test_1_launchd_start_result.status.current_state
|
||||||
|
- test_1_launchd_restart_result.status.current_state == 'started'
|
||||||
|
- test_1_launchd_restart_result.status.current_pid != '-'
|
||||||
|
- test_1_launchd_restart_result.status.status_code == '0'
|
|
@ -0,0 +1,32 @@
|
||||||
|
---
|
||||||
|
# -----------------------------------------------------------
|
||||||
|
|
||||||
|
- name: "[{{ item }}] Given a started service with RunAtLoad set to true..."
|
||||||
|
launchd:
|
||||||
|
name: "{{ launchd_service_name }}"
|
||||||
|
state: started
|
||||||
|
enabled: yes
|
||||||
|
become: yes
|
||||||
|
register: test_1_launchd_start_result
|
||||||
|
|
||||||
|
- name: "[{{ item }}] Validate that service was started"
|
||||||
|
assert:
|
||||||
|
that:
|
||||||
|
- test_1_launchd_start_result is success
|
||||||
|
- test_1_launchd_start_result is changed
|
||||||
|
- test_1_launchd_start_result.status.previous_pid == '-'
|
||||||
|
- test_1_launchd_start_result.status.previous_state == 'unloaded'
|
||||||
|
- test_1_launchd_start_result.status.current_state == 'started'
|
||||||
|
- test_1_launchd_start_result.status.current_pid != '-'
|
||||||
|
- test_1_launchd_start_result.status.status_code == '0'
|
||||||
|
|
||||||
|
- name: "[{{ item }}] Validate that RunAtLoad is set to true"
|
||||||
|
replace:
|
||||||
|
path: "{{ launchd_plist_location }}"
|
||||||
|
regexp: |
|
||||||
|
\s+<key>RunAtLoad</key>
|
||||||
|
\s+<true/>
|
||||||
|
replace: found_run_at_load
|
||||||
|
check_mode: yes
|
||||||
|
register: contents_would_have
|
||||||
|
failed_when: not contents_would_have is changed
|
|
@ -0,0 +1,112 @@
|
||||||
|
---
|
||||||
|
|
||||||
|
# -----------------------------------------------------------
|
||||||
|
|
||||||
|
- name: "[{{ item }}] Given a started service in check mode"
|
||||||
|
launchd:
|
||||||
|
name: "{{ launchd_service_name }}"
|
||||||
|
state: started
|
||||||
|
become: yes
|
||||||
|
register: "test_1_launchd_start_result_check_mode"
|
||||||
|
check_mode: yes
|
||||||
|
|
||||||
|
|
||||||
|
- name: "[{{ item }}] Validate that service was started in check mode"
|
||||||
|
assert:
|
||||||
|
that:
|
||||||
|
- test_1_launchd_start_result_check_mode is success
|
||||||
|
- test_1_launchd_start_result_check_mode is changed
|
||||||
|
|
||||||
|
|
||||||
|
- name: "[{{ item }}] Given a started service..."
|
||||||
|
launchd:
|
||||||
|
name: "{{ launchd_service_name }}"
|
||||||
|
state: started
|
||||||
|
become: yes
|
||||||
|
register: "test_1_launchd_start_result"
|
||||||
|
|
||||||
|
|
||||||
|
- name: "[{{ item }}] Validate that service was started"
|
||||||
|
assert:
|
||||||
|
that:
|
||||||
|
- test_1_launchd_start_result is success
|
||||||
|
- test_1_launchd_start_result is changed
|
||||||
|
- test_1_launchd_start_result.status.previous_pid == '-'
|
||||||
|
- test_1_launchd_start_result.status.previous_state == 'unloaded'
|
||||||
|
- test_1_launchd_start_result.status.current_state == 'started'
|
||||||
|
- test_1_launchd_start_result.status.current_pid != '-'
|
||||||
|
- test_1_launchd_start_result.status.status_code == '0'
|
||||||
|
|
||||||
|
# -----------------------------------------------------------
|
||||||
|
|
||||||
|
- name: "[{{ item }}] Given a stopped service..."
|
||||||
|
launchd:
|
||||||
|
name: "{{ launchd_service_name }}"
|
||||||
|
state: stopped
|
||||||
|
become: yes
|
||||||
|
register: "test_2_launchd_stop_result"
|
||||||
|
|
||||||
|
- name: "[{{ item }}] Validate that service was stopped after it was started"
|
||||||
|
assert:
|
||||||
|
that:
|
||||||
|
- test_2_launchd_stop_result is success
|
||||||
|
- test_2_launchd_stop_result is changed
|
||||||
|
- test_2_launchd_stop_result.status.previous_pid == test_1_launchd_start_result.status.current_pid
|
||||||
|
- test_2_launchd_stop_result.status.previous_state == test_1_launchd_start_result.status.current_state
|
||||||
|
- test_2_launchd_stop_result.status.current_state == 'stopped'
|
||||||
|
- test_2_launchd_stop_result.status.current_pid == '-'
|
||||||
|
|
||||||
|
# -----------------------------------------------------------
|
||||||
|
|
||||||
|
- name: "[{{ item }}] Given a stopped service..."
|
||||||
|
launchd:
|
||||||
|
name: "{{ launchd_service_name }}"
|
||||||
|
state: stopped
|
||||||
|
become: yes
|
||||||
|
register: "test_3_launchd_stop_result"
|
||||||
|
|
||||||
|
- name: "[{{ item }}] Validate that service can be stopped after being already stopped"
|
||||||
|
assert:
|
||||||
|
that:
|
||||||
|
- test_3_launchd_stop_result is success
|
||||||
|
- not test_3_launchd_stop_result is changed
|
||||||
|
- test_3_launchd_stop_result.status.previous_pid == '-'
|
||||||
|
- test_3_launchd_stop_result.status.previous_state == 'stopped'
|
||||||
|
- test_3_launchd_stop_result.status.current_state == 'stopped'
|
||||||
|
- test_3_launchd_stop_result.status.current_pid == '-'
|
||||||
|
|
||||||
|
# -----------------------------------------------------------
|
||||||
|
|
||||||
|
- name: "[{{ item }}] Given a started service..."
|
||||||
|
launchd:
|
||||||
|
name: "{{ launchd_service_name }}"
|
||||||
|
state: started
|
||||||
|
become: yes
|
||||||
|
register: "test_4_launchd_start_result"
|
||||||
|
|
||||||
|
- name: "[{{ item }}] Validate that service was started..."
|
||||||
|
assert:
|
||||||
|
that:
|
||||||
|
- test_4_launchd_start_result is success
|
||||||
|
- test_4_launchd_start_result is changed
|
||||||
|
- test_4_launchd_start_result.status.previous_pid == '-'
|
||||||
|
- test_4_launchd_start_result.status.previous_state == 'stopped'
|
||||||
|
- test_4_launchd_start_result.status.current_state == 'started'
|
||||||
|
- test_4_launchd_start_result.status.current_pid != '-'
|
||||||
|
- test_4_launchd_start_result.status.status_code == '0'
|
||||||
|
|
||||||
|
- name: "[{{ item }}] And when service is started again..."
|
||||||
|
launchd:
|
||||||
|
name: "{{ launchd_service_name }}"
|
||||||
|
state: started
|
||||||
|
become: yes
|
||||||
|
register: "test_5_launchd_start_result"
|
||||||
|
|
||||||
|
- name: "[{{ item }}] Validate that service is still in the same state as before"
|
||||||
|
assert:
|
||||||
|
that:
|
||||||
|
- test_5_launchd_start_result is success
|
||||||
|
- not test_5_launchd_start_result is changed
|
||||||
|
- test_5_launchd_start_result.status.previous_pid == test_4_launchd_start_result.status.current_pid
|
||||||
|
- test_5_launchd_start_result.status.previous_state == test_4_launchd_start_result.status.current_state
|
||||||
|
- test_5_launchd_start_result.status.status_code == '0'
|
|
@ -0,0 +1,11 @@
|
||||||
|
---
|
||||||
|
|
||||||
|
# -----------------------------------------------------------
|
||||||
|
|
||||||
|
- name: "[{{ item }}] Expect that an error occurs when an unknown service is used."
|
||||||
|
launchd:
|
||||||
|
name: com.acme.unknownservice
|
||||||
|
state: started
|
||||||
|
register: result
|
||||||
|
failed_when:
|
||||||
|
- not '"Unable to infer the path of com.acme.unknownservice service plist file and it was not found among active services" in result.msg'
|
|
@ -0,0 +1,62 @@
|
||||||
|
---
|
||||||
|
|
||||||
|
# -----------------------------------------------------------
|
||||||
|
|
||||||
|
- name: "[{{ item }}] Given a started service..."
|
||||||
|
launchd:
|
||||||
|
name: "{{ launchd_service_name }}"
|
||||||
|
state: started
|
||||||
|
become: yes
|
||||||
|
register: "test_1_launchd_start_result"
|
||||||
|
|
||||||
|
|
||||||
|
- name: "[{{ item }}] When unloading the service in check mode"
|
||||||
|
launchd:
|
||||||
|
name: "{{ launchd_service_name }}"
|
||||||
|
state: unloaded
|
||||||
|
become: yes
|
||||||
|
register: "test_1_launchd_unloaded_result_check_mode"
|
||||||
|
check_mode: yes
|
||||||
|
|
||||||
|
- name: "[{{ item }}] Validate that service was unloaded in check mode"
|
||||||
|
assert:
|
||||||
|
that:
|
||||||
|
- test_1_launchd_unloaded_result_check_mode is success
|
||||||
|
- test_1_launchd_unloaded_result_check_mode is changed
|
||||||
|
|
||||||
|
|
||||||
|
- name: "[{{ item }}] When unloading the service..."
|
||||||
|
launchd:
|
||||||
|
name: "{{ launchd_service_name }}"
|
||||||
|
state: unloaded
|
||||||
|
become: yes
|
||||||
|
register: "test_1_launchd_unloaded_result"
|
||||||
|
|
||||||
|
- name: "[{{ item }}] Validate that service was unloaded"
|
||||||
|
assert:
|
||||||
|
that:
|
||||||
|
- test_1_launchd_unloaded_result is success
|
||||||
|
- test_1_launchd_unloaded_result is changed
|
||||||
|
- test_1_launchd_unloaded_result.status.previous_pid == test_1_launchd_start_result.status.current_pid
|
||||||
|
- test_1_launchd_unloaded_result.status.previous_state == test_1_launchd_start_result.status.current_state
|
||||||
|
- test_1_launchd_unloaded_result.status.current_state == 'unloaded'
|
||||||
|
- test_1_launchd_unloaded_result.status.current_pid == '-'
|
||||||
|
|
||||||
|
# -----------------------------------------------------------
|
||||||
|
|
||||||
|
- name: "[{{ item }}] Given an unloaded service on an unloaded service..."
|
||||||
|
launchd:
|
||||||
|
name: "{{ launchd_service_name }}"
|
||||||
|
state: unloaded
|
||||||
|
become: yes
|
||||||
|
register: "test_2_launchd_unloaded_result"
|
||||||
|
|
||||||
|
- name: "[{{ item }}] Validate that service did not change and is still unloaded"
|
||||||
|
assert:
|
||||||
|
that:
|
||||||
|
- test_2_launchd_unloaded_result is success
|
||||||
|
- not test_2_launchd_unloaded_result is changed
|
||||||
|
- test_2_launchd_unloaded_result.status.previous_pid == '-'
|
||||||
|
- test_2_launchd_unloaded_result.status.previous_state == 'unloaded'
|
||||||
|
- test_2_launchd_unloaded_result.status.current_state == 'unloaded'
|
||||||
|
- test_2_launchd_unloaded_result.status.current_pid == '-'
|
|
@ -0,0 +1,13 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||||
|
<plist version="1.0">
|
||||||
|
<dict>
|
||||||
|
<key>Label</key>
|
||||||
|
<string>{{ launchd_service_name }}</string>
|
||||||
|
<key>ProgramArguments</key>
|
||||||
|
<array>
|
||||||
|
<string>/usr/local/sbin/ansible_test_service</string>
|
||||||
|
<string>21212</string>
|
||||||
|
</array>
|
||||||
|
</dict>
|
||||||
|
</plist>
|
|
@ -0,0 +1,13 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||||
|
<plist version="1.0">
|
||||||
|
<dict>
|
||||||
|
<key>Label</key>
|
||||||
|
<string>{{ launchd_service_name }}</string>
|
||||||
|
<key>ProgramArguments</key>
|
||||||
|
<array>
|
||||||
|
<string>/usr/local/sbin/ansible_test_service</string>
|
||||||
|
<string>21213</string>
|
||||||
|
</array>
|
||||||
|
</dict>
|
||||||
|
</plist>
|
4
tests/integration/targets/launchd/vars/main.yml
Normal file
4
tests/integration/targets/launchd/vars/main.yml
Normal file
|
@ -0,0 +1,4 @@
|
||||||
|
---
|
||||||
|
|
||||||
|
launchd_service_name: launchd.test.service
|
||||||
|
launchd_plist_location: /Library/LaunchDaemons/{{ launchd_service_name }}.plist
|
|
@ -1235,6 +1235,7 @@ plugins/modules/system/java_keystore.py validate-modules:doc-missing-type
|
||||||
plugins/modules/system/java_keystore.py validate-modules:parameter-type-not-in-doc
|
plugins/modules/system/java_keystore.py validate-modules:parameter-type-not-in-doc
|
||||||
plugins/modules/system/java_keystore.py validate-modules:undocumented-parameter
|
plugins/modules/system/java_keystore.py validate-modules:undocumented-parameter
|
||||||
plugins/modules/system/kernel_blacklist.py validate-modules:parameter-type-not-in-doc
|
plugins/modules/system/kernel_blacklist.py validate-modules:parameter-type-not-in-doc
|
||||||
|
plugins/modules/system/launchd.py use-argspec-type-path # False positive
|
||||||
plugins/modules/system/lbu.py validate-modules:doc-elements-mismatch
|
plugins/modules/system/lbu.py validate-modules:doc-elements-mismatch
|
||||||
plugins/modules/system/locale_gen.py validate-modules:parameter-type-not-in-doc
|
plugins/modules/system/locale_gen.py validate-modules:parameter-type-not-in-doc
|
||||||
plugins/modules/system/lvg.py pylint:blacklisted-name
|
plugins/modules/system/lvg.py pylint:blacklisted-name
|
||||||
|
|
|
@ -1235,6 +1235,7 @@ plugins/modules/system/java_keystore.py validate-modules:doc-missing-type
|
||||||
plugins/modules/system/java_keystore.py validate-modules:parameter-type-not-in-doc
|
plugins/modules/system/java_keystore.py validate-modules:parameter-type-not-in-doc
|
||||||
plugins/modules/system/java_keystore.py validate-modules:undocumented-parameter
|
plugins/modules/system/java_keystore.py validate-modules:undocumented-parameter
|
||||||
plugins/modules/system/kernel_blacklist.py validate-modules:parameter-type-not-in-doc
|
plugins/modules/system/kernel_blacklist.py validate-modules:parameter-type-not-in-doc
|
||||||
|
plugins/modules/system/launchd.py use-argspec-type-path # False positive
|
||||||
plugins/modules/system/lbu.py validate-modules:doc-elements-mismatch
|
plugins/modules/system/lbu.py validate-modules:doc-elements-mismatch
|
||||||
plugins/modules/system/locale_gen.py validate-modules:parameter-type-not-in-doc
|
plugins/modules/system/locale_gen.py validate-modules:parameter-type-not-in-doc
|
||||||
plugins/modules/system/lvg.py pylint:blacklisted-name
|
plugins/modules/system/lvg.py pylint:blacklisted-name
|
||||||
|
|
|
@ -955,6 +955,7 @@ plugins/modules/system/interfaces_file.py validate-modules:parameter-type-not-in
|
||||||
plugins/modules/system/java_cert.py pylint:blacklisted-name
|
plugins/modules/system/java_cert.py pylint:blacklisted-name
|
||||||
plugins/modules/system/java_keystore.py validate-modules:doc-missing-type
|
plugins/modules/system/java_keystore.py validate-modules:doc-missing-type
|
||||||
plugins/modules/system/kernel_blacklist.py validate-modules:parameter-type-not-in-doc
|
plugins/modules/system/kernel_blacklist.py validate-modules:parameter-type-not-in-doc
|
||||||
|
plugins/modules/system/launchd.py use-argspec-type-path # False positive
|
||||||
plugins/modules/system/locale_gen.py validate-modules:parameter-type-not-in-doc
|
plugins/modules/system/locale_gen.py validate-modules:parameter-type-not-in-doc
|
||||||
plugins/modules/system/lvg.py pylint:blacklisted-name
|
plugins/modules/system/lvg.py pylint:blacklisted-name
|
||||||
plugins/modules/system/lvol.py pylint:blacklisted-name
|
plugins/modules/system/lvol.py pylint:blacklisted-name
|
||||||
|
|
Loading…
Add table
Reference in a new issue