Add support for policy types, 'erasure_coding' policy
This patch extends the StoragePolicy class for non-replication storage policies, the first one being "erasure coding". Changes: - Add 'policy_type' support to BaseStoragePolicy class - Disallow direct instantiation of BaseStoragePolicy class - Subclass BaseStoragePolicy - "StoragePolicy": . Replication policy, default . policy_type = 'replication' - "ECStoragePolicy": . Erasure Coding policy . policy_type = 'erasure_coding' . Private member variables ec_type (EC backend), ec_num_data_fragments (number of fragments original data split into after erasure coding operation), ec_num_parity_fragments (number of parity fragments generated during erasure coding) . Private methods EC specific attributes and ring validator methods. - Swift will use PyECLib, a Python Erasure Coding library, for erasure coding operations. PyECLib is already an approved OpenStack core requirement. (https://bitbucket.org/kmgreen2/pyeclib/) - Add test cases for - 'policy_type' StoragePolicy member - policy_type == 'erasure_coding' DocImpact Co-Authored-By: Alistair Coles <alistair.coles@hp.com> Co-Authored-By: Thiago da Silva <thiago@redhat.com> Co-Authored-By: Clay Gerrard <clay.gerrard@gmail.com> Co-Authored-By: Paul Luse <paul.e.luse@intel.com> Co-Authored-By: Samuel Merritt <sam@swiftstack.com> Co-Authored-By: Christian Schwede <christian.schwede@enovance.com> Co-Authored-By: Yuan Zhou <yuan.zhou@intel.com> Change-Id: Ie0e09796e3ec45d3e656fb7540d0e5a5709b8386 Implements: blueprint ec-proxy-work
This commit is contained in:
parent
ce596684f6
commit
ed54066288
@ -22,9 +22,13 @@ swift_hash_path_prefix = changeme
|
|||||||
# defined you must define a policy with index 0 and you must specify a
|
# defined you must define a policy with index 0 and you must specify a
|
||||||
# default. It is recommended you always define a section for
|
# default. It is recommended you always define a section for
|
||||||
# storage-policy:0.
|
# storage-policy:0.
|
||||||
|
#
|
||||||
|
# A 'policy_type' argument is also supported but is not mandatory. Default
|
||||||
|
# policy type 'replication' is used when 'policy_type' is unspecified.
|
||||||
[storage-policy:0]
|
[storage-policy:0]
|
||||||
name = Policy-0
|
name = Policy-0
|
||||||
default = yes
|
default = yes
|
||||||
|
#policy_type = replication
|
||||||
|
|
||||||
# the following section would declare a policy called 'silver', the number of
|
# the following section would declare a policy called 'silver', the number of
|
||||||
# replicas will be determined by how the ring is built. In this example the
|
# replicas will be determined by how the ring is built. In this example the
|
||||||
@ -39,6 +43,42 @@ default = yes
|
|||||||
# current default.
|
# current default.
|
||||||
#[storage-policy:1]
|
#[storage-policy:1]
|
||||||
#name = silver
|
#name = silver
|
||||||
|
#policy_type = replication
|
||||||
|
|
||||||
|
# The following declares a storage policy of type 'erasure_coding' which uses
|
||||||
|
# Erasure Coding for data reliability. The 'erasure_coding' storage policy in
|
||||||
|
# Swift is available as a "beta". Please refer to Swift documentation for
|
||||||
|
# details on how the 'erasure_coding' storage policy is implemented.
|
||||||
|
#
|
||||||
|
# Swift uses PyECLib, a Python Erasure coding API library, for encode/decode
|
||||||
|
# operations. Please refer to Swift documentation for details on how to
|
||||||
|
# install PyECLib.
|
||||||
|
#
|
||||||
|
# When defining an EC policy, 'policy_type' needs to be 'erasure_coding' and
|
||||||
|
# EC configuration parameters 'ec_type', 'ec_num_data_fragments' and
|
||||||
|
# 'ec_num_parity_fragments' must be specified. 'ec_type' is chosen from the
|
||||||
|
# list of EC backends supported by PyECLib. The ring configured for the
|
||||||
|
# storage policy must have it's "replica" count configured to
|
||||||
|
# 'ec_num_data_fragments' + 'ec_num_parity_fragments' - this requirement is
|
||||||
|
# validated when services start. 'ec_object_segment_size' is the amount of
|
||||||
|
# data that will be buffered up before feeding a segment into the
|
||||||
|
# encoder/decoder. More information about these configuration options and
|
||||||
|
# supported `ec_type` schemes is available in the Swift documentation. Please
|
||||||
|
# refer to Swift documentation for details on how to configure EC policies.
|
||||||
|
#
|
||||||
|
# The example 'deepfreeze10-4' policy defined below is a _sample_
|
||||||
|
# configuration with 10 'data' and 4 'parity' fragments. 'ec_type'
|
||||||
|
# defines the Erasure Coding scheme. 'jerasure_rs_vand' (Reed-Solomon
|
||||||
|
# Vandermonde) is used as an example below.
|
||||||
|
#
|
||||||
|
#[storage-policy:2]
|
||||||
|
#name = deepfreeze10-4
|
||||||
|
#policy_type = erasure_coding
|
||||||
|
#ec_type = jerasure_rs_vand
|
||||||
|
#ec_num_data_fragments = 10
|
||||||
|
#ec_num_parity_fragments = 4
|
||||||
|
#ec_object_segment_size = 1048576
|
||||||
|
|
||||||
|
|
||||||
# The swift-constraints section sets the basic constraints on data
|
# The swift-constraints section sets the basic constraints on data
|
||||||
# saved in the swift cluster. These constraints are automatically
|
# saved in the swift cluster. These constraints are automatically
|
||||||
|
@ -17,10 +17,18 @@ import string
|
|||||||
|
|
||||||
from swift.common.utils import config_true_value, SWIFT_CONF_FILE
|
from swift.common.utils import config_true_value, SWIFT_CONF_FILE
|
||||||
from swift.common.ring import Ring
|
from swift.common.ring import Ring
|
||||||
|
from swift.common.utils import quorum_size
|
||||||
|
from swift.common.exceptions import RingValidationError
|
||||||
|
from pyeclib.ec_iface import ECDriver, ECDriverError, VALID_EC_TYPES
|
||||||
|
|
||||||
LEGACY_POLICY_NAME = 'Policy-0'
|
LEGACY_POLICY_NAME = 'Policy-0'
|
||||||
VALID_CHARS = '-' + string.letters + string.digits
|
VALID_CHARS = '-' + string.letters + string.digits
|
||||||
|
|
||||||
|
DEFAULT_POLICY_TYPE = REPL_POLICY = 'replication'
|
||||||
|
EC_POLICY = 'erasure_coding'
|
||||||
|
|
||||||
|
DEFAULT_EC_OBJECT_SEGMENT_SIZE = 1048576
|
||||||
|
|
||||||
|
|
||||||
class PolicyError(ValueError):
|
class PolicyError(ValueError):
|
||||||
|
|
||||||
@ -38,36 +46,73 @@ def _get_policy_string(base, policy_index):
|
|||||||
return return_string
|
return return_string
|
||||||
|
|
||||||
|
|
||||||
def get_policy_string(base, policy_index):
|
def get_policy_string(base, policy_or_index):
|
||||||
"""
|
"""
|
||||||
Helper function to construct a string from a base and the policy
|
Helper function to construct a string from a base and the policy.
|
||||||
index. Used to encode the policy index into either a file name
|
Used to encode the policy index into either a file name or a
|
||||||
or a directory name by various modules.
|
directory name by various modules.
|
||||||
|
|
||||||
:param base: the base string
|
:param base: the base string
|
||||||
:param policy_index: the storage policy index
|
:param policy_or_index: StoragePolicy instance, or an index
|
||||||
|
(string or int), if None the legacy
|
||||||
|
storage Policy-0 is assumed.
|
||||||
|
|
||||||
:returns: base name with policy index added
|
:returns: base name with policy index added
|
||||||
|
:raises: PolicyError if no policy exists with the given policy_index
|
||||||
"""
|
"""
|
||||||
if POLICIES.get_by_index(policy_index) is None:
|
if isinstance(policy_or_index, BaseStoragePolicy):
|
||||||
raise PolicyError("No policy with index %r" % policy_index)
|
policy = policy_or_index
|
||||||
return _get_policy_string(base, policy_index)
|
else:
|
||||||
|
policy = POLICIES.get_by_index(policy_or_index)
|
||||||
|
if policy is None:
|
||||||
|
raise PolicyError("Unknown policy", index=policy_or_index)
|
||||||
|
return _get_policy_string(base, int(policy))
|
||||||
|
|
||||||
|
|
||||||
class StoragePolicy(object):
|
def split_policy_string(policy_string):
|
||||||
"""
|
"""
|
||||||
Represents a storage policy.
|
Helper function to convert a string representing a base and a
|
||||||
Not meant to be instantiated directly; use
|
policy. Used to decode the policy from either a file name or
|
||||||
:func:`~swift.common.storage_policy.reload_storage_policies` to load
|
a directory name by various modules.
|
||||||
POLICIES from ``swift.conf``.
|
|
||||||
|
:param policy_string: base name with policy index added
|
||||||
|
|
||||||
|
:raises: PolicyError if given index does not map to a valid policy
|
||||||
|
:returns: a tuple, in the form (base, policy) where base is the base
|
||||||
|
string and policy is the StoragePolicy instance for the
|
||||||
|
index encoded in the policy_string.
|
||||||
|
"""
|
||||||
|
if '-' in policy_string:
|
||||||
|
base, policy_index = policy_string.rsplit('-', 1)
|
||||||
|
else:
|
||||||
|
base, policy_index = policy_string, None
|
||||||
|
policy = POLICIES.get_by_index(policy_index)
|
||||||
|
if get_policy_string(base, policy) != policy_string:
|
||||||
|
raise PolicyError("Unknown policy", index=policy_index)
|
||||||
|
return base, policy
|
||||||
|
|
||||||
|
|
||||||
|
class BaseStoragePolicy(object):
|
||||||
|
"""
|
||||||
|
Represents a storage policy. Not meant to be instantiated directly;
|
||||||
|
implement a derived subclasses (e.g. StoragePolicy, ECStoragePolicy, etc)
|
||||||
|
or use :func:`~swift.common.storage_policy.reload_storage_policies` to
|
||||||
|
load POLICIES from ``swift.conf``.
|
||||||
|
|
||||||
The object_ring property is lazy loaded once the service's ``swift_dir``
|
The object_ring property is lazy loaded once the service's ``swift_dir``
|
||||||
is known via :meth:`~StoragePolicyCollection.get_object_ring`, but it may
|
is known via :meth:`~StoragePolicyCollection.get_object_ring`, but it may
|
||||||
be over-ridden via object_ring kwarg at create time for testing or
|
be over-ridden via object_ring kwarg at create time for testing or
|
||||||
actively loaded with :meth:`~StoragePolicy.load_ring`.
|
actively loaded with :meth:`~StoragePolicy.load_ring`.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
policy_type_to_policy_cls = {}
|
||||||
|
|
||||||
def __init__(self, idx, name='', is_default=False, is_deprecated=False,
|
def __init__(self, idx, name='', is_default=False, is_deprecated=False,
|
||||||
object_ring=None):
|
object_ring=None):
|
||||||
|
# do not allow BaseStoragePolicy class to be instantiated directly
|
||||||
|
if type(self) == BaseStoragePolicy:
|
||||||
|
raise TypeError("Can't instantiate BaseStoragePolicy directly")
|
||||||
|
# policy parameter validation
|
||||||
try:
|
try:
|
||||||
self.idx = int(idx)
|
self.idx = int(idx)
|
||||||
except ValueError:
|
except ValueError:
|
||||||
@ -88,6 +133,8 @@ class StoragePolicy(object):
|
|||||||
self.name = name
|
self.name = name
|
||||||
self.is_deprecated = config_true_value(is_deprecated)
|
self.is_deprecated = config_true_value(is_deprecated)
|
||||||
self.is_default = config_true_value(is_default)
|
self.is_default = config_true_value(is_default)
|
||||||
|
if self.policy_type not in BaseStoragePolicy.policy_type_to_policy_cls:
|
||||||
|
raise PolicyError('Invalid type', self.policy_type)
|
||||||
if self.is_deprecated and self.is_default:
|
if self.is_deprecated and self.is_default:
|
||||||
raise PolicyError('Deprecated policy can not be default. '
|
raise PolicyError('Deprecated policy can not be default. '
|
||||||
'Invalid config', self.idx)
|
'Invalid config', self.idx)
|
||||||
@ -101,8 +148,80 @@ class StoragePolicy(object):
|
|||||||
return cmp(self.idx, int(other))
|
return cmp(self.idx, int(other))
|
||||||
|
|
||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
return ("StoragePolicy(%d, %r, is_default=%s, is_deprecated=%s)") % (
|
return ("%s(%d, %r, is_default=%s, "
|
||||||
self.idx, self.name, self.is_default, self.is_deprecated)
|
"is_deprecated=%s, policy_type=%r)") % \
|
||||||
|
(self.__class__.__name__, self.idx, self.name,
|
||||||
|
self.is_default, self.is_deprecated, self.policy_type)
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def register(cls, policy_type):
|
||||||
|
"""
|
||||||
|
Decorator for Storage Policy implementations to register
|
||||||
|
their StoragePolicy class. This will also set the policy_type
|
||||||
|
attribute on the registered implementation.
|
||||||
|
"""
|
||||||
|
def register_wrapper(policy_cls):
|
||||||
|
if policy_type in cls.policy_type_to_policy_cls:
|
||||||
|
raise PolicyError(
|
||||||
|
'%r is already registered for the policy_type %r' % (
|
||||||
|
cls.policy_type_to_policy_cls[policy_type],
|
||||||
|
policy_type))
|
||||||
|
cls.policy_type_to_policy_cls[policy_type] = policy_cls
|
||||||
|
policy_cls.policy_type = policy_type
|
||||||
|
return policy_cls
|
||||||
|
return register_wrapper
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def _config_options_map(cls):
|
||||||
|
"""
|
||||||
|
Map config option name to StoragePolicy parameter name.
|
||||||
|
"""
|
||||||
|
return {
|
||||||
|
'name': 'name',
|
||||||
|
'policy_type': 'policy_type',
|
||||||
|
'default': 'is_default',
|
||||||
|
'deprecated': 'is_deprecated',
|
||||||
|
}
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def from_config(cls, policy_index, options):
|
||||||
|
config_to_policy_option_map = cls._config_options_map()
|
||||||
|
policy_options = {}
|
||||||
|
for config_option, value in options.items():
|
||||||
|
try:
|
||||||
|
policy_option = config_to_policy_option_map[config_option]
|
||||||
|
except KeyError:
|
||||||
|
raise PolicyError('Invalid option %r in '
|
||||||
|
'storage-policy section' % config_option,
|
||||||
|
index=policy_index)
|
||||||
|
policy_options[policy_option] = value
|
||||||
|
return cls(policy_index, **policy_options)
|
||||||
|
|
||||||
|
def get_info(self, config=False):
|
||||||
|
"""
|
||||||
|
Return the info dict and conf file options for this policy.
|
||||||
|
|
||||||
|
:param config: boolean, if True all config options are returned
|
||||||
|
"""
|
||||||
|
info = {}
|
||||||
|
for config_option, policy_attribute in \
|
||||||
|
self._config_options_map().items():
|
||||||
|
info[config_option] = getattr(self, policy_attribute)
|
||||||
|
if not config:
|
||||||
|
# remove some options for public consumption
|
||||||
|
if not self.is_default:
|
||||||
|
info.pop('default')
|
||||||
|
if not self.is_deprecated:
|
||||||
|
info.pop('deprecated')
|
||||||
|
info.pop('policy_type')
|
||||||
|
return info
|
||||||
|
|
||||||
|
def _validate_ring(self):
|
||||||
|
"""
|
||||||
|
Hook, called when the ring is loaded. Can be used to
|
||||||
|
validate the ring against the StoragePolicy configuration.
|
||||||
|
"""
|
||||||
|
pass
|
||||||
|
|
||||||
def load_ring(self, swift_dir):
|
def load_ring(self, swift_dir):
|
||||||
"""
|
"""
|
||||||
@ -114,11 +233,194 @@ class StoragePolicy(object):
|
|||||||
return
|
return
|
||||||
self.object_ring = Ring(swift_dir, ring_name=self.ring_name)
|
self.object_ring = Ring(swift_dir, ring_name=self.ring_name)
|
||||||
|
|
||||||
def get_options(self):
|
# Validate ring to make sure it conforms to policy requirements
|
||||||
"""Return the valid conf file options for this policy."""
|
self._validate_ring()
|
||||||
return {'name': self.name,
|
|
||||||
'default': self.is_default,
|
@property
|
||||||
'deprecated': self.is_deprecated}
|
def quorum(self):
|
||||||
|
"""
|
||||||
|
Number of successful backend requests needed for the proxy to
|
||||||
|
consider the client request successful.
|
||||||
|
"""
|
||||||
|
raise NotImplementedError()
|
||||||
|
|
||||||
|
|
||||||
|
@BaseStoragePolicy.register(REPL_POLICY)
|
||||||
|
class StoragePolicy(BaseStoragePolicy):
|
||||||
|
"""
|
||||||
|
Represents a storage policy of type 'replication'. Default storage policy
|
||||||
|
class unless otherwise overridden from swift.conf.
|
||||||
|
|
||||||
|
Not meant to be instantiated directly; use
|
||||||
|
:func:`~swift.common.storage_policy.reload_storage_policies` to load
|
||||||
|
POLICIES from ``swift.conf``.
|
||||||
|
"""
|
||||||
|
|
||||||
|
@property
|
||||||
|
def quorum(self):
|
||||||
|
"""
|
||||||
|
Quorum concept in the replication case:
|
||||||
|
floor(number of replica / 2) + 1
|
||||||
|
"""
|
||||||
|
if not self.object_ring:
|
||||||
|
raise PolicyError('Ring is not loaded')
|
||||||
|
return quorum_size(self.object_ring.replica_count)
|
||||||
|
|
||||||
|
|
||||||
|
@BaseStoragePolicy.register(EC_POLICY)
|
||||||
|
class ECStoragePolicy(BaseStoragePolicy):
|
||||||
|
"""
|
||||||
|
Represents a storage policy of type 'erasure_coding'.
|
||||||
|
|
||||||
|
Not meant to be instantiated directly; use
|
||||||
|
:func:`~swift.common.storage_policy.reload_storage_policies` to load
|
||||||
|
POLICIES from ``swift.conf``.
|
||||||
|
"""
|
||||||
|
def __init__(self, idx, name='', is_default=False,
|
||||||
|
is_deprecated=False, object_ring=None,
|
||||||
|
ec_segment_size=DEFAULT_EC_OBJECT_SEGMENT_SIZE,
|
||||||
|
ec_type=None, ec_ndata=None, ec_nparity=None):
|
||||||
|
|
||||||
|
super(ECStoragePolicy, self).__init__(
|
||||||
|
idx, name, is_default, is_deprecated, object_ring)
|
||||||
|
|
||||||
|
# Validate erasure_coding policy specific members
|
||||||
|
# ec_type is one of the EC implementations supported by PyEClib
|
||||||
|
if ec_type is None:
|
||||||
|
raise PolicyError('Missing ec_type')
|
||||||
|
if ec_type not in VALID_EC_TYPES:
|
||||||
|
raise PolicyError('Wrong ec_type %s for policy %s, should be one'
|
||||||
|
' of "%s"' % (ec_type, self.name,
|
||||||
|
', '.join(VALID_EC_TYPES)))
|
||||||
|
self._ec_type = ec_type
|
||||||
|
|
||||||
|
# Define _ec_ndata as the number of EC data fragments
|
||||||
|
# Accessible as the property "ec_ndata"
|
||||||
|
try:
|
||||||
|
value = int(ec_ndata)
|
||||||
|
if value <= 0:
|
||||||
|
raise ValueError
|
||||||
|
self._ec_ndata = value
|
||||||
|
except (TypeError, ValueError):
|
||||||
|
raise PolicyError('Invalid ec_num_data_fragments %r' %
|
||||||
|
ec_ndata, index=self.idx)
|
||||||
|
|
||||||
|
# Define _ec_nparity as the number of EC parity fragments
|
||||||
|
# Accessible as the property "ec_nparity"
|
||||||
|
try:
|
||||||
|
value = int(ec_nparity)
|
||||||
|
if value <= 0:
|
||||||
|
raise ValueError
|
||||||
|
self._ec_nparity = value
|
||||||
|
except (TypeError, ValueError):
|
||||||
|
raise PolicyError('Invalid ec_num_parity_fragments %r'
|
||||||
|
% ec_nparity, index=self.idx)
|
||||||
|
|
||||||
|
# Define _ec_segment_size as the encode segment unit size
|
||||||
|
# Accessible as the property "ec_segment_size"
|
||||||
|
try:
|
||||||
|
value = int(ec_segment_size)
|
||||||
|
if value <= 0:
|
||||||
|
raise ValueError
|
||||||
|
self._ec_segment_size = value
|
||||||
|
except (TypeError, ValueError):
|
||||||
|
raise PolicyError('Invalid ec_object_segment_size %r' %
|
||||||
|
ec_segment_size, index=self.idx)
|
||||||
|
|
||||||
|
# Initialize PyECLib EC backend
|
||||||
|
try:
|
||||||
|
self.pyeclib_driver = \
|
||||||
|
ECDriver(k=self._ec_ndata, m=self._ec_nparity,
|
||||||
|
ec_type=self._ec_type)
|
||||||
|
except ECDriverError as e:
|
||||||
|
raise PolicyError("Error creating EC policy (%s)" % e,
|
||||||
|
index=self.idx)
|
||||||
|
|
||||||
|
# quorum size in the EC case depends on the choice of EC scheme.
|
||||||
|
self._ec_quorum_size = \
|
||||||
|
self._ec_ndata + self.pyeclib_driver.min_parity_fragments_needed()
|
||||||
|
|
||||||
|
@property
|
||||||
|
def ec_type(self):
|
||||||
|
return self._ec_type
|
||||||
|
|
||||||
|
@property
|
||||||
|
def ec_ndata(self):
|
||||||
|
return self._ec_ndata
|
||||||
|
|
||||||
|
@property
|
||||||
|
def ec_nparity(self):
|
||||||
|
return self._ec_nparity
|
||||||
|
|
||||||
|
@property
|
||||||
|
def ec_segment_size(self):
|
||||||
|
return self._ec_segment_size
|
||||||
|
|
||||||
|
def __repr__(self):
|
||||||
|
return ("%s, EC config(ec_type=%s, ec_segment_size=%d, "
|
||||||
|
"ec_ndata=%d, ec_nparity=%d)") % (
|
||||||
|
super(ECStoragePolicy, self).__repr__(), self.ec_type,
|
||||||
|
self.ec_segment_size, self.ec_ndata, self.ec_nparity)
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def _config_options_map(cls):
|
||||||
|
options = super(ECStoragePolicy, cls)._config_options_map()
|
||||||
|
options.update({
|
||||||
|
'ec_type': 'ec_type',
|
||||||
|
'ec_object_segment_size': 'ec_segment_size',
|
||||||
|
'ec_num_data_fragments': 'ec_ndata',
|
||||||
|
'ec_num_parity_fragments': 'ec_nparity',
|
||||||
|
})
|
||||||
|
return options
|
||||||
|
|
||||||
|
def get_info(self, config=False):
|
||||||
|
info = super(ECStoragePolicy, self).get_info(config=config)
|
||||||
|
if not config:
|
||||||
|
info.pop('ec_object_segment_size')
|
||||||
|
info.pop('ec_num_data_fragments')
|
||||||
|
info.pop('ec_num_parity_fragments')
|
||||||
|
info.pop('ec_type')
|
||||||
|
return info
|
||||||
|
|
||||||
|
def _validate_ring(self):
|
||||||
|
"""
|
||||||
|
EC specific validation
|
||||||
|
|
||||||
|
Replica count check - we need _at_least_ (#data + #parity) replicas
|
||||||
|
configured. Also if the replica count is larger than exactly that
|
||||||
|
number there's a non-zero risk of error for code that is considering
|
||||||
|
the number of nodes in the primary list from the ring.
|
||||||
|
"""
|
||||||
|
if not self.object_ring:
|
||||||
|
raise PolicyError('Ring is not loaded')
|
||||||
|
nodes_configured = self.object_ring.replica_count
|
||||||
|
if nodes_configured != (self.ec_ndata + self.ec_nparity):
|
||||||
|
raise RingValidationError(
|
||||||
|
'EC ring for policy %s needs to be configured with '
|
||||||
|
'exactly %d nodes. Got %d.' % (self.name,
|
||||||
|
self.ec_ndata + self.ec_nparity, nodes_configured))
|
||||||
|
|
||||||
|
@property
|
||||||
|
def quorum(self):
|
||||||
|
"""
|
||||||
|
Number of successful backend requests needed for the proxy to consider
|
||||||
|
the client request successful.
|
||||||
|
|
||||||
|
The quorum size for EC policies defines the minimum number
|
||||||
|
of data + parity elements required to be able to guarantee
|
||||||
|
the desired fault tolerance, which is the number of data
|
||||||
|
elements supplemented by the minimum number of parity
|
||||||
|
elements required by the chosen erasure coding scheme.
|
||||||
|
|
||||||
|
For example, for Reed-Solomon, the minimum number parity
|
||||||
|
elements required is 1, and thus the quorum_size requirement
|
||||||
|
is ec_ndata + 1.
|
||||||
|
|
||||||
|
Given the number of parity elements required is not the same
|
||||||
|
for every erasure coding scheme, consult PyECLib for
|
||||||
|
min_parity_fragments_needed()
|
||||||
|
"""
|
||||||
|
return self._ec_quorum_size
|
||||||
|
|
||||||
|
|
||||||
class StoragePolicyCollection(object):
|
class StoragePolicyCollection(object):
|
||||||
@ -236,9 +538,19 @@ class StoragePolicyCollection(object):
|
|||||||
:returns: storage policy, or None if no such policy
|
:returns: storage policy, or None if no such policy
|
||||||
"""
|
"""
|
||||||
# makes it easier for callers to just pass in a header value
|
# makes it easier for callers to just pass in a header value
|
||||||
index = int(index) if index else 0
|
if index in ('', None):
|
||||||
|
index = 0
|
||||||
|
else:
|
||||||
|
try:
|
||||||
|
index = int(index)
|
||||||
|
except ValueError:
|
||||||
|
return None
|
||||||
return self.by_index.get(index)
|
return self.by_index.get(index)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def legacy(self):
|
||||||
|
return self.get_by_index(None)
|
||||||
|
|
||||||
def get_object_ring(self, policy_idx, swift_dir):
|
def get_object_ring(self, policy_idx, swift_dir):
|
||||||
"""
|
"""
|
||||||
Get the ring object to use to handle a request based on its policy.
|
Get the ring object to use to handle a request based on its policy.
|
||||||
@ -267,10 +579,7 @@ class StoragePolicyCollection(object):
|
|||||||
# delete from /info if deprecated
|
# delete from /info if deprecated
|
||||||
if pol.is_deprecated:
|
if pol.is_deprecated:
|
||||||
continue
|
continue
|
||||||
policy_entry = {}
|
policy_entry = pol.get_info()
|
||||||
policy_entry['name'] = pol.name
|
|
||||||
if pol.is_default:
|
|
||||||
policy_entry['default'] = pol.is_default
|
|
||||||
policy_info.append(policy_entry)
|
policy_info.append(policy_entry)
|
||||||
return policy_info
|
return policy_info
|
||||||
|
|
||||||
@ -287,22 +596,10 @@ def parse_storage_policies(conf):
|
|||||||
if not section.startswith('storage-policy:'):
|
if not section.startswith('storage-policy:'):
|
||||||
continue
|
continue
|
||||||
policy_index = section.split(':', 1)[1]
|
policy_index = section.split(':', 1)[1]
|
||||||
# map config option name to StoragePolicy parameter name
|
config_options = dict(conf.items(section))
|
||||||
config_to_policy_option_map = {
|
policy_type = config_options.pop('policy_type', DEFAULT_POLICY_TYPE)
|
||||||
'name': 'name',
|
policy_cls = BaseStoragePolicy.policy_type_to_policy_cls[policy_type]
|
||||||
'default': 'is_default',
|
policy = policy_cls.from_config(policy_index, config_options)
|
||||||
'deprecated': 'is_deprecated',
|
|
||||||
}
|
|
||||||
policy_options = {}
|
|
||||||
for config_option, value in conf.items(section):
|
|
||||||
try:
|
|
||||||
policy_option = config_to_policy_option_map[config_option]
|
|
||||||
except KeyError:
|
|
||||||
raise PolicyError('Invalid option %r in '
|
|
||||||
'storage-policy section %r' % (
|
|
||||||
config_option, section))
|
|
||||||
policy_options[policy_option] = value
|
|
||||||
policy = StoragePolicy(policy_index, **policy_options)
|
|
||||||
policies.append(policy)
|
policies.append(policy)
|
||||||
|
|
||||||
return StoragePolicyCollection(policies)
|
return StoragePolicyCollection(policies)
|
||||||
|
@ -63,7 +63,7 @@ from swift.common.exceptions import DiskFileQuarantined, DiskFileNotExist, \
|
|||||||
DiskFileDeleted, DiskFileError, DiskFileNotOpen, PathNotDir, \
|
DiskFileDeleted, DiskFileError, DiskFileNotOpen, PathNotDir, \
|
||||||
ReplicationLockTimeout, DiskFileExpired, DiskFileXattrNotSupported
|
ReplicationLockTimeout, DiskFileExpired, DiskFileXattrNotSupported
|
||||||
from swift.common.swob import multi_range_iterator
|
from swift.common.swob import multi_range_iterator
|
||||||
from swift.common.storage_policy import get_policy_string, POLICIES
|
from swift.common.storage_policy import get_policy_string, split_policy_string
|
||||||
from functools import partial
|
from functools import partial
|
||||||
|
|
||||||
|
|
||||||
@ -178,11 +178,7 @@ def extract_policy_index(obj_path):
|
|||||||
obj_dirname = obj_portion[:obj_portion.index('/')]
|
obj_dirname = obj_portion[:obj_portion.index('/')]
|
||||||
except Exception:
|
except Exception:
|
||||||
return policy_idx
|
return policy_idx
|
||||||
if '-' in obj_dirname:
|
return int(split_policy_string(obj_dirname)[1])
|
||||||
base, policy_idx = obj_dirname.split('-', 1)
|
|
||||||
if POLICIES.get_by_index(policy_idx) is None:
|
|
||||||
policy_idx = 0
|
|
||||||
return int(policy_idx)
|
|
||||||
|
|
||||||
|
|
||||||
def quarantine_renamer(device_path, corrupted_file_path):
|
def quarantine_renamer(device_path, corrupted_file_path):
|
||||||
@ -474,11 +470,8 @@ def object_audit_location_generator(devices, mount_check=True, logger=None,
|
|||||||
if dir.startswith(DATADIR_BASE)]:
|
if dir.startswith(DATADIR_BASE)]:
|
||||||
datadir_path = os.path.join(devices, device, dir)
|
datadir_path = os.path.join(devices, device, dir)
|
||||||
# warn if the object dir doesn't match with a policy
|
# warn if the object dir doesn't match with a policy
|
||||||
policy_idx = 0
|
|
||||||
if '-' in dir:
|
|
||||||
base, policy_idx = dir.split('-', 1)
|
|
||||||
try:
|
try:
|
||||||
get_data_dir(policy_idx)
|
base, policy = split_policy_string(dir)
|
||||||
except ValueError:
|
except ValueError:
|
||||||
if logger:
|
if logger:
|
||||||
logger.warn(_('Directory %s does not map to a '
|
logger.warn(_('Directory %s does not map to a '
|
||||||
|
@ -223,7 +223,7 @@ def _in_process_setup_ring(swift_conf, conf_src_dir, testdir):
|
|||||||
# make policy_to_test be policy index 0 and default for the test config
|
# make policy_to_test be policy index 0 and default for the test config
|
||||||
sp_zero_section = sp_prefix + '0'
|
sp_zero_section = sp_prefix + '0'
|
||||||
conf.add_section(sp_zero_section)
|
conf.add_section(sp_zero_section)
|
||||||
for (k, v) in policy_to_test.get_options().items():
|
for (k, v) in policy_to_test.get_info(config=True).items():
|
||||||
conf.set(sp_zero_section, k, v)
|
conf.set(sp_zero_section, k, v)
|
||||||
conf.set(sp_zero_section, 'default', True)
|
conf.set(sp_zero_section, 'default', True)
|
||||||
|
|
||||||
|
@ -235,19 +235,20 @@ class TestInternalClient(unittest.TestCase):
|
|||||||
write_fake_ring(object_ring_path)
|
write_fake_ring(object_ring_path)
|
||||||
with patch_policies([StoragePolicy(0, 'legacy', True)]):
|
with patch_policies([StoragePolicy(0, 'legacy', True)]):
|
||||||
client = internal_client.InternalClient(conf_path, 'test', 1)
|
client = internal_client.InternalClient(conf_path, 'test', 1)
|
||||||
self.assertEqual(client.account_ring, client.app.app.app.account_ring)
|
self.assertEqual(client.account_ring,
|
||||||
self.assertEqual(client.account_ring.serialized_path,
|
client.app.app.app.account_ring)
|
||||||
account_ring_path)
|
self.assertEqual(client.account_ring.serialized_path,
|
||||||
self.assertEqual(client.container_ring,
|
account_ring_path)
|
||||||
client.app.app.app.container_ring)
|
self.assertEqual(client.container_ring,
|
||||||
self.assertEqual(client.container_ring.serialized_path,
|
client.app.app.app.container_ring)
|
||||||
container_ring_path)
|
self.assertEqual(client.container_ring.serialized_path,
|
||||||
object_ring = client.app.app.app.get_object_ring(0)
|
container_ring_path)
|
||||||
self.assertEqual(client.get_object_ring(0),
|
object_ring = client.app.app.app.get_object_ring(0)
|
||||||
object_ring)
|
self.assertEqual(client.get_object_ring(0),
|
||||||
self.assertEqual(object_ring.serialized_path,
|
object_ring)
|
||||||
object_ring_path)
|
self.assertEqual(object_ring.serialized_path,
|
||||||
self.assertEquals(client.auto_create_account_prefix, '-')
|
object_ring_path)
|
||||||
|
self.assertEquals(client.auto_create_account_prefix, '-')
|
||||||
|
|
||||||
def test_init(self):
|
def test_init(self):
|
||||||
class App(object):
|
class App(object):
|
||||||
|
@ -19,8 +19,23 @@ import mock
|
|||||||
from tempfile import NamedTemporaryFile
|
from tempfile import NamedTemporaryFile
|
||||||
from test.unit import patch_policies, FakeRing
|
from test.unit import patch_policies, FakeRing
|
||||||
from swift.common.storage_policy import (
|
from swift.common.storage_policy import (
|
||||||
StoragePolicy, StoragePolicyCollection, POLICIES, PolicyError,
|
StoragePolicyCollection, POLICIES, PolicyError, parse_storage_policies,
|
||||||
parse_storage_policies, reload_storage_policies, get_policy_string)
|
reload_storage_policies, get_policy_string, split_policy_string,
|
||||||
|
BaseStoragePolicy, StoragePolicy, ECStoragePolicy, REPL_POLICY, EC_POLICY,
|
||||||
|
VALID_EC_TYPES, DEFAULT_EC_OBJECT_SEGMENT_SIZE)
|
||||||
|
from swift.common.exceptions import RingValidationError
|
||||||
|
|
||||||
|
|
||||||
|
@BaseStoragePolicy.register('fake')
|
||||||
|
class FakeStoragePolicy(BaseStoragePolicy):
|
||||||
|
"""
|
||||||
|
Test StoragePolicy class - the only user at the moment is
|
||||||
|
test_validate_policies_type_invalid()
|
||||||
|
"""
|
||||||
|
def __init__(self, idx, name='', is_default=False, is_deprecated=False,
|
||||||
|
object_ring=None):
|
||||||
|
super(FakeStoragePolicy, self).__init__(
|
||||||
|
idx, name, is_default, is_deprecated, object_ring)
|
||||||
|
|
||||||
|
|
||||||
class TestStoragePolicies(unittest.TestCase):
|
class TestStoragePolicies(unittest.TestCase):
|
||||||
@ -31,15 +46,35 @@ class TestStoragePolicies(unittest.TestCase):
|
|||||||
conf.readfp(StringIO.StringIO(conf_str))
|
conf.readfp(StringIO.StringIO(conf_str))
|
||||||
return conf
|
return conf
|
||||||
|
|
||||||
@patch_policies([StoragePolicy(0, 'zero', True),
|
def assertRaisesWithMessage(self, exc_class, message, f, *args, **kwargs):
|
||||||
StoragePolicy(1, 'one', False),
|
try:
|
||||||
StoragePolicy(2, 'two', False),
|
f(*args, **kwargs)
|
||||||
StoragePolicy(3, 'three', False, is_deprecated=True)])
|
except exc_class as err:
|
||||||
|
err_msg = str(err)
|
||||||
|
self.assert_(message in err_msg, 'Error message %r did not '
|
||||||
|
'have expected substring %r' % (err_msg, message))
|
||||||
|
else:
|
||||||
|
self.fail('%r did not raise %s' % (message, exc_class.__name__))
|
||||||
|
|
||||||
|
def test_policy_baseclass_instantiate(self):
|
||||||
|
self.assertRaisesWithMessage(TypeError,
|
||||||
|
"Can't instantiate BaseStoragePolicy",
|
||||||
|
BaseStoragePolicy, 1, 'one')
|
||||||
|
|
||||||
|
@patch_policies([
|
||||||
|
StoragePolicy(0, 'zero', is_default=True),
|
||||||
|
StoragePolicy(1, 'one'),
|
||||||
|
StoragePolicy(2, 'two'),
|
||||||
|
StoragePolicy(3, 'three', is_deprecated=True),
|
||||||
|
ECStoragePolicy(10, 'ten', ec_type='jerasure_rs_vand',
|
||||||
|
ec_ndata=10, ec_nparity=4),
|
||||||
|
])
|
||||||
def test_swift_info(self):
|
def test_swift_info(self):
|
||||||
# the deprecated 'three' should not exist in expect
|
# the deprecated 'three' should not exist in expect
|
||||||
expect = [{'default': True, 'name': 'zero'},
|
expect = [{'default': True, 'name': 'zero'},
|
||||||
{'name': 'two'},
|
{'name': 'two'},
|
||||||
{'name': 'one'}]
|
{'name': 'one'},
|
||||||
|
{'name': 'ten'}]
|
||||||
swift_info = POLICIES.get_policy_info()
|
swift_info = POLICIES.get_policy_info()
|
||||||
self.assertEquals(sorted(expect, key=lambda k: k['name']),
|
self.assertEquals(sorted(expect, key=lambda k: k['name']),
|
||||||
sorted(swift_info, key=lambda k: k['name']))
|
sorted(swift_info, key=lambda k: k['name']))
|
||||||
@ -48,10 +83,48 @@ class TestStoragePolicies(unittest.TestCase):
|
|||||||
def test_get_policy_string(self):
|
def test_get_policy_string(self):
|
||||||
self.assertEquals(get_policy_string('something', 0), 'something')
|
self.assertEquals(get_policy_string('something', 0), 'something')
|
||||||
self.assertEquals(get_policy_string('something', None), 'something')
|
self.assertEquals(get_policy_string('something', None), 'something')
|
||||||
|
self.assertEquals(get_policy_string('something', ''), 'something')
|
||||||
self.assertEquals(get_policy_string('something', 1),
|
self.assertEquals(get_policy_string('something', 1),
|
||||||
'something' + '-1')
|
'something' + '-1')
|
||||||
self.assertRaises(PolicyError, get_policy_string, 'something', 99)
|
self.assertRaises(PolicyError, get_policy_string, 'something', 99)
|
||||||
|
|
||||||
|
@patch_policies
|
||||||
|
def test_split_policy_string(self):
|
||||||
|
expectations = {
|
||||||
|
'something': ('something', POLICIES[0]),
|
||||||
|
'something-1': ('something', POLICIES[1]),
|
||||||
|
'tmp': ('tmp', POLICIES[0]),
|
||||||
|
'objects': ('objects', POLICIES[0]),
|
||||||
|
'tmp-1': ('tmp', POLICIES[1]),
|
||||||
|
'objects-1': ('objects', POLICIES[1]),
|
||||||
|
'objects-': PolicyError,
|
||||||
|
'objects-0': PolicyError,
|
||||||
|
'objects--1': ('objects-', POLICIES[1]),
|
||||||
|
'objects-+1': PolicyError,
|
||||||
|
'objects--': PolicyError,
|
||||||
|
'objects-foo': PolicyError,
|
||||||
|
'objects--bar': PolicyError,
|
||||||
|
'objects-+bar': PolicyError,
|
||||||
|
# questionable, demonstrated as inverse of get_policy_string
|
||||||
|
'objects+0': ('objects+0', POLICIES[0]),
|
||||||
|
'': ('', POLICIES[0]),
|
||||||
|
'0': ('0', POLICIES[0]),
|
||||||
|
'-1': ('', POLICIES[1]),
|
||||||
|
}
|
||||||
|
for policy_string, expected in expectations.items():
|
||||||
|
if expected == PolicyError:
|
||||||
|
try:
|
||||||
|
invalid = split_policy_string(policy_string)
|
||||||
|
except PolicyError:
|
||||||
|
continue # good
|
||||||
|
else:
|
||||||
|
self.fail('The string %r returned %r '
|
||||||
|
'instead of raising a PolicyError' %
|
||||||
|
(policy_string, invalid))
|
||||||
|
self.assertEqual(expected, split_policy_string(policy_string))
|
||||||
|
# should be inverse of get_policy_string
|
||||||
|
self.assertEqual(policy_string, get_policy_string(*expected))
|
||||||
|
|
||||||
def test_defaults(self):
|
def test_defaults(self):
|
||||||
self.assertTrue(len(POLICIES) > 0)
|
self.assertTrue(len(POLICIES) > 0)
|
||||||
|
|
||||||
@ -66,7 +139,9 @@ class TestStoragePolicies(unittest.TestCase):
|
|||||||
def test_storage_policy_repr(self):
|
def test_storage_policy_repr(self):
|
||||||
test_policies = [StoragePolicy(0, 'aay', True),
|
test_policies = [StoragePolicy(0, 'aay', True),
|
||||||
StoragePolicy(1, 'bee', False),
|
StoragePolicy(1, 'bee', False),
|
||||||
StoragePolicy(2, 'cee', False)]
|
StoragePolicy(2, 'cee', False),
|
||||||
|
ECStoragePolicy(10, 'ten', ec_type='jerasure_rs_vand',
|
||||||
|
ec_ndata=10, ec_nparity=3)]
|
||||||
policies = StoragePolicyCollection(test_policies)
|
policies = StoragePolicyCollection(test_policies)
|
||||||
for policy in policies:
|
for policy in policies:
|
||||||
policy_repr = repr(policy)
|
policy_repr = repr(policy)
|
||||||
@ -75,6 +150,13 @@ class TestStoragePolicies(unittest.TestCase):
|
|||||||
self.assert_('is_deprecated=%s' % policy.is_deprecated in
|
self.assert_('is_deprecated=%s' % policy.is_deprecated in
|
||||||
policy_repr)
|
policy_repr)
|
||||||
self.assert_(policy.name in policy_repr)
|
self.assert_(policy.name in policy_repr)
|
||||||
|
if policy.policy_type == EC_POLICY:
|
||||||
|
self.assert_('ec_type=%s' % policy.ec_type in policy_repr)
|
||||||
|
self.assert_('ec_ndata=%s' % policy.ec_ndata in policy_repr)
|
||||||
|
self.assert_('ec_nparity=%s' %
|
||||||
|
policy.ec_nparity in policy_repr)
|
||||||
|
self.assert_('ec_segment_size=%s' %
|
||||||
|
policy.ec_segment_size in policy_repr)
|
||||||
collection_repr = repr(policies)
|
collection_repr = repr(policies)
|
||||||
collection_repr_lines = collection_repr.splitlines()
|
collection_repr_lines = collection_repr.splitlines()
|
||||||
self.assert_(policies.__class__.__name__ in collection_repr_lines[0])
|
self.assert_(policies.__class__.__name__ in collection_repr_lines[0])
|
||||||
@ -157,15 +239,16 @@ class TestStoragePolicies(unittest.TestCase):
|
|||||||
def test_validate_policy_params(self):
|
def test_validate_policy_params(self):
|
||||||
StoragePolicy(0, 'name') # sanity
|
StoragePolicy(0, 'name') # sanity
|
||||||
# bogus indexes
|
# bogus indexes
|
||||||
self.assertRaises(PolicyError, StoragePolicy, 'x', 'name')
|
self.assertRaises(PolicyError, FakeStoragePolicy, 'x', 'name')
|
||||||
self.assertRaises(PolicyError, StoragePolicy, -1, 'name')
|
self.assertRaises(PolicyError, FakeStoragePolicy, -1, 'name')
|
||||||
|
|
||||||
# non-zero Policy-0
|
# non-zero Policy-0
|
||||||
self.assertRaisesWithMessage(PolicyError, 'reserved', StoragePolicy,
|
self.assertRaisesWithMessage(PolicyError, 'reserved',
|
||||||
1, 'policy-0')
|
FakeStoragePolicy, 1, 'policy-0')
|
||||||
# deprecate default
|
# deprecate default
|
||||||
self.assertRaisesWithMessage(
|
self.assertRaisesWithMessage(
|
||||||
PolicyError, 'Deprecated policy can not be default',
|
PolicyError, 'Deprecated policy can not be default',
|
||||||
StoragePolicy, 1, 'Policy-1', is_default=True,
|
FakeStoragePolicy, 1, 'Policy-1', is_default=True,
|
||||||
is_deprecated=True)
|
is_deprecated=True)
|
||||||
# weird names
|
# weird names
|
||||||
names = (
|
names = (
|
||||||
@ -178,7 +261,7 @@ class TestStoragePolicies(unittest.TestCase):
|
|||||||
)
|
)
|
||||||
for name in names:
|
for name in names:
|
||||||
self.assertRaisesWithMessage(PolicyError, 'Invalid name',
|
self.assertRaisesWithMessage(PolicyError, 'Invalid name',
|
||||||
StoragePolicy, 1, name)
|
FakeStoragePolicy, 1, name)
|
||||||
|
|
||||||
def test_validate_policies_names(self):
|
def test_validate_policies_names(self):
|
||||||
# duplicate names
|
# duplicate names
|
||||||
@ -188,6 +271,40 @@ class TestStoragePolicies(unittest.TestCase):
|
|||||||
self.assertRaises(PolicyError, StoragePolicyCollection,
|
self.assertRaises(PolicyError, StoragePolicyCollection,
|
||||||
test_policies)
|
test_policies)
|
||||||
|
|
||||||
|
def test_validate_policies_type_default(self):
|
||||||
|
# no type specified - make sure the policy is initialized to
|
||||||
|
# DEFAULT_POLICY_TYPE
|
||||||
|
test_policy = FakeStoragePolicy(0, 'zero', True)
|
||||||
|
self.assertEquals(test_policy.policy_type, 'fake')
|
||||||
|
|
||||||
|
def test_validate_policies_type_invalid(self):
|
||||||
|
class BogusStoragePolicy(FakeStoragePolicy):
|
||||||
|
policy_type = 'bogus'
|
||||||
|
# unsupported policy type - initialization with FakeStoragePolicy
|
||||||
|
self.assertRaisesWithMessage(PolicyError, 'Invalid type',
|
||||||
|
BogusStoragePolicy, 1, 'one')
|
||||||
|
|
||||||
|
def test_policies_type_attribute(self):
|
||||||
|
test_policies = [
|
||||||
|
StoragePolicy(0, 'zero', is_default=True),
|
||||||
|
StoragePolicy(1, 'one'),
|
||||||
|
StoragePolicy(2, 'two'),
|
||||||
|
StoragePolicy(3, 'three', is_deprecated=True),
|
||||||
|
ECStoragePolicy(10, 'ten', ec_type='jerasure_rs_vand',
|
||||||
|
ec_ndata=10, ec_nparity=3),
|
||||||
|
]
|
||||||
|
policies = StoragePolicyCollection(test_policies)
|
||||||
|
self.assertEquals(policies.get_by_index(0).policy_type,
|
||||||
|
REPL_POLICY)
|
||||||
|
self.assertEquals(policies.get_by_index(1).policy_type,
|
||||||
|
REPL_POLICY)
|
||||||
|
self.assertEquals(policies.get_by_index(2).policy_type,
|
||||||
|
REPL_POLICY)
|
||||||
|
self.assertEquals(policies.get_by_index(3).policy_type,
|
||||||
|
REPL_POLICY)
|
||||||
|
self.assertEquals(policies.get_by_index(10).policy_type,
|
||||||
|
EC_POLICY)
|
||||||
|
|
||||||
def test_names_are_normalized(self):
|
def test_names_are_normalized(self):
|
||||||
test_policies = [StoragePolicy(0, 'zero', True),
|
test_policies = [StoragePolicy(0, 'zero', True),
|
||||||
StoragePolicy(1, 'ZERO', False)]
|
StoragePolicy(1, 'ZERO', False)]
|
||||||
@ -207,16 +324,6 @@ class TestStoragePolicies(unittest.TestCase):
|
|||||||
self.assertEqual(pol1, policies.get_by_name(name))
|
self.assertEqual(pol1, policies.get_by_name(name))
|
||||||
self.assertEqual(policies.get_by_name(name).name, 'One')
|
self.assertEqual(policies.get_by_name(name).name, 'One')
|
||||||
|
|
||||||
def assertRaisesWithMessage(self, exc_class, message, f, *args, **kwargs):
|
|
||||||
try:
|
|
||||||
f(*args, **kwargs)
|
|
||||||
except exc_class as err:
|
|
||||||
err_msg = str(err)
|
|
||||||
self.assert_(message in err_msg, 'Error message %r did not '
|
|
||||||
'have expected substring %r' % (err_msg, message))
|
|
||||||
else:
|
|
||||||
self.fail('%r did not raise %s' % (message, exc_class.__name__))
|
|
||||||
|
|
||||||
def test_deprecated_default(self):
|
def test_deprecated_default(self):
|
||||||
bad_conf = self._conf("""
|
bad_conf = self._conf("""
|
||||||
[storage-policy:1]
|
[storage-policy:1]
|
||||||
@ -395,6 +502,133 @@ class TestStoragePolicies(unittest.TestCase):
|
|||||||
self.assertRaisesWithMessage(PolicyError, 'Invalid name',
|
self.assertRaisesWithMessage(PolicyError, 'Invalid name',
|
||||||
parse_storage_policies, bad_conf)
|
parse_storage_policies, bad_conf)
|
||||||
|
|
||||||
|
# policy_type = erasure_coding
|
||||||
|
|
||||||
|
# missing ec_type, ec_num_data_fragments and ec_num_parity_fragments
|
||||||
|
bad_conf = self._conf("""
|
||||||
|
[storage-policy:0]
|
||||||
|
name = zero
|
||||||
|
[storage-policy:1]
|
||||||
|
name = ec10-4
|
||||||
|
policy_type = erasure_coding
|
||||||
|
""")
|
||||||
|
|
||||||
|
self.assertRaisesWithMessage(PolicyError, 'Missing ec_type',
|
||||||
|
parse_storage_policies, bad_conf)
|
||||||
|
|
||||||
|
# missing ec_type, but other options valid...
|
||||||
|
bad_conf = self._conf("""
|
||||||
|
[storage-policy:0]
|
||||||
|
name = zero
|
||||||
|
[storage-policy:1]
|
||||||
|
name = ec10-4
|
||||||
|
policy_type = erasure_coding
|
||||||
|
ec_num_data_fragments = 10
|
||||||
|
ec_num_parity_fragments = 4
|
||||||
|
""")
|
||||||
|
|
||||||
|
self.assertRaisesWithMessage(PolicyError, 'Missing ec_type',
|
||||||
|
parse_storage_policies, bad_conf)
|
||||||
|
|
||||||
|
# ec_type specified, but invalid...
|
||||||
|
bad_conf = self._conf("""
|
||||||
|
[storage-policy:0]
|
||||||
|
name = zero
|
||||||
|
default = yes
|
||||||
|
[storage-policy:1]
|
||||||
|
name = ec10-4
|
||||||
|
policy_type = erasure_coding
|
||||||
|
ec_type = garbage_alg
|
||||||
|
ec_num_data_fragments = 10
|
||||||
|
ec_num_parity_fragments = 4
|
||||||
|
""")
|
||||||
|
|
||||||
|
self.assertRaisesWithMessage(PolicyError,
|
||||||
|
'Wrong ec_type garbage_alg for policy '
|
||||||
|
'ec10-4, should be one of "%s"' %
|
||||||
|
(', '.join(VALID_EC_TYPES)),
|
||||||
|
parse_storage_policies, bad_conf)
|
||||||
|
|
||||||
|
# missing and invalid ec_num_parity_fragments
|
||||||
|
bad_conf = self._conf("""
|
||||||
|
[storage-policy:0]
|
||||||
|
name = zero
|
||||||
|
[storage-policy:1]
|
||||||
|
name = ec10-4
|
||||||
|
policy_type = erasure_coding
|
||||||
|
ec_type = jerasure_rs_vand
|
||||||
|
ec_num_data_fragments = 10
|
||||||
|
""")
|
||||||
|
|
||||||
|
self.assertRaisesWithMessage(PolicyError,
|
||||||
|
'Invalid ec_num_parity_fragments',
|
||||||
|
parse_storage_policies, bad_conf)
|
||||||
|
|
||||||
|
for num_parity in ('-4', '0', 'x'):
|
||||||
|
bad_conf = self._conf("""
|
||||||
|
[storage-policy:0]
|
||||||
|
name = zero
|
||||||
|
[storage-policy:1]
|
||||||
|
name = ec10-4
|
||||||
|
policy_type = erasure_coding
|
||||||
|
ec_type = jerasure_rs_vand
|
||||||
|
ec_num_data_fragments = 10
|
||||||
|
ec_num_parity_fragments = %s
|
||||||
|
""" % num_parity)
|
||||||
|
|
||||||
|
self.assertRaisesWithMessage(PolicyError,
|
||||||
|
'Invalid ec_num_parity_fragments',
|
||||||
|
parse_storage_policies, bad_conf)
|
||||||
|
|
||||||
|
# missing and invalid ec_num_data_fragments
|
||||||
|
bad_conf = self._conf("""
|
||||||
|
[storage-policy:0]
|
||||||
|
name = zero
|
||||||
|
[storage-policy:1]
|
||||||
|
name = ec10-4
|
||||||
|
policy_type = erasure_coding
|
||||||
|
ec_type = jerasure_rs_vand
|
||||||
|
ec_num_parity_fragments = 4
|
||||||
|
""")
|
||||||
|
|
||||||
|
self.assertRaisesWithMessage(PolicyError,
|
||||||
|
'Invalid ec_num_data_fragments',
|
||||||
|
parse_storage_policies, bad_conf)
|
||||||
|
|
||||||
|
for num_data in ('-10', '0', 'x'):
|
||||||
|
bad_conf = self._conf("""
|
||||||
|
[storage-policy:0]
|
||||||
|
name = zero
|
||||||
|
[storage-policy:1]
|
||||||
|
name = ec10-4
|
||||||
|
policy_type = erasure_coding
|
||||||
|
ec_type = jerasure_rs_vand
|
||||||
|
ec_num_data_fragments = %s
|
||||||
|
ec_num_parity_fragments = 4
|
||||||
|
""" % num_data)
|
||||||
|
|
||||||
|
self.assertRaisesWithMessage(PolicyError,
|
||||||
|
'Invalid ec_num_data_fragments',
|
||||||
|
parse_storage_policies, bad_conf)
|
||||||
|
|
||||||
|
# invalid ec_object_segment_size
|
||||||
|
for segment_size in ('-4', '0', 'x'):
|
||||||
|
bad_conf = self._conf("""
|
||||||
|
[storage-policy:0]
|
||||||
|
name = zero
|
||||||
|
[storage-policy:1]
|
||||||
|
name = ec10-4
|
||||||
|
policy_type = erasure_coding
|
||||||
|
ec_object_segment_size = %s
|
||||||
|
ec_type = jerasure_rs_vand
|
||||||
|
ec_num_data_fragments = 10
|
||||||
|
ec_num_parity_fragments = 4
|
||||||
|
""" % segment_size)
|
||||||
|
|
||||||
|
self.assertRaisesWithMessage(PolicyError,
|
||||||
|
'Invalid ec_object_segment_size',
|
||||||
|
parse_storage_policies, bad_conf)
|
||||||
|
|
||||||
# Additional section added to ensure parser ignores other sections
|
# Additional section added to ensure parser ignores other sections
|
||||||
conf = self._conf("""
|
conf = self._conf("""
|
||||||
[some-other-section]
|
[some-other-section]
|
||||||
@ -430,6 +664,8 @@ class TestStoragePolicies(unittest.TestCase):
|
|||||||
self.assertEquals("zero", policies.get_by_index(None).name)
|
self.assertEquals("zero", policies.get_by_index(None).name)
|
||||||
self.assertEquals("zero", policies.get_by_index('').name)
|
self.assertEquals("zero", policies.get_by_index('').name)
|
||||||
|
|
||||||
|
self.assertEqual(policies.get_by_index(0), policies.legacy)
|
||||||
|
|
||||||
def test_reload_invalid_storage_policies(self):
|
def test_reload_invalid_storage_policies(self):
|
||||||
conf = self._conf("""
|
conf = self._conf("""
|
||||||
[storage-policy:0]
|
[storage-policy:0]
|
||||||
@ -512,18 +748,124 @@ class TestStoragePolicies(unittest.TestCase):
|
|||||||
for policy in POLICIES:
|
for policy in POLICIES:
|
||||||
self.assertEqual(POLICIES[int(policy)], policy)
|
self.assertEqual(POLICIES[int(policy)], policy)
|
||||||
|
|
||||||
def test_storage_policy_get_options(self):
|
def test_quorum_size_replication(self):
|
||||||
policy = StoragePolicy(1, 'gold', True, False)
|
expected_sizes = {1: 1,
|
||||||
self.assertEqual({'name': 'gold',
|
2: 2,
|
||||||
'default': True,
|
3: 2,
|
||||||
'deprecated': False},
|
4: 3,
|
||||||
policy.get_options())
|
5: 3}
|
||||||
|
for n, expected in expected_sizes.items():
|
||||||
|
policy = StoragePolicy(0, 'zero',
|
||||||
|
object_ring=FakeRing(replicas=n))
|
||||||
|
self.assertEqual(policy.quorum, expected)
|
||||||
|
|
||||||
policy = StoragePolicy(1, 'gold', False, True)
|
def test_quorum_size_erasure_coding(self):
|
||||||
self.assertEqual({'name': 'gold',
|
test_ec_policies = [
|
||||||
'default': False,
|
ECStoragePolicy(10, 'ec8-2', ec_type='jerasure_rs_vand',
|
||||||
'deprecated': True},
|
ec_ndata=8, ec_nparity=2),
|
||||||
policy.get_options())
|
ECStoragePolicy(11, 'df10-6', ec_type='flat_xor_hd_4',
|
||||||
|
ec_ndata=10, ec_nparity=6),
|
||||||
|
]
|
||||||
|
for ec_policy in test_ec_policies:
|
||||||
|
k = ec_policy.ec_ndata
|
||||||
|
expected_size = \
|
||||||
|
k + ec_policy.pyeclib_driver.min_parity_fragments_needed()
|
||||||
|
self.assertEqual(expected_size, ec_policy.quorum)
|
||||||
|
|
||||||
|
def test_validate_ring(self):
|
||||||
|
test_policies = [
|
||||||
|
ECStoragePolicy(0, 'ec8-2', ec_type='jerasure_rs_vand',
|
||||||
|
ec_ndata=8, ec_nparity=2,
|
||||||
|
object_ring=FakeRing(replicas=8),
|
||||||
|
is_default=True),
|
||||||
|
ECStoragePolicy(1, 'ec10-4', ec_type='jerasure_rs_vand',
|
||||||
|
ec_ndata=10, ec_nparity=4,
|
||||||
|
object_ring=FakeRing(replicas=10)),
|
||||||
|
ECStoragePolicy(2, 'ec4-2', ec_type='jerasure_rs_vand',
|
||||||
|
ec_ndata=4, ec_nparity=2,
|
||||||
|
object_ring=FakeRing(replicas=7)),
|
||||||
|
]
|
||||||
|
policies = StoragePolicyCollection(test_policies)
|
||||||
|
|
||||||
|
for policy in policies:
|
||||||
|
msg = 'EC ring for policy %s needs to be configured with ' \
|
||||||
|
'exactly %d nodes.' % \
|
||||||
|
(policy.name, policy.ec_ndata + policy.ec_nparity)
|
||||||
|
self.assertRaisesWithMessage(
|
||||||
|
RingValidationError, msg,
|
||||||
|
policy._validate_ring)
|
||||||
|
|
||||||
|
def test_storage_policy_get_info(self):
|
||||||
|
test_policies = [
|
||||||
|
StoragePolicy(0, 'zero', is_default=True),
|
||||||
|
StoragePolicy(1, 'one', is_deprecated=True),
|
||||||
|
ECStoragePolicy(10, 'ten',
|
||||||
|
ec_type='jerasure_rs_vand',
|
||||||
|
ec_ndata=10, ec_nparity=3),
|
||||||
|
ECStoragePolicy(11, 'done', is_deprecated=True,
|
||||||
|
ec_type='jerasure_rs_vand',
|
||||||
|
ec_ndata=10, ec_nparity=3),
|
||||||
|
]
|
||||||
|
policies = StoragePolicyCollection(test_policies)
|
||||||
|
expected = {
|
||||||
|
# default replication
|
||||||
|
(0, True): {
|
||||||
|
'name': 'zero',
|
||||||
|
'default': True,
|
||||||
|
'deprecated': False,
|
||||||
|
'policy_type': REPL_POLICY
|
||||||
|
},
|
||||||
|
(0, False): {
|
||||||
|
'name': 'zero',
|
||||||
|
'default': True,
|
||||||
|
},
|
||||||
|
# deprecated replication
|
||||||
|
(1, True): {
|
||||||
|
'name': 'one',
|
||||||
|
'default': False,
|
||||||
|
'deprecated': True,
|
||||||
|
'policy_type': REPL_POLICY
|
||||||
|
},
|
||||||
|
(1, False): {
|
||||||
|
'name': 'one',
|
||||||
|
'deprecated': True,
|
||||||
|
},
|
||||||
|
# enabled ec
|
||||||
|
(10, True): {
|
||||||
|
'name': 'ten',
|
||||||
|
'default': False,
|
||||||
|
'deprecated': False,
|
||||||
|
'policy_type': EC_POLICY,
|
||||||
|
'ec_type': 'jerasure_rs_vand',
|
||||||
|
'ec_num_data_fragments': 10,
|
||||||
|
'ec_num_parity_fragments': 3,
|
||||||
|
'ec_object_segment_size': DEFAULT_EC_OBJECT_SEGMENT_SIZE,
|
||||||
|
},
|
||||||
|
(10, False): {
|
||||||
|
'name': 'ten',
|
||||||
|
},
|
||||||
|
# deprecated ec
|
||||||
|
(11, True): {
|
||||||
|
'name': 'done',
|
||||||
|
'default': False,
|
||||||
|
'deprecated': True,
|
||||||
|
'policy_type': EC_POLICY,
|
||||||
|
'ec_type': 'jerasure_rs_vand',
|
||||||
|
'ec_num_data_fragments': 10,
|
||||||
|
'ec_num_parity_fragments': 3,
|
||||||
|
'ec_object_segment_size': DEFAULT_EC_OBJECT_SEGMENT_SIZE,
|
||||||
|
},
|
||||||
|
(11, False): {
|
||||||
|
'name': 'done',
|
||||||
|
'deprecated': True,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
self.maxDiff = None
|
||||||
|
for policy in policies:
|
||||||
|
expected_info = expected[(int(policy), True)]
|
||||||
|
self.assertEqual(policy.get_info(config=True), expected_info)
|
||||||
|
expected_info = expected[(int(policy), False)]
|
||||||
|
self.assertEqual(policy.get_info(config=False), expected_info)
|
||||||
|
|
||||||
|
|
||||||
if __name__ == '__main__':
|
if __name__ == '__main__':
|
||||||
|
@ -140,9 +140,10 @@ class TestDiskFileModuleMethods(unittest.TestCase):
|
|||||||
pn = '/objects-1/0/606/198452b6ef6247c78606/1401379842.14643.data'
|
pn = '/objects-1/0/606/198452b6ef6247c78606/1401379842.14643.data'
|
||||||
self.assertEqual(diskfile.extract_policy_index(pn), 1)
|
self.assertEqual(diskfile.extract_policy_index(pn), 1)
|
||||||
|
|
||||||
# bad policy index
|
# well formatted but, unknown policy index
|
||||||
pn = 'objects-2/0/606/198427efcff042c78606/1401379842.14643.data'
|
pn = 'objects-2/0/606/198427efcff042c78606/1401379842.14643.data'
|
||||||
self.assertEqual(diskfile.extract_policy_index(pn), 0)
|
self.assertRaises(ValueError,
|
||||||
|
diskfile.extract_policy_index, pn)
|
||||||
bad_path = '/srv/node/sda1/objects-t/1/abc/def/1234.data'
|
bad_path = '/srv/node/sda1/objects-t/1/abc/def/1234.data'
|
||||||
self.assertRaises(ValueError,
|
self.assertRaises(ValueError,
|
||||||
diskfile.extract_policy_index, bad_path)
|
diskfile.extract_policy_index, bad_path)
|
||||||
|
@ -4510,6 +4510,7 @@ class TestObjectServer(unittest.TestCase):
|
|||||||
resp.close()
|
resp.close()
|
||||||
|
|
||||||
|
|
||||||
|
@patch_policies
|
||||||
class TestZeroCopy(unittest.TestCase):
|
class TestZeroCopy(unittest.TestCase):
|
||||||
"""Test the object server's zero-copy functionality"""
|
"""Test the object server's zero-copy functionality"""
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user