Skip to content

Commit

Permalink
SSH connection plugin creates ControlPersist socket files in a secure…
Browse files Browse the repository at this point in the history
… directory

Files were being created in /tmp, but will now be created in $HOME/.ansible/cp/
Addresses CVE-2013-4259: ansible uses a socket with predictable filename in /tmp
  • Loading branch information
jimi-c committed Aug 21, 2013
1 parent ae98a02 commit 6bf5d19
Show file tree
Hide file tree
Showing 2 changed files with 33 additions and 5 deletions.
14 changes: 13 additions & 1 deletion lib/ansible/runner/connection_plugins/ssh.py
Expand Up @@ -43,6 +43,7 @@ def __init__(self, runner, host, port, user, password, private_key_file, *args,
self.user = user
self.password = password
self.private_key_file = private_key_file
self.cp_dir = utils.prepare_writeable_dir('$HOME/.ansible/cp',mode=0700)
self.HASHED_KEY_MAGIC = "|1|"

def connect(self):
Expand All @@ -57,7 +58,18 @@ def connect(self):
else:
self.common_args += ["-o", "ControlMaster=auto",
"-o", "ControlPersist=60s",
"-o", "ControlPath=/tmp/ansible-ssh-%h-%p-%r"]
"-o", "ControlPath=%s/ansible-ssh-%%h-%%p-%%r" % self.cp_dir]

cp_in_use = False
cp_path_set = False
for arg in self.common_args:
if arg.find("ControlPersist") != -1:
cp_in_use = True
if arg.find("ControlPath") != -1:
cp_path_set = True

if cp_in_use and not cp_path_set:
self.common_args += ["-o", "ControlPath=%s/ansible-ssh-%%h-%%p-%%r" % self.cp_dir]

if not C.HOST_KEY_CHECKING:
self.common_args += ["-o", "StrictHostKeyChecking=no"]
Expand Down
24 changes: 20 additions & 4 deletions lib/ansible/utils/__init__.py
Expand Up @@ -193,18 +193,34 @@ def is_executable(path):
or stat.S_IXGRP & os.stat(path)[stat.ST_MODE]
or stat.S_IXOTH & os.stat(path)[stat.ST_MODE])

def prepare_writeable_dir(tree):
def unfrackpath(path):
'''
returns a path that is free of symlinks, environment
variables, relative path traversals and symbols (~)
example:
'$HOME/../../var/mail' becomes '/var/spool/mail'
'''
return os.path.normpath(os.path.realpath(os.path.expandvars(os.path.expanduser(path))))

def prepare_writeable_dir(tree,mode=0777):
''' make sure a directory exists and is writeable '''

if tree != '/':
tree = os.path.realpath(os.path.expanduser(tree))
# modify the mode to ensure the owner at least
# has read/write access to this directory
mode |= 0700

# make sure the tree path is always expanded
# and normalized and free of symlinks
tree = unfrackpath(tree)

if not os.path.exists(tree):
try:
os.makedirs(tree)
os.makedirs(tree, mode)
except (IOError, OSError), e:
exit("Could not make dir %s: %s" % (tree, e))
if not os.access(tree, os.W_OK):
exit("Cannot write to path %s" % tree)
return tree

def path_dwim(basedir, given):
'''
Expand Down

0 comments on commit 6bf5d19

Please sign in to comment.