Fix bug in obj updater run_once().

The "not" in front of the ismount() call got accidentally dropped in a
recent change.  This patch adds it back along with a few more tests.

Note that this bug only showed up on an SAIO during probe tests because
I used actually-mounted (virtual) "disks".  So keep that in mind when
building SAIOs for development/testing.

Change-Id: Ia193f3c4b73203605954036863575c22ddab6b03
This commit is contained in:
Darrell Bishop 2013-11-26 15:08:13 -05:00
parent c15f98f3b3
commit 0ba071f27c
3 changed files with 89 additions and 8 deletions

View File

@ -71,8 +71,8 @@ class ObjectUpdater(Daemon):
# read from container ring to ensure it's fresh # read from container ring to ensure it's fresh
self.get_container_ring().get_nodes('') self.get_container_ring().get_nodes('')
for device in os.listdir(self.devices): for device in os.listdir(self.devices):
if self.mount_check and not \ if self.mount_check and \
ismount(os.path.join(self.devices, device)): not ismount(os.path.join(self.devices, device)):
self.logger.increment('errors') self.logger.increment('errors')
self.logger.warn( self.logger.warn(
_('Skipping %s as it is not mounted'), device) _('Skipping %s as it is not mounted'), device)
@ -115,7 +115,7 @@ class ObjectUpdater(Daemon):
self.failures = 0 self.failures = 0
for device in os.listdir(self.devices): for device in os.listdir(self.devices):
if self.mount_check and \ if self.mount_check and \
ismount(os.path.join(self.devices, device)): not ismount(os.path.join(self.devices, device)):
self.logger.increment('errors') self.logger.increment('errors')
self.logger.warn( self.logger.warn(
_('Skipping %s as it is not mounted'), device) _('Skipping %s as it is not mounted'), device)

View File

@ -256,6 +256,7 @@ class FakeLogger(logging.Logger):
error = _store_and_log_in('error') error = _store_and_log_in('error')
info = _store_and_log_in('info') info = _store_and_log_in('info')
warning = _store_and_log_in('warning') warning = _store_and_log_in('warning')
warn = _store_and_log_in('warning')
debug = _store_and_log_in('debug') debug = _store_and_log_in('debug')
def exception(self, *args, **kwargs): def exception(self, *args, **kwargs):

View File

