summaryrefslogtreecommitdiffstats
path: root/lib/ansible/modules/expect.py
diff options
context:
space:
mode:
Diffstat (limited to 'lib/ansible/modules/expect.py')
-rw-r--r--lib/ansible/modules/expect.py258
1 files changed, 258 insertions, 0 deletions
diff --git a/lib/ansible/modules/expect.py b/lib/ansible/modules/expect.py
new file mode 100644
index 0000000..99ffe9f
--- /dev/null
+++ b/lib/ansible/modules/expect.py
@@ -0,0 +1,258 @@
+# -*- coding: utf-8 -*-
+
+# (c) 2015, Matt Martz <matt@sivel.net>
+# 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: expect
+version_added: '2.0'
+short_description: Executes a command and responds to prompts
+description:
+ - The C(expect) module executes a command and responds to prompts.
+ - The given command will be executed on all selected nodes. It will not be
+ processed through the shell, so variables like C($HOME) and operations
+ like C("<"), C(">"), C("|"), and C("&") will not work.
+options:
+ command:
+ description:
+ - The command module takes command to run.
+ required: true
+ type: str
+ creates:
+ type: path
+ description:
+ - A filename, when it already exists, this step will B(not) be run.
+ removes:
+ type: path
+ description:
+ - A filename, when it does not exist, this step will B(not) be run.
+ chdir:
+ type: path
+ description:
+ - Change into this directory before running the command.
+ responses:
+ type: dict
+ description:
+ - Mapping of expected string/regex and string to respond with. If the
+ response is a list, successive matches return successive
+ responses. List functionality is new in 2.1.
+ required: true
+ timeout:
+ type: int
+ description:
+ - Amount of time in seconds to wait for the expected strings. Use
+ C(null) to disable timeout.
+ default: 30
+ echo:
+ description:
+ - Whether or not to echo out your response strings.
+ default: false
+ type: bool
+requirements:
+ - python >= 2.6
+ - pexpect >= 3.3
+extends_documentation_fragment: action_common_attributes
+attributes:
+ check_mode:
+ support: none
+ diff_mode:
+ support: none
+ platform:
+ support: full
+ platforms: posix
+notes:
+ - If you want to run a command through the shell (say you are using C(<),
+ C(>), C(|), and so on), you must specify a shell in the command such as
+ C(/bin/bash -c "/path/to/something | grep else").
+ - The question, or key, under I(responses) is a python regex match. Case
+ insensitive searches are indicated with a prefix of C(?i).
+ - The C(pexpect) library used by this module operates with a search window
+ of 2000 bytes, and does not use a multiline regex match. To perform a
+ start of line bound match, use a pattern like ``(?m)^pattern``
+ - By default, if a question is encountered multiple times, its string
+ response will be repeated. If you need different responses for successive
+ question matches, instead of a string response, use a list of strings as
+ the response. The list functionality is new in 2.1.
+ - The M(ansible.builtin.expect) module is designed for simple scenarios.
+ For more complex needs, consider the use of expect code with the M(ansible.builtin.shell)
+ or M(ansible.builtin.script) modules. (An example is part of the M(ansible.builtin.shell) module documentation).
+seealso:
+- module: ansible.builtin.script
+- module: ansible.builtin.shell
+author: "Matt Martz (@sivel)"
+'''
+
+EXAMPLES = r'''
+- name: Case insensitive password string match
+ ansible.builtin.expect:
+ command: passwd username
+ responses:
+ (?i)password: "MySekretPa$$word"
+ # you don't want to show passwords in your logs
+ no_log: true
+
+- name: Generic question with multiple different responses
+ ansible.builtin.expect:
+ command: /path/to/custom/command
+ responses:
+ Question:
+ - response1
+ - response2
+ - response3
+'''
+
+import datetime
+import os
+import traceback
+
+PEXPECT_IMP_ERR = None
+try:
+ import pexpect
+ HAS_PEXPECT = True
+except ImportError:
+ PEXPECT_IMP_ERR = traceback.format_exc()
+ HAS_PEXPECT = False
+
+from ansible.module_utils.basic import AnsibleModule, missing_required_lib
+from ansible.module_utils._text import to_bytes, to_native, to_text
+
+
+def response_closure(module, question, responses):
+ resp_gen = (b'%s\n' % to_bytes(r).rstrip(b'\n') for r in responses)
+
+ def wrapped(info):
+ try:
+ return next(resp_gen)
+ except StopIteration:
+ module.fail_json(msg="No remaining responses for '%s', "
+ "output was '%s'" %
+ (question,
+ info['child_result_list'][-1]))
+
+ return wrapped
+
+
+def main():
+ module = AnsibleModule(
+ argument_spec=dict(
+ command=dict(required=True),
+ chdir=dict(type='path'),
+ creates=dict(type='path'),
+ removes=dict(type='path'),
+ responses=dict(type='dict', required=True),
+ timeout=dict(type='int', default=30),
+ echo=dict(type='bool', default=False),
+ )
+ )
+
+ if not HAS_PEXPECT:
+ module.fail_json(msg=missing_required_lib("pexpect"),
+ exception=PEXPECT_IMP_ERR)
+
+ chdir = module.params['chdir']
+ args = module.params['command']
+ creates = module.params['creates']
+ removes = module.params['removes']
+ responses = module.params['responses']
+ timeout = module.params['timeout']
+ echo = module.params['echo']
+
+ events = dict()
+ for key, value in responses.items():
+ if isinstance(value, list):
+ response = response_closure(module, key, value)
+ else:
+ response = b'%s\n' % to_bytes(value).rstrip(b'\n')
+
+ events[to_bytes(key)] = response
+
+ if args.strip() == '':
+ module.fail_json(rc=256, msg="no command given")
+
+ if chdir:
+ chdir = os.path.abspath(chdir)
+ os.chdir(chdir)
+
+ if creates:
+ # do not run the command if the line contains creates=filename
+ # and the filename already exists. This allows idempotence
+ # of command executions.
+ if os.path.exists(creates):
+ module.exit_json(
+ cmd=args,
+ stdout="skipped, since %s exists" % creates,
+ changed=False,
+ rc=0
+ )
+
+ if removes:
+ # do not run the command if the line contains removes=filename
+ # and the filename does not exist. This allows idempotence
+ # of command executions.
+ if not os.path.exists(removes):
+ module.exit_json(
+ cmd=args,
+ stdout="skipped, since %s does not exist" % removes,
+ changed=False,
+ rc=0
+ )
+
+ startd = datetime.datetime.now()
+
+ try:
+ try:
+ # Prefer pexpect.run from pexpect>=4
+ b_out, rc = pexpect.run(args, timeout=timeout, withexitstatus=True,
+ events=events, cwd=chdir, echo=echo,
+ encoding=None)
+ except TypeError:
+ # Use pexpect._run in pexpect>=3.3,<4
+ # pexpect.run doesn't support `echo`
+ # pexpect.runu doesn't support encoding=None
+ b_out, rc = pexpect._run(args, timeout=timeout, withexitstatus=True,
+ events=events, extra_args=None, logfile=None,
+ cwd=chdir, env=None, _spawn=pexpect.spawn,
+ echo=echo)
+
+ except (TypeError, AttributeError) as e:
+ # This should catch all insufficient versions of pexpect
+ # We deem them insufficient for their lack of ability to specify
+ # to not echo responses via the run/runu functions, which would
+ # potentially leak sensitive information
+ module.fail_json(msg='Insufficient version of pexpect installed '
+ '(%s), this module requires pexpect>=3.3. '
+ 'Error was %s' % (pexpect.__version__, to_native(e)))
+ except pexpect.ExceptionPexpect as e:
+ module.fail_json(msg='%s' % to_native(e), exception=traceback.format_exc())
+
+ endd = datetime.datetime.now()
+ delta = endd - startd
+
+ if b_out is None:
+ b_out = b''
+
+ result = dict(
+ cmd=args,
+ stdout=to_native(b_out).rstrip('\r\n'),
+ rc=rc,
+ start=str(startd),
+ end=str(endd),
+ delta=str(delta),
+ changed=True,
+ )
+
+ if rc is None:
+ module.fail_json(msg='command exceeded timeout', **result)
+ elif rc != 0:
+ module.fail_json(msg='non-zero return code', **result)
+
+ module.exit_json(**result)
+
+
+if __name__ == '__main__':
+ main()