mirror of
				https://github.com/ansible-collections/community.general.git
				synced 2025-10-25 21:44:00 -07:00 
			
		
		
		
	
		
			
				
	
	
		
			202 lines
		
	
	
	
		
			5.7 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			202 lines
		
	
	
	
		
			5.7 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| # Copyright (c) 2018, Ansible Project
 | |
| # Simplified BSD License (see licenses/simplified_bsd.txt or https://opensource.org/licenses/BSD-2-Clause)
 | |
| 
 | |
| from __future__ import (absolute_import, division, print_function)
 | |
| __metaclass__ = type
 | |
| 
 | |
| import errno
 | |
| import os
 | |
| import stat
 | |
| import re
 | |
| import pwd
 | |
| import grp
 | |
| import time
 | |
| import shutil
 | |
| import traceback
 | |
| import fcntl
 | |
| import sys
 | |
| 
 | |
| from contextlib import contextmanager
 | |
| from ansible.module_utils._text import to_bytes, to_native, to_text
 | |
| from ansible.module_utils.six import b, binary_type
 | |
| 
 | |
| try:
 | |
|     import selinux
 | |
|     HAVE_SELINUX = True
 | |
| except ImportError:
 | |
|     HAVE_SELINUX = False
 | |
| 
 | |
| 
 | |
| FILE_ATTRIBUTES = {
 | |
|     'A': 'noatime',
 | |
|     'a': 'append',
 | |
|     'c': 'compressed',
 | |
|     'C': 'nocow',
 | |
|     'd': 'nodump',
 | |
|     'D': 'dirsync',
 | |
|     'e': 'extents',
 | |
|     'E': 'encrypted',
 | |
|     'h': 'blocksize',
 | |
|     'i': 'immutable',
 | |
|     'I': 'indexed',
 | |
|     'j': 'journalled',
 | |
|     'N': 'inline',
 | |
|     's': 'zero',
 | |
|     'S': 'synchronous',
 | |
|     't': 'notail',
 | |
|     'T': 'blockroot',
 | |
|     'u': 'undelete',
 | |
|     'X': 'compressedraw',
 | |
|     'Z': 'compresseddirty',
 | |
| }
 | |
| 
 | |
| 
 | |
| # Used for parsing symbolic file perms
 | |
| MODE_OPERATOR_RE = re.compile(r'[+=-]')
 | |
| USERS_RE = re.compile(r'[^ugo]')
 | |
| PERMS_RE = re.compile(r'[^rwxXstugo]')
 | |
| 
 | |
| 
 | |
| _PERM_BITS = 0o7777          # file mode permission bits
 | |
| _EXEC_PERM_BITS = 0o0111     # execute permission bits
 | |
| _DEFAULT_PERM = 0o0666       # default file permission bits
 | |
| 
 | |
| 
 | |
| def is_executable(path):
 | |
|     # This function's signature needs to be repeated
 | |
|     # as the first line of its docstring.
 | |
|     # This method is reused by the basic module,
 | |
|     # the repetion helps the basic module's html documentation come out right.
 | |
|     # http://www.sphinx-doc.org/en/master/usage/extensions/autodoc.html#confval-autodoc_docstring_signature
 | |
|     '''is_executable(path)
 | |
| 
 | |
|     is the given path executable?
 | |
| 
 | |
|     :arg path: The path of the file to check.
 | |
| 
 | |
|     Limitations:
 | |
| 
 | |
|     * Does not account for FSACLs.
 | |
|     * Most times we really want to know "Can the current user execute this
 | |
|       file".  This function does not tell us that, only if any execute bit is set.
 | |
|     '''
 | |
|     # These are all bitfields so first bitwise-or all the permissions we're
 | |
|     # looking for, then bitwise-and with the file's mode to determine if any
 | |
|     # execute bits are set.
 | |
|     return ((stat.S_IXUSR | stat.S_IXGRP | stat.S_IXOTH) & os.stat(path)[stat.ST_MODE])
 | |
| 
 | |
| 
 | |
| def format_attributes(attributes):
 | |
|     attribute_list = [FILE_ATTRIBUTES.get(attr) for attr in attributes if attr in FILE_ATTRIBUTES]
 | |
|     return attribute_list
 | |
| 
 | |
| 
 | |
| def get_flags_from_attributes(attributes):
 | |
|     flags = [key for key, attr in FILE_ATTRIBUTES.items() if attr in attributes]
 | |
|     return ''.join(flags)
 | |
| 
 | |
| 
 | |