@ -14,6 +14,7 @@
# limitations under the License. # limitations under the License.
import cPickle as pickle import cPickle as pickle
import mock
import os import os
import unittest import unittest
from contextlib import closing from contextlib import closing
@ -45,11 +46,15 @@ class TestObjectUpdater(unittest.TestCase):
ring_file = os.path.join(self.testdir, 'container.ring.gz') ring_file = os.path.join(self.testdir, 'container.ring.gz')
with closing(GzipFile(ring_file, 'wb')) as f: with closing(GzipFile(ring_file, 'wb')) as f:
pickle.dump( pickle.dump(
RingData([[0, 1, 0, 1], [1, 0, 1, 0]], RingData([[0, 1, 2, 0, 1, 2],
[1, 2, 0, 1, 2, 0],
[2, 3, 1, 2, 3, 1]],
[{'id': 0, 'ip': '127.0.0.1', 'port': 1, [{'id': 0, 'ip': '127.0.0.1', 'port': 1,
'device': 'sda1', 'zone': 0}, 'device': 'sda1', 'zone': 0},
{'id': 1, 'ip': '127.0.0.1', 'port': 1, {'id': 1, 'ip': '127.0.0.1', 'port': 1,
'device': 'sda1', 'zone': 2}], 30), 'device': 'sda1', 'zone': 2},
{'id': 2, 'ip': '127.0.0.1', 'port': 1,
'device': 'sda1', 'zone': 4}], 30),
f) f)
self.devices_dir = os.path.join(self.testdir, 'devices') self.devices_dir = os.path.join(self.testdir, 'devices')
os.mkdir(self.devices_dir) os.mkdir(self.devices_dir)
@ -80,6 +85,11 @@ class TestObjectUpdater(unittest.TestCase):
prefix_dir = os.path.join(self.sda1, ASYNCDIR, 'abc') prefix_dir = os.path.join(self.sda1, ASYNCDIR, 'abc')
mkpath(prefix_dir) mkpath(prefix_dir)
# A non-directory where directory is expected should just be skipped...
not_a_dir_path = os.path.join(self.sda1, ASYNCDIR, 'not_a_dir')
with open(not_a_dir_path, 'w'):
pass
objects = { objects = {
'a': [1089.3, 18.37, 12.83, 1.3], 'a': [1089.3, 18.37, 12.83, 1.3],
'b': [49.4, 49.3, 49.2, 49.1], 'b': [49.4, 49.3, 49.2, 49.1],
@ -112,9 +122,12 @@ class TestObjectUpdater(unittest.TestCase):
'node_timeout': '5'}) 'node_timeout': '5'})
cu.object_sweep(self.sda1) cu.object_sweep(self.sda1)
self.assert_(not os.path.exists(prefix_dir)) self.assert_(not os.path.exists(prefix_dir))
self.assert_(os.path.exists(not_a_dir_path))
self.assertEqual(expected, seen) self.assertEqual(expected, seen)
def test_run_once(self): @mock.patch.object(object_updater, 'ismount')
def test_run_once_with_disk_unmounted(self, mock_ismount):
mock_ismount.return_value = False
cu = object_updater.ObjectUpdater({ cu = object_updater.ObjectUpdater({
'devices': self.devices_dir, 'devices': self.devices_dir,
'mount_check': 'false', 'mount_check': 'false',
@ -127,12 +140,62 @@ class TestObjectUpdater(unittest.TestCase):
os.mkdir(async_dir) os.mkdir(async_dir)
cu.run_once() cu.run_once()
self.assert_(os.path.exists(async_dir)) self.assert_(os.path.exists(async_dir))
# mount_check == False means no call to ismount
self.assertEqual([], mock_ismount.mock_calls)
cu = object_updater.ObjectUpdater({
'devices': self.devices_dir,
'mount_check': 'TrUe',
'swift_dir': self.testdir,
'interval': '1',
'concurrency': '1',
'node_timeout': '15'})
odd_dir = os.path.join(async_dir, 'not really supposed to be here')
os.mkdir(odd_dir)
cu.logger = FakeLogger()
cu.run_once()
self.assert_(os.path.exists(async_dir))
self.assert_(os.path.exists(odd_dir)) # skipped because not mounted!
# mount_check == True means ismount was checked
self.assertEqual([
mock.call(self.sda1),
], mock_ismount.mock_calls)
self.assertEqual(cu.logger.get_increment_counts(), {'errors': 1})
@mock.patch.object(object_updater, 'ismount')
def test_run_once(self, mock_ismount):
mock_ismount.return_value = True
cu = object_updater.ObjectUpdater({
'devices': self.devices_dir,
'mount_check': 'false',
'swift_dir': self.testdir,
'interval': '1',
'concurrency': '1',
'node_timeout': '15'})
cu.run_once()
async_dir = os.path.join(self.sda1, ASYNCDIR)
os.mkdir(async_dir)
cu.run_once()
self.assert_(os.path.exists(async_dir))
# mount_check == False means no call to ismount
self.assertEqual([], mock_ismount.mock_calls)
cu = object_updater.ObjectUpdater({
'devices': self.devices_dir,
'mount_check': 'TrUe',
'swift_dir': self.testdir,
'interval': '1',
'concurrency': '1',
'node_timeout': '15'})
odd_dir = os.path.join(async_dir, 'not really supposed to be here') odd_dir = os.path.join(async_dir, 'not really supposed to be here')
os.mkdir(odd_dir) os.mkdir(odd_dir)
cu.run_once() cu.run_once()
self.assert_(os.path.exists(async_dir)) self.assert_(os.path.exists(async_dir))
self.assert_(not os.path.exists(odd_dir)) self.assert_(not os.path.exists(odd_dir))
# mount_check == True means ismount was checked
self.assertEqual([
mock.call(self.sda1),
], mock_ismount.mock_calls)
ohash = hash_path('a', 'c', 'o') ohash = hash_path('a', 'c', 'o')
odir = os.path.join(async_dir, ohash[-3:]) odir = os.path.join(async_dir, ohash[-3:])
@ -147,7 +210,8 @@ class TestObjectUpdater(unittest.TestCase):
with open(path, 'wb') as async_pending: with open(path, 'wb') as async_pending:
pickle.dump({'op': 'PUT', 'account': 'a', 'container': 'c', pickle.dump({'op': 'PUT', 'account': 'a', 'container': 'c',
'obj': 'o', 'headers': { 'obj': 'o', 'headers': {
'X-Container-Timestamp': normalize_timestamp(0)}}, 'X-Container-Timestamp':
normalize_timestamp(0)}},
async_pending) async_pending)
cu.logger = FakeLogger() cu.logger = FakeLogger()
cu.run_once() cu.run_once()
@ -155,6 +219,8 @@ class TestObjectUpdater(unittest.TestCase):
self.assert_(os.path.exists(op_path)) self.assert_(os.path.exists(op_path))
self.assertEqual(cu.logger.get_increment_counts(), self.assertEqual(cu.logger.get_increment_counts(),
{'failures': 1, 'unlinks': 1}) {'failures': 1, 'unlinks': 1})
self.assertEqual(None,
pickle.load(open(op_path)).get('successes'))
bindsock = listen(('127.0.0.1', 0)) bindsock = listen(('127.0.0.1', 0))
@ -196,7 +262,7 @@ class TestObjectUpdater(unittest.TestCase):
return err return err
return None return None
event = spawn(accept, [201, 500]) event = spawn(accept, [201, 500, 500])
for dev in cu.get_container_ring().devs: for dev in cu.get_container_ring().devs:
if dev is not None: if dev is not None:
dev['port'] = bindsock.getsockname()[1] dev['port'] = bindsock.getsockname()[1]
@ -209,6 +275,20 @@ class TestObjectUpdater(unittest.TestCase):
self.assert_(os.path.exists(op_path)) self.assert_(os.path.exists(op_path))
self.assertEqual(cu.logger.get_increment_counts(), self.assertEqual(cu.logger.get_increment_counts(),
{'failures': 1}) {'failures': 1})
self.assertEqual([0],
pickle.load(open(op_path)).get('successes'))
event = spawn(accept, [404, 500])
cu.logger = FakeLogger()
cu.run_once()
err = event.wait()
if err:
raise err
self.assert_(os.path.exists(op_path))
self.assertEqual(cu.logger.get_increment_counts(),
{'failures': 1})
self.assertEqual([0, 1],
pickle.load(open(op_path)).get('successes'))
event = spawn(accept, [201]) event = spawn(accept, [201])
cu.logger = FakeLogger() cu.logger = FakeLogger()