fixed fetch traversal from slurp (#68720)
* fixed fetch traversal from slurp * ignore slurp result for dest * fixed naming when source is relative * fixed bug in local connection plugin * added tests with fake slurp * moved existing role tests into runme.sh * normalized on action excepts * moved dest transform down to when needed * added is_subpath check * fixed bug in local connection fixes #67793 CVE-2019-3828
This commit is contained in:
parent
087be1da50
commit
ba87c225cd
12 changed files with 121 additions and 34 deletions
2
changelogs/fragments/fetch_no_slurp.yml
Normal file
2
changelogs/fragments/fetch_no_slurp.yml
Normal file
|
@ -0,0 +1,2 @@
|
|||
bugfixes:
|
||||
- In fetch action, avoid using slurp return to set up dest, also ensure no dir traversal CVE-2019-3828.
|
|
@ -20,14 +20,14 @@ __metaclass__ = type
|
|||
import os
|
||||
import base64
|
||||
|
||||
from ansible.errors import AnsibleError
|
||||
from ansible.errors import AnsibleActionFail, AnsibleActionSkip
|
||||
from ansible.module_utils._text import to_bytes
|
||||
from ansible.module_utils.six import string_types
|
||||
from ansible.module_utils.parsing.convert_bool import boolean
|
||||
from ansible.plugins.action import ActionBase
|
||||
from ansible.utils.display import Display
|
||||
from ansible.utils.hashing import checksum, checksum_s, md5, secure_hash
|
||||
from ansible.utils.path import makedirs_safe
|
||||
from ansible.utils.path import makedirs_safe, is_subpath
|
||||
|
||||
display = Display()
|
||||
|
||||
|
@ -44,29 +44,27 @@ class ActionModule(ActionBase):
|
|||
|
||||
try:
|
||||
if self._play_context.check_mode:
|
||||
result['skipped'] = True
|
||||
result['msg'] = 'check mode not (yet) supported for this module'
|
||||
return result
|
||||
raise AnsibleActionSkip('check mode not (yet) supported for this module')
|
||||
|
||||
source = self._task.args.get('src', None)
|
||||
dest = self._task.args.get('dest', None)
|
||||
original_dest = dest = self._task.args.get('dest', None)
|
||||
flat = boolean(self._task.args.get('flat'), strict=False)
|
||||
fail_on_missing = boolean(self._task.args.get('fail_on_missing', True), strict=False)
|
||||
validate_checksum = boolean(self._task.args.get('validate_checksum', True), strict=False)
|
||||
|
||||
msg = ''
|
||||
# validate source and dest are strings FIXME: use basic.py and module specs
|
||||
if not isinstance(source, string_types):
|
||||
result['msg'] = "Invalid type supplied for source option, it must be a string"
|
||||
msg = "Invalid type supplied for source option, it must be a string"
|
||||
|
||||
if not isinstance(dest, string_types):
|
||||
result['msg'] = "Invalid type supplied for dest option, it must be a string"
|
||||
msg = "Invalid type supplied for dest option, it must be a string"
|
||||
|
||||
if source is None or dest is None:
|
||||
result['msg'] = "src and dest are required"
|
||||
msg = "src and dest are required"
|
||||
|
||||
if result.get('msg'):
|
||||
result['failed'] = True
|
||||
return result
|
||||
if msg:
|
||||
raise AnsibleActionFail(msg)
|
||||
|
||||
source = self._connection._shell.join_path(source)
|
||||
source = self._remote_expand_user(source)
|
||||
|
@ -94,12 +92,6 @@ class ActionModule(ActionBase):
|
|||
remote_data = base64.b64decode(slurpres['content'])
|
||||
if remote_data is not None:
|
||||
remote_checksum = checksum_s(remote_data)
|
||||
# the source path may have been expanded on the
|
||||
# target system, so we compare it here and use the
|
||||
# expanded version if it's different
|
||||
remote_source = slurpres.get('source')
|
||||
if remote_source and remote_source != source:
|
||||
source = remote_source
|
||||
|
||||
# calculate the destination name
|
||||
if os.path.sep not in self._connection._shell.join_path('a', ''):
|
||||
|
@ -108,13 +100,14 @@ class ActionModule(ActionBase):
|
|||
else:
|
||||
source_local = source
|
||||
|
||||
dest = os.path.expanduser(dest)
|
||||
# ensure we only use file name, avoid relative paths
|
||||
if not is_subpath(dest, original_dest):
|
||||
# TODO: ? dest = os.path.expanduser(dest.replace(('../','')))
|
||||
raise AnsibleActionFail("Detected directory traversal, expected to be contained in '%s' but got '%s'" % (original_dest, dest))
|
||||
|
||||
if flat:
|
||||
if os.path.isdir(to_bytes(dest, errors='surrogate_or_strict')) and not dest.endswith(os.sep):
|
||||
result['msg'] = "dest is an existing directory, use a trailing slash if you want to fetch src into that directory"
|
||||
result['file'] = dest
|
||||
result['failed'] = True
|
||||
return result
|
||||
raise AnsibleActionFail("dest is an existing directory, use a trailing slash if you want to fetch src into that directory")
|
||||
if dest.endswith(os.sep):
|
||||
# if the path ends with "/", we'll use the source filename as the
|
||||
# destination filename
|
||||
|
@ -131,8 +124,6 @@ class ActionModule(ActionBase):
|
|||
target_name = self._play_context.remote_addr
|
||||
dest = "%s/%s/%s" % (self._loader.path_dwim(dest), target_name, source_local)
|
||||
|
||||
dest = dest.replace("//", "/")
|
||||
|
||||
if remote_checksum in ('0', '1', '2', '3', '4', '5'):
|
||||
result['changed'] = False
|
||||
result['file'] = source
|
||||
|
@ -160,6 +151,8 @@ class ActionModule(ActionBase):
|
|||
result['msg'] += ", not transferring, ignored"
|
||||
return result
|
||||
|
||||
dest = os.path.normpath(dest)
|
||||
|
||||
# calculate checksum for the local file
|
||||
local_checksum = checksum(dest)
|
||||
|
||||
|
@ -176,7 +169,7 @@ class ActionModule(ActionBase):
|
|||
f.write(remote_data)
|
||||
f.close()
|
||||
except (IOError, OSError) as e:
|
||||
raise AnsibleError("Failed to fetch the file: %s" % e)
|
||||
raise AnsibleActionFail("Failed to fetch the file: %s" % e)
|
||||
new_checksum = secure_hash(dest)
|
||||
# For backwards compatibility. We'll return None on FIPS enabled systems
|
||||
try:
|
||||
|
|
|
@ -29,6 +29,7 @@ from ansible.module_utils.six import text_type, binary_type
|
|||
from ansible.module_utils._text import to_bytes, to_native, to_text
|
||||
from ansible.plugins.connection import ConnectionBase
|
||||
from ansible.utils.display import Display
|
||||
from ansible.utils.path import unfrackpath
|
||||
|
||||
display = Display()
|
||||
|
||||
|
@ -130,18 +131,13 @@ class Connection(ConnectionBase):
|
|||
display.debug("done with local.exec_command()")
|
||||
return (p.returncode, stdout, stderr)
|
||||
|
||||
def _ensure_abs(self, path):
|
||||
if not os.path.isabs(path) and self.cwd is not None:
|
||||
path = os.path.normpath(os.path.join(self.cwd, path))
|
||||
return path
|
||||
|
||||
def put_file(self, in_path, out_path):
|
||||
''' transfer a file from local to local '''
|
||||
|
||||
super(Connection, self).put_file(in_path, out_path)
|
||||
|
||||
in_path = self._ensure_abs(in_path)
|
||||
out_path = self._ensure_abs(out_path)
|
||||
in_path = unfrackpath(in_path, basedir=self.cwd)
|
||||
out_path = unfrackpath(out_path, basedir=self.cwd)
|
||||
|
||||
display.vvv(u"PUT {0} TO {1}".format(in_path, out_path), host=self._play_context.remote_addr)
|
||||
if not os.path.exists(to_bytes(in_path, errors='surrogate_or_strict')):
|
||||
|
|
|
@ -132,3 +132,26 @@ def cleanup_tmp_file(path, warn=False):
|
|||
display.display(u'Unable to remove temporary file {0}'.format(to_text(e)))
|
||||
except Exception:
|
||||
pass
|
||||
|
||||
|
||||
def is_subpath(child, parent):
|
||||
"""
|
||||
Compares paths to check if one is contained in the other
|
||||
:arg: child: Path to test
|
||||
:arg parent; Path to test against
|
||||
"""
|
||||
test = False
|
||||
|
||||
abs_child = unfrackpath(child, follow=False)
|
||||
abs_parent = unfrackpath(parent, follow=False)
|
||||
|
||||
c = abs_child.split(os.path.sep)
|
||||
p = abs_parent.split(os.path.sep)
|
||||
|
||||
try:
|
||||
test = c[:len(p)] == p
|
||||
except IndexError:
|
||||
# child is shorter than parent so cannot be subpath
|
||||
pass
|
||||
|
||||
return test
|
||||
|
|
|
@ -1 +1,2 @@
|
|||
shippable/posix/group2
|
||||
needs/target/setup_remote_tmp_dir
|
||||
|
|
|
@ -0,0 +1,26 @@
|
|||
- name: ensure that 'fake slurp' does not poison fetch source
|
||||
hosts: localhost
|
||||
gather_facts: False
|
||||
tasks:
|
||||
- name: fetch with relative source path
|
||||
fetch: src=../injection/here.txt dest={{output_dir}}
|
||||
become: true
|
||||
register: islurp
|
||||
|
||||
- name: fetch with normal source path
|
||||
fetch: src=here.txt dest={{output_dir}}
|
||||
become: true
|
||||
register: islurp2
|
||||
|
||||
- name: ensure all is good in hollywood
|
||||
assert:
|
||||
that:
|
||||
- "'..' not in islurp['dest']"
|
||||
- "'..' not in islurp2['dest']"
|
||||
- "'foo' not in islurp['dest']"
|
||||
- "'foo' not in islurp2['dest']"
|
||||
|
||||
- name: try to trip dest anyways
|
||||
fetch: src=../injection/here.txt dest={{output_dir}}
|
||||
become: true
|
||||
register: islurp2
|
1
test/integration/targets/fetch/injection/here.txt
Normal file
1
test/integration/targets/fetch/injection/here.txt
Normal file
|
@ -0,0 +1 @@
|
|||
this is a test file
|
29
test/integration/targets/fetch/injection/library/slurp.py
Normal file
29
test/integration/targets/fetch/injection/library/slurp.py
Normal file
|
@ -0,0 +1,29 @@
|
|||
#!/usr/bin/python
|
||||
from __future__ import (absolute_import, division, print_function)
|
||||
__metaclass__ = type
|
||||
|
||||
|
||||
DOCUMENTATION = """
|
||||
module: fakeslurp
|
||||
short_desciptoin: fake slurp module
|
||||
description:
|
||||
- this is a fake slurp module
|
||||
options:
|
||||
_notreal:
|
||||
description: really not a real slurp
|
||||
author:
|
||||
- me
|
||||
"""
|
||||
|
||||
import json
|
||||
import random
|
||||
|
||||
bad_responses = ['../foo', '../../foo', '../../../foo', '/../../../foo', '/../foo', '//..//foo', '..//..//foo']
|
||||
|
||||
|
||||
def main():
|
||||
print(json.dumps(dict(changed=False, content='', encoding='base64', source=random.choice(bad_responses))))
|
||||
|
||||
|
||||
if __name__ == '__main__':
|
||||
main()
|
|
@ -1,3 +1,2 @@
|
|||
dependencies:
|
||||
- prepare_tests
|
||||
- setup_remote_tmp_dir
|
5
test/integration/targets/fetch/run_fetch_tests.yml
Normal file
5
test/integration/targets/fetch/run_fetch_tests.yml
Normal file
|
@ -0,0 +1,5 @@
|
|||
- name: call fetch_tests role
|
||||
hosts: testhost
|
||||
gather_facts: false
|
||||
roles:
|
||||
- fetch_tests
|
12
test/integration/targets/fetch/runme.sh
Executable file
12
test/integration/targets/fetch/runme.sh
Executable file
|
@ -0,0 +1,12 @@
|
|||
#!/usr/bin/env bash
|
||||
|
||||
set -eux
|
||||
|
||||
# setup required roles
|
||||
ln -s ../../setup_remote_tmp_dir roles/setup_remote_tmp_dir
|
||||
|
||||
# run old type role tests
|
||||
ansible-playbook -i ../../inventory run_fetch_tests.yml -e "output_dir=${OUTPUT_DIR}" -v "$@"
|
||||
|
||||
# run tests to avoid path injection from slurp when fetch uses become
|
||||
ansible-playbook -i ../../inventory injection/avoid_slurp_return.yml -e "output_dir=${OUTPUT_DIR}" -v "$@"
|
Loading…
Add table
Reference in a new issue