Merge "Update test infrastructure" into feature/ec_review
This commit is contained in:
commit
8a58bbf75f
@ -1317,7 +1317,12 @@ class TestFile(Base):
|
|||||||
self.assertEqual(file_types, file_types_read)
|
self.assertEqual(file_types, file_types_read)
|
||||||
|
|
||||||
def testRangedGets(self):
|
def testRangedGets(self):
|
||||||
file_length = 10000
|
# We set the file_length to a strange multiple here. This is to check
|
||||||
|
# that ranges still work in the EC case when the requested range
|
||||||
|
# spans EC segment boundaries. The 1 MiB base value is chosen because
|
||||||
|
# that's a common EC segment size. The 1.33 multiple is to ensure we
|
||||||
|
# aren't aligned on segment boundaries
|
||||||
|
file_length = int(1048576 * 1.33)
|
||||||
range_size = file_length / 10
|
range_size = file_length / 10
|
||||||
file_item = self.env.container.file(Utils.create_name())
|
file_item = self.env.container.file(Utils.create_name())
|
||||||
data = file_item.write_random(file_length)
|
data = file_item.write_random(file_length)
|
||||||
|
@ -22,24 +22,30 @@ import errno
|
|||||||
import sys
|
import sys
|
||||||
from contextlib import contextmanager, closing
|
from contextlib import contextmanager, closing
|
||||||
from collections import defaultdict, Iterable
|
from collections import defaultdict, Iterable
|
||||||
|
import itertools
|
||||||
from numbers import Number
|
from numbers import Number
|
||||||
from tempfile import NamedTemporaryFile
|
from tempfile import NamedTemporaryFile
|
||||||
import time
|
import time
|
||||||
|
import eventlet
|
||||||
from eventlet.green import socket
|
from eventlet.green import socket
|
||||||
from tempfile import mkdtemp
|
from tempfile import mkdtemp
|
||||||
from shutil import rmtree
|
from shutil import rmtree
|
||||||
|
from swift.common.utils import Timestamp
|
||||||
from test import get_config
|
from test import get_config
|
||||||
from swift.common import swob, utils
|
from swift.common import swob, utils
|
||||||
from swift.common.ring import Ring, RingData
|
from swift.common.ring import Ring, RingData
|
||||||
from hashlib import md5
|
from hashlib import md5
|
||||||
from eventlet import sleep, Timeout
|
|
||||||
import logging.handlers
|
import logging.handlers
|
||||||
from httplib import HTTPException
|
from httplib import HTTPException
|
||||||
from swift.common import storage_policy
|
from swift.common import storage_policy
|
||||||
|
from swift.common.storage_policy import StoragePolicy, ECStoragePolicy
|
||||||
import functools
|
import functools
|
||||||
import cPickle as pickle
|
import cPickle as pickle
|
||||||
from gzip import GzipFile
|
from gzip import GzipFile
|
||||||
import mock as mocklib
|
import mock as mocklib
|
||||||
|
import inspect
|
||||||
|
|
||||||
|
EMPTY_ETAG = md5().hexdigest()
|
||||||
|
|
||||||
# try not to import this module from swift
|
# try not to import this module from swift
|
||||||
if not os.path.basename(sys.argv[0]).startswith('swift'):
|
if not os.path.basename(sys.argv[0]).startswith('swift'):
|
||||||
@ -47,26 +53,40 @@ if not os.path.basename(sys.argv[0]).startswith('swift'):
|
|||||||
utils.HASH_PATH_SUFFIX = 'endcap'
|
utils.HASH_PATH_SUFFIX = 'endcap'
|
||||||
|
|
||||||
|
|
||||||
def patch_policies(thing_or_policies=None, legacy_only=False):
|
def patch_policies(thing_or_policies=None, legacy_only=False,
|
||||||
if legacy_only:
|
with_ec_default=False, fake_ring_args=None):
|
||||||
default_policies = [storage_policy.StoragePolicy(
|
|
||||||
0, 'legacy', True, object_ring=FakeRing())]
|
|
||||||
else:
|
|
||||||
default_policies = [
|
|
||||||
storage_policy.StoragePolicy(
|
|
||||||
0, 'nulo', True, object_ring=FakeRing()),
|
|
||||||
storage_policy.StoragePolicy(
|
|
||||||
1, 'unu', object_ring=FakeRing()),
|
|
||||||
]
|
|
||||||
|
|
||||||
thing_or_policies = thing_or_policies or default_policies
|
|
||||||
|
|
||||||
if isinstance(thing_or_policies, (
|
if isinstance(thing_or_policies, (
|
||||||
Iterable, storage_policy.StoragePolicyCollection)):
|
Iterable, storage_policy.StoragePolicyCollection)):
|
||||||
return PatchPolicies(thing_or_policies)
|
return PatchPolicies(thing_or_policies, fake_ring_args=fake_ring_args)
|
||||||
|
|
||||||
|
if legacy_only:
|
||||||
|
default_policies = [
|
||||||
|
StoragePolicy(0, name='legacy', is_default=True),
|
||||||
|
]
|
||||||
|
default_ring_args = [{}]
|
||||||
|
elif with_ec_default:
|
||||||
|
default_policies = [
|
||||||
|
ECStoragePolicy(0, name='ec', is_default=True,
|
||||||
|
ec_type='jerasure_rs_vand', ec_ndata=4,
|
||||||
|
ec_nparity=2, ec_segment_size=4096),
|
||||||
|
StoragePolicy(1, name='unu'),
|
||||||
|
]
|
||||||
|
default_ring_args = [{'replicas': 6}, {}]
|
||||||
else:
|
else:
|
||||||
# it's a thing!
|
default_policies = [
|
||||||
return PatchPolicies(default_policies)(thing_or_policies)
|
StoragePolicy(0, name='nulo', is_default=True),
|
||||||
|
StoragePolicy(1, name='unu'),
|
||||||
|
]
|
||||||
|
default_ring_args = [{}, {}]
|
||||||
|
|
||||||
|
fake_ring_args = fake_ring_args or default_ring_args
|
||||||
|
decorator = PatchPolicies(default_policies, fake_ring_args=fake_ring_args)
|
||||||
|
|
||||||
|
if not thing_or_policies:
|
||||||
|
return decorator
|
||||||
|
else:
|
||||||
|
# it's a thing, we return the wrapped thing instead of the decorator
|
||||||
|
return decorator(thing_or_policies)
|
||||||
|
|
||||||
|
|
||||||
class PatchPolicies(object):
|
class PatchPolicies(object):
|
||||||
@ -76,11 +96,33 @@ class PatchPolicies(object):
|
|||||||
patched yet)
|
patched yet)
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def __init__(self, policies):
|
def __init__(self, policies, fake_ring_args=None):
|
||||||
if isinstance(policies, storage_policy.StoragePolicyCollection):
|
if isinstance(policies, storage_policy.StoragePolicyCollection):
|
||||||
self.policies = policies
|
self.policies = policies
|
||||||
else:
|
else:
|
||||||
self.policies = storage_policy.StoragePolicyCollection(policies)
|
self.policies = storage_policy.StoragePolicyCollection(policies)
|
||||||
|
self.fake_ring_args = fake_ring_args or [None] * len(self.policies)
|
||||||
|
|
||||||
|
def _setup_rings(self):
|
||||||
|
"""
|
||||||
|
Our tests tend to use the policies rings like their own personal
|
||||||
|
playground - which can be a problem in the particular case of a
|
||||||
|
patched TestCase class where the FakeRing objects are scoped in the
|
||||||
|
call to the patch_policies wrapper outside of the TestCase instance
|
||||||
|
which can lead to some bled state.
|
||||||
|
|
||||||
|
To help tests get better isolation without having to think about it,
|
||||||
|
here we're capturing the args required to *build* a new FakeRing
|
||||||
|
instances so we can ensure each test method gets a clean ring setup.
|
||||||
|
|
||||||
|
The TestCase can always "tweak" these fresh rings in setUp - or if
|
||||||
|
they'd prefer to get the same "reset" behavior with custom FakeRing's
|
||||||
|
they can pass in their own fake_ring_args to patch_policies instead of
|
||||||
|
setting the object_ring on the policy definitions.
|
||||||
|
"""
|
||||||
|
for policy, fake_ring_arg in zip(self.policies, self.fake_ring_args):
|
||||||
|
if fake_ring_arg is not None:
|
||||||
|
policy.object_ring = FakeRing(**fake_ring_arg)
|
||||||
|
|
||||||
def __call__(self, thing):
|
def __call__(self, thing):
|
||||||
if isinstance(thing, type):
|
if isinstance(thing, type):
|
||||||
@ -89,24 +131,33 @@ class PatchPolicies(object):
|
|||||||
return self._patch_method(thing)
|
return self._patch_method(thing)
|
||||||
|
|
||||||
def _patch_class(self, cls):
|
def _patch_class(self, cls):
|
||||||
|
"""
|
||||||
|
Creating a new class that inherits from decorated class is the more
|
||||||
|
common way I've seen class decorators done - but it seems to cause
|
||||||
|
infinite recursion when super is called from inside methods in the
|
||||||
|
decorated class.
|
||||||
|
"""
|
||||||
|
|
||||||
class NewClass(cls):
|
orig_setUp = cls.setUp
|
||||||
|
orig_tearDown = cls.tearDown
|
||||||
|
|
||||||
already_patched = False
|
def setUp(cls_self):
|
||||||
|
self._orig_POLICIES = storage_policy._POLICIES
|
||||||
|
if not getattr(cls_self, '_policies_patched', False):
|
||||||
|
storage_policy._POLICIES = self.policies
|
||||||
|
self._setup_rings()
|
||||||
|
cls_self._policies_patched = True
|
||||||
|
|
||||||
def setUp(cls_self):
|
orig_setUp(cls_self)
|
||||||
self._orig_POLICIES = storage_policy._POLICIES
|
|
||||||
if not cls_self.already_patched:
|
|
||||||
storage_policy._POLICIES = self.policies
|
|
||||||
cls_self.already_patched = True
|
|
||||||
super(NewClass, cls_self).setUp()
|
|
||||||
|
|
||||||
def tearDown(cls_self):
|
def tearDown(cls_self):
|
||||||
super(NewClass, cls_self).tearDown()
|
orig_tearDown(cls_self)
|
||||||
storage_policy._POLICIES = self._orig_POLICIES
|
storage_policy._POLICIES = self._orig_POLICIES
|
||||||
|
|
||||||
NewClass.__name__ = cls.__name__
|
cls.setUp = setUp
|
||||||
return NewClass
|
cls.tearDown = tearDown
|
||||||
|
|
||||||
|
return cls
|
||||||
|
|
||||||
def _patch_method(self, f):
|
def _patch_method(self, f):
|
||||||
@functools.wraps(f)
|
@functools.wraps(f)
|
||||||
@ -114,6 +165,7 @@ class PatchPolicies(object):
|
|||||||
self._orig_POLICIES = storage_policy._POLICIES
|
self._orig_POLICIES = storage_policy._POLICIES
|
||||||
try:
|
try:
|
||||||
storage_policy._POLICIES = self.policies
|
storage_policy._POLICIES = self.policies
|
||||||
|
self._setup_rings()
|
||||||
return f(*args, **kwargs)
|
return f(*args, **kwargs)
|
||||||
finally:
|
finally:
|
||||||
storage_policy._POLICIES = self._orig_POLICIES
|
storage_policy._POLICIES = self._orig_POLICIES
|
||||||
@ -178,7 +230,9 @@ class FakeRing(Ring):
|
|||||||
for x in xrange(self.replicas, min(self.replicas + self.max_more_nodes,
|
for x in xrange(self.replicas, min(self.replicas + self.max_more_nodes,
|
||||||
self.replicas * self.replicas)):
|
self.replicas * self.replicas)):
|
||||||
yield {'ip': '10.0.0.%s' % x,
|
yield {'ip': '10.0.0.%s' % x,
|
||||||
|
'replication_ip': '10.0.0.%s' % x,
|
||||||
'port': self._base_port + x,
|
'port': self._base_port + x,
|
||||||
|
'replication_port': self._base_port + x,
|
||||||
'device': 'sda',
|
'device': 'sda',
|
||||||
'zone': x % 3,
|
'zone': x % 3,
|
||||||
'region': x % 2,
|
'region': x % 2,
|
||||||
@ -206,6 +260,48 @@ def write_fake_ring(path, *devs):
|
|||||||
pickle.dump(RingData(replica2part2dev_id, devs, part_shift), f)
|
pickle.dump(RingData(replica2part2dev_id, devs, part_shift), f)
|
||||||
|
|
||||||
|
|
||||||
|
class FabricatedRing(Ring):
|
||||||
|
"""
|
||||||
|
When a FakeRing just won't do - you can fabricate one to meet
|
||||||
|
your tests needs.
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __init__(self, replicas=6, devices=8, nodes=4, port=6000,
|
||||||
|
part_power=4):
|
||||||
|
self.devices = devices
|
||||||
|
self.nodes = nodes
|
||||||
|
self.port = port
|
||||||
|
self.replicas = 6
|
||||||
|
self.part_power = part_power
|
||||||
|
self._part_shift = 32 - self.part_power
|
||||||
|
self._reload()
|
||||||
|
|
||||||
|
def _reload(self, *args, **kwargs):
|
||||||
|
self._rtime = time.time() * 2
|
||||||
|
if hasattr(self, '_replica2part2dev_id'):
|
||||||
|
return
|
||||||
|
self._devs = [{
|
||||||
|
'region': 1,
|
||||||
|
'zone': 1,
|
||||||
|
'weight': 1.0,
|
||||||
|
'id': i,
|
||||||
|
'device': 'sda%d' % i,
|
||||||
|
'ip': '10.0.0.%d' % (i % self.nodes),
|
||||||
|
'replication_ip': '10.0.0.%d' % (i % self.nodes),
|
||||||
|
'port': self.port,
|
||||||
|
'replication_port': self.port,
|
||||||
|
} for i in range(self.devices)]
|
||||||
|
|
||||||
|
self._replica2part2dev_id = [
|
||||||
|
[None] * 2 ** self.part_power
|
||||||
|
for i in range(self.replicas)
|
||||||
|
]
|
||||||
|
dev_ids = itertools.cycle(range(self.devices))
|
||||||
|
for p in range(2 ** self.part_power):
|
||||||
|
for r in range(self.replicas):
|
||||||
|
self._replica2part2dev_id[r][p] = next(dev_ids)
|
||||||
|
|
||||||
|
|
||||||
class FakeMemcache(object):
|
class FakeMemcache(object):
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
@ -363,8 +459,8 @@ class UnmockTimeModule(object):
|
|||||||
logging.time = UnmockTimeModule()
|
logging.time = UnmockTimeModule()
|
||||||
|
|
||||||
|
|
||||||
class FakeLogger(logging.Logger):
|
class FakeLogger(logging.Logger, object):
|
||||||
# a thread safe logger
|
# a thread safe fake logger
|
||||||
|
|
||||||
def __init__(self, *args, **kwargs):
|
def __init__(self, *args, **kwargs):
|
||||||
self._clear()
|
self._clear()
|
||||||
@ -376,22 +472,31 @@ class FakeLogger(logging.Logger):
|
|||||||
self.thread_locals = None
|
self.thread_locals = None
|
||||||
self.parent = None
|
self.parent = None
|
||||||
|
|
||||||
|
store_in = {
|
||||||
|
logging.ERROR: 'error',
|
||||||
|
logging.WARNING: 'warning',
|
||||||
|
logging.INFO: 'info',
|
||||||
|
logging.DEBUG: 'debug',
|
||||||
|
logging.CRITICAL: 'critical',
|
||||||
|
}
|
||||||
|
|
||||||
|
def _log(self, level, msg, *args, **kwargs):
|
||||||
|
store_name = self.store_in[level]
|
||||||
|
cargs = [msg]
|
||||||
|
if any(args):
|
||||||
|
cargs.extend(args)
|
||||||
|
captured = dict(kwargs)
|
||||||
|
if 'exc_info' in kwargs and \
|
||||||
|
not isinstance(kwargs['exc_info'], tuple):
|
||||||
|
captured['exc_info'] = sys.exc_info()
|
||||||
|
self.log_dict[store_name].append((tuple(cargs), captured))
|
||||||
|
super(FakeLogger, self)._log(level, msg, *args, **kwargs)
|
||||||
|
|
||||||
def _clear(self):
|
def _clear(self):
|
||||||
self.log_dict = defaultdict(list)
|
self.log_dict = defaultdict(list)
|
||||||
self.lines_dict = {'critical': [], 'error': [], 'info': [],
|
self.lines_dict = {'critical': [], 'error': [], 'info': [],
|
||||||
'warning': [], 'debug': []}
|
'warning': [], 'debug': []}
|
||||||
|
|
||||||
def _store_in(store_name):
|
|
||||||
def stub_fn(self, *args, **kwargs):
|
|
||||||
self.log_dict[store_name].append((args, kwargs))
|
|
||||||
return stub_fn
|
|
||||||
|
|
||||||
def _store_and_log_in(store_name, level):
|
|
||||||
def stub_fn(self, *args, **kwargs):
|
|
||||||
self.log_dict[store_name].append((args, kwargs))
|
|
||||||
self._log(level, args[0], args[1:], **kwargs)
|
|
||||||
return stub_fn
|
|
||||||
|
|
||||||
def get_lines_for_level(self, level):
|
def get_lines_for_level(self, level):
|
||||||
if level not in self.lines_dict:
|
if level not in self.lines_dict:
|
||||||
raise KeyError(
|
raise KeyError(
|
||||||
@ -404,16 +509,10 @@ class FakeLogger(logging.Logger):
|
|||||||
return dict((level, msgs) for level, msgs in self.lines_dict.items()
|
return dict((level, msgs) for level, msgs in self.lines_dict.items()
|
||||||
if len(msgs) > 0)
|
if len(msgs) > 0)
|
||||||
|
|
||||||
error = _store_and_log_in('error', logging.ERROR)
|
def _store_in(store_name):
|
||||||
info = _store_and_log_in('info', logging.INFO)
|
def stub_fn(self, *args, **kwargs):
|
||||||
warning = _store_and_log_in('warning', logging.WARNING)
|
self.log_dict[store_name].append((args, kwargs))
|
||||||
warn = _store_and_log_in('warning', logging.WARNING)
|
return stub_fn
|
||||||
debug = _store_and_log_in('debug', logging.DEBUG)
|
|
||||||
|
|
||||||
def exception(self, *args, **kwargs):
|
|
||||||
self.log_dict['exception'].append((args, kwargs,
|
|
||||||
str(sys.exc_info()[1])))
|
|
||||||
print 'FakeLogger Exception: %s' % self.log_dict
|
|
||||||
|
|
||||||
# mock out the StatsD logging methods:
|
# mock out the StatsD logging methods:
|
||||||
update_stats = _store_in('update_stats')
|
update_stats = _store_in('update_stats')
|
||||||
@ -605,19 +704,53 @@ def mock(update):
|
|||||||
delattr(module, attr)
|
delattr(module, attr)
|
||||||
|
|
||||||
|
|
||||||
|
class SlowBody(object):
|
||||||
|
"""
|
||||||
|
This will work with our fake_http_connect, if you hand in these
|
||||||
|
instead of strings it will make reads take longer by the given
|
||||||
|
amount. It should be a little bit easier to extend than the
|
||||||
|
current slow kwarg - which inserts whitespace in the response.
|
||||||
|
Also it should be easy to detect if you have one of these (or a
|
||||||
|
subclass) for the body inside of FakeConn if we wanted to do
|
||||||
|
something smarter than just duck-type the str/buffer api
|
||||||
|
enough to get by.
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __init__(self, body, slowness):
|
||||||
|
self.body = body
|
||||||
|
self.slowness = slowness
|
||||||
|
|
||||||
|
def slowdown(self):
|
||||||
|
eventlet.sleep(self.slowness)
|
||||||
|
|
||||||
|
def __getitem__(self, s):
|
||||||
|
return SlowBody(self.body[s], self.slowness)
|
||||||
|
|
||||||
|
def __len__(self):
|
||||||
|
return len(self.body)
|
||||||
|
|
||||||
|
def __radd__(self, other):
|
||||||
|
self.slowdown()
|
||||||
|
return other + self.body
|
||||||
|
|
||||||
|
|
||||||
def fake_http_connect(*code_iter, **kwargs):
|
def fake_http_connect(*code_iter, **kwargs):
|
||||||
|
|
||||||
class FakeConn(object):
|
class FakeConn(object):
|
||||||
|
|
||||||
def __init__(self, status, etag=None, body='', timestamp='1',
|
def __init__(self, status, etag=None, body='', timestamp='1',
|
||||||
headers=None):
|
headers=None, expect_headers=None, connection_id=None,
|
||||||
|
give_send=None):
|
||||||
# connect exception
|
# connect exception
|
||||||
if isinstance(status, (Exception, Timeout)):
|
if isinstance(status, (Exception, eventlet.Timeout)):
|
||||||
raise status
|
raise status
|
||||||
if isinstance(status, tuple):
|
if isinstance(status, tuple):
|
||||||
self.expect_status, self.status = status
|
self.expect_status = list(status[:-1])
|
||||||
|
self.status = status[-1]
|
||||||
|
self.explicit_expect_list = True
|
||||||
else:
|
else:
|
||||||
self.expect_status, self.status = (None, status)
|
self.expect_status, self.status = ([], status)
|
||||||
|
self.explicit_expect_list = False
|
||||||
if not self.expect_status:
|
if not self.expect_status:
|
||||||
# when a swift backend service returns a status before reading
|
# when a swift backend service returns a status before reading
|
||||||
# from the body (mostly an error response) eventlet.wsgi will
|
# from the body (mostly an error response) eventlet.wsgi will
|
||||||
@ -628,9 +761,9 @@ def fake_http_connect(*code_iter, **kwargs):
|
|||||||
# our backend services and return certain types of responses
|
# our backend services and return certain types of responses
|
||||||
# as expect statuses just like a real backend server would do.
|
# as expect statuses just like a real backend server would do.
|
||||||
if self.status in (507, 412, 409):
|
if self.status in (507, 412, 409):
|
||||||
self.expect_status = status
|
self.expect_status = [status]
|
||||||
else:
|
else:
|
||||||
self.expect_status = 100
|
self.expect_status = [100, 100]
|
||||||
self.reason = 'Fake'
|
self.reason = 'Fake'
|
||||||
self.host = '1.2.3.4'
|
self.host = '1.2.3.4'
|
||||||
self.port = '1234'
|
self.port = '1234'
|
||||||
@ -639,32 +772,41 @@ def fake_http_connect(*code_iter, **kwargs):
|
|||||||
self.etag = etag
|
self.etag = etag
|
||||||
self.body = body
|
self.body = body
|
||||||
self.headers = headers or {}
|
self.headers = headers or {}
|
||||||
|
self.expect_headers = expect_headers or {}
|
||||||
self.timestamp = timestamp
|
self.timestamp = timestamp
|
||||||
|
self.connection_id = connection_id
|
||||||
|
self.give_send = give_send
|
||||||
if 'slow' in kwargs and isinstance(kwargs['slow'], list):
|
if 'slow' in kwargs and isinstance(kwargs['slow'], list):
|
||||||
try:
|
try:
|
||||||
self._next_sleep = kwargs['slow'].pop(0)
|
self._next_sleep = kwargs['slow'].pop(0)
|
||||||
except IndexError:
|
except IndexError:
|
||||||
self._next_sleep = None
|
self._next_sleep = None
|
||||||
|
# be nice to trixy bits with node_iter's
|
||||||
|
eventlet.sleep()
|
||||||
|
|
||||||
def getresponse(self):
|
def getresponse(self):
|
||||||
if isinstance(self.status, (Exception, Timeout)):
|
if self.expect_status and self.explicit_expect_list:
|
||||||
|
raise Exception('Test did not consume all fake '
|
||||||
|
'expect status: %r' % (self.expect_status,))
|
||||||
|
if isinstance(self.status, (Exception, eventlet.Timeout)):
|
||||||
raise self.status
|
raise self.status
|
||||||
exc = kwargs.get('raise_exc')
|
exc = kwargs.get('raise_exc')
|
||||||
if exc:
|
if exc:
|
||||||
if isinstance(exc, (Exception, Timeout)):
|
if isinstance(exc, (Exception, eventlet.Timeout)):
|
||||||
raise exc
|
raise exc
|
||||||
raise Exception('test')
|
raise Exception('test')
|
||||||
if kwargs.get('raise_timeout_exc'):
|
if kwargs.get('raise_timeout_exc'):
|
||||||
raise Timeout()
|
raise eventlet.Timeout()
|
||||||
return self
|
return self
|
||||||
|
|
||||||
def getexpect(self):
|
def getexpect(self):
|
||||||
if isinstance(self.expect_status, (Exception, Timeout)):
|
expect_status = self.expect_status.pop(0)
|
||||||
|
if isinstance(self.expect_status, (Exception, eventlet.Timeout)):
|
||||||
raise self.expect_status
|
raise self.expect_status
|
||||||
headers = {}
|
headers = dict(self.expect_headers)
|
||||||
if self.expect_status == 409:
|
if expect_status == 409:
|
||||||
headers['X-Backend-Timestamp'] = self.timestamp
|
headers['X-Backend-Timestamp'] = self.timestamp
|
||||||
return FakeConn(self.expect_status, headers=headers)
|
return FakeConn(expect_status, headers=headers)
|
||||||
|
|
||||||
def getheaders(self):
|
def getheaders(self):
|
||||||
etag = self.etag
|
etag = self.etag
|
||||||
@ -717,18 +859,20 @@ def fake_http_connect(*code_iter, **kwargs):
|
|||||||
if am_slow:
|
if am_slow:
|
||||||
if self.sent < 4:
|
if self.sent < 4:
|
||||||
self.sent += 1
|
self.sent += 1
|
||||||
sleep(value)
|
eventlet.sleep(value)
|
||||||
return ' '
|
return ' '
|
||||||
rv = self.body[:amt]
|
rv = self.body[:amt]
|
||||||
self.body = self.body[amt:]
|
self.body = self.body[amt:]
|
||||||
return rv
|
return rv
|
||||||
|
|
||||||
def send(self, amt=None):
|
def send(self, amt=None):
|
||||||
|
if self.give_send:
|
||||||
|
self.give_send(self.connection_id, amt)
|
||||||
am_slow, value = self.get_slow()
|
am_slow, value = self.get_slow()
|
||||||
if am_slow:
|
if am_slow:
|
||||||
if self.received < 4:
|
if self.received < 4:
|
||||||
self.received += 1
|
self.received += 1
|
||||||
sleep(value)
|
eventlet.sleep(value)
|
||||||
|
|
||||||
def getheader(self, name, default=None):
|
def getheader(self, name, default=None):
|
||||||
return swob.HeaderKeyDict(self.getheaders()).get(name, default)
|
return swob.HeaderKeyDict(self.getheaders()).get(name, default)
|
||||||
@ -738,16 +882,22 @@ def fake_http_connect(*code_iter, **kwargs):
|
|||||||
|
|
||||||
timestamps_iter = iter(kwargs.get('timestamps') or ['1'] * len(code_iter))
|
timestamps_iter = iter(kwargs.get('timestamps') or ['1'] * len(code_iter))
|
||||||
etag_iter = iter(kwargs.get('etags') or [None] * len(code_iter))
|
etag_iter = iter(kwargs.get('etags') or [None] * len(code_iter))
|
||||||
if isinstance(kwargs.get('headers'), list):
|
if isinstance(kwargs.get('headers'), (list, tuple)):
|
||||||
headers_iter = iter(kwargs['headers'])
|
headers_iter = iter(kwargs['headers'])
|
||||||
else:
|
else:
|
||||||
headers_iter = iter([kwargs.get('headers', {})] * len(code_iter))
|
headers_iter = iter([kwargs.get('headers', {})] * len(code_iter))
|
||||||
|
if isinstance(kwargs.get('expect_headers'), (list, tuple)):
|
||||||
|
expect_headers_iter = iter(kwargs['expect_headers'])
|
||||||
|
else:
|
||||||
|
expect_headers_iter = iter([kwargs.get('expect_headers', {})] *
|
||||||
|
len(code_iter))
|
||||||
|
|
||||||
x = kwargs.get('missing_container', [False] * len(code_iter))
|
x = kwargs.get('missing_container', [False] * len(code_iter))
|
||||||
if not isinstance(x, (tuple, list)):
|
if not isinstance(x, (tuple, list)):
|
||||||
x = [x] * len(code_iter)
|
x = [x] * len(code_iter)
|
||||||
container_ts_iter = iter(x)
|
container_ts_iter = iter(x)
|
||||||
code_iter = iter(code_iter)
|
code_iter = iter(code_iter)
|
||||||
|
conn_id_and_code_iter = enumerate(code_iter)
|
||||||
static_body = kwargs.get('body', None)
|
static_body = kwargs.get('body', None)
|
||||||
body_iter = kwargs.get('body_iter', None)
|
body_iter = kwargs.get('body_iter', None)
|
||||||
if body_iter:
|
if body_iter:
|
||||||
@ -755,17 +905,22 @@ def fake_http_connect(*code_iter, **kwargs):
|
|||||||
|
|
||||||
def connect(*args, **ckwargs):
|
def connect(*args, **ckwargs):
|
||||||
if kwargs.get('slow_connect', False):
|
if kwargs.get('slow_connect', False):
|
||||||
sleep(0.1)
|
eventlet.sleep(0.1)
|
||||||
if 'give_content_type' in kwargs:
|
if 'give_content_type' in kwargs:
|
||||||
if len(args) >= 7 and 'Content-Type' in args[6]:
|
if len(args) >= 7 and 'Content-Type' in args[6]:
|
||||||
kwargs['give_content_type'](args[6]['Content-Type'])
|
kwargs['give_content_type'](args[6]['Content-Type'])
|
||||||
else:
|
else:
|
||||||
kwargs['give_content_type']('')
|
kwargs['give_content_type']('')
|
||||||
|
i, status = conn_id_and_code_iter.next()
|
||||||
if 'give_connect' in kwargs:
|
if 'give_connect' in kwargs:
|
||||||
kwargs['give_connect'](*args, **ckwargs)
|
give_conn_fn = kwargs['give_connect']
|
||||||
status = code_iter.next()
|
argspec = inspect.getargspec(give_conn_fn)
|
||||||
|
if argspec.keywords or 'connection_id' in argspec.args:
|
||||||
|
ckwargs['connection_id'] = i
|
||||||
|
give_conn_fn(*args, **ckwargs)
|
||||||
etag = etag_iter.next()
|
etag = etag_iter.next()
|
||||||
headers = headers_iter.next()
|
headers = headers_iter.next()
|
||||||
|
expect_headers = expect_headers_iter.next()
|
||||||
timestamp = timestamps_iter.next()
|
timestamp = timestamps_iter.next()
|
||||||
|
|
||||||
if status <= 0:
|
if status <= 0:
|
||||||
@ -775,7 +930,8 @@ def fake_http_connect(*code_iter, **kwargs):
|
|||||||
else:
|
else:
|
||||||
body = body_iter.next()
|
body = body_iter.next()
|
||||||
return FakeConn(status, etag, body=body, timestamp=timestamp,
|
return FakeConn(status, etag, body=body, timestamp=timestamp,
|
||||||
headers=headers)
|
headers=headers, expect_headers=expect_headers,
|
||||||
|
connection_id=i, give_send=kwargs.get('give_send'))
|
||||||
|
|
||||||
connect.code_iter = code_iter
|
connect.code_iter = code_iter
|
||||||
|
|
||||||
@ -806,3 +962,7 @@ def mocked_http_conn(*args, **kwargs):
|
|||||||
left_over_status = list(fake_conn.code_iter)
|
left_over_status = list(fake_conn.code_iter)
|
||||||
if left_over_status:
|
if left_over_status:
|
||||||
raise AssertionError('left over status %r' % left_over_status)
|
raise AssertionError('left over status %r' % left_over_status)
|
||||||
|
|
||||||
|
|
||||||
|
def make_timestamp_iter():
|
||||||
|
return iter(Timestamp(t) for t in itertools.count(int(time.time())))
|
||||||
|
@ -564,9 +564,10 @@ class TestDloGetManifest(DloTestCase):
|
|||||||
environ={'REQUEST_METHOD': 'GET'})
|
environ={'REQUEST_METHOD': 'GET'})
|
||||||
status, headers, body = self.call_dlo(req)
|
status, headers, body = self.call_dlo(req)
|
||||||
self.assertEqual(status, "409 Conflict")
|
self.assertEqual(status, "409 Conflict")
|
||||||
err_log = self.dlo.logger.log_dict['exception'][0][0][0]
|
err_lines = self.dlo.logger.get_lines_for_level('error')
|
||||||
self.assertTrue(err_log.startswith('ERROR: An error occurred '
|
self.assertEqual(len(err_lines), 1)
|
||||||
'while retrieving segments'))
|
self.assertTrue(err_lines[0].startswith(
|
||||||
|
'ERROR: An error occurred while retrieving segments'))
|
||||||
|
|
||||||
def test_error_fetching_second_segment(self):
|
def test_error_fetching_second_segment(self):
|
||||||
self.app.register(
|
self.app.register(
|
||||||
@ -581,9 +582,10 @@ class TestDloGetManifest(DloTestCase):
|
|||||||
self.assertTrue(isinstance(exc, exceptions.SegmentError))
|
self.assertTrue(isinstance(exc, exceptions.SegmentError))
|
||||||
self.assertEqual(status, "200 OK")
|
self.assertEqual(status, "200 OK")
|
||||||
self.assertEqual(''.join(body), "aaaaa") # first segment made it out
|
self.assertEqual(''.join(body), "aaaaa") # first segment made it out
|
||||||
err_log = self.dlo.logger.log_dict['exception'][0][0][0]
|
err_lines = self.dlo.logger.get_lines_for_level('error')
|
||||||
self.assertTrue(err_log.startswith('ERROR: An error occurred '
|
self.assertEqual(len(err_lines), 1)
|
||||||
'while retrieving segments'))
|
self.assertTrue(err_lines[0].startswith(
|
||||||
|
'ERROR: An error occurred while retrieving segments'))
|
||||||
|
|
||||||
def test_error_listing_container_first_listing_request(self):
|
def test_error_listing_container_first_listing_request(self):
|
||||||
self.app.register(
|
self.app.register(
|
||||||
|
@ -1431,9 +1431,10 @@ class TestSloGetManifest(SloTestCase):
|
|||||||
|
|
||||||
self.assertEqual(status, '409 Conflict')
|
self.assertEqual(status, '409 Conflict')
|
||||||
self.assertEqual(self.app.call_count, 10)
|
self.assertEqual(self.app.call_count, 10)
|
||||||
err_log = self.slo.logger.log_dict['exception'][0][0][0]
|
error_lines = self.slo.logger.get_lines_for_level('error')
|
||||||
self.assertTrue(err_log.startswith('ERROR: An error occurred '
|
self.assertEqual(len(error_lines), 1)
|
||||||
'while retrieving segments'))
|
self.assertTrue(error_lines[0].startswith(
|
||||||
|
'ERROR: An error occurred while retrieving segments'))
|
||||||
|
|
||||||
def test_get_with_if_modified_since(self):
|
def test_get_with_if_modified_since(self):
|
||||||
# It's important not to pass the If-[Un]Modified-Since header to the
|
# It's important not to pass the If-[Un]Modified-Since header to the
|
||||||
@ -1508,9 +1509,10 @@ class TestSloGetManifest(SloTestCase):
|
|||||||
status, headers, body = self.call_slo(req)
|
status, headers, body = self.call_slo(req)
|
||||||
|
|
||||||
self.assertEqual('409 Conflict', status)
|
self.assertEqual('409 Conflict', status)
|
||||||
err_log = self.slo.logger.log_dict['exception'][0][0][0]
|
error_lines = self.slo.logger.get_lines_for_level('error')
|
||||||
self.assertTrue(err_log.startswith('ERROR: An error occurred '
|
self.assertEqual(len(error_lines), 1)
|
||||||
'while retrieving segments'))
|
self.assertTrue(error_lines[0].startswith(
|
||||||
|
'ERROR: An error occurred while retrieving segments'))
|
||||||
|
|
||||||
def test_invalid_json_submanifest(self):
|
def test_invalid_json_submanifest(self):
|
||||||
self.app.register(
|
self.app.register(
|
||||||
@ -1585,9 +1587,10 @@ class TestSloGetManifest(SloTestCase):
|
|||||||
status, headers, body = self.call_slo(req)
|
status, headers, body = self.call_slo(req)
|
||||||
|
|
||||||
self.assertEqual('409 Conflict', status)
|
self.assertEqual('409 Conflict', status)
|
||||||
err_log = self.slo.logger.log_dict['exception'][0][0][0]
|
error_lines = self.slo.logger.get_lines_for_level('error')
|
||||||
self.assertTrue(err_log.startswith('ERROR: An error occurred '
|
self.assertEqual(len(error_lines), 1)
|
||||||
'while retrieving segments'))
|
self.assertTrue(error_lines[0].startswith(
|
||||||
|
'ERROR: An error occurred while retrieving segments'))
|
||||||
|
|
||||||
def test_first_segment_mismatched_size(self):
|
def test_first_segment_mismatched_size(self):
|
||||||
self.app.register('GET', '/v1/AUTH_test/gettest/manifest-badsize',
|
self.app.register('GET', '/v1/AUTH_test/gettest/manifest-badsize',
|
||||||
@ -1603,9 +1606,10 @@ class TestSloGetManifest(SloTestCase):
|
|||||||
status, headers, body = self.call_slo(req)
|
status, headers, body = self.call_slo(req)
|
||||||
|
|
||||||
self.assertEqual('409 Conflict', status)
|
self.assertEqual('409 Conflict', status)
|
||||||
err_log = self.slo.logger.log_dict['exception'][0][0][0]
|
error_lines = self.slo.logger.get_lines_for_level('error')
|
||||||
self.assertTrue(err_log.startswith('ERROR: An error occurred '
|
self.assertEqual(len(error_lines), 1)
|
||||||
'while retrieving segments'))
|
self.assertTrue(error_lines[0].startswith(
|
||||||
|
'ERROR: An error occurred while retrieving segments'))
|
||||||
|
|
||||||
def test_download_takes_too_long(self):
|
def test_download_takes_too_long(self):
|
||||||
the_time = [time.time()]
|
the_time = [time.time()]
|
||||||
@ -1657,9 +1661,10 @@ class TestSloGetManifest(SloTestCase):
|
|||||||
status, headers, body = self.call_slo(req)
|
status, headers, body = self.call_slo(req)
|
||||||
|
|
||||||
self.assertEqual('409 Conflict', status)
|
self.assertEqual('409 Conflict', status)
|
||||||
err_log = self.slo.logger.log_dict['exception'][0][0][0]
|
error_lines = self.slo.logger.get_lines_for_level('error')
|
||||||
self.assertTrue(err_log.startswith('ERROR: An error occurred '
|
self.assertEqual(len(error_lines), 1)
|
||||||
'while retrieving segments'))
|
self.assertTrue(error_lines[0].startswith(
|
||||||
|
'ERROR: An error occurred while retrieving segments'))
|
||||||
|
|
||||||
|
|
||||||
class TestSloBulkLogger(unittest.TestCase):
|
class TestSloBulkLogger(unittest.TestCase):
|
||||||
|
@ -939,9 +939,9 @@ class TestContainerSync(unittest.TestCase):
|
|||||||
{'account': 'a', 'container': 'c', 'storage_policy_index': 0},
|
{'account': 'a', 'container': 'c', 'storage_policy_index': 0},
|
||||||
realm, realm_key))
|
realm, realm_key))
|
||||||
self.assertEquals(cs.container_puts, 2)
|
self.assertEquals(cs.container_puts, 2)
|
||||||
self.assertTrue(
|
error_lines = cs.logger.get_lines_for_level('error')
|
||||||
cs.logger.log_dict['exception'][0][0][0].startswith(
|
self.assertEqual(len(error_lines), 1)
|
||||||
'ERROR Syncing '))
|
self.assertTrue(error_lines[0].startswith('ERROR Syncing '))
|
||||||
finally:
|
finally:
|
||||||
sync.uuid = orig_uuid
|
sync.uuid = orig_uuid
|
||||||
sync.shuffle = orig_shuffle
|
sync.shuffle = orig_shuffle
|
||||||
|
@ -2484,8 +2484,9 @@ class TestDiskFile(unittest.TestCase):
|
|||||||
self.fail("Expected exception DiskFileNoSpace")
|
self.fail("Expected exception DiskFileNoSpace")
|
||||||
self.assertTrue(_m_fallocate.called)
|
self.assertTrue(_m_fallocate.called)
|
||||||
self.assertTrue(_m_unlink.called)
|
self.assertTrue(_m_unlink.called)
|
||||||
self.assert_(self.df_mgr.logger.log_dict['exception'][0][0][0].
|
error_lines = self.df_mgr.logger.get_lines_for_level('error')
|
||||||
startswith("Error removing tempfile:"))
|
for line in error_lines:
|
||||||
|
self.assertTrue(line.startswith("Error removing tempfile:"))
|
||||||
|
|
||||||
if __name__ == '__main__':
|
if __name__ == '__main__':
|
||||||
unittest.main()
|
unittest.main()
|
||||||
|
@ -16,7 +16,7 @@
|
|||||||
import urllib
|
import urllib
|
||||||
from time import time
|
from time import time
|
||||||
from unittest import main, TestCase
|
from unittest import main, TestCase
|
||||||
from test.unit import FakeLogger, FakeRing, mocked_http_conn
|
from test.unit import FakeRing, mocked_http_conn, debug_logger
|
||||||
from copy import deepcopy
|
from copy import deepcopy
|
||||||
from tempfile import mkdtemp
|
from tempfile import mkdtemp
|
||||||
from shutil import rmtree
|
from shutil import rmtree
|
||||||
@ -53,7 +53,8 @@ class TestObjectExpirer(TestCase):
|
|||||||
internal_client.sleep = not_sleep
|
internal_client.sleep = not_sleep
|
||||||
|
|
||||||
self.rcache = mkdtemp()
|
self.rcache = mkdtemp()
|
||||||
self.logger = FakeLogger()
|
self.conf = {'recon_cache_path': self.rcache}
|
||||||
|
self.logger = debug_logger('test-recon')
|
||||||
|
|
||||||
def tearDown(self):
|
def tearDown(self):
|
||||||
rmtree(self.rcache)
|
rmtree(self.rcache)
|
||||||
@ -167,7 +168,7 @@ class TestObjectExpirer(TestCase):
|
|||||||
'2': set('5-five 6-six'.split()),
|
'2': set('5-five 6-six'.split()),
|
||||||
'3': set(u'7-seven\u2661'.split()),
|
'3': set(u'7-seven\u2661'.split()),
|
||||||
}
|
}
|
||||||
x = ObjectExpirer({})
|
x = ObjectExpirer(self.conf)
|
||||||
x.swift = InternalClient(containers)
|
x.swift = InternalClient(containers)
|
||||||
|
|
||||||
deleted_objects = {}
|
deleted_objects = {}
|
||||||
@ -233,31 +234,32 @@ class TestObjectExpirer(TestCase):
|
|||||||
x = expirer.ObjectExpirer({}, logger=self.logger)
|
x = expirer.ObjectExpirer({}, logger=self.logger)
|
||||||
|
|
||||||
x.report()
|
x.report()
|
||||||
self.assertEqual(x.logger.log_dict['info'], [])
|
self.assertEqual(x.logger.get_lines_for_level('info'), [])
|
||||||
|
|
||||||
x.logger._clear()
|
x.logger._clear()
|
||||||
x.report(final=True)
|
x.report(final=True)
|
||||||
self.assertTrue('completed' in x.logger.log_dict['info'][-1][0][0],
|
self.assertTrue(
|
||||||
x.logger.log_dict['info'])
|
'completed' in str(x.logger.get_lines_for_level('info')))
|
||||||
self.assertTrue('so far' not in x.logger.log_dict['info'][-1][0][0],
|
self.assertTrue(
|
||||||
x.logger.log_dict['info'])
|
'so far' not in str(x.logger.get_lines_for_level('info')))
|
||||||
|
|
||||||
x.logger._clear()
|
x.logger._clear()
|
||||||
x.report_last_time = time() - x.report_interval
|
x.report_last_time = time() - x.report_interval
|
||||||
x.report()
|
x.report()
|
||||||
self.assertTrue('completed' not in x.logger.log_dict['info'][-1][0][0],
|
self.assertTrue(
|
||||||
x.logger.log_dict['info'])
|
'completed' not in str(x.logger.get_lines_for_level('info')))
|
||||||
self.assertTrue('so far' in x.logger.log_dict['info'][-1][0][0],
|
self.assertTrue(
|
||||||
x.logger.log_dict['info'])
|
'so far' in str(x.logger.get_lines_for_level('info')))
|
||||||
|
|
||||||
def test_run_once_nothing_to_do(self):
|
def test_run_once_nothing_to_do(self):
|
||||||
x = expirer.ObjectExpirer({}, logger=self.logger)
|
x = expirer.ObjectExpirer(self.conf, logger=self.logger)
|
||||||
x.swift = 'throw error because a string does not have needed methods'
|
x.swift = 'throw error because a string does not have needed methods'
|
||||||
x.run_once()
|
x.run_once()
|
||||||
self.assertEqual(x.logger.log_dict['exception'],
|
self.assertEqual(x.logger.get_lines_for_level('error'),
|
||||||
[(("Unhandled exception",), {},
|
["Unhandled exception: "])
|
||||||
"'str' object has no attribute "
|
log_args, log_kwargs = x.logger.log_dict['error'][0]
|
||||||
"'get_account_info'")])
|
self.assertEqual(str(log_kwargs['exc_info'][1]),
|
||||||
|
"'str' object has no attribute 'get_account_info'")
|
||||||
|
|
||||||
def test_run_once_calls_report(self):
|
def test_run_once_calls_report(self):
|
||||||
class InternalClient(object):
|
class InternalClient(object):
|
||||||
@ -267,14 +269,14 @@ class TestObjectExpirer(TestCase):
|
|||||||
def iter_containers(*a, **kw):
|
def iter_containers(*a, **kw):
|
||||||
return []
|
return []
|
||||||
|
|
||||||
x = expirer.ObjectExpirer({}, logger=self.logger)
|
x = expirer.ObjectExpirer(self.conf, logger=self.logger)
|
||||||
x.swift = InternalClient()
|
x.swift = InternalClient()
|
||||||
x.run_once()
|
x.run_once()
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
x.logger.log_dict['info'],
|
x.logger.get_lines_for_level('info'), [
|
||||||
[(('Pass beginning; 1 possible containers; '
|
'Pass beginning; 1 possible containers; 2 possible objects',
|
||||||
'2 possible objects',), {}),
|
'Pass completed in 0s; 0 objects expired',
|
||||||
(('Pass completed in 0s; 0 objects expired',), {})])
|
])
|
||||||
|
|
||||||
def test_run_once_unicode_problem(self):
|
def test_run_once_unicode_problem(self):
|
||||||
class InternalClient(object):
|
class InternalClient(object):
|
||||||
@ -296,7 +298,7 @@ class TestObjectExpirer(TestCase):
|
|||||||
def delete_container(*a, **kw):
|
def delete_container(*a, **kw):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
x = expirer.ObjectExpirer({}, logger=self.logger)
|
x = expirer.ObjectExpirer(self.conf, logger=self.logger)
|
||||||
x.swift = InternalClient()
|
x.swift = InternalClient()
|
||||||
|
|
||||||
requests = []
|
requests = []
|
||||||
@ -323,27 +325,28 @@ class TestObjectExpirer(TestCase):
|
|||||||
def iter_objects(*a, **kw):
|
def iter_objects(*a, **kw):
|
||||||
raise Exception('This should not have been called')
|
raise Exception('This should not have been called')
|
||||||
|
|
||||||
x = expirer.ObjectExpirer({'recon_cache_path': self.rcache},
|
x = expirer.ObjectExpirer(self.conf,
|
||||||
logger=self.logger)
|
logger=self.logger)
|
||||||
x.swift = InternalClient([{'name': str(int(time() + 86400))}])
|
x.swift = InternalClient([{'name': str(int(time() + 86400))}])
|
||||||
x.run_once()
|
x.run_once()
|
||||||
for exccall in x.logger.log_dict['exception']:
|
logs = x.logger.all_log_lines()
|
||||||
self.assertTrue(
|
self.assertEqual(logs['info'], [
|
||||||
'This should not have been called' not in exccall[0][0])
|
'Pass beginning; 1 possible containers; 2 possible objects',
|
||||||
self.assertEqual(
|
'Pass completed in 0s; 0 objects expired',
|
||||||
x.logger.log_dict['info'],
|
])
|
||||||
[(('Pass beginning; 1 possible containers; '
|
self.assertTrue('error' not in logs)
|
||||||
'2 possible objects',), {}),
|
|
||||||
(('Pass completed in 0s; 0 objects expired',), {})])
|
|
||||||
|
|
||||||
# Reverse test to be sure it still would blow up the way expected.
|
# Reverse test to be sure it still would blow up the way expected.
|
||||||
fake_swift = InternalClient([{'name': str(int(time() - 86400))}])
|
fake_swift = InternalClient([{'name': str(int(time() - 86400))}])
|
||||||
x = expirer.ObjectExpirer({}, logger=self.logger, swift=fake_swift)
|
x = expirer.ObjectExpirer(self.conf, logger=self.logger,
|
||||||
|
swift=fake_swift)
|
||||||
x.run_once()
|
x.run_once()
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
x.logger.log_dict['exception'],
|
x.logger.get_lines_for_level('error'), [
|
||||||
[(('Unhandled exception',), {},
|
'Unhandled exception: '])
|
||||||
str(Exception('This should not have been called')))])
|
log_args, log_kwargs = x.logger.log_dict['error'][-1]
|
||||||
|
self.assertEqual(str(log_kwargs['exc_info'][1]),
|
||||||
|
'This should not have been called')
|
||||||
|
|
||||||
def test_object_timestamp_break(self):
|
def test_object_timestamp_break(self):
|
||||||
class InternalClient(object):
|
class InternalClient(object):
|
||||||
@ -369,33 +372,27 @@ class TestObjectExpirer(TestCase):
|
|||||||
fake_swift = InternalClient(
|
fake_swift = InternalClient(
|
||||||
[{'name': str(int(time() - 86400))}],
|
[{'name': str(int(time() - 86400))}],
|
||||||
[{'name': '%d-actual-obj' % int(time() + 86400)}])
|
[{'name': '%d-actual-obj' % int(time() + 86400)}])
|
||||||
x = expirer.ObjectExpirer({}, logger=self.logger, swift=fake_swift)
|
x = expirer.ObjectExpirer(self.conf, logger=self.logger,
|
||||||
|
swift=fake_swift)
|
||||||
x.run_once()
|
x.run_once()
|
||||||
for exccall in x.logger.log_dict['exception']:
|
self.assertTrue('error' not in x.logger.all_log_lines())
|
||||||
self.assertTrue(
|
self.assertEqual(x.logger.get_lines_for_level('info'), [
|
||||||
'This should not have been called' not in exccall[0][0])
|
'Pass beginning; 1 possible containers; 2 possible objects',
|
||||||
self.assertEqual(
|
'Pass completed in 0s; 0 objects expired',
|
||||||
x.logger.log_dict['info'],
|
])
|
||||||
[(('Pass beginning; 1 possible containers; '
|
|
||||||
'2 possible objects',), {}),
|
|
||||||
(('Pass completed in 0s; 0 objects expired',), {})])
|
|
||||||
|
|
||||||
# Reverse test to be sure it still would blow up the way expected.
|
# Reverse test to be sure it still would blow up the way expected.
|
||||||
ts = int(time() - 86400)
|
ts = int(time() - 86400)
|
||||||
fake_swift = InternalClient(
|
fake_swift = InternalClient(
|
||||||
[{'name': str(int(time() - 86400))}],
|
[{'name': str(int(time() - 86400))}],
|
||||||
[{'name': '%d-actual-obj' % ts}])
|
[{'name': '%d-actual-obj' % ts}])
|
||||||
x = expirer.ObjectExpirer({}, logger=self.logger, swift=fake_swift)
|
x = expirer.ObjectExpirer(self.conf, logger=self.logger,
|
||||||
|
swift=fake_swift)
|
||||||
x.delete_actual_object = should_not_be_called
|
x.delete_actual_object = should_not_be_called
|
||||||
x.run_once()
|
x.run_once()
|
||||||
excswhiledeleting = []
|
|
||||||
for exccall in x.logger.log_dict['exception']:
|
|
||||||
if exccall[0][0].startswith('Exception while deleting '):
|
|
||||||
excswhiledeleting.append(exccall[0][0])
|
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
excswhiledeleting,
|
x.logger.get_lines_for_level('error'),
|
||||||
['Exception while deleting object %d %d-actual-obj '
|
['Exception while deleting object %d %d-actual-obj '
|
||||||
'This should not have been called' % (ts, ts)])
|
'This should not have been called: ' % (ts, ts)])
|
||||||
|
|
||||||
def test_failed_delete_keeps_entry(self):
|
def test_failed_delete_keeps_entry(self):
|
||||||
class InternalClient(object):
|
class InternalClient(object):
|
||||||
@ -428,24 +425,22 @@ class TestObjectExpirer(TestCase):
|
|||||||
fake_swift = InternalClient(
|
fake_swift = InternalClient(
|
||||||
[{'name': str(int(time() - 86400))}],
|
[{'name': str(int(time() - 86400))}],
|
||||||
[{'name': '%d-actual-obj' % ts}])
|
[{'name': '%d-actual-obj' % ts}])
|
||||||
x = expirer.ObjectExpirer({}, logger=self.logger, swift=fake_swift)
|
x = expirer.ObjectExpirer(self.conf, logger=self.logger,
|
||||||
|
swift=fake_swift)
|
||||||
x.iter_containers = lambda: [str(int(time() - 86400))]
|
x.iter_containers = lambda: [str(int(time() - 86400))]
|
||||||
x.delete_actual_object = deliberately_blow_up
|
x.delete_actual_object = deliberately_blow_up
|
||||||
x.pop_queue = should_not_get_called
|
x.pop_queue = should_not_get_called
|
||||||
x.run_once()
|
x.run_once()
|
||||||
excswhiledeleting = []
|
error_lines = x.logger.get_lines_for_level('error')
|
||||||
for exccall in x.logger.log_dict['exception']:
|
|
||||||
if exccall[0][0].startswith('Exception while deleting '):
|
|
||||||
excswhiledeleting.append(exccall[0][0])
|
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
excswhiledeleting,
|
error_lines,
|
||||||
['Exception while deleting object %d %d-actual-obj '
|
['Exception while deleting object %d %d-actual-obj '
|
||||||
'failed to delete actual object' % (ts, ts)])
|
'failed to delete actual object: ' % (ts, ts)])
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
x.logger.log_dict['info'],
|
x.logger.get_lines_for_level('info'), [
|
||||||
[(('Pass beginning; 1 possible containers; '
|
'Pass beginning; 1 possible containers; 2 possible objects',
|
||||||
'2 possible objects',), {}),
|
'Pass completed in 0s; 0 objects expired',
|
||||||
(('Pass completed in 0s; 0 objects expired',), {})])
|
])
|
||||||
|
|
||||||
# Reverse test to be sure it still would blow up the way expected.
|
# Reverse test to be sure it still would blow up the way expected.
|
||||||
ts = int(time() - 86400)
|
ts = int(time() - 86400)
|
||||||
@ -453,18 +448,15 @@ class TestObjectExpirer(TestCase):
|
|||||||
[{'name': str(int(time() - 86400))}],
|
[{'name': str(int(time() - 86400))}],
|
||||||
[{'name': '%d-actual-obj' % ts}])
|
[{'name': '%d-actual-obj' % ts}])
|
||||||
self.logger._clear()
|
self.logger._clear()
|
||||||
x = expirer.ObjectExpirer({}, logger=self.logger, swift=fake_swift)
|
x = expirer.ObjectExpirer(self.conf, logger=self.logger,
|
||||||
|
swift=fake_swift)
|
||||||
x.delete_actual_object = lambda o, t: None
|
x.delete_actual_object = lambda o, t: None
|
||||||
x.pop_queue = should_not_get_called
|
x.pop_queue = should_not_get_called
|
||||||
x.run_once()
|
x.run_once()
|
||||||
excswhiledeleting = []
|
|
||||||
for exccall in x.logger.log_dict['exception']:
|
|
||||||
if exccall[0][0].startswith('Exception while deleting '):
|
|
||||||
excswhiledeleting.append(exccall[0][0])
|
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
excswhiledeleting,
|
self.logger.get_lines_for_level('error'),
|
||||||
['Exception while deleting object %d %d-actual-obj This should '
|
['Exception while deleting object %d %d-actual-obj This should '
|
||||||
'not have been called' % (ts, ts)])
|
'not have been called: ' % (ts, ts)])
|
||||||
|
|
||||||
def test_success_gets_counted(self):
|
def test_success_gets_counted(self):
|
||||||
class InternalClient(object):
|
class InternalClient(object):
|
||||||
@ -493,7 +485,8 @@ class TestObjectExpirer(TestCase):
|
|||||||
fake_swift = InternalClient(
|
fake_swift = InternalClient(
|
||||||
[{'name': str(int(time() - 86400))}],
|
[{'name': str(int(time() - 86400))}],
|
||||||
[{'name': '%d-acc/c/actual-obj' % int(time() - 86400)}])
|
[{'name': '%d-acc/c/actual-obj' % int(time() - 86400)}])
|
||||||
x = expirer.ObjectExpirer({}, logger=self.logger, swift=fake_swift)
|
x = expirer.ObjectExpirer(self.conf, logger=self.logger,
|
||||||
|
swift=fake_swift)
|
||||||
x.delete_actual_object = lambda o, t: None
|
x.delete_actual_object = lambda o, t: None
|
||||||
x.pop_queue = lambda c, o: None
|
x.pop_queue = lambda c, o: None
|
||||||
self.assertEqual(x.report_objects, 0)
|
self.assertEqual(x.report_objects, 0)
|
||||||
@ -501,10 +494,9 @@ class TestObjectExpirer(TestCase):
|
|||||||
x.run_once()
|
x.run_once()
|
||||||
self.assertEqual(x.report_objects, 1)
|
self.assertEqual(x.report_objects, 1)
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
x.logger.log_dict['info'],
|
x.logger.get_lines_for_level('info'),
|
||||||
[(('Pass beginning; 1 possible containers; '
|
['Pass beginning; 1 possible containers; 2 possible objects',
|
||||||
'2 possible objects',), {}),
|
'Pass completed in 0s; 1 objects expired'])
|
||||||
(('Pass completed in 0s; 1 objects expired',), {})])
|
|
||||||
|
|
||||||
def test_delete_actual_object_does_not_get_unicode(self):
|
def test_delete_actual_object_does_not_get_unicode(self):
|
||||||
class InternalClient(object):
|
class InternalClient(object):
|
||||||
@ -539,17 +531,18 @@ class TestObjectExpirer(TestCase):
|
|||||||
fake_swift = InternalClient(
|
fake_swift = InternalClient(
|
||||||
[{'name': str(int(time() - 86400))}],
|
[{'name': str(int(time() - 86400))}],
|
||||||
[{'name': u'%d-actual-obj' % int(time() - 86400)}])
|
[{'name': u'%d-actual-obj' % int(time() - 86400)}])
|
||||||
x = expirer.ObjectExpirer({}, logger=self.logger, swift=fake_swift)
|
x = expirer.ObjectExpirer(self.conf, logger=self.logger,
|
||||||
|
swift=fake_swift)
|
||||||
x.delete_actual_object = delete_actual_object_test_for_unicode
|
x.delete_actual_object = delete_actual_object_test_for_unicode
|
||||||
x.pop_queue = lambda c, o: None
|
x.pop_queue = lambda c, o: None
|
||||||
self.assertEqual(x.report_objects, 0)
|
self.assertEqual(x.report_objects, 0)
|
||||||
x.run_once()
|
x.run_once()
|
||||||
self.assertEqual(x.report_objects, 1)
|
self.assertEqual(x.report_objects, 1)
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
x.logger.log_dict['info'],
|
x.logger.get_lines_for_level('info'), [
|
||||||
[(('Pass beginning; 1 possible containers; '
|
'Pass beginning; 1 possible containers; 2 possible objects',
|
||||||
'2 possible objects',), {}),
|
'Pass completed in 0s; 1 objects expired',
|
||||||
(('Pass completed in 0s; 1 objects expired',), {})])
|
])
|
||||||
self.assertFalse(got_unicode[0])
|
self.assertFalse(got_unicode[0])
|
||||||
|
|
||||||
def test_failed_delete_continues_on(self):
|
def test_failed_delete_continues_on(self):
|
||||||
@ -579,7 +572,7 @@ class TestObjectExpirer(TestCase):
|
|||||||
def fail_delete_actual_object(actual_obj, timestamp):
|
def fail_delete_actual_object(actual_obj, timestamp):
|
||||||
raise Exception('failed to delete actual object')
|
raise Exception('failed to delete actual object')
|
||||||
|
|
||||||
x = expirer.ObjectExpirer({}, logger=self.logger)
|
x = expirer.ObjectExpirer(self.conf, logger=self.logger)
|
||||||
|
|
||||||
cts = int(time() - 86400)
|
cts = int(time() - 86400)
|
||||||
ots = int(time() - 86400)
|
ots = int(time() - 86400)
|
||||||
@ -597,28 +590,24 @@ class TestObjectExpirer(TestCase):
|
|||||||
x.swift = InternalClient(containers, objects)
|
x.swift = InternalClient(containers, objects)
|
||||||
x.delete_actual_object = fail_delete_actual_object
|
x.delete_actual_object = fail_delete_actual_object
|
||||||
x.run_once()
|
x.run_once()
|
||||||
excswhiledeleting = []
|
error_lines = x.logger.get_lines_for_level('error')
|
||||||
for exccall in x.logger.log_dict['exception']:
|
self.assertEqual(sorted(error_lines), sorted([
|
||||||
if exccall[0][0].startswith('Exception while deleting '):
|
|
||||||
excswhiledeleting.append(exccall[0][0])
|
|
||||||
self.assertEqual(sorted(excswhiledeleting), sorted([
|
|
||||||
'Exception while deleting object %d %d-actual-obj failed to '
|
'Exception while deleting object %d %d-actual-obj failed to '
|
||||||
'delete actual object' % (cts, ots),
|
'delete actual object: ' % (cts, ots),
|
||||||
'Exception while deleting object %d %d-next-obj failed to '
|
'Exception while deleting object %d %d-next-obj failed to '
|
||||||
'delete actual object' % (cts, ots),
|
'delete actual object: ' % (cts, ots),
|
||||||
'Exception while deleting object %d %d-actual-obj failed to '
|
'Exception while deleting object %d %d-actual-obj failed to '
|
||||||
'delete actual object' % (cts + 1, ots),
|
'delete actual object: ' % (cts + 1, ots),
|
||||||
'Exception while deleting object %d %d-next-obj failed to '
|
'Exception while deleting object %d %d-next-obj failed to '
|
||||||
'delete actual object' % (cts + 1, ots),
|
'delete actual object: ' % (cts + 1, ots),
|
||||||
'Exception while deleting container %d failed to delete '
|
'Exception while deleting container %d failed to delete '
|
||||||
'container' % (cts,),
|
'container: ' % (cts,),
|
||||||
'Exception while deleting container %d failed to delete '
|
'Exception while deleting container %d failed to delete '
|
||||||
'container' % (cts + 1,)]))
|
'container: ' % (cts + 1,)]))
|
||||||
self.assertEqual(
|
self.assertEqual(x.logger.get_lines_for_level('info'), [
|
||||||
x.logger.log_dict['info'],
|
'Pass beginning; 1 possible containers; 2 possible objects',
|
||||||
[(('Pass beginning; 1 possible containers; '
|
'Pass completed in 0s; 0 objects expired',
|
||||||
'2 possible objects',), {}),
|
])
|
||||||
(('Pass completed in 0s; 0 objects expired',), {})])
|
|
||||||
|
|
||||||
def test_run_forever_initial_sleep_random(self):
|
def test_run_forever_initial_sleep_random(self):
|
||||||
global last_not_sleep
|
global last_not_sleep
|
||||||
@ -664,9 +653,11 @@ class TestObjectExpirer(TestCase):
|
|||||||
finally:
|
finally:
|
||||||
expirer.sleep = orig_sleep
|
expirer.sleep = orig_sleep
|
||||||
self.assertEqual(str(err), 'exiting exception 2')
|
self.assertEqual(str(err), 'exiting exception 2')
|
||||||
self.assertEqual(x.logger.log_dict['exception'],
|
self.assertEqual(x.logger.get_lines_for_level('error'),
|
||||||
[(('Unhandled exception',), {},
|
['Unhandled exception: '])
|
||||||
'exception 1')])
|
log_args, log_kwargs = x.logger.log_dict['error'][0]
|
||||||
|
self.assertEqual(str(log_kwargs['exc_info'][1]),
|
||||||
|
'exception 1')
|
||||||
|
|
||||||
def test_delete_actual_object(self):
|
def test_delete_actual_object(self):
|
||||||
got_env = [None]
|
got_env = [None]
|
||||||
|
@ -27,7 +27,7 @@ from errno import ENOENT, ENOTEMPTY, ENOTDIR
|
|||||||
from eventlet.green import subprocess
|
from eventlet.green import subprocess
|
||||||
from eventlet import Timeout, tpool
|
from eventlet import Timeout, tpool
|
||||||
|
|
||||||
from test.unit import FakeLogger, debug_logger, patch_policies
|
from test.unit import debug_logger, patch_policies
|
||||||
from swift.common import utils
|
from swift.common import utils
|
||||||
from swift.common.utils import hash_path, mkdirs, normalize_timestamp, \
|
from swift.common.utils import hash_path, mkdirs, normalize_timestamp, \
|
||||||
storage_directory
|
storage_directory
|
||||||
@ -190,7 +190,7 @@ class TestObjectReplicator(unittest.TestCase):
|
|||||||
swift_dir=self.testdir, devices=self.devices, mount_check='false',
|
swift_dir=self.testdir, devices=self.devices, mount_check='false',
|
||||||
timeout='300', stats_interval='1', sync_method='rsync')
|
timeout='300', stats_interval='1', sync_method='rsync')
|
||||||
self.replicator = object_replicator.ObjectReplicator(self.conf)
|
self.replicator = object_replicator.ObjectReplicator(self.conf)
|
||||||
self.replicator.logger = FakeLogger()
|
self.logger = self.replicator.logger = debug_logger('test-replicator')
|
||||||
self.df_mgr = diskfile.DiskFileManager(self.conf,
|
self.df_mgr = diskfile.DiskFileManager(self.conf,
|
||||||
self.replicator.logger)
|
self.replicator.logger)
|
||||||
|
|
||||||
@ -280,22 +280,20 @@ class TestObjectReplicator(unittest.TestCase):
|
|||||||
|
|
||||||
def test_collect_jobs_mkdirs_error(self):
|
def test_collect_jobs_mkdirs_error(self):
|
||||||
|
|
||||||
|
non_local = {}
|
||||||
|
|
||||||
def blowup_mkdirs(path):
|
def blowup_mkdirs(path):
|
||||||
|
non_local['path'] = path
|
||||||
raise OSError('Ow!')
|
raise OSError('Ow!')
|
||||||
|
|
||||||
with mock.patch.object(object_replicator, 'mkdirs', blowup_mkdirs):
|
with mock.patch.object(object_replicator, 'mkdirs', blowup_mkdirs):
|
||||||
rmtree(self.objects, ignore_errors=1)
|
rmtree(self.objects, ignore_errors=1)
|
||||||
object_replicator.mkdirs = blowup_mkdirs
|
object_replicator.mkdirs = blowup_mkdirs
|
||||||
self.replicator.collect_jobs()
|
self.replicator.collect_jobs()
|
||||||
self.assertTrue('exception' in self.replicator.logger.log_dict)
|
self.assertEqual(self.logger.get_lines_for_level('error'), [
|
||||||
self.assertEquals(
|
'ERROR creating %s: ' % non_local['path']])
|
||||||
len(self.replicator.logger.log_dict['exception']), 1)
|
log_args, log_kwargs = self.logger.log_dict['error'][0]
|
||||||
exc_args, exc_kwargs, exc_str = \
|
self.assertEqual(str(log_kwargs['exc_info'][1]), 'Ow!')
|
||||||
self.replicator.logger.log_dict['exception'][0]
|
|
||||||
self.assertEquals(len(exc_args), 1)
|
|
||||||
self.assertTrue(exc_args[0].startswith('ERROR creating '))
|
|
||||||
self.assertEquals(exc_kwargs, {})
|
|
||||||
self.assertEquals(exc_str, 'Ow!')
|
|
||||||
|
|
||||||
def test_collect_jobs(self):
|
def test_collect_jobs(self):
|
||||||
jobs = self.replicator.collect_jobs()
|
jobs = self.replicator.collect_jobs()
|
||||||
@ -383,14 +381,13 @@ class TestObjectReplicator(unittest.TestCase):
|
|||||||
|
|
||||||
self.assertFalse(os.path.exists(pol_0_part_1_path))
|
self.assertFalse(os.path.exists(pol_0_part_1_path))
|
||||||
self.assertFalse(os.path.exists(pol_1_part_1_path))
|
self.assertFalse(os.path.exists(pol_1_part_1_path))
|
||||||
|
self.assertEqual(
|
||||||
logged_warnings = sorted(self.replicator.logger.log_dict['warning'])
|
sorted(self.logger.get_lines_for_level('warning')), [
|
||||||
self.assertEquals(
|
('Removing partition directory which was a file: %s'
|
||||||
(('Removing partition directory which was a file: %s',
|
% pol_1_part_1_path),
|
||||||
pol_1_part_1_path), {}), logged_warnings[0])
|
('Removing partition directory which was a file: %s'
|
||||||
self.assertEquals(
|
% pol_0_part_1_path),
|
||||||
(('Removing partition directory which was a file: %s',
|
])
|
||||||
pol_0_part_1_path), {}), logged_warnings[1])
|
|
||||||
|
|
||||||
def test_delete_partition(self):
|
def test_delete_partition(self):
|
||||||
with mock.patch('swift.obj.replicator.http_connect',
|
with mock.patch('swift.obj.replicator.http_connect',
|
||||||
@ -833,7 +830,8 @@ class TestObjectReplicator(unittest.TestCase):
|
|||||||
def test_delete_partition_ssync_with_cleanup_failure(self):
|
def test_delete_partition_ssync_with_cleanup_failure(self):
|
||||||
with mock.patch('swift.obj.replicator.http_connect',
|
with mock.patch('swift.obj.replicator.http_connect',
|
||||||
mock_http_connect(200)):
|
mock_http_connect(200)):
|
||||||
self.replicator.logger = mock_logger = mock.MagicMock()
|
self.replicator.logger = mock_logger = \
|
||||||
|
debug_logger('test-replicator')
|
||||||
df = self.df_mgr.get_diskfile('sda', '1', 'a', 'c', 'o')
|
df = self.df_mgr.get_diskfile('sda', '1', 'a', 'c', 'o')
|
||||||
mkdirs(df._datadir)
|
mkdirs(df._datadir)
|
||||||
f = open(os.path.join(df._datadir,
|
f = open(os.path.join(df._datadir,
|
||||||
@ -886,7 +884,7 @@ class TestObjectReplicator(unittest.TestCase):
|
|||||||
with mock.patch('os.rmdir',
|
with mock.patch('os.rmdir',
|
||||||
raise_exception_rmdir(OSError, ENOENT)):
|
raise_exception_rmdir(OSError, ENOENT)):
|
||||||
self.replicator.replicate()
|
self.replicator.replicate()
|
||||||
self.assertEquals(mock_logger.exception.call_count, 0)
|
self.assertFalse(mock_logger.get_lines_for_level('error'))
|
||||||
self.assertFalse(os.access(whole_path_from, os.F_OK))
|
self.assertFalse(os.access(whole_path_from, os.F_OK))
|
||||||
self.assertTrue(os.access(suffix_dir_path, os.F_OK))
|
self.assertTrue(os.access(suffix_dir_path, os.F_OK))
|
||||||
self.assertTrue(os.access(part_path, os.F_OK))
|
self.assertTrue(os.access(part_path, os.F_OK))
|
||||||
@ -895,7 +893,7 @@ class TestObjectReplicator(unittest.TestCase):
|
|||||||
with mock.patch('os.rmdir',
|
with mock.patch('os.rmdir',
|
||||||
raise_exception_rmdir(OSError, ENOTEMPTY)):
|
raise_exception_rmdir(OSError, ENOTEMPTY)):
|
||||||
self.replicator.replicate()
|
self.replicator.replicate()
|
||||||
self.assertEquals(mock_logger.exception.call_count, 0)
|
self.assertFalse(mock_logger.get_lines_for_level('error'))
|
||||||
self.assertFalse(os.access(whole_path_from, os.F_OK))
|
self.assertFalse(os.access(whole_path_from, os.F_OK))
|
||||||
self.assertTrue(os.access(suffix_dir_path, os.F_OK))
|
self.assertTrue(os.access(suffix_dir_path, os.F_OK))
|
||||||
self.assertTrue(os.access(part_path, os.F_OK))
|
self.assertTrue(os.access(part_path, os.F_OK))
|
||||||
@ -904,7 +902,7 @@ class TestObjectReplicator(unittest.TestCase):
|
|||||||
with mock.patch('os.rmdir',
|
with mock.patch('os.rmdir',
|
||||||
raise_exception_rmdir(OSError, ENOTDIR)):
|
raise_exception_rmdir(OSError, ENOTDIR)):
|
||||||
self.replicator.replicate()
|
self.replicator.replicate()
|
||||||
self.assertEquals(mock_logger.exception.call_count, 1)
|
self.assertEqual(len(mock_logger.get_lines_for_level('error')), 1)
|
||||||
self.assertFalse(os.access(whole_path_from, os.F_OK))
|
self.assertFalse(os.access(whole_path_from, os.F_OK))
|
||||||
self.assertTrue(os.access(suffix_dir_path, os.F_OK))
|
self.assertTrue(os.access(suffix_dir_path, os.F_OK))
|
||||||
self.assertTrue(os.access(part_path, os.F_OK))
|
self.assertTrue(os.access(part_path, os.F_OK))
|
||||||
|
@ -77,6 +77,8 @@ class TestObjectController(unittest.TestCase):
|
|||||||
self.df_mgr = diskfile.DiskFileManager(conf,
|
self.df_mgr = diskfile.DiskFileManager(conf,
|
||||||
self.object_controller.logger)
|
self.object_controller.logger)
|
||||||
|
|
||||||
|
self.logger = debug_logger('test-object-controller')
|
||||||
|
|
||||||
def tearDown(self):
|
def tearDown(self):
|
||||||
"""Tear down for testing swift.object.server.ObjectController"""
|
"""Tear down for testing swift.object.server.ObjectController"""
|
||||||
rmtree(self.tmpdir)
|
rmtree(self.tmpdir)
|
||||||
@ -3305,7 +3307,7 @@ class TestObjectController(unittest.TestCase):
|
|||||||
given_args.extend(args)
|
given_args.extend(args)
|
||||||
|
|
||||||
self.object_controller.async_update = fake_async_update
|
self.object_controller.async_update = fake_async_update
|
||||||
self.object_controller.logger = FakeLogger()
|
self.object_controller.logger = self.logger
|
||||||
req = Request.blank(
|
req = Request.blank(
|
||||||
'/v1/a/c/o',
|
'/v1/a/c/o',
|
||||||
environ={'REQUEST_METHOD': 'PUT'},
|
environ={'REQUEST_METHOD': 'PUT'},
|
||||||
@ -3317,10 +3319,10 @@ class TestObjectController(unittest.TestCase):
|
|||||||
self.object_controller.delete_at_update('PUT', 2, 'a', 'c', 'o',
|
self.object_controller.delete_at_update('PUT', 2, 'a', 'c', 'o',
|
||||||
req, 'sda1', 0)
|
req, 'sda1', 0)
|
||||||
self.assertEquals(
|
self.assertEquals(
|
||||||
self.object_controller.logger.log_dict['warning'],
|
self.logger.get_lines_for_level('warning'),
|
||||||
[(('X-Delete-At-Container header must be specified for expiring '
|
['X-Delete-At-Container header must be specified for expiring '
|
||||||
'objects background PUT to work properly. Making best guess as '
|
'objects background PUT to work properly. Making best guess as '
|
||||||
'to the container name for now.',), {})])
|
'to the container name for now.'])
|
||||||
|
|
||||||
def test_delete_at_update_delete(self):
|
def test_delete_at_update_delete(self):
|
||||||
given_args = []
|
given_args = []
|
||||||
@ -4167,7 +4169,7 @@ class TestObjectController(unittest.TestCase):
|
|||||||
outbuf = StringIO()
|
outbuf = StringIO()
|
||||||
self.object_controller = object_server.ObjectController(
|
self.object_controller = object_server.ObjectController(
|
||||||
{'devices': self.testdir, 'mount_check': 'false',
|
{'devices': self.testdir, 'mount_check': 'false',
|
||||||
'replication_server': 'false'}, logger=FakeLogger())
|
'replication_server': 'false'}, logger=self.logger)
|
||||||
|
|
||||||
def start_response(*args):
|
def start_response(*args):
|
||||||
# Sends args to outbuf
|
# Sends args to outbuf
|
||||||
@ -4207,11 +4209,10 @@ class TestObjectController(unittest.TestCase):
|
|||||||
env, start_response)
|
env, start_response)
|
||||||
self.assertEqual(response, answer)
|
self.assertEqual(response, answer)
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
self.object_controller.logger.log_dict['info'],
|
self.logger.get_lines_for_level('info'),
|
||||||
[(('None - - [01/Jan/1970:02:46:41 +0000] "PUT'
|
['None - - [01/Jan/1970:02:46:41 +0000] "PUT'
|
||||||
' /sda1/p/a/c/o" 405 - "-" "-" "-" 1.0000 "-"'
|
' /sda1/p/a/c/o" 405 - "-" "-" "-" 1.0000 "-"'
|
||||||
' 1234 -',),
|
' 1234 -'])
|
||||||
{})])
|
|
||||||
|
|
||||||
def test_call_incorrect_replication_method(self):
|
def test_call_incorrect_replication_method(self):
|
||||||
inbuf = StringIO()
|
inbuf = StringIO()
|
||||||
@ -4281,17 +4282,17 @@ class TestObjectController(unittest.TestCase):
|
|||||||
new=mock_method):
|
new=mock_method):
|
||||||
response = self.object_controller.__call__(env, start_response)
|
response = self.object_controller.__call__(env, start_response)
|
||||||
self.assertEqual(response, answer)
|
self.assertEqual(response, answer)
|
||||||
self.assertEqual(self.object_controller.logger.log_dict['info'],
|
self.assertEqual(self.logger.get_lines_for_level('info'), [])
|
||||||
[])
|
|
||||||
|
|
||||||
def test__call__returns_500(self):
|
def test__call__returns_500(self):
|
||||||
inbuf = StringIO()
|
inbuf = StringIO()
|
||||||
errbuf = StringIO()
|
errbuf = StringIO()
|
||||||
outbuf = StringIO()
|
outbuf = StringIO()
|
||||||
|
self.logger = debug_logger('test')
|
||||||
self.object_controller = object_server.ObjectController(
|
self.object_controller = object_server.ObjectController(
|
||||||
{'devices': self.testdir, 'mount_check': 'false',
|
{'devices': self.testdir, 'mount_check': 'false',
|
||||||
'replication_server': 'false', 'log_requests': 'false'},
|
'replication_server': 'false', 'log_requests': 'false'},
|
||||||
logger=FakeLogger())
|
logger=self.logger)
|
||||||
|
|
||||||
def start_response(*args):
|
def start_response(*args):
|
||||||
# Sends args to outbuf
|
# Sends args to outbuf
|
||||||
@ -4323,14 +4324,11 @@ class TestObjectController(unittest.TestCase):
|
|||||||
response = self.object_controller.__call__(env, start_response)
|
response = self.object_controller.__call__(env, start_response)
|
||||||
self.assertTrue(response[0].startswith(
|
self.assertTrue(response[0].startswith(
|
||||||
'Traceback (most recent call last):'))
|
'Traceback (most recent call last):'))
|
||||||
self.assertEqual(
|
self.assertEqual(self.logger.get_lines_for_level('error'), [
|
||||||
self.object_controller.logger.log_dict['exception'],
|
'ERROR __call__ error with %(method)s %(path)s : ' % {
|
||||||
[(('ERROR __call__ error with %(method)s %(path)s ',
|
'method': 'PUT', 'path': '/sda1/p/a/c/o'},
|
||||||
{'method': 'PUT', 'path': '/sda1/p/a/c/o'}),
|
])
|
||||||
{},
|
self.assertEqual(self.logger.get_lines_for_level('info'), [])
|
||||||
'')])
|
|
||||||
self.assertEqual(self.object_controller.logger.log_dict['INFO'],
|
|
||||||
[])
|
|
||||||
|
|
||||||
def test_PUT_slow(self):
|
def test_PUT_slow(self):
|
||||||
inbuf = StringIO()
|
inbuf = StringIO()
|
||||||
@ -4340,7 +4338,7 @@ class TestObjectController(unittest.TestCase):
|
|||||||
{'devices': self.testdir, 'mount_check': 'false',
|
{'devices': self.testdir, 'mount_check': 'false',
|
||||||
'replication_server': 'false', 'log_requests': 'false',
|
'replication_server': 'false', 'log_requests': 'false',
|
||||||
'slow': '10'},
|
'slow': '10'},
|
||||||
logger=FakeLogger())
|
logger=self.logger)
|
||||||
|
|
||||||
def start_response(*args):
|
def start_response(*args):
|
||||||
# Sends args to outbuf
|
# Sends args to outbuf
|
||||||
@ -4373,14 +4371,14 @@ class TestObjectController(unittest.TestCase):
|
|||||||
mock.MagicMock()) as ms:
|
mock.MagicMock()) as ms:
|
||||||
self.object_controller.__call__(env, start_response)
|
self.object_controller.__call__(env, start_response)
|
||||||
ms.assert_called_with(9)
|
ms.assert_called_with(9)
|
||||||
self.assertEqual(
|
self.assertEqual(self.logger.get_lines_for_level('info'),
|
||||||
self.object_controller.logger.log_dict['info'], [])
|
[])
|
||||||
|
|
||||||
def test_log_line_format(self):
|
def test_log_line_format(self):
|
||||||
req = Request.blank(
|
req = Request.blank(
|
||||||
'/sda1/p/a/c/o',
|
'/sda1/p/a/c/o',
|
||||||
environ={'REQUEST_METHOD': 'HEAD', 'REMOTE_ADDR': '1.2.3.4'})
|
environ={'REQUEST_METHOD': 'HEAD', 'REMOTE_ADDR': '1.2.3.4'})
|
||||||
self.object_controller.logger = FakeLogger()
|
self.object_controller.logger = self.logger
|
||||||
with mock.patch(
|
with mock.patch(
|
||||||
'time.gmtime', mock.MagicMock(side_effect=[gmtime(10001.0)])):
|
'time.gmtime', mock.MagicMock(side_effect=[gmtime(10001.0)])):
|
||||||
with mock.patch(
|
with mock.patch(
|
||||||
@ -4390,9 +4388,9 @@ class TestObjectController(unittest.TestCase):
|
|||||||
'os.getpid', mock.MagicMock(return_value=1234)):
|
'os.getpid', mock.MagicMock(return_value=1234)):
|
||||||
req.get_response(self.object_controller)
|
req.get_response(self.object_controller)
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
self.object_controller.logger.log_dict['info'],
|
self.logger.get_lines_for_level('info'),
|
||||||
[(('1.2.3.4 - - [01/Jan/1970:02:46:41 +0000] "HEAD /sda1/p/a/c/o" '
|
['1.2.3.4 - - [01/Jan/1970:02:46:41 +0000] "HEAD /sda1/p/a/c/o" '
|
||||||
'404 - "-" "-" "-" 2.0000 "-" 1234 -',), {})])
|
'404 - "-" "-" "-" 2.0000 "-" 1234 -'])
|
||||||
|
|
||||||
@patch_policies([storage_policy.StoragePolicy(0, 'zero', True),
|
@patch_policies([storage_policy.StoragePolicy(0, 'zero', True),
|
||||||
storage_policy.StoragePolicy(1, 'one', False)])
|
storage_policy.StoragePolicy(1, 'one', False)])
|
||||||
|
Loading…
x
Reference in New Issue
Block a user