Sign Up
Log In
Log In
or
Sign Up
Places
All Projects
Status Monitor
Collapse sidebar
SUSE:SLE-12-SP1:GA
salt.3514
0034-Fix-git_pillar-race-condition.patch
Overview
Repositories
Revisions
Requests
Users
Attributes
Meta
File 0034-Fix-git_pillar-race-condition.patch of Package salt.3514
From 17bd6cf1edbcab0e343bc8fe0382756e1fd6c2a0 Mon Sep 17 00:00:00 2001 From: Erik Johnson <palehose@gmail.com> Date: Fri, 11 Mar 2016 15:05:57 -0600 Subject: [PATCH 34/35] Fix git_pillar race condition - Strip whitespace when splitting - Add GitLockError exception class - salt.utils.gitfs: rewrite locking code This does a few things: 1. Introduces the concept of a checkout lock, to prevent concurrent "_pillar" master funcs from trying to checkout the repo at the same time. 2. Refrains from checking out unless the SHA has changed. 3. Cleans up some usage of the GitProvider subclass' "url" attribute when "id" should be used. - salt.runners.cache: Add ability to clear checkout locks - Pass through the lock_type This is necessary for "salt-run fileserver.clear_lock" to work - salt.fileserver: Add ability to clear checkout locks - Fix duplicate output - Use remote_ref instead of local_ref to see if checkout is necessary --- salt/exceptions.py | 33 +++ salt/fileserver/__init__.py | 10 +- salt/fileserver/gitfs.py | 4 +- salt/runners/cache.py | 31 ++- salt/runners/fileserver.py | 12 +- salt/utils/__init__.py | 12 ++ salt/utils/gitfs.py | 499 +++++++++++++++++++++++++++++++++----------- 7 files changed, 463 insertions(+), 138 deletions(-) diff --git a/salt/exceptions.py b/salt/exceptions.py index 67bf323255ee..ed52f8c3622b 100644 --- a/salt/exceptions.py +++ b/salt/exceptions.py @@ -98,6 +98,39 @@ class FileserverConfigError(SaltException): ''' +class FileLockError(SaltException): + ''' + Used when an error occurs obtaining a file lock + ''' + def __init__(self, msg, time_start=None, *args, **kwargs): + super(FileLockError, self).__init__(msg, *args, **kwargs) + if time_start is None: + log.warning( + 'time_start should be provided when raising a FileLockError. ' + 'Defaulting to current time as a fallback, but this may ' + 'result in an inaccurate timeout.' + ) + self.time_start = time.time() + else: + self.time_start = time_start + + +class GitLockError(SaltException): + ''' + Raised when an uncaught error occurs in the midst of obtaining an + update/checkout lock in salt.utils.gitfs. + + NOTE: While this uses the errno param similar to an OSError, this exception + class is *not* as subclass of OSError. This is done intentionally, so that + this exception class can be caught in a try/except without being caught as + an OSError. + ''' + def __init__(self, errno, strerror, *args, **kwargs): + super(GitLockError, self).__init__(strerror, *args, **kwargs) + self.errno = errno + self.strerror = strerror + + class SaltInvocationError(SaltException, TypeError): ''' Used when the wrong number of arguments are sent to modules or invalid diff --git a/salt/fileserver/__init__.py b/salt/fileserver/__init__.py index c40e512d940c..8ff6f223a5eb 100644 --- a/salt/fileserver/__init__.py +++ b/salt/fileserver/__init__.py @@ -272,7 +272,7 @@ def is_file_ignored(opts, fname): return False -def clear_lock(clear_func, lock_type, remote=None): +def clear_lock(clear_func, role, remote=None, lock_type='update'): ''' Function to allow non-fileserver functions to clear update locks @@ -282,7 +282,7 @@ def clear_lock(clear_func, lock_type, remote=None): lists, one containing messages describing successfully cleared locks, and one containing messages describing errors encountered. - lock_type + role What type of lock is being cleared (gitfs, git_pillar, etc.). Used solely for logging purposes. @@ -290,14 +290,16 @@ def clear_lock(clear_func, lock_type, remote=None): Optional string which should be used in ``func`` to pattern match so that a subset of remotes can be targeted. + lock_type : update + Which type of lock to clear Returns the return data from ``clear_func``. ''' - msg = 'Clearing update lock for {0} remotes'.format(lock_type) + msg = 'Clearing {0} lock for {1} remotes'.format(lock_type, role) if remote: msg += ' matching {0}'.format(remote) log.debug(msg) - return clear_func(remote=remote) + return clear_func(remote=remote, lock_type=lock_type) class Fileserver(object): diff --git a/salt/fileserver/gitfs.py b/salt/fileserver/gitfs.py index fc92964334e5..8f74e92c8649 100644 --- a/salt/fileserver/gitfs.py +++ b/salt/fileserver/gitfs.py @@ -93,13 +93,13 @@ def clear_cache(): return gitfs.clear_cache() -def clear_lock(remote=None): +def clear_lock(remote=None, lock_type='update'): ''' Clear update.lk ''' gitfs = salt.utils.gitfs.GitFS(__opts__) gitfs.init_remotes(__opts__['gitfs_remotes'], PER_REMOTE_OVERRIDES) - return gitfs.clear_lock(remote=remote) + return gitfs.clear_lock(remote=remote, lock_type=lock_type) def lock(remote=None): diff --git a/salt/runners/cache.py b/salt/runners/cache.py index 674a85e9e75f..b96489773b8d 100644 --- a/salt/runners/cache.py +++ b/salt/runners/cache.py @@ -238,7 +238,7 @@ def clear_all(tgt=None, expr_form='glob'): clear_mine_flag=True) -def clear_git_lock(role, remote=None): +def clear_git_lock(role, remote=None, **kwargs): ''' .. versionadded:: 2015.8.2 @@ -261,12 +261,23 @@ def clear_git_lock(role, remote=None): have their lock cleared. For example, a ``remote`` value of **github** will remove the lock from all github.com remotes. + type : update,checkout + The types of lock to clear. Can be ``update``, ``checkout``, or both of + et (either comma-separated or as a Python list). + + .. versionadded:: 2015.8.9 + CLI Example: .. code-block:: bash salt-run cache.clear_git_lock git_pillar ''' + kwargs = salt.utils.clean_kwargs(**kwargs) + type_ = salt.utils.split_input(kwargs.pop('type', ['update', 'checkout'])) + if kwargs: + salt.utils.invalid_kwargs(kwargs) + if role == 'gitfs': git_objects = [salt.utils.gitfs.GitFS(__opts__)] git_objects[0].init_remotes(__opts__['gitfs_remotes'], @@ -315,11 +326,15 @@ def clear_git_lock(role, remote=None): ret = {} for obj in git_objects: - cleared, errors = _clear_lock(obj.clear_lock, role, remote) - if cleared: - ret.setdefault('cleared', []).extend(cleared) - if errors: - ret.setdefault('errors', []).extend(errors) + for lock_type in type_: + cleared, errors = _clear_lock(obj.clear_lock, + role, + remote=remote, + lock_type=lock_type) + if cleared: + ret.setdefault('cleared', []).extend(cleared) + if errors: + ret.setdefault('errors', []).extend(errors) if not ret: - ret = 'No locks were removed' - salt.output.display_output(ret, 'nested', opts=__opts__) + return 'No locks were removed' + return ret diff --git a/salt/runners/fileserver.py b/salt/runners/fileserver.py index c6efe0221a3c..7ec3d5e3e2a9 100644 --- a/salt/runners/fileserver.py +++ b/salt/runners/fileserver.py @@ -293,8 +293,8 @@ def clear_cache(backend=None): if errors: ret['errors'] = errors if not ret: - ret = 'No cache was cleared' - salt.output.display_output(ret, 'nested', opts=__opts__) + return 'No cache was cleared' + return ret def clear_lock(backend=None, remote=None): @@ -334,8 +334,8 @@ def clear_lock(backend=None, remote=None): if errors: ret['errors'] = errors if not ret: - ret = 'No locks were removed' - salt.output.display_output(ret, 'nested', opts=__opts__) + return 'No locks were removed' + return ret def lock(backend=None, remote=None): @@ -376,5 +376,5 @@ def lock(backend=None, remote=None): if errors: ret['errors'] = errors if not ret: - ret = 'No locks were set' - salt.output.display_output(ret, 'nested', opts=__opts__) + return 'No locks were set' + return ret diff --git a/salt/utils/__init__.py b/salt/utils/__init__.py index 4e40cafd25d7..5ee73b168349 100644 --- a/salt/utils/__init__.py +++ b/salt/utils/__init__.py @@ -2875,3 +2875,15 @@ def invalid_kwargs(invalid_kwargs, raise_exc=True): raise SaltInvocationError(msg) else: return msg + + +def split_input(val): + ''' + Take an input value and split it into a list, returning the resulting list + ''' + if isinstance(val, list): + return val + try: + return [x.strip() for x in val.split(',')] + except AttributeError: + return [x.strip() for x in str(val).split(',')] diff --git a/salt/utils/gitfs.py b/salt/utils/gitfs.py index 411da5a2e8cd..289aa9ee5288 100644 --- a/salt/utils/gitfs.py +++ b/salt/utils/gitfs.py @@ -3,6 +3,7 @@ # Import python libs from __future__ import absolute_import import copy +import contextlib import distutils.version # pylint: disable=import-error,no-name-in-module import errno import fnmatch @@ -15,6 +16,7 @@ import shlex import shutil import stat import subprocess +import time from datetime import datetime VALID_PROVIDERS = ('gitpython', 'pygit2', 'dulwich') @@ -57,7 +59,7 @@ import salt.utils import salt.utils.itertools import salt.utils.url import salt.fileserver -from salt.exceptions import FileserverConfigError +from salt.exceptions import FileserverConfigError, GitLockError from salt.utils.event import tagify # Import third party libs @@ -298,29 +300,8 @@ class GitProvider(object): _check_ref(ret, base_ref, rname) return ret - def check_lock(self): - ''' - Used by the provider-specific fetch() function to check the existence - of an update lock, and set the lock if not present. If the lock exists - already, or if there was a problem setting the lock, this function - returns False. If the lock was successfully set, return True. - ''' - if os.path.exists(self.lockfile): - log.warning( - 'Update lockfile is present for {0} remote \'{1}\', ' - 'skipping. If this warning persists, it is possible that the ' - 'update process was interrupted. Removing {2} or running ' - '\'salt-run cache.clear_git_lock {0}\' will allow updates to ' - 'continue for this remote.' - .format(self.role, self.id, self.lockfile) - ) - return False - errors = self.lock()[-1] - if errors: - log.error('Unable to set update lock for {0} remote \'{1}\', ' - 'skipping.'.format(self.role, self.id)) - return False - return True + def _get_lock_file(self, lock_type='update'): + return os.path.join(self.gitdir, lock_type + '.lk') def check_root(self): ''' @@ -344,65 +325,143 @@ class GitProvider(object): ''' return [] - def clear_lock(self): + def clear_lock(self, lock_type='update'): ''' Clear update.lk ''' + lock_file = self._get_lock_file(lock_type=lock_type) + def _add_error(errlist, exc): msg = ('Unable to remove update lock for {0} ({1}): {2} ' - .format(self.url, self.lockfile, exc)) + .format(self.url, lock_file, exc)) log.debug(msg) errlist.append(msg) success = [] failed = [] - if os.path.exists(self.lockfile): - try: - os.remove(self.lockfile) - except OSError as exc: - if exc.errno == errno.EISDIR: - # Somehow this path is a directory. Should never happen - # unless some wiseguy manually creates a directory at this - # path, but just in case, handle it. - try: - shutil.rmtree(self.lockfile) - except OSError as exc: - _add_error(failed, exc) - else: + + try: + os.remove(lock_file) + except OSError as exc: + if exc.errno == errno.ENOENT: + # No lock file present + pass + elif exc.errno == errno.EISDIR: + # Somehow this path is a directory. Should never happen + # unless some wiseguy manually creates a directory at this + # path, but just in case, handle it. + try: + shutil.rmtree(lock_file) + except OSError as exc: _add_error(failed, exc) else: - msg = 'Removed lock for {0} remote \'{1}\''.format( + _add_error(failed, exc) + else: + msg = 'Removed {0} lock for {1} remote \'{2}\''.format( + lock_type, + self.role, + self.id + ) + log.debug(msg) + success.append(msg) + return success, failed + + def fetch(self): + ''' + Fetch the repo. If the local copy was updated, return True. If the + local copy was already up-to-date, return False. + + This function requires that a _fetch() function be implemented in a + sub-class. + ''' + try: + with self.gen_lock(lock_type='update'): + log.debug('Fetching %s remote \'%s\'', self.role, self.id) + # Run provider-specific fetch code + return self._fetch() + except GitLockError as exc: + if exc.errno == errno.EEXIST: + log.warning( + 'Update lock file is present for %s remote \'%s\', ' + 'skipping. If this warning persists, it is possible that ' + 'the update process was interrupted, but the lock could ' + 'also have been manually set. Removing %s or running ' + '\'salt-run cache.clear_git_lock %s type=update\' will ' + 'allow updates to continue for this remote.', + self.role, + self.id, + self._get_lock_file(lock_type='update'), self.role, - self.id ) - log.debug(msg) - success.append(msg) - return success, failed + return False + + def _lock(self, lock_type='update', failhard=False): + ''' + Place a lock file if (and only if) it does not already exist. + ''' + try: + fh_ = os.open(self._get_lock_file(lock_type), + os.O_CREAT | os.O_EXCL | os.O_WRONLY) + with os.fdopen(fh_, 'w'): + # Write the lock file and close the filehandle + pass + except (OSError, IOError) as exc: + if exc.errno == errno.EEXIST: + if failhard: + raise + return None + else: + msg = 'Unable to set {0} lock for {1} ({2}): {3} '.format( + lock_type, + self.id, + self._get_lock_file(lock_type), + exc + ) + log.error(msg) + raise GitLockError(exc.errno, msg) + msg = 'Set {0} lock for {1} remote \'{2}\''.format( + lock_type, + self.role, + self.id + ) + log.debug(msg) + return msg def lock(self): ''' - Place an update.lk + Place an lock file and report on the success/failure. This is an + interface to be used by the fileserver runner, so it is hard-coded to + perform an update lock. We aren't using the gen_lock() + contextmanager here because the lock is meant to stay and not be + automatically removed. ''' success = [] failed = [] - if not os.path.exists(self.lockfile): - try: - with salt.utils.fopen(self.lockfile, 'w+') as fp_: - fp_.write('') - except (IOError, OSError) as exc: - msg = ('Unable to set update lock for {0} ({1}): {2} ' - .format(self.url, self.lockfile, exc)) - log.error(msg) - failed.append(msg) - else: - msg = 'Set lock for {0} remote \'{1}\''.format( - self.role, - self.id - ) - log.debug(msg) - success.append(msg) + try: + result = self._lock(lock_type='update') + except GitLockError as exc: + failed.append(exc.strerror) + else: + if result is not None: + success.append(result) return success, failed + @contextlib.contextmanager + def gen_lock(self, lock_type='update'): + ''' + Set and automatically clear a lock + ''' + lock_set = False + try: + self._lock(lock_type=lock_type, failhard=True) + lock_set = True + yield + except (OSError, IOError, GitLockError) as exc: + raise GitLockError(exc.errno, exc.strerror) + finally: + if lock_set: + self.clear_lock(lock_type=lock_type) + def init_remote(self): ''' This function must be overridden in a sub-class @@ -432,13 +491,14 @@ class GitProvider(object): blacklist=self.env_blacklist ) - def envs(self): + def _fetch(self): ''' - This function must be overridden in a sub-class + Provider-specific code for fetching, must be implemented in a + sub-class. ''' raise NotImplementedError() - def fetch(self): + def envs(self): ''' This function must be overridden in a sub-class ''' @@ -504,17 +564,67 @@ class GitPython(GitProvider): def checkout(self): ''' - Checkout the configured branch/tag + Checkout the configured branch/tag. We catch an "Exception" class here + instead of a specific exception class because the exceptions raised by + GitPython when running these functions vary in different versions of + GitPython. ''' - for ref in ('origin/' + self.branch, self.branch): + try: + head_sha = self.repo.rev_parse('HEAD').hexsha + except Exception: + # Should only happen the first time we are checking out, since + # we fetch first before ever checking anything out. + head_sha = None + + # 'origin/' + self.branch ==> matches a branch head + # 'tags/' + self.branch + '@{commit}' ==> matches tag's commit + for rev_parse_target, checkout_ref in ( + ('origin/' + self.branch, 'origin/' + self.branch), + ('tags/' + self.branch + '@{commit}', 'tags/' + self.branch)): try: - self.repo.git.checkout(ref) + target_sha = self.repo.rev_parse(rev_parse_target).hexsha + except Exception: + # ref does not exist + continue + else: + if head_sha == target_sha: + # No need to checkout, we're already up-to-date + return self.check_root() + + try: + with self.gen_lock(lock_type='checkout'): + self.repo.git.checkout(checkout_ref) + log.debug( + '%s remote \'%s\' has been checked out to %s', + self.role, + self.id, + checkout_ref + ) + except GitLockError as exc: + if exc.errno == errno.EEXIST: + # Re-raise with a different strerror containing a + # more meaningful error message for the calling + # function. + raise GitLockError( + exc.errno, + 'Checkout lock exists for {0} remote \'{1}\'' + .format(self.role, self.id) + ) + else: + log.error( + 'Error %d encountered obtaining checkout lock ' + 'for %s remote \'%s\'', + exc.errno, + self.role, + self.id + ) + return None except Exception: continue return self.check_root() log.error( - 'Failed to checkout {0} from {1} remote \'{2}\': remote ref does ' - 'not exist'.format(self.branch, self.role, self.id) + 'Failed to checkout %s from %s remote \'%s\': remote ref does ' + 'not exist', self.branch, self.role, self.id ) return None @@ -555,7 +665,7 @@ class GitPython(GitProvider): log.error(_INVALID_REPO.format(self.cachedir, self.url)) return new - self.lockfile = os.path.join(self.repo.working_dir, 'update.lk') + self.gitdir = os.path.join(self.repo.working_dir, '.git') if not self.repo.remotes: try: @@ -604,13 +714,11 @@ class GitPython(GitProvider): ref_paths = [x.path for x in self.repo.refs] return self._get_envs_from_ref_paths(ref_paths) - def fetch(self): + def _fetch(self): ''' Fetch the repo. If the local copy was updated, return True. If the local copy was already up-to-date, return False. ''' - if not self.check_lock(): - return False origin = self.repo.remotes[0] try: fetch_results = origin.fetch() @@ -772,7 +880,61 @@ class Pygit2(GitProvider): remote_ref = 'refs/remotes/origin/' + self.branch tag_ref = 'refs/tags/' + self.branch + try: + local_head = self.repo.lookup_reference('HEAD') + except KeyError: + log.warning( + 'HEAD not present in %s remote \'%s\'', self.role, self.id + ) + return None + + try: + head_sha = local_head.get_object().hex + except AttributeError: + # Shouldn't happen, but just in case a future pygit2 API change + # breaks things, avoid a traceback and log an error. + log.error( + 'Unable to get SHA of HEAD for %s remote \'%s\'', + self.role, self.id + ) + return None + except KeyError: + head_sha = None + refs = self.repo.listall_references() + + def _perform_checkout(checkout_ref, branch=True): + ''' + DRY function for checking out either a branch or a tag + ''' + try: + with self.gen_lock(lock_type='checkout'): + # Checkout the local branch corresponding to the + # remote ref. + self.repo.checkout(checkout_ref) + if branch: + self.repo.reset(oid, pygit2.GIT_RESET_HARD) + return True + except GitLockError as exc: + if exc.errno == errno.EEXIST: + # Re-raise with a different strerror containing a + # more meaningful error message for the calling + # function. + raise GitLockError( + exc.errno, + 'Checkout lock exists for {0} remote \'{1}\'' + .format(self.role, self.id) + ) + else: + log.error( + 'Error %d encountered obtaining checkout lock ' + 'for %s remote \'%s\'', + exc.errno, + self.role, + self.id + ) + return False + try: if remote_ref in refs: # Get commit id for the remote ref @@ -782,41 +944,99 @@ class Pygit2(GitProvider): # it at the commit id of the remote ref self.repo.create_reference(local_ref, oid) - # Check HEAD ref existence (checking out local_ref when HEAD - # ref doesn't exist will raise an exception in pygit2 >= 0.21), - # and create the HEAD ref if it is missing. - head_ref = self.repo.lookup_reference('HEAD').target - if head_ref not in refs and head_ref != local_ref: - branch_name = head_ref.partition('refs/heads/')[-1] - if not branch_name: - # Shouldn't happen, but log an error if it does - log.error( - 'pygit2 was unable to resolve branch name from ' - 'HEAD ref \'{0}\' in {1} remote \'{2}\''.format( - head_ref, self.role, self.id + try: + target_sha = \ + self.repo.lookup_reference(remote_ref).get_object().hex + except KeyError: + log.error( + 'pygit2 was unable to get SHA for %s in %s remote ' + '\'%s\'', local_ref, self.role, self.id + ) + return None + + # Only perform a checkout if HEAD and target are not pointing + # at the same SHA1. + if head_sha != target_sha: + # Check existence of the ref in refs/heads/ which + # corresponds to the local HEAD. Checking out local_ref + # below when no local ref for HEAD is missing will raise an + # exception in pygit2 >= 0.21. If this ref is not present, + # create it. The "head_ref != local_ref" check ensures we + # don't try to add this ref if it is not necessary, as it + # would have been added above already. head_ref would be + # the same as local_ref if the branch name was changed but + # the cachedir was not (for example if a "name" parameter + # was used in a git_pillar remote, or if we are using + # winrepo which takes the basename of the repo as the + # cachedir). + head_ref = local_head.target + # If head_ref is not a string, it will point to a + # pygit2.Oid object and we are in detached HEAD mode. + # Therefore, there is no need to add a local reference. If + # head_ref == local_ref, then the local reference for HEAD + # in refs/heads/ already exists and again, no need to add. + if isinstance(head_ref, six.string_types) \ + and head_ref not in refs and head_ref != local_ref: + branch_name = head_ref.partition('refs/heads/')[-1] + if not branch_name: + # Shouldn't happen, but log an error if it does + log.error( + 'pygit2 was unable to resolve branch name from ' + 'HEAD ref \'{0}\' in {1} remote \'{2}\''.format( + head_ref, self.role, self.id + ) ) + return None + remote_head = 'refs/remotes/origin/' + branch_name + if remote_head not in refs: + log.error( + 'Unable to find remote ref \'{0}\' in {1} remote ' + '\'{2}\''.format(head_ref, self.role, self.id) + ) + return None + self.repo.create_reference( + head_ref, + self.repo.lookup_reference(remote_head).target ) + + if not _perform_checkout(local_ref, branch=True): return None - remote_head = 'refs/remotes/origin/' + branch_name - if remote_head not in refs: - log.error( - 'Unable to find remote ref \'{0}\' in {1} remote ' - '\'{2}\''.format(head_ref, self.role, self.id) - ) - return None - self.repo.create_reference( - head_ref, - self.repo.lookup_reference(remote_head).target - ) - # Point HEAD at the local ref - self.repo.checkout(local_ref) - # Reset HEAD to the commit id of the remote ref - self.repo.reset(oid, pygit2.GIT_RESET_HARD) + # Return the relative root, if present return self.check_root() + elif tag_ref in refs: - self.repo.checkout(tag_ref) - return self.check_root() + tag_obj = self.repo.revparse_single(tag_ref) + if not isinstance(tag_obj, pygit2.Tag): + log.error( + '%s does not correspond to pygit2.Tag object', + tag_ref + ) + else: + try: + # If no AttributeError raised, this is an annotated tag + tag_sha = tag_obj.target.hex + except AttributeError: + try: + tag_sha = tag_obj.hex + except AttributeError: + # Shouldn't happen, but could if a future pygit2 + # API change breaks things. + log.error( + 'Unable to resolve %s from %s remote \'%s\' ' + 'to either an annotated or non-annotated tag', + tag_ref, self.role, self.id + ) + return None + + if head_sha != target_sha: + if not _perform_checkout(local_ref, branch=False): + return None + + # Return the relative root, if present + return self.check_root() + except GitLockError: + raise except Exception as exc: log.error( 'Failed to checkout {0} from {1} remote \'{2}\': {3}'.format( @@ -921,7 +1141,7 @@ class Pygit2(GitProvider): log.error(_INVALID_REPO.format(self.cachedir, self.url)) return new - self.lockfile = os.path.join(self.repo.workdir, 'update.lk') + self.gitdir = os.path.join(self.repo.workdir, '.git') if not self.repo.remotes: try: @@ -997,13 +1217,11 @@ class Pygit2(GitProvider): ref_paths = self.repo.listall_references() return self._get_envs_from_ref_paths(ref_paths) - def fetch(self): + def _fetch(self): ''' Fetch the repo. If the local copy was updated, return True. If the local copy was already up-to-date, return False. ''' - if not self.check_lock(): - return False origin = self.repo.remotes[0] refs_pre = self.repo.listall_references() fetch_kwargs = {} @@ -1345,13 +1563,11 @@ class Dulwich(GitProvider): # pylint: disable=abstract-method ref_paths = self.get_env_refs(self.repo.get_refs()) return self._get_envs_from_ref_paths(ref_paths) - def fetch(self): + def _fetch(self): ''' Fetch the repo. If the local copy was updated, return True. If the local copy was already up-to-date, return False. ''' - if not self.check_lock(): - return False # origin is just a url here, there is no origin object origin = self.url client, path = \ @@ -1613,6 +1829,23 @@ class Dulwich(GitProvider): # pylint: disable=abstract-method new = False if not os.listdir(self.cachedir): # Repo cachedir is empty, initialize a new repo there + self.repo = dulwich.repo.Repo.init(self.cachedir) + new = True + else: + # Repo cachedir exists, try to attach + try: + self.repo = dulwich.repo.Repo(self.cachedir) + except dulwich.repo.NotGitRepository: + log.error(_INVALID_REPO.format(self.cachedir, self.url)) + return new + + self.gitdir = os.path.join(self.repo.path, '.git') + + # Read in config file and look for the remote + try: + conf = self.get_conf() + conf.get(('remote', 'origin'), 'url') + except KeyError: try: self.repo = dulwich.repo.Repo.init(self.cachedir) new = True @@ -1827,9 +2060,9 @@ class GitBase(object): ) return errors - def clear_lock(self, remote=None): + def clear_lock(self, remote=None, lock_type='update'): ''' - Clear update.lk + Clear update.lk for all remotes ''' cleared = [] errors = [] @@ -1844,7 +2077,7 @@ class GitBase(object): # remote was non-string, try again if not fnmatch.fnmatch(repo.url, six.text_type(remote)): continue - success, failed = repo.clear_lock() + success, failed = repo.clear_lock(lock_type=lock_type) cleared.extend(success) errors.extend(failed) return cleared, errors @@ -1870,8 +2103,6 @@ class GitBase(object): '\'{2}\''.format(exc, self.role, repo.id), exc_info_on_loglevel=logging.DEBUG ) - finally: - repo.clear_lock() return changed def lock(self, remote=None): @@ -1936,7 +2167,7 @@ class GitBase(object): self.hash_cachedir, self.find_file ) - except (IOError, OSError): + except (OSError, IOError): # Hash file won't exist if no files have yet been served up pass @@ -2166,6 +2397,38 @@ class GitBase(object): ) ) + def do_checkout(self, repo): + ''' + Common code for git_pillar/winrepo to handle locking and checking out + of a repo. + ''' + time_start = time.time() + while time.time() - time_start <= 5: + try: + return repo.checkout() + except GitLockError as exc: + if exc.errno == errno.EEXIST: + time.sleep(0.1) + continue + else: + log.error( + 'Error %d encountered while obtaining checkout ' + 'lock for %s remote \'%s\': %s', + exc.errno, + repo.role, + repo.id, + exc + ) + break + else: + log.error( + 'Timed out waiting for checkout lock to be released for ' + '%s remote \'%s\'. If this error persists, run \'salt-run ' + 'cache.clear_git_lock %s type=checkout\' to clear it.', + self.role, repo.id, self.role + ) + return None + class GitFS(GitBase): ''' @@ -2460,7 +2723,7 @@ class GitPillar(GitBase): ''' self.pillar_dirs = {} for repo in self.remotes: - cachedir = repo.checkout() + cachedir = self.do_checkout(repo) if cachedir is not None: # Figure out which environment this remote should be assigned if repo.env: @@ -2502,6 +2765,6 @@ class WinRepo(GitBase): ''' self.winrepo_dirs = {} for repo in self.remotes: - cachedir = repo.checkout() + cachedir = self.do_checkout(repo) if cachedir is not None: - self.winrepo_dirs[repo.url] = cachedir + self.winrepo_dirs[repo.id] = cachedir -- 2.7.2
Locations
Projects
Search
Status Monitor
Help
OpenBuildService.org
Documentation
API Documentation
Code of Conduct
Contact
Support
@OBShq
Terms
openSUSE Build Service is sponsored by
The Open Build Service is an
openSUSE project
.
Sign Up
Log In
Places
Places
All Projects
Status Monitor