2012-03-15 21:53:14 -04:00
|
|
|
#!/usr/bin/python
|
2012-08-02 21:29:10 -04:00
|
|
|
# -*- coding: utf-8 -*-
|
2012-03-15 21:53:14 -04:00
|
|
|
|
|
|
|
# (c) 2012, Michael DeHaan <michael.dehaan@gmail.com>
|
|
|
|
#
|
|
|
|
# This file is part of Ansible
|
|
|
|
#
|
|
|
|
# Ansible is free software: you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU General Public License as published by
|
|
|
|
# the Free Software Foundation, either version 3 of the License, or
|
|
|
|
# (at your option) any later version.
|
|
|
|
#
|
|
|
|
# Ansible is distributed in the hope that it will be useful,
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
# GNU General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU General Public License
|
|
|
|
# along with Ansible. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
import shutil
|
|
|
|
import stat
|
|
|
|
import grp
|
|
|
|
import pwd
|
2012-04-12 10:33:10 -07:00
|
|
|
try:
|
|
|
|
import selinux
|
|
|
|
HAVE_SELINUX=True
|
|
|
|
except ImportError:
|
|
|
|
HAVE_SELINUX=False
|
2012-03-15 21:53:14 -04:00
|
|
|
|
2012-09-19 16:09:26 +02:00
|
|
|
DOCUMENTATION = '''
|
|
|
|
---
|
|
|
|
module: file
|
|
|
|
short_description: Sets attributes of files
|
|
|
|
description:
|
|
|
|
- Sets attributes of files, symlinks, and directories, or removes
|
|
|
|
files/symlinks/directories. Many other modules support the same options as
|
2012-11-22 07:23:10 +01:00
|
|
|
the M(file) module - including M(copy), M(template), and M(assemble).
|
2012-09-19 16:09:26 +02:00
|
|
|
options:
|
2012-10-12 22:08:07 -04:00
|
|
|
path:
|
2012-09-27 21:06:31 -04:00
|
|
|
description:
|
2012-11-21 18:49:30 +01:00
|
|
|
- defines the file being managed, unless when used with C(state=link), and then sets the destination to create a symbolic link to using I(src)
|
2012-09-27 21:06:31 -04:00
|
|
|
required: true
|
|
|
|
default: []
|
2013-06-14 20:45:58 +02:00
|
|
|
aliases: ['dest', 'name']
|
2012-09-27 21:06:31 -04:00
|
|
|
state:
|
|
|
|
description:
|
2012-10-01 09:18:54 +02:00
|
|
|
- If C(directory), all immediate subdirectories will be created if they
|
|
|
|
do not exist. If C(file), the file will NOT be created if it does not
|
|
|
|
exist, see the M(copy) or M(template) module if you want that behavior.
|
2013-06-15 11:14:34 -04:00
|
|
|
If C(link), the symbolic link will be created or changed. Use C(hard)
|
|
|
|
for hardlinks. If C(absent), directories will be recursively deleted,
|
|
|
|
and files or symlinks will be unlinked.
|
2012-09-27 21:06:31 -04:00
|
|
|
required: false
|
|
|
|
default: file
|
2013-06-15 11:14:34 -04:00
|
|
|
choices: [ file, link, directory, hard, absent ]
|
2012-09-27 21:06:31 -04:00
|
|
|
mode:
|
2012-10-01 09:18:54 +02:00
|
|
|
required: false
|
|
|
|
default: null
|
|
|
|
choices: []
|
|
|
|
description:
|
2012-11-21 18:49:30 +01:00
|
|
|
- mode the file or directory should be, such as 0644 as would be fed to I(chmod)
|
2012-10-01 09:18:54 +02:00
|
|
|
owner:
|
|
|
|
required: false
|
|
|
|
default: null
|
|
|
|
choices: []
|
|
|
|
description:
|
|
|
|
- name of the user that should own the file/directory, as would be fed to I(chown)
|
|
|
|
group:
|
|
|
|
required: false
|
|
|
|
default: null
|
|
|
|
choices: []
|
|
|
|
description:
|
|
|
|
- name of the group that should own the file/directory, as would be fed to I(chown)
|
|
|
|
src:
|
|
|
|
required: false
|
|
|
|
default: null
|
|
|
|
choices: []
|
|
|
|
description:
|
|
|
|
- path of the file to link to (applies only to C(state=link)).
|
|
|
|
seuser:
|
|
|
|
required: false
|
|
|
|
default: null
|
|
|
|
choices: []
|
|
|
|
description:
|
|
|
|
- user part of SELinux file context. Will default to system policy, if
|
|
|
|
applicable. If set to C(_default), it will use the C(user) portion of the
|
2012-11-21 18:49:30 +01:00
|
|
|
policy if available
|
2012-10-01 09:18:54 +02:00
|
|
|
serole:
|
|
|
|
required: false
|
|
|
|
default: null
|
|
|
|
choices: []
|
|
|
|
description:
|
|
|
|
- role part of SELinux file context, C(_default) feature works as for I(seuser).
|
|
|
|
setype:
|
|
|
|
required: false
|
|
|
|
default: null
|
|
|
|
choices: []
|
|
|
|
description:
|
|
|
|
- type part of SELinux file context, C(_default) feature works as for I(seuser).
|
|
|
|
selevel:
|
|
|
|
required: false
|
|
|
|
default: "s0"
|
|
|
|
choices: []
|
|
|
|
description:
|
|
|
|
- level part of the SELinux file context. This is the MLS/MCS attribute,
|
|
|
|
sometimes known as the C(range). C(_default) feature works as for
|
|
|
|
I(seuser).
|
|
|
|
context:
|
|
|
|
required: false
|
|
|
|
default: null
|
|
|
|
choices: [ "default" ]
|
2012-09-27 21:06:31 -04:00
|
|
|
description:
|
2012-10-01 09:18:54 +02:00
|
|
|
- accepts only C(default) as value. This will restore a file's SELinux context
|
|
|
|
in the policy. Does nothing if no default value is available.
|
2013-02-09 12:01:11 -05:00
|
|
|
recurse:
|
|
|
|
required: false
|
2013-03-12 13:18:12 +01:00
|
|
|
default: "no"
|
2013-02-09 12:01:11 -05:00
|
|
|
choices: [ "yes", "no" ]
|
2013-02-17 18:48:17 +01:00
|
|
|
version_added: "1.1"
|
2013-02-09 12:01:11 -05:00
|
|
|
description:
|
|
|
|
- recursively set the specified file attributes (applies only to state=directory)
|
2012-09-19 16:09:26 +02:00
|
|
|
notes:
|
|
|
|
- See also M(copy), M(template), M(assemble)
|
|
|
|
requirements: [ ]
|
2012-10-01 09:18:54 +02:00
|
|
|
author: Michael DeHaan
|
2012-09-19 16:09:26 +02:00
|
|
|
'''
|
|
|
|
|
2013-06-14 11:53:43 +02:00
|
|
|
EXAMPLES = '''
|
|
|
|
- file: path=/etc/foo.conf owner=foo group=foo mode=0644
|
|
|
|
- file: src=/file/to/link/to dest=/path/to/symlink owner=foo group=foo state=link
|
|
|
|
'''
|
|
|
|
|
2013-06-15 11:14:34 -04:00
|
|
|
def dolink(src, path, state, module):
|
|
|
|
try:
|
|
|
|
if state == 'hard':
|
|
|
|
os.link(src,path)
|
|
|
|
else:
|
|
|
|
os.symlink(src, path)
|
|
|
|
except OSError, e:
|
|
|
|
module.fail_json(path=path, msg='Error while linking: %s' % str(e))
|
|
|
|
|
2012-07-30 21:50:32 -04:00
|
|
|
def main():
|
|
|
|
|
2012-08-11 12:35:58 -04:00
|
|
|
# FIXME: pass this around, should not use global
|
2012-07-30 21:50:32 -04:00
|
|
|
global module
|
2012-08-11 12:35:58 -04:00
|
|
|
|
2012-08-01 19:42:31 -04:00
|
|
|
module = AnsibleModule(
|
2012-07-30 21:50:32 -04:00
|
|
|
argument_spec = dict(
|
2013-06-15 11:14:34 -04:00
|
|
|
state = dict(choices=['file','directory','link','hard','absent'], default='file'),
|
2012-07-30 21:50:32 -04:00
|
|
|
path = dict(aliases=['dest', 'name'], required=True),
|
2013-02-26 20:57:36 -05:00
|
|
|
recurse = dict(default='no', type='bool'),
|
2013-05-30 12:53:24 +02:00
|
|
|
force = dict(required=False,default=False,type='bool'),
|
2013-03-26 23:12:56 -04:00
|
|
|
diff_peek = dict(default=None),
|
|
|
|
validate = dict(required=False, default=None),
|
2012-10-20 22:51:36 -04:00
|
|
|
),
|
2013-02-03 19:46:25 -05:00
|
|
|
add_file_common_args=True,
|
|
|
|
supports_check_mode=True
|
2012-07-30 21:50:32 -04:00
|
|
|
)
|
|
|
|
|
|
|
|
params = module.params
|
|
|
|
state = params['state']
|
2013-05-30 12:53:24 +02:00
|
|
|
force = params['force']
|
2013-01-07 16:30:29 +10:00
|
|
|
params['path'] = path = os.path.expanduser(params['path'])
|
2012-10-12 20:07:05 -04:00
|
|
|
|
2013-02-25 23:32:52 +01:00
|
|
|
# short-circuit for diff_peek
|
|
|
|
if params.get('diff_peek', None) is not None:
|
|
|
|
appears_binary = False
|
|
|
|
try:
|
|
|
|
f = open(path)
|
|
|
|
b = f.read(8192)
|
|
|
|
f.close()
|
|
|
|
if b.find("\x00") != -1:
|
|
|
|
appears_binary = True
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
module.exit_json(path=path, changed=False, appears_binary=appears_binary)
|
|
|
|
|
2012-10-12 20:07:05 -04:00
|
|
|
# source is both the source of a symlink or an informational passing of the src for a template module
|
|
|
|
# or copy module, even if this module never uses it, it is needed to key off some things
|
|
|
|
|
2012-10-20 22:51:36 -04:00
|
|
|
src = params.get('src', None)
|
2012-07-30 21:50:32 -04:00
|
|
|
if src:
|
|
|
|
src = os.path.expanduser(src)
|
|
|
|
|
2012-10-18 20:08:57 +08:00
|
|
|
if src is not None and os.path.isdir(path) and state != "link":
|
2012-10-12 20:07:05 -04:00
|
|
|
params['path'] = path = os.path.join(path, os.path.basename(src))
|
|
|
|
|
2012-10-20 22:51:36 -04:00
|
|
|
file_args = module.load_file_common_arguments(params)
|
2012-07-30 21:50:32 -04:00
|
|
|
|
2013-06-15 11:14:34 -04:00
|
|
|
if state in ['link','hard'] and (src is None or path is None):
|
|
|
|
module.fail_json(msg='src and dest are required for creating links')
|
2012-07-30 21:50:32 -04:00
|
|
|
elif path is None:
|
2012-10-20 22:51:36 -04:00
|
|
|
module.fail_json(msg='path is required')
|
2012-07-30 21:50:32 -04:00
|
|
|
|
|
|
|
changed = False
|
|
|
|
|
|
|
|
prev_state = 'absent'
|
2012-10-12 20:07:05 -04:00
|
|
|
|
2012-07-30 21:50:32 -04:00
|
|
|
if os.path.lexists(path):
|
|
|
|
if os.path.islink(path):
|
|
|
|
prev_state = 'link'
|
2012-08-21 19:36:48 +02:00
|
|
|
elif os.path.isdir(path):
|
2012-07-30 21:50:32 -04:00
|
|
|
prev_state = 'directory'
|
2012-08-21 19:36:48 +02:00
|
|
|
else:
|
|
|
|
prev_state = 'file'
|
2012-03-15 21:53:14 -04:00
|
|
|
|
2012-07-30 21:50:32 -04:00
|
|
|
if prev_state != 'absent' and state == 'absent':
|
|
|
|
try:
|
|
|
|
if prev_state == 'directory':
|
|
|
|
if os.path.islink(path):
|
2013-05-11 15:04:42 -04:00
|
|
|
if module.check_mode:
|
|
|
|
module.exit_json(changed=True)
|
2012-07-30 21:50:32 -04:00
|
|
|
os.unlink(path)
|
|
|
|
else:
|
2012-10-20 22:51:36 -04:00
|
|
|
try:
|
2013-05-11 15:04:42 -04:00
|
|
|
if module.check_mode:
|
|
|
|
module.exit_json(changed=True)
|
2012-10-20 22:51:36 -04:00
|
|
|
shutil.rmtree(path, ignore_errors=False)
|
|
|
|
except:
|
|
|
|
module.exit_json(msg="rmtree failed")
|
2012-07-30 21:50:32 -04:00
|
|
|
else:
|
2013-04-25 21:33:47 -04:00
|
|
|
if module.check_mode:
|
|
|
|
module.exit_json(changed=True)
|
2012-07-30 21:50:32 -04:00
|
|
|
os.unlink(path)
|
|
|
|
except Exception, e:
|
2012-10-20 22:51:36 -04:00
|
|
|
module.fail_json(path=path, msg=str(e))
|
|
|
|
module.exit_json(path=path, changed=True)
|
2012-03-15 21:53:14 -04:00
|
|
|
|
2012-10-16 18:14:42 -04:00
|
|
|
if prev_state != 'absent' and prev_state != state:
|
2013-05-30 12:53:24 +02:00
|
|
|
if force and prev_state == 'file' and state == 'link':
|
|
|
|
pass
|
|
|
|
else:
|
|
|
|
module.fail_json(path=path, msg='refusing to convert between %s and %s for %s' % (prev_state, state, src))
|
2012-03-15 21:53:14 -04:00
|
|
|
|
2012-07-30 21:50:32 -04:00
|
|
|
if prev_state == 'absent' and state == 'absent':
|
2012-10-20 22:51:36 -04:00
|
|
|
module.exit_json(path=path, changed=False)
|
2012-03-15 21:53:14 -04:00
|
|
|
|
2012-07-30 21:50:32 -04:00
|
|
|
if state == 'file':
|
2012-08-06 20:07:02 -04:00
|
|
|
|
2012-09-18 21:02:16 -04:00
|
|
|
if prev_state != 'file':
|
2013-06-15 11:14:34 -04:00
|
|
|
module.fail_json(path=path, msg='file (%s) does not exist, use copy or template module to create' % path)
|
2012-03-15 21:53:14 -04:00
|
|
|
|
2012-10-20 22:51:36 -04:00
|
|
|
changed = module.set_file_attributes_if_different(file_args, changed)
|
|
|
|
module.exit_json(path=path, changed=changed)
|
2012-03-15 21:53:14 -04:00
|
|
|
|
2012-07-30 21:50:32 -04:00
|
|
|
elif state == 'directory':
|
|
|
|
if prev_state == 'absent':
|
2013-04-25 21:33:47 -04:00
|
|
|
if module.check_mode:
|
|
|
|
module.exit_json(changed=True)
|
2012-07-30 21:50:32 -04:00
|
|
|
os.makedirs(path)
|
|
|
|
changed = True
|
2012-08-06 20:07:02 -04:00
|
|
|
|
2012-10-20 22:51:36 -04:00
|
|
|
changed = module.set_directory_attributes_if_different(file_args, changed)
|
2013-02-23 19:59:52 +01:00
|
|
|
recurse = params['recurse']
|
2013-02-09 12:01:11 -05:00
|
|
|
if recurse:
|
|
|
|
for root,dirs,files in os.walk( file_args['path'] ):
|
|
|
|
for dir in dirs:
|
|
|
|
dirname=os.path.join(root,dir)
|
|
|
|
tmp_file_args = file_args.copy()
|
|
|
|
tmp_file_args['path']=dirname
|
|
|
|
changed = module.set_directory_attributes_if_different(tmp_file_args, changed)
|
|
|
|
for file in files:
|
|
|
|
filename=os.path.join(root,file)
|
|
|
|
tmp_file_args = file_args.copy()
|
|
|
|
tmp_file_args['path']=filename
|
|
|
|
changed = module.set_file_attributes_if_different(tmp_file_args, changed)
|
2012-10-20 22:51:36 -04:00
|
|
|
module.exit_json(path=path, changed=changed)
|
2012-03-15 21:53:14 -04:00
|
|
|
|
2013-06-15 11:14:34 -04:00
|
|
|
elif state in ['link','hard']:
|
2012-08-06 20:07:02 -04:00
|
|
|
|
2012-07-30 21:50:32 -04:00
|
|
|
if os.path.isabs(src):
|
|
|
|
abs_src = src
|
|
|
|
else:
|
2012-08-09 21:46:03 -04:00
|
|
|
module.fail_json(msg="absolute paths are required")
|
2013-05-30 13:05:10 +02:00
|
|
|
|
|
|
|
if not os.path.exists(abs_src) and not force:
|
2012-10-20 22:51:36 -04:00
|
|
|
module.fail_json(path=path, src=src, msg='src file does not exist')
|
2012-08-06 20:07:02 -04:00
|
|
|
|
2012-07-30 21:50:32 -04:00
|
|
|
if prev_state == 'absent':
|
2013-04-25 21:33:47 -04:00
|
|
|
if module.check_mode:
|
|
|
|
module.exit_json(changed=True)
|
2013-06-15 11:14:34 -04:00
|
|
|
dolink(src, path, state, module)
|
2012-07-30 21:50:32 -04:00
|
|
|
changed = True
|
|
|
|
elif prev_state == 'link':
|
|
|
|
old_src = os.readlink(path)
|
|
|
|
if not os.path.isabs(old_src):
|
|
|
|
old_src = os.path.join(os.path.dirname(path), old_src)
|
|
|
|
if old_src != src:
|
2013-04-25 21:33:47 -04:00
|
|
|
if module.check_mode:
|
|
|
|
module.exit_json(changed=True)
|
2012-07-30 21:50:32 -04:00
|
|
|
os.unlink(path)
|
2013-06-15 11:14:34 -04:00
|
|
|
dolink(src, path, state, module)
|
2012-09-18 21:02:16 -04:00
|
|
|
changed = True
|
2013-06-15 11:14:34 -04:00
|
|
|
elif prev_state == 'file':
|
2013-05-30 12:53:24 +02:00
|
|
|
if not force:
|
|
|
|
module.fail_json(dest=path, src=src, msg='Cannot link, file exists at destination')
|
|
|
|
else:
|
|
|
|
if module.check_mode:
|
|
|
|
module.exit_json(changed=True)
|
|
|
|
os.unlink(path)
|
|
|
|
dolink(src, path, state, module)
|
|
|
|
changed = True
|
2012-07-30 21:50:32 -04:00
|
|
|
else:
|
2012-10-20 22:51:36 -04:00
|
|
|
module.fail_json(dest=path, src=src, msg='unexpected position reached')
|
2012-07-30 21:50:32 -04:00
|
|
|
|
|
|
|
# set modes owners and context as needed
|
2012-10-20 22:51:36 -04:00
|
|
|
|
|
|
|
file_args = module.load_file_common_arguments(module.params)
|
2013-04-25 21:33:47 -04:00
|
|
|
changed = module.set_context_if_different(path, file_args['secontext'], changed)
|
|
|
|
changed = module.set_owner_if_different(path, file_args['owner'], changed)
|
|
|
|
changed = module.set_group_if_different(path, file_args['group'], changed)
|
|
|
|
changed = module.set_mode_if_different(path, file_args['mode'], changed)
|
2012-04-01 16:10:23 -07:00
|
|
|
|
2012-07-30 21:50:32 -04:00
|
|
|
module.exit_json(dest=path, src=src, changed=changed)
|
2012-04-01 16:10:23 -07:00
|
|
|
|
2012-10-20 22:51:36 -04:00
|
|
|
module.fail_json(path=path, msg='unexpected position reached')
|
2012-04-01 16:10:23 -07:00
|
|
|
|
2012-07-30 21:50:32 -04:00
|
|
|
# this is magic, see lib/ansible/module_common.py
|
|
|
|
#<<INCLUDE_ANSIBLE_MODULE_COMMON>>
|
2012-08-01 00:21:36 -04:00
|
|
|
main()
|
2012-03-15 21:53:14 -04:00
|
|
|
|