mirror of https://github.com/ansible/ansible.git
You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
585 lines
15 KiB
YAML
585 lines
15 KiB
YAML
# Test code for the command and shell modules.
|
|
|
|
# Copyright: (c) 2014, Richard Isaacson <richard.c.isaacson@gmail.com>
|
|
# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
|
|
|
|
- name: use command with unsupported executable arg
|
|
command: ls /dev/null
|
|
args:
|
|
executable: /bogus
|
|
register: executable
|
|
|
|
- name: assert executable warning was reported
|
|
assert:
|
|
that:
|
|
- executable.stdout == '/dev/null'
|
|
- executable.warnings | length() == 1
|
|
- "'no longer supported' in executable.warnings[0]"
|
|
|
|
- name: use command with no command
|
|
command:
|
|
args:
|
|
chdir: /
|
|
register: no_command
|
|
ignore_errors: true
|
|
|
|
- name: assert executable fails with no command
|
|
assert:
|
|
that:
|
|
- no_command is failed
|
|
- no_command.msg == 'no command given'
|
|
- no_command.rc == 256
|
|
|
|
- name: use argv
|
|
command:
|
|
argv:
|
|
- echo
|
|
- testing
|
|
register: argv_command
|
|
ignore_errors: true
|
|
|
|
- name: assert executable works with argv
|
|
assert:
|
|
that:
|
|
- "argv_command.stdout == 'testing'"
|
|
|
|
- name: use argv and command string
|
|
command: echo testing
|
|
args:
|
|
argv:
|
|
- echo
|
|
- testing
|
|
register: argv_and_string_command
|
|
ignore_errors: true
|
|
|
|
- name: assert executable fails with both argv and command string
|
|
assert:
|
|
that:
|
|
- argv_and_string_command is failed
|
|
- argv_and_string_command.msg == 'only command or argv can be given, not both'
|
|
- argv_and_string_command.rc == 256
|
|
|
|
- set_fact:
|
|
remote_tmp_dir_test: "{{ remote_tmp_dir }}/test_command_shell"
|
|
|
|
- name: make sure our testing sub-directory does not exist
|
|
file:
|
|
path: "{{ remote_tmp_dir_test }}"
|
|
state: absent
|
|
|
|
- name: create our testing sub-directory
|
|
file:
|
|
path: "{{ remote_tmp_dir_test }}"
|
|
state: directory
|
|
|
|
- name: prep our test script
|
|
copy:
|
|
src: test.sh
|
|
dest: "{{ remote_tmp_dir_test }}"
|
|
mode: '0755'
|
|
|
|
- name: prep our test script
|
|
copy:
|
|
src: create_afile.sh
|
|
dest: "{{ remote_tmp_dir_test }}"
|
|
mode: '0755'
|
|
|
|
- name: prep our test script
|
|
copy:
|
|
src: remove_afile.sh
|
|
dest: "{{ remote_tmp_dir_test }}"
|
|
mode: '0755'
|
|
|
|
- name: locate bash
|
|
shell: which bash
|
|
register: bash
|
|
|
|
##
|
|
## command
|
|
##
|
|
|
|
- name: execute the test.sh script via command
|
|
command: "{{ remote_tmp_dir_test }}/test.sh"
|
|
register: command_result0
|
|
|
|
- name: assert that the script executed correctly
|
|
assert:
|
|
that:
|
|
- command_result0.rc == 0
|
|
- command_result0.stderr == ''
|
|
- command_result0.stdout == 'win'
|
|
|
|
# executable
|
|
|
|
# FIXME doesn't have the expected stdout.
|
|
|
|
#- name: execute the test.sh script with executable via command
|
|
# command: "{{remote_tmp_dir_test }}/test.sh executable={{ bash.stdout }}"
|
|
# register: command_result1
|
|
#
|
|
#- name: assert that the script executed correctly with command
|
|
# assert:
|
|
# that:
|
|
# - "command_result1.rc == 0"
|
|
# - "command_result1.stderr == ''"
|
|
# - "command_result1.stdout == 'win'"
|
|
|
|
# chdir
|
|
|
|
- name: execute the test.sh script with chdir via command
|
|
command: ./test.sh
|
|
args:
|
|
chdir: "{{ remote_tmp_dir_test }}"
|
|
register: command_result2
|
|
|
|
- name: Check invalid chdir
|
|
command: echo
|
|
args:
|
|
chdir: "{{ remote_tmp_dir }}/nope"
|
|
ignore_errors: yes
|
|
register: chdir_invalid
|
|
|
|
- name: assert that the script executed correctly with chdir
|
|
assert:
|
|
that:
|
|
- command_result2.rc == 0
|
|
- command_result2.stderr == ''
|
|
- command_result2.stdout == 'win'
|
|
- chdir_invalid is failed
|
|
- chdir_invalid.msg is search('Unable to change directory')
|
|
|
|
# creates
|
|
|
|
- name: verify that afile.txt is absent
|
|
file:
|
|
path: "{{ remote_tmp_dir_test }}/afile.txt"
|
|
state: absent
|
|
|
|
- name: create afile.txt with create_afile.sh via command (check mode)
|
|
command: "{{ remote_tmp_dir_test }}/create_afile.sh {{remote_tmp_dir_test }}/afile.txt"
|
|
args:
|
|
creates: "{{ remote_tmp_dir_test }}/afile.txt"
|
|
register: check_mode_result
|
|
check_mode: yes
|
|
|
|
- assert:
|
|
that:
|
|
- check_mode_result.changed
|
|
- "'skipped' not in check_mode_result"
|
|
|
|
- name: verify that afile.txt still does not exist
|
|
stat:
|
|
path: "{{remote_tmp_dir_test}}/afile.txt"
|
|
register: stat_result
|
|
failed_when: stat_result.stat.exists
|
|
|
|
- name: create afile.txt with create_afile.sh via command
|
|
command: "{{ remote_tmp_dir_test }}/create_afile.sh {{remote_tmp_dir_test }}/afile.txt"
|
|
args:
|
|
creates: "{{ remote_tmp_dir_test }}/afile.txt"
|
|
|
|
- name: verify that afile.txt is present
|
|
file:
|
|
path: "{{ remote_tmp_dir_test }}/afile.txt"
|
|
state: file
|
|
|
|
- name: re-run previous command using creates with globbing (check mode)
|
|
command: "{{ remote_tmp_dir_test }}/create_afile.sh {{ remote_tmp_dir_test }}/afile.txt"
|
|
args:
|
|
creates: "{{ remote_tmp_dir_test }}/afile.*"
|
|
register: check_mode_result
|
|
check_mode: yes
|
|
|
|
- assert:
|
|
that:
|
|
- not check_mode_result.changed
|
|
- "'skipped' not in check_mode_result"
|
|
|
|
- name: re-run previous command using creates with globbing
|
|
command: "{{ remote_tmp_dir_test }}/create_afile.sh {{ remote_tmp_dir_test }}/afile.txt"
|
|
args:
|
|
creates: "{{ remote_tmp_dir_test }}/afile.*"
|
|
register: command_result3
|
|
|
|
- name: assert that creates with globbing is working
|
|
assert:
|
|
that:
|
|
- command_result3 is not changed
|
|
|
|
# removes
|
|
|
|
- name: remove afile.txt with remote_afile.sh via command (check mode)
|
|
command: "{{ remote_tmp_dir_test }}/remove_afile.sh {{ remote_tmp_dir_test }}/afile.txt"
|
|
args:
|
|
removes: "{{ remote_tmp_dir_test }}/afile.txt"
|
|
register: check_mode_result
|
|
check_mode: yes
|
|
|
|
- assert:
|
|
that:
|
|
- check_mode_result.changed
|
|
- "'skipped' not in check_mode_result"
|
|
|
|
- name: verify that afile.txt still exists
|
|
stat:
|
|
path: "{{remote_tmp_dir_test}}/afile.txt"
|
|
register: stat_result
|
|
failed_when: not stat_result.stat.exists
|
|
|
|
- name: remove afile.txt with remote_afile.sh via command
|
|
command: "{{ remote_tmp_dir_test }}/remove_afile.sh {{ remote_tmp_dir_test }}/afile.txt"
|
|
args:
|
|
removes: "{{ remote_tmp_dir_test }}/afile.txt"
|
|
|
|
- name: verify that afile.txt is absent
|
|
file: path={{remote_tmp_dir_test}}/afile.txt state=absent
|
|
|
|
- name: re-run previous command using removes with globbing (check mode)
|
|
command: "{{ remote_tmp_dir_test }}/remove_afile.sh {{ remote_tmp_dir_test }}/afile.txt"
|
|
args:
|
|
removes: "{{ remote_tmp_dir_test }}/afile.*"
|
|
register: check_mode_result
|
|
check_mode: yes
|
|
|
|
- assert:
|
|
that:
|
|
- not check_mode_result.changed
|
|
- "'skipped' not in check_mode_result"
|
|
|
|
- name: re-run previous command using removes with globbing
|
|
command: "{{ remote_tmp_dir_test }}/remove_afile.sh {{ remote_tmp_dir_test }}/afile.txt"
|
|
args:
|
|
removes: "{{ remote_tmp_dir_test }}/afile.*"
|
|
register: command_result4
|
|
|
|
- name: assert that removes with globbing is working
|
|
assert:
|
|
that:
|
|
- command_result4.changed != True
|
|
|
|
- name: pass stdin to cat via command
|
|
command: cat
|
|
args:
|
|
stdin: 'foobar'
|
|
register: command_result5
|
|
|
|
- name: assert that stdin is passed
|
|
assert:
|
|
that:
|
|
- command_result5.stdout == 'foobar'
|
|
|
|
- name: send to stdin literal multiline block
|
|
command: "{{ ansible_python.executable }} -c 'import hashlib, sys; print(hashlib.sha1((sys.stdin.buffer if hasattr(sys.stdin, \"buffer\") else sys.stdin).read()).hexdigest())'"
|
|
args:
|
|
stdin: |-
|
|
this is the first line
|
|
this is the second line
|
|
|
|
this line is after an empty line
|
|
this line is the last line
|
|
register: command_result6
|
|
|
|
- name: assert the multiline input was passed correctly
|
|
assert:
|
|
that:
|
|
- "command_result6.stdout == '9cd0697c6a9ff6689f0afb9136fa62e0b3fee903'"
|
|
|
|
- name: check default var expansion
|
|
command: /bin/sh -c 'echo "\$TEST"'
|
|
environment:
|
|
TEST: z
|
|
register: command_result7
|
|
|
|
- name: assert vars were expanded
|
|
assert:
|
|
that:
|
|
- command_result7.stdout == '\\z'
|
|
|
|
- name: check disabled var expansion
|
|
command: /bin/sh -c 'echo "\$TEST"'
|
|
args:
|
|
expand_argument_vars: false
|
|
environment:
|
|
TEST: z
|
|
register: command_result8
|
|
|
|
- name: assert vars were not expanded
|
|
assert:
|
|
that:
|
|
- command_result8.stdout == '$TEST'
|
|
|
|
##
|
|
## shell
|
|
##
|
|
|
|
- name: Execute the test.sh script
|
|
shell: "{{ remote_tmp_dir_test }}/test.sh"
|
|
register: shell_result0
|
|
|
|
- name: Assert that the script executed correctly
|
|
assert:
|
|
that:
|
|
- shell_result0 is changed
|
|
- shell_result0.cmd == '{{ remote_tmp_dir_test }}/test.sh'
|
|
- shell_result0.rc == 0
|
|
- shell_result0.stderr == ''
|
|
- shell_result0.stdout == 'win'
|
|
|
|
# executable
|
|
|
|
# FIXME doesn't pass the expected stdout
|
|
|
|
#- name: execute the test.sh script
|
|
# shell: "{{remote_tmp_dir_test }}/test.sh executable={{ bash.stdout }}"
|
|
# register: shell_result1
|
|
#
|
|
#- name: assert that the shell executed correctly
|
|
# assert:
|
|
# that:
|
|
# - "shell_result1.rc == 0"
|
|
# - "shell_result1.stderr == ''"
|
|
# - "shell_result1.stdout == 'win'"
|
|
|
|
# chdir
|
|
|
|
- name: Execute the test.sh script with chdir
|
|
shell: ./test.sh
|
|
args:
|
|
chdir: "{{ remote_tmp_dir_test }}"
|
|
register: shell_result2
|
|
|
|
- name: Assert that the shell executed correctly with chdir
|
|
assert:
|
|
that:
|
|
- shell_result2 is changed
|
|
- shell_result2.cmd == './test.sh'
|
|
- shell_result2.rc == 0
|
|
- shell_result2.stderr == ''
|
|
- shell_result2.stdout == 'win'
|
|
|
|
# creates
|
|
|
|
- name: Verify that afile.txt is absent
|
|
file:
|
|
path: "{{ remote_tmp_dir_test }}/afile.txt"
|
|
state: absent
|
|
|
|
- name: Execute the test.sh script with chdir
|
|
shell: "{{ remote_tmp_dir_test }}/test.sh > {{ remote_tmp_dir_test }}/afile.txt"
|
|
args:
|
|
chdir: "{{ remote_tmp_dir_test }}"
|
|
creates: "{{ remote_tmp_dir_test }}/afile.txt"
|
|
|
|
- name: Verify that afile.txt is present
|
|
file:
|
|
path: "{{ remote_tmp_dir_test }}/afile.txt"
|
|
state: file
|
|
|
|
# multiline
|
|
|
|
- name: Remove test file previously created
|
|
file:
|
|
path: "{{ remote_tmp_dir_test }}/afile.txt"
|
|
state: absent
|
|
|
|
- name: Execute a shell command using a literal multiline block
|
|
args:
|
|
executable: "{{ bash.stdout }}"
|
|
shell: |
|
|
echo this is a \
|
|
"multiline echo" \
|
|
"with a new line
|
|
in quotes" \
|
|
| {{ ansible_python.executable }} -c 'import hashlib, sys; print(hashlib.sha1((sys.stdin.buffer if hasattr(sys.stdin, "buffer") else sys.stdin).read()).hexdigest())'
|
|
echo "this is a second line"
|
|
register: shell_result5
|
|
|
|
- name: Assert the multiline shell command ran as expected
|
|
assert:
|
|
that:
|
|
- shell_result5 is changed
|
|
- shell_result5.rc == 0
|
|
- shell_result5.cmd == 'echo this is a "multiline echo" "with a new line\nin quotes" | ' + ansible_python.executable + ' -c \'import hashlib, sys; print(hashlib.sha1((sys.stdin.buffer if hasattr(sys.stdin, "buffer") else sys.stdin).read()).hexdigest())\'\necho "this is a second line"\n'
|
|
- shell_result5.stdout == '5575bb6b71c9558db0b6fbbf2f19909eeb4e3b98\nthis is a second line'
|
|
|
|
- name: Execute a shell command using a literal multiline block with arguments in it
|
|
shell: |
|
|
executable="{{ bash.stdout }}"
|
|
creates={{ remote_tmp_dir_test }}/afile.txt
|
|
echo "test"
|
|
register: shell_result6
|
|
|
|
- name: Assert the multiline shell command with arguments in it run as expected
|
|
assert:
|
|
that:
|
|
- shell_result6 is changed
|
|
- shell_result6.rc == 0
|
|
- shell_result6.cmd == 'echo "test"\n'
|
|
- shell_result6.stdout == 'test'
|
|
|
|
- name: Execute a shell command using a multiline block where whitespaces matter
|
|
shell: |
|
|
cat <<EOF
|
|
One
|
|
Two
|
|
Three
|
|
EOF
|
|
register: shell_result7
|
|
|
|
- name: Assert the multiline shell command outputs with whitespaces preserved
|
|
assert:
|
|
that:
|
|
- shell_result7 is changed
|
|
- shell_result7.rc == 0
|
|
- shell_result7.cmd == 'cat <<EOF\nOne\n Two\n Three\nEOF\n'
|
|
- shell_result7.stdout == 'One\n Two\n Three'
|
|
|
|
- name: execute a shell command with no trailing newline to stdin
|
|
shell: cat > {{remote_tmp_dir_test }}/afile.txt
|
|
args:
|
|
stdin: test
|
|
stdin_add_newline: no
|
|
|
|
- name: make sure content matches expected
|
|
copy:
|
|
dest: "{{remote_tmp_dir_test }}/afile.txt"
|
|
content: test
|
|
register: shell_result7
|
|
failed_when:
|
|
- shell_result7 is failed or
|
|
shell_result7 is changed
|
|
|
|
- name: execute a shell command with trailing newline to stdin
|
|
shell: cat > {{remote_tmp_dir_test }}/afile.txt
|
|
args:
|
|
stdin: test
|
|
stdin_add_newline: yes
|
|
|
|
- name: make sure content matches expected
|
|
copy:
|
|
dest: "{{remote_tmp_dir_test }}/afile.txt"
|
|
content: |
|
|
test
|
|
register: shell_result8
|
|
failed_when:
|
|
- shell_result8 is failed or
|
|
shell_result8 is changed
|
|
|
|
- name: execute a shell command with trailing newline to stdin, default
|
|
shell: cat > {{remote_tmp_dir_test }}/afile.txt
|
|
args:
|
|
stdin: test
|
|
|
|
- name: make sure content matches expected
|
|
copy:
|
|
dest: "{{remote_tmp_dir_test }}/afile.txt"
|
|
content: |
|
|
test
|
|
register: shell_result9
|
|
failed_when:
|
|
- shell_result9 is failed or
|
|
shell_result9 is changed
|
|
|
|
- name: remove the previously created file
|
|
file:
|
|
path: "{{ remote_tmp_dir_test }}/afile.txt"
|
|
state: absent
|
|
|
|
- name: test check mode skip message
|
|
command:
|
|
cmd: "true"
|
|
check_mode: yes
|
|
register: result
|
|
|
|
- name: assert check message exists
|
|
assert:
|
|
that:
|
|
- "'Command would have run if not in check mode' in result.msg"
|
|
- result.skipped
|
|
- not result.changed
|
|
|
|
- name: test check mode creates/removes message
|
|
command:
|
|
cmd: "true"
|
|
creates: yes
|
|
check_mode: yes
|
|
register: result
|
|
|
|
- name: assert check message exists
|
|
assert:
|
|
that:
|
|
- "'Command would have run if not in check mode' in result.msg"
|
|
- "'skipped' not in result"
|
|
- result.changed
|
|
|
|
- name: command symlink handling
|
|
block:
|
|
- name: Create target folders
|
|
file:
|
|
path: '{{remote_tmp_dir}}/www_root/site'
|
|
state: directory
|
|
|
|
- name: Create symlink
|
|
file:
|
|
path: '{{remote_tmp_dir}}/www'
|
|
state: link
|
|
src: '{{remote_tmp_dir}}/www_root'
|
|
|
|
- name: check parent using chdir
|
|
shell: dirname "$PWD"
|
|
args:
|
|
chdir: '{{remote_tmp_dir}}/www/site'
|
|
register: parent_dir_chdir
|
|
|
|
- name: check parent using cd
|
|
shell: cd "{{remote_tmp_dir}}/www/site" && dirname "$PWD"
|
|
register: parent_dir_cd
|
|
|
|
- name: check expected outputs
|
|
assert:
|
|
that:
|
|
- parent_dir_chdir.stdout != parent_dir_cd.stdout
|
|
# These tests use endswith, to get around /private/tmp on macos
|
|
- 'parent_dir_cd.stdout.endswith(remote_tmp_dir ~ "/www")'
|
|
- 'parent_dir_chdir.stdout.endswith(remote_tmp_dir ~ "/www_root")'
|
|
|
|
- name: Set print error command for Python 3
|
|
set_fact:
|
|
print_error_command: print(msg, file=sys.stderr)
|
|
|
|
- name: run command with strip
|
|
command: '{{ ansible_python_interpreter }} -c "import sys; msg=''hello \n \r''; print(msg); {{ print_error_command }}"'
|
|
register: command_strip
|
|
|
|
- name: run command without strip
|
|
command: '{{ ansible_python_interpreter }} -c "import sys; msg=''hello \n \r''; print(msg); {{ print_error_command }}"'
|
|
args:
|
|
strip_empty_ends: no
|
|
register: command_no_strip
|
|
|
|
- name: Verify strip behavior worked as expected
|
|
assert:
|
|
that:
|
|
- command_strip.stdout == 'hello \n '
|
|
- command_strip.stderr == 'hello \n '
|
|
- command_no_strip.stdout== 'hello \n \r\n'
|
|
- command_no_strip.stderr == 'hello \n \r\n'
|
|
|
|
- name: run shell with expand_argument_vars
|
|
shell: echo 'hi'
|
|
args:
|
|
expand_argument_vars: false
|
|
register: shell_expand_failure
|
|
ignore_errors: true
|
|
|
|
- name: assert shell with expand_arguments_vars failed
|
|
assert:
|
|
that:
|
|
- shell_expand_failure is failed
|
|
- "shell_expand_failure.msg == 'Unsupported parameters for (shell) module: expand_argument_vars'"
|
|
|
|
- name: Run command that backgrounds, to ensure no hang
|
|
shell: '{{ role_path }}/scripts/yoink.sh &'
|
|
delegate_to: localhost
|
|
timeout: 5
|