Accepting request 375270 from systemsmanagement:saltstack:testing
- Use SHA256 by default in master, minion and proxy (bsc#955373) Add: * 0036-Use-SHA256-hash-type-by-default.patch - 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 OBS-URL: https://build.opensuse.org/request/show/375270 OBS-URL: https://build.opensuse.org/package/show/systemsmanagement:saltstack/salt?expand=0&rev=64
This commit is contained in:
parent
8bc8eaf64f
commit
eac5c0b1cc
963
0034-Fix-git_pillar-race-condition.patch
Normal file
963
0034-Fix-git_pillar-race-condition.patch
Normal 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
|
||||||
|
|
67
0035-Fix-the-always-false-behavior-on-checking-state.patch
Normal file
67
0035-Fix-the-always-false-behavior-on-checking-state.patch
Normal 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
|
||||||
|
|
53
0036-Use-SHA256-hash-type-by-default.patch
Normal file
53
0036-Use-SHA256-hash-type-by-default.patch
Normal 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
|
||||||
|
|
17
salt.changes
17
salt.changes
@ -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
|
Sat Mar 12 17:08:03 UTC 2016 - mc@suse.com
|
||||||
|
|
||||||
|
@ -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
|
Patch32: 0032-Add-error-check-when-retcode-is-0-but-stderr-is-pres.patch
|
||||||
# PATCH-FIX-UPSTREAM https://github.com/saltstack/salt/pull/31793
|
# PATCH-FIX-UPSTREAM https://github.com/saltstack/salt/pull/31793
|
||||||
Patch33: 0033-fixing-init-system-dectection-on-sles-11-refs-31617.patch
|
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
|
BuildRoot: %{_tmppath}/%{name}-%{version}-build
|
||||||
BuildRequires: logrotate
|
BuildRequires: logrotate
|
||||||
@ -480,6 +486,9 @@ cp %{S:1} .
|
|||||||
%patch31 -p1
|
%patch31 -p1
|
||||||
%patch32 -p1
|
%patch32 -p1
|
||||||
%patch33 -p1
|
%patch33 -p1
|
||||||
|
%patch34 -p1
|
||||||
|
%patch35 -p1
|
||||||
|
%patch36 -p1
|
||||||
|
|
||||||
%build
|
%build
|
||||||
python setup.py --salt-transport=both build
|
python setup.py --salt-transport=both build
|
||||||
|
Loading…
Reference in New Issue
Block a user