delegated_serial_command.py 8.6 KB

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