mirror of
				https://github.com/ansible-collections/community.general.git
				synced 2025-10-25 21:44:00 -07:00 
			
		
		
		
	
		
			
				
	
	
		
			524 lines
		
	
	
	
		
			17 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			524 lines
		
	
	
	
		
			17 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| #!/usr/bin/python
 | |
| 
 | |
| # Copyright (c) 2018, Martin Migasiewicz <migasiew.nk@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 annotations
 | |
| 
 | |
| 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.
 | |
| extends_documentation_fragment:
 | |
|   - community.general.attributes
 | |
| attributes:
 | |
|   check_mode:
 | |
|     support: full
 | |
|   diff_mode:
 | |
|     support: none
 | |
| options:
 | |
|   name:
 | |
|     description:
 | |
|       - Name of the service.
 | |
|     type: str
 | |
|     required: true
 | |
|   plist:
 | |
|     description:
 | |
|       - Name of the V(.plist) file for the service.
 | |
|       - Defaults to V({name}.plist).
 | |
|     type: str
 | |
|     version_added: 10.1.0
 | |
|   state:
 | |
|     description:
 | |
|       - V(started)/V(stopped) are idempotent actions that do not run commands unless necessary.
 | |
|       - C(launchd) does not support V(restarted) nor V(reloaded) natively. These states trigger a stop/start (restarted) or
 | |
|         an unload/load (reloaded).
 | |
|       - V(restarted) unloads and loads the service before start to ensure that the latest job definition (plist) is used.
 | |
|       - V(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 C(KeepAlive) attribute set to V(true) in a launchd configuration. In case this is set to V(true),
 | |
|         stopping a service causes that C(launchd) starts the service again.
 | |
|       - Set this option to V(true) to let this module change the C(KeepAlive) attribute to V(false).
 | |
|     type: bool
 | |
|     default: false
 | |
| 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: true
 | |
| 
 | |
| - name: Restart memcached
 | |
|   community.general.launchd:
 | |
|     name: org.memcached
 | |
|     state: restarted
 | |
| 
 | |
| - name: Unload memcached
 | |
|   community.general.launchd:
 | |
|     name: org.memcached
 | |
|     state: unloaded
 | |
| 
 | |
| - name: restart sshd
 | |
|   community.general.launchd:
 | |
|     name: com.openssh.sshd
 | |
|     plist: ssh.plist
 | |
|     state: restarted
 | |
| """
 | |
| 
 | |
| 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
 | |
| from ansible.module_utils.common.text.converters import to_native
 | |
| 
 | |
| 
 | |
| 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, filename=None):
 | |
|         self.__changed = False
 | |
|         self.__service = service
 | |
|         if filename is not None:
 | |
|             self.__filename = filename
 | |
|         else:
 | |
|             self.__filename = '%s.plist' % service
 | |
| 
 | |
|         state, pid, dummy, dummy = LaunchCtlList(module, self.__service).run()
 | |
| 
 | |
|         self.__file = self.__find_service_plist(self.__filename)
 | |
|         if self.__file is None:
 | |
|             msg = 'Unable to find the plist file %s for service %s' % (
 | |
|                 self.__filename, self.__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)
 | |
| 
 | |
|     @staticmethod
 | |
|     def __find_service_plist(filename):
 | |
|         """Finds the plist file associated with a service"""
 | |
| 
 | |
|         launchd_paths = [
 | |
|             os.path.join(os.getenv('HOME'), 'Library/LaunchAgents'),
 | |
|             '/Library/LaunchAgents',
 | |
|             '/Library/LaunchDaemons',
 | |
|             '/System/Library/LaunchAgents',
 | |
|             '/System/Library/LaunchDaemons'
 | |
|         ]
 | |
| 
 | |
|         for path in launchd_paths:
 | |
|             try:
 | |
|                 files = os.listdir(path)
 | |
|             except OSError:
 | |
|                 continue
 | |
| 
 | |
|             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 __read_plist_file(self, module):
 | |
|         service_plist = {}
 | |
|         try:
 | |
|             with open(self.__file, 'rb') as plist_fp:
 | |
|                 service_plist = plistlib.load(plist_fp)
 | |
|         except Exception as e:
 | |
|             module.fail_json(msg="Failed to read plist file "
 | |
|                                  "%s due to %s" % (self.__file, to_native(e)))
 | |
|         return service_plist
 | |
| 
 | |
|     def __write_plist_file(self, module, service_plist=None):
 | |
|         if not service_plist:
 | |
|             service_plist = {}
 | |
| 
 | |
|         try:
 | |
|             with open(self.__file, 'wb') as plist_fp:
 | |
|                 plistlib.dump(service_plist, plist_fp)
 | |
|         except Exception as e:
 | |
|             module.fail_json(msg="Failed to write to plist file "
 | |
|                                  " %s due to %s" % (self.__file, to_native(e)))
 | |
| 
 | |
|     def __handle_param_enabled(self, module):
 | |
|         if module.params['enabled'] is not None:
 | |
|             service_plist = self.__read_plist_file(module)
 | |
| 
 | |
|             # 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:
 | |
|                         self.__write_plist_file(module, service_plist)
 | |
|                         self.__changed = True
 | |
| 
 | |
|     def __handle_param_force_stop(self, module):
 | |
|         if module.params['force_stop'] is not None:
 | |
|             service_plist = self.__read_plist_file(module)
 | |
| 
 | |
|             # 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:
 | |
|                         self.__write_plist_file(module, service_plist)
 | |
|                         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 in (ServiceState.STOPPED, 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 in (ServiceState.STARTED, 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),
 | |
|             plist=dict(type='str'),
 | |
|             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']
 | |
|     plist_filename = module.params['plist']
 | |
|     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, plist_filename)
 | |
|     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:
 | |
|         if result['status']['current_state'] != action:
 | |
|             result['changed'] = True
 | |
|     module.exit_json(**result)
 | |
| 
 | |
| 
 | |
| if __name__ == '__main__':
 | |
|     main()
 |