delegated_serial_command.py 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274
  1. #!/usr/bin/python
  2. # -*- coding: utf-8 -*-
  3. # (c) 2012, Michael DeHaan <michael.dehaan@gmail.com>, and others
  4. # (c) 2016, Andrew Butcher <abutcher@redhat.com>
  5. #
  6. # This module is derrived from the Ansible command module.
  7. #
  8. # Ansible is free software: you can redistribute it and/or modify
  9. # it under the terms of the GNU General Public License as published by
  10. # the Free Software Foundation, either version 3 of the License, or
  11. # (at your option) any later version.
  12. #
  13. # Ansible is distributed in the hope that it will be useful,
  14. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  15. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  16. # GNU General Public License for more details.
  17. #
  18. # You should have received a copy of the GNU General Public License
  19. # along with Ansible. If not, see <http://www.gnu.org/licenses/>.
  20. # pylint: disable=unused-wildcard-import,wildcard-import,unused-import,redefined-builtin
  21. ''' delegated_serial_command '''
  22. import datetime
  23. import errno
  24. import glob
  25. import shlex
  26. import os
  27. import fcntl
  28. import time
  29. DOCUMENTATION = '''
  30. ---
  31. module: delegated_serial_command
  32. short_description: Executes a command on a remote node
  33. version_added: historical
  34. description:
  35. - The M(command) module takes the command name followed by a list
  36. of space-delimited arguments.
  37. - The given command will be executed on all selected nodes. It
  38. will not be processed through the shell, so variables like
  39. C($HOME) and operations like C("<"), C(">"), C("|"), and C("&")
  40. will not work (use the M(shell) module if you need these
  41. features).
  42. - Creates and maintains a lockfile such that this module will
  43. wait for other invocations to proceed.
  44. options:
  45. command:
  46. description:
  47. - the command to run
  48. required: true
  49. default: null
  50. creates:
  51. description:
  52. - a filename or (since 2.0) glob pattern, when it already
  53. exists, this step will B(not) be run.
  54. required: no
  55. default: null
  56. removes:
  57. description:
  58. - a filename or (since 2.0) glob pattern, when it does not
  59. exist, this step will B(not) be run.
  60. version_added: "0.8"
  61. required: no
  62. default: null
  63. chdir:
  64. description:
  65. - cd into this directory before running the command
  66. version_added: "0.6"
  67. required: false
  68. default: null
  69. executable:
  70. description:
  71. - change the shell used to execute the command. Should be an
  72. absolute path to the executable.
  73. required: false
  74. default: null
  75. version_added: "0.9"
  76. warn:
  77. version_added: "1.8"
  78. default: yes
  79. description:
  80. - if command warnings are on in ansible.cfg, do not warn about
  81. this particular line if set to no/false.
  82. required: false
  83. lockfile:
  84. default: yes
  85. description:
  86. - the lockfile that will be created
  87. timeout:
  88. default: yes
  89. description:
  90. - time in milliseconds to wait to obtain the lock
  91. notes:
  92. - If you want to run a command through the shell (say you are using C(<),
  93. C(>), C(|), etc), you actually want the M(shell) module instead. The
  94. M(command) module is much more secure as it's not affected by the user's
  95. environment.
  96. - " C(creates), C(removes), and C(chdir) can be specified after
  97. the command. For instance, if you only want to run a command if
  98. a certain file does not exist, use this."
  99. author:
  100. - Ansible Core Team
  101. - Michael DeHaan
  102. - Andrew Butcher
  103. '''
  104. EXAMPLES = '''
  105. # Example from Ansible Playbooks.
  106. - delegated_serial_command:
  107. command: /sbin/shutdown -t now
  108. # Run the command if the specified file does not exist.
  109. - delegated_serial_command:
  110. command: /usr/bin/make_database.sh arg1 arg2
  111. creates: /path/to/database
  112. '''
  113. # Dict of options and their defaults
  114. OPTIONS = {'chdir': None,
  115. 'creates': None,
  116. 'command': None,
  117. 'executable': None,
  118. 'NO_LOG': None,
  119. 'removes': None,
  120. 'warn': True,
  121. 'lockfile': None,
  122. 'timeout': None}
  123. def check_command(commandline):
  124. ''' Check provided command '''
  125. arguments = {'chown': 'owner', 'chmod': 'mode', 'chgrp': 'group',
  126. 'ln': 'state=link', 'mkdir': 'state=directory',
  127. 'rmdir': 'state=absent', 'rm': 'state=absent', 'touch': 'state=touch'}
  128. commands = {'git': 'git', 'hg': 'hg', 'curl': 'get_url or uri', 'wget': 'get_url or uri',
  129. 'svn': 'subversion', 'service': 'service',
  130. 'mount': 'mount', 'rpm': 'yum, dnf or zypper', 'yum': 'yum', 'apt-get': 'apt',
  131. 'tar': 'unarchive', 'unzip': 'unarchive', 'sed': 'template or lineinfile',
  132. 'rsync': 'synchronize', 'dnf': 'dnf', 'zypper': 'zypper'}
  133. become = ['sudo', 'su', 'pbrun', 'pfexec', 'runas']
  134. warnings = list()
  135. command = os.path.basename(commandline.split()[0])
  136. # pylint: disable=line-too-long
  137. if command in arguments:
  138. warnings.append("Consider using file module with {0} rather than running {1}".format(arguments[command], command))
  139. if command in commands:
  140. warnings.append("Consider using {0} module rather than running {1}".format(commands[command], command))
  141. if command in become:
  142. warnings.append(
  143. "Consider using 'become', 'become_method', and 'become_user' rather than running {0}".format(command,))
  144. return warnings
  145. # pylint: disable=too-many-statements,too-many-branches,too-many-locals
  146. def main():
  147. ''' Main module function '''
  148. module = AnsibleModule( # noqa: F405
  149. argument_spec=dict(
  150. _uses_shell=dict(type='bool', default=False),
  151. command=dict(required=True),
  152. chdir=dict(),
  153. executable=dict(),
  154. creates=dict(),
  155. removes=dict(),
  156. warn=dict(type='bool', default=True),
  157. lockfile=dict(default='/tmp/delegated_serial_command.lock'),
  158. timeout=dict(type='int', default=30)
  159. )
  160. )
  161. shell = module.params['_uses_shell']
  162. chdir = module.params['chdir']
  163. executable = module.params['executable']
  164. command = module.params['command']
  165. creates = module.params['creates']
  166. removes = module.params['removes']
  167. warn = module.params['warn']
  168. lockfile = module.params['lockfile']
  169. timeout = module.params['timeout']
  170. if command.strip() == '':
  171. module.fail_json(rc=256, msg="no command given")
  172. iterated = 0
  173. lockfd = open(lockfile, 'w+')
  174. while iterated < timeout:
  175. try:
  176. fcntl.flock(lockfd, fcntl.LOCK_EX | fcntl.LOCK_NB)
  177. break
  178. # pylint: disable=invalid-name
  179. except IOError as e:
  180. if e.errno != errno.EAGAIN:
  181. module.fail_json(msg="I/O Error {0}: {1}".format(e.errno, e.strerror))
  182. else:
  183. iterated += 1
  184. time.sleep(0.1)
  185. if chdir:
  186. chdir = os.path.abspath(os.path.expanduser(chdir))
  187. os.chdir(chdir)
  188. if creates:
  189. # do not run the command if the line contains creates=filename
  190. # and the filename already exists. This allows idempotence
  191. # of command executions.
  192. path = os.path.expanduser(creates)
  193. if glob.glob(path):
  194. module.exit_json(
  195. cmd=command,
  196. stdout="skipped, since %s exists" % path,
  197. changed=False,
  198. stderr=False,
  199. rc=0
  200. )
  201. if removes:
  202. # do not run the command if the line contains removes=filename
  203. # and the filename does not exist. This allows idempotence
  204. # of command executions.
  205. path = os.path.expanduser(removes)
  206. if not glob.glob(path):
  207. module.exit_json(
  208. cmd=command,
  209. stdout="skipped, since %s does not exist" % path,
  210. changed=False,
  211. stderr=False,
  212. rc=0
  213. )
  214. warnings = list()
  215. if warn:
  216. warnings = check_command(command)
  217. if not shell:
  218. command = shlex.split(command)
  219. startd = datetime.datetime.now()
  220. # pylint: disable=invalid-name
  221. rc, out, err = module.run_command(command, executable=executable, use_unsafe_shell=shell)
  222. fcntl.flock(lockfd, fcntl.LOCK_UN)
  223. lockfd.close()
  224. endd = datetime.datetime.now()
  225. delta = endd - startd
  226. if out is None:
  227. out = ''
  228. if err is None:
  229. err = ''
  230. module.exit_json(
  231. cmd=command,
  232. stdout=out.rstrip("\r\n"),
  233. stderr=err.rstrip("\r\n"),
  234. rc=rc,
  235. start=str(startd),
  236. end=str(endd),
  237. delta=str(delta),
  238. changed=True,
  239. warnings=warnings,
  240. iterated=iterated
  241. )
  242. # import module snippets
  243. # pylint: disable=wrong-import-position
  244. from ansible.module_utils.basic import * # noqa: F402,F403
  245. main()