| def get_file_arg_spec():
 | |
|     arg_spec = dict(
 | |
|         mode=dict(type='raw'),
 | |
|         owner=dict(),
 | |
|         group=dict(),
 | |
|         seuser=dict(),
 | |
|         serole=dict(),
 | |
|         selevel=dict(),
 | |
|         setype=dict(),
 | |
|         attributes=dict(aliases=['attr']),
 | |
|     )
 | |
|     return arg_spec
 | |
| 
 | |
| 
 | |
| class LockTimeout(Exception):
 | |
|     pass
 | |
| 
 | |
| 
 | |
| class FileLock:
 | |
|     '''
 | |
|     Currently FileLock is implemented via fcntl.flock on a lock file, however this
 | |
|     behaviour may change in the future. Avoid mixing lock types fcntl.flock,
 | |
|     fcntl.lockf and module_utils.common.file.FileLock as it will certainly cause
 | |
|     unwanted and/or unexpected behaviour
 | |
|     '''
 | |
|     def __init__(self):
 | |
|         self.lockfd = None
 | |
| 
 | |
|     @contextmanager
 | |
|     def lock_file(self, path, tmpdir, lock_timeout=None):
 | |
|         '''
 | |
|         Context for lock acquisition
 | |
|         '''
 | |
|         try:
 | |
|             self.set_lock(path, tmpdir, lock_timeout)
 | |
|             yield
 | |
|         finally:
 | |
|             self.unlock()
 | |
| 
 | |
|     def set_lock(self, path, tmpdir, lock_timeout=None):
 | |
|         '''
 | |
|         Create a lock file based on path with flock to prevent other processes
 | |
|         using given path.
 | |
|         Please note that currently file locking only works when it's executed by
 | |
|         the same user, I.E single user scenarios
 | |
| 
 | |
|         :kw path: Path (file) to lock
 | |
|         :kw tmpdir: Path where to place the temporary .lock file
 | |
|         :kw lock_timeout:
 | |
|             Wait n seconds for lock acquisition, fail if timeout is reached.
 | |
|             0 = Do not wait, fail if lock cannot be acquired immediately,
 | |
|             Default is None, wait indefinitely until lock is released.
 | |
|         :returns: True
 | |
|         '''
 | |
|         lock_path = os.path.join(tmpdir, 'ansible-{0}.lock'.format(os.path.basename(path)))
 | |
|         l_wait = 0.1
 | |
|         r_exception = IOError
 | |
|         if sys.version_info[0] == 3:
 | |
|             r_exception = BlockingIOError
 | |
| 
 | |
|         self.lockfd = open(lock_path, 'w')
 | |
| 
 | |
|         if lock_timeout <= 0:
 | |
|             fcntl.flock(self.lockfd, fcntl.LOCK_EX | fcntl.LOCK_NB)
 | |
|             os.chmod(lock_path, stat.S_IWRITE | stat.S_IREAD)
 | |
|             return True
 | |
| 
 | |
|         if lock_timeout:
 | |
|             e_secs = 0
 | |
|             while e_secs < lock_timeout:
 | |
|                 try:
 | |
|                     fcntl.flock(self.lockfd, fcntl.LOCK_EX | fcntl.LOCK_NB)
 | |
|                     os.chmod(lock_path, stat.S_IWRITE | stat.S_IREAD)
 | |
|                     return True
 | |
|                 except r_exception:
 | |
|                     time.sleep(l_wait)
 | |
|                     e_secs += l_wait
 | |
|                     continue
 | |
| 
 | |
|             self.lockfd.close()
 | |
|             raise LockTimeout('{0} sec'.format(lock_timeout))
 | |
| 
 | |
|         fcntl.flock(self.lockfd, fcntl.LOCK_EX)
 | |
|         os.chmod(lock_path, stat.S_IWRITE | stat.S_IREAD)
 | |
| 
 | |
|         return True
 | |
| 
 | |
|     def unlock(self):
 | |
|         '''
 | |
|         Make sure lock file is available for everyone and Unlock the file descriptor
 | |
|         locked by set_lock
 | |
| 
 | |
|         :returns: True
 | |
|         '''
 | |
|         if not self.lockfd:
 | |
|             return True
 | |
| 
 | |
|         try:
 | |
|             fcntl.flock(self.lockfd, fcntl.LOCK_UN)
 | |
|             self.lockfd.close()
 | |
|         except ValueError:  # file wasn't opened, let context manager fail gracefully
 | |
|             pass
 | |
| 
 | |
|         return True
 |