Accepting request 375690 from systemsmanagement:saltstack

1

OBS-URL: https://build.opensuse.org/request/show/375690
OBS-URL: https://build.opensuse.org/package/show/openSUSE:Factory/salt?expand=0&rev=58
This commit is contained in:
Dominique Leuenberger 2016-03-26 14:24:17 +00:00 committed by Git OBS Bridge
commit 25f81ef1c4
5 changed files with 1109 additions and 0 deletions

View File

@ -0,0 +1,963 @@
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

View File

@ -0,0 +1,67 @@
From bb8048d4bd842746b09dbafe3a610e0d7c3e1bc2 Mon Sep 17 00:00:00 2001
From: Bo Maryniuk <bo@suse.de>
Date: Tue, 8 Mar 2016 16:00:26 +0100
Subject: [PATCH 35/35] Fix the always-false behavior on checking state
- Fix PEP8 continuation
- Keep first level away from lists.
- Adjust test
---
salt/utils/__init__.py | 15 +++++----------
tests/unit/utils/utils_test.py | 2 +-
2 files changed, 6 insertions(+), 11 deletions(-)
diff --git a/salt/utils/__init__.py b/salt/utils/__init__.py
index 5ee73b168349..8c8309e99f95 100644
--- a/salt/utils/__init__.py
+++ b/salt/utils/__init__.py
@@ -1741,7 +1741,7 @@ def gen_state_tag(low):
return '{0[state]}_|-{0[__id__]}_|-{0[name]}_|-{0[fun]}'.format(low)
-def check_state_result(running):
+def check_state_result(running, recurse=False):
'''
Check the total return value of the run and determine if the running
dict has any issues
@@ -1754,20 +1754,15 @@ def check_state_result(running):
ret = True
for state_result in six.itervalues(running):
- if not isinstance(state_result, dict):
- # return false when hosts return a list instead of a dict
+ if not recurse and not isinstance(state_result, dict):
ret = False
- if ret:
+ if ret and isinstance(state_result, dict):
result = state_result.get('result', _empty)
if result is False:
ret = False
# only override return value if we are not already failed
- elif (
- result is _empty
- and isinstance(state_result, dict)
- and ret
- ):
- ret = check_state_result(state_result)
+ elif result is _empty and isinstance(state_result, dict) and ret:
+ ret = check_state_result(state_result, recurse=True)
# return as soon as we got a failure
if not ret:
break
diff --git a/tests/unit/utils/utils_test.py b/tests/unit/utils/utils_test.py
index 611bfce0ed4b..261af69b59fc 100644
--- a/tests/unit/utils/utils_test.py
+++ b/tests/unit/utils/utils_test.py
@@ -406,7 +406,7 @@ class UtilsTestCase(TestCase):
('test_state0', {'result': True}),
('test_state', {'result': True}),
])),
- ('host2', [])
+ ('host2', OrderedDict([]))
]))
])
}
--
2.7.2

View File

@ -0,0 +1,53 @@
From eea48a283a184a02223fc440fec54a47a5b47b62 Mon Sep 17 00:00:00 2001
From: Bo Maryniuk <bo@suse.de>
Date: Thu, 17 Mar 2016 12:30:23 +0100
Subject: [PATCH 36/36] Use SHA256 hash type by default
---
conf/master | 2 +-
conf/minion | 2 +-
conf/proxy | 2 +-
3 files changed, 3 insertions(+), 3 deletions(-)
diff --git a/conf/master b/conf/master
index cf05ec4..5f6beaa 100644
--- a/conf/master
+++ b/conf/master
@@ -474,7 +474,7 @@ syndic_user: salt
#
# Prior to changing this value, the master should be stopped and all Salt
# caches should be cleared.
-#hash_type: md5
+hash_type: sha256
# The buffer size in the file server can be adjusted here:
#file_buffer_size: 1048576
diff --git a/conf/minion b/conf/minion
index e17ec61..ba4111a 100644
--- a/conf/minion
+++ b/conf/minion
@@ -447,7 +447,7 @@
#
# Warning: Prior to changing this value, the minion should be stopped and all
# Salt caches should be cleared.
-#hash_type: sha256
+hash_type: sha256
# The Salt pillar is searched for locally if file_client is set to local. If
# this is the case, and pillar data is defined, then the pillar_roots need to
diff --git a/conf/proxy b/conf/proxy
index 0de6af8..77ecf3b 100644
--- a/conf/proxy
+++ b/conf/proxy
@@ -427,7 +427,7 @@
#
# Warning: Prior to changing this value, the minion should be stopped and all
# Salt caches should be cleared.
-#hash_type: sha256
+hash_type: sha256
# The Salt pillar is searched for locally if file_client is set to local. If
# this is the case, and pillar data is defined, then the pillar_roots need to
--
2.7.3

View File

@ -1,3 +1,20 @@
-------------------------------------------------------------------
Thu Mar 17 12:09:14 UTC 2016 - bmaryniuk@suse.com
- Use SHA256 by default in master, minion and proxy (bsc#955373)
Add:
* 0036-Use-SHA256-hash-type-by-default.patch
-------------------------------------------------------------------
Wed Mar 16 15:34:26 UTC 2016 - bmaryniuk@suse.com
- Fix state structure compilation
Add:
* 0035-Fix-the-always-false-behavior-on-checking-state.patch
- Fix git_pillar race condition
Add:
* 0034-Fix-git_pillar-race-condition.patch
-------------------------------------------------------------------
Sat Mar 12 17:08:03 UTC 2016 - mc@suse.com

View File

@ -105,6 +105,12 @@ Patch31: 0031-Only-use-LONGSIZE-in-rpm.info-if-available.-Otherwis.patch
Patch32: 0032-Add-error-check-when-retcode-is-0-but-stderr-is-pres.patch
# PATCH-FIX-UPSTREAM https://github.com/saltstack/salt/pull/31793
Patch33: 0033-fixing-init-system-dectection-on-sles-11-refs-31617.patch
# PATCH-FIX-UPSTREAM https://github.com/saltstack/salt/pull/31836
Patch34: 0034-Fix-git_pillar-race-condition.patch
# PATCH-FIX-UPSTREAM https://github.com/saltstack/salt/pull/31745
Patch35: 0035-Fix-the-always-false-behavior-on-checking-state.patch
# PATCH-FIX-OPENSUSE - Upstream default hash type is set to MD5, while we require SHA256 (bsc#955373)
Patch36: 0036-Use-SHA256-hash-type-by-default.patch
BuildRoot: %{_tmppath}/%{name}-%{version}-build
BuildRequires: logrotate
@ -480,6 +486,9 @@ cp %{S:1} .
%patch31 -p1
%patch32 -p1
%patch33 -p1
%patch34 -p1
%patch35 -p1
%patch36 -p1
%build
python setup.py --salt-transport=both build