From 0ba071f27c009e1d028189e812f722e8583a07ee Mon Sep 17 00:00:00 2001 From: Darrell Bishop Date: Tue, 26 Nov 2013 15:08:13 -0500 Subject: [PATCH] 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 --- swift/obj/updater.py | 6 +-- test/unit/__init__.py | 1 + test/unit/obj/test_updater.py | 90 +++++++++++++++++++++++++++++++++-- 3 files changed, 89 insertions(+), 8 deletions(-) diff --git a/swift/obj/updater.py b/swift/obj/updater.py index 1592088c54..e2db10c91e 100644 --- a/swift/obj/updater.py +++ b/swift/obj/updater.py @@ -71,8 +71,8 @@ class ObjectUpdater(Daemon): # read from container ring to ensure it's fresh self.get_container_ring().get_nodes('') for device in os.listdir(self.devices): - if self.mount_check and not \ - ismount(os.path.join(self.devices, device)): + if self.mount_check and \ + not ismount(os.path.join(self.devices, device)): self.logger.increment('errors') self.logger.warn( _('Skipping %s as it is not mounted'), device) @@ -115,7 +115,7 @@ class ObjectUpdater(Daemon): self.failures = 0 for device in os.listdir(self.devices): 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.warn( _('Skipping %s as it is not mounted'), device) diff --git a/test/unit/__init__.py b/test/unit/__init__.py index 8ad04cbc5e..b764f16596 100644 --- a/test/unit/__init__.py +++ b/test/unit/__init__.py @@ -256,6 +256,7 @@ class FakeLogger(logging.Logger): error = _store_and_log_in('error') info = _store_and_log_in('info') warning = _store_and_log_in('warning') + warn = _store_and_log_in('warning') debug = _store_and_log_in('debug') def exception(self, *args, **kwargs): diff --git a/test/unit/obj/test_updater.py b/test/unit/obj/test_updater.py index eb9f07c0ac..41bec125f2 100644 --- a/test/unit/obj/test_updater.py +++ b/test/unit/obj/test_updater.py @@ -14,6 +14,7 @@ # limitations under the License. import cPickle as pickle +import mock import os import unittest from contextlib import closing @@ -45,11 +46,15 @@ class TestObjectUpdater(unittest.TestCase): ring_file = os.path.join(self.testdir, 'container.ring.gz') with closing(GzipFile(ring_file, 'wb')) as f: 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, 'device': 'sda1', 'zone': 0}, {'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) self.devices_dir = os.path.join(self.testdir, 'devices') os.mkdir(self.devices_dir) @@ -80,6 +85,11 @@ class TestObjectUpdater(unittest.TestCase): prefix_dir = os.path.join(self.sda1, ASYNCDIR, 'abc') 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 = { 'a': [1089.3, 18.37, 12.83, 1.3], 'b': [49.4, 49.3, 49.2, 49.1], @@ -112,9 +122,12 @@ class TestObjectUpdater(unittest.TestCase): 'node_timeout': '5'}) cu.object_sweep(self.sda1) self.assert_(not os.path.exists(prefix_dir)) + self.assert_(os.path.exists(not_a_dir_path)) 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({ 'devices': self.devices_dir, 'mount_check': 'false', @@ -127,12 +140,62 @@ class TestObjectUpdater(unittest.TestCase): 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') + 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') os.mkdir(odd_dir) cu.run_once() self.assert_(os.path.exists(async_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') odir = os.path.join(async_dir, ohash[-3:]) @@ -147,7 +210,8 @@ class TestObjectUpdater(unittest.TestCase): with open(path, 'wb') as async_pending: pickle.dump({'op': 'PUT', 'account': 'a', 'container': 'c', 'obj': 'o', 'headers': { - 'X-Container-Timestamp': normalize_timestamp(0)}}, + 'X-Container-Timestamp': + normalize_timestamp(0)}}, async_pending) cu.logger = FakeLogger() cu.run_once() @@ -155,6 +219,8 @@ class TestObjectUpdater(unittest.TestCase): self.assert_(os.path.exists(op_path)) self.assertEqual(cu.logger.get_increment_counts(), {'failures': 1, 'unlinks': 1}) + self.assertEqual(None, + pickle.load(open(op_path)).get('successes')) bindsock = listen(('127.0.0.1', 0)) @@ -196,7 +262,7 @@ class TestObjectUpdater(unittest.TestCase): return err return None - event = spawn(accept, [201, 500]) + event = spawn(accept, [201, 500, 500]) for dev in cu.get_container_ring().devs: if dev is not None: dev['port'] = bindsock.getsockname()[1] @@ -209,6 +275,20 @@ class TestObjectUpdater(unittest.TestCase): self.assert_(os.path.exists(op_path)) self.assertEqual(cu.logger.get_increment_counts(), {'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]) cu.logger = FakeLogger()