reconstructor: include partially reverted handoffs in handoffs_remaining

For a reconstructor revert job, if sync'd to sufficient other nodes,
the handoff partition is considered done and handoffs_remaining is not
incremented. With the new max_objects_per_revert option [1], a ssync
job may appear to be complete but not all objects have yet been
reverted, so handoffs remaining should be incremented.

[1] Related-Change: If81760c80a4692212e3774e73af5ce37c02e8aff
Change-Id: I59572f75b9b0ba331369eb7358932943b7935ff0
This commit is contained in:
Alistair Coles 2021-12-03 14:35:09 +00:00
parent 8ee631ccee
commit 1b3879e0da
4 changed files with 78 additions and 27 deletions

View File

@ -1088,19 +1088,22 @@ class ObjectReconstructor(Daemon):
with df_mgr.partition_lock(job['device'], job['policy'], with df_mgr.partition_lock(job['device'], job['policy'],
job['partition'], name='replication', job['partition'], name='replication',
timeout=0.2): timeout=0.2):
limited_by_max_objects = False
for node in job['sync_to']: for node in job['sync_to']:
node['backend_index'] = job['policy'].get_backend_index( node['backend_index'] = job['policy'].get_backend_index(
node['index']) node['index'])
success, in_sync_objs = ssync_sender( sender = ssync_sender(
self, node, job, job['suffixes'], self, node, job, job['suffixes'],
include_non_durable=True, include_non_durable=True,
max_objects=self.max_objects_per_revert)() max_objects=self.max_objects_per_revert)
success, in_sync_objs = sender()
limited_by_max_objects |= sender.limited_by_max_objects
if success: if success:
syncd_with += 1 syncd_with += 1
reverted_objs.update(in_sync_objs) reverted_objs.update(in_sync_objs)
if syncd_with >= len(job['sync_to']): if syncd_with >= len(job['sync_to']):
self.delete_reverted_objs(job, reverted_objs) self.delete_reverted_objs(job, reverted_objs)
else: if syncd_with < len(job['sync_to']) or limited_by_max_objects:
self.handoffs_remaining += 1 self.handoffs_remaining += 1
except PartitionLockTimeout: except PartitionLockTimeout:
self.logger.info("Unable to lock handoff partition %d for revert " self.logger.info("Unable to lock handoff partition %d for revert "

View File

@ -155,6 +155,7 @@ class Sender(object):
self.remote_check_objs = remote_check_objs self.remote_check_objs = remote_check_objs
self.include_non_durable = include_non_durable self.include_non_durable = include_non_durable
self.max_objects = max_objects self.max_objects = max_objects
self.limited_by_max_objects = False
def __call__(self): def __call__(self):
""" """
@ -285,6 +286,7 @@ class Sender(object):
Full documentation of this can be found at Full documentation of this can be found at
:py:meth:`.Receiver.missing_check`. :py:meth:`.Receiver.missing_check`.
""" """
self.limited_by_max_objects = False
available_map = {} available_map = {}
send_map = {} send_map = {}
# First, send our list. # First, send our list.
@ -307,6 +309,7 @@ class Sender(object):
self.remote_check_objs, hash_gen) self.remote_check_objs, hash_gen)
nlines = 0 nlines = 0
nbytes = 0 nbytes = 0
object_hash = None
for object_hash, timestamps in hash_gen: for object_hash, timestamps in hash_gen:
available_map[object_hash] = timestamps available_map[object_hash] = timestamps
with exceptions.MessageTimeout( with exceptions.MessageTimeout(
@ -321,8 +324,10 @@ class Sender(object):
nlines += 1 nlines += 1
nbytes += len(msg) nbytes += len(msg)
if 0 < self.max_objects <= nlines: if 0 < self.max_objects <= nlines:
break
for _ in hash_gen: for _ in hash_gen:
# only log truncation if there were more hashes to come... # only log truncation if there were more hashes to come...
self.limited_by_max_objects = True
self.daemon.logger.info( self.daemon.logger.info(
'ssync missing_check truncated after %d objects: ' 'ssync missing_check truncated after %d objects: '
'device: %s, part: %s, policy: %s, last object hash: ' 'device: %s, part: %s, policy: %s, last object hash: '
@ -330,7 +335,6 @@ class Sender(object):
self.job['partition'], int(self.job['policy']), self.job['partition'], int(self.job['policy']),
object_hash) object_hash)
break break
break
with exceptions.MessageTimeout( with exceptions.MessageTimeout(
self.daemon.node_timeout, 'missing_check end'): self.daemon.node_timeout, 'missing_check end'):
msg = b':MISSING_CHECK: END\r\n' msg = b':MISSING_CHECK: END\r\n'

View File

@ -52,21 +52,33 @@ from test.unit import (patch_policies, mocked_http_conn, FabricatedRing,
from test.unit.obj.common import write_diskfile from test.unit.obj.common import write_diskfile
@contextmanager class FakeSsyncSender(object):
def mock_ssync_sender(ssync_calls=None, response_callback=None, **kwargs): def __init__(self, daemon, node, job, suffixes, ssync_calls=None,
def fake_ssync(daemon, node, job, suffixes, **kwargs): response_callback=None, **kwargs):
if ssync_calls is not None: if ssync_calls is not None:
call_args = {'node': node, 'job': job, 'suffixes': suffixes} call_args = {'node': node, 'job': job, 'suffixes': suffixes}
call_args.update(kwargs) call_args.update(kwargs)
ssync_calls.append(call_args) ssync_calls.append(call_args)
self.response_callback = response_callback
self.node = node
self.job = job
self.suffixes = suffixes
self.limited_by_max_objects = False
def fake_call(): def __call__(self):
if response_callback: if self.response_callback:
response = response_callback(node, job, suffixes) response = self.response_callback(
self.node, self.job, self.suffixes)
else: else:
response = True, {} response = True, {}
return response return response
return fake_call
@contextmanager
def mock_ssync_sender(ssync_calls=None, response_callback=None, **kwargs):
def fake_ssync(daemon, node, job, suffixes, **kwargs):
return FakeSsyncSender(daemon, node, job, suffixes, ssync_calls,
response_callback, **kwargs)
with mock.patch('swift.obj.reconstructor.ssync_sender', fake_ssync): with mock.patch('swift.obj.reconstructor.ssync_sender', fake_ssync):
yield fake_ssync yield fake_ssync
@ -1126,13 +1138,16 @@ class TestGlobalSetupObjectReconstructor(unittest.TestCase):
frag_index=self.job.get('frag_index'), frag_index=self.job.get('frag_index'),
frag_prefs=frag_prefs) frag_prefs=frag_prefs)
self.available_map = {} self.available_map = {}
self.limited_by_max_objects = False
nlines = 0 nlines = 0
for hash_, timestamps in hash_gen: for hash_, timestamps in hash_gen:
self.available_map[hash_] = timestamps self.available_map[hash_] = timestamps
nlines += 1 nlines += 1
if 0 < max_objects <= nlines: if 0 < max_objects <= nlines:
break break
for _ in hash_gen:
self.limited_by_max_objects = True
break
context['available_map'] = self.available_map context['available_map'] = self.available_map
ssync_calls.append(context) ssync_calls.append(context)
self.success = True self.success = True
@ -1459,7 +1474,7 @@ class TestGlobalSetupObjectReconstructor(unittest.TestCase):
dev['replication_port'] == dev['replication_port'] ==
self.reconstructor.port] self.reconstructor.port]
partition = (local_devs[0]['id'] + 1) % 3 partition = (local_devs[0]['id'] + 1) % 3
# 2 durable objects # three durable objects
df_0 = self._create_diskfile( df_0 = self._create_diskfile(
object_name='zero', part=partition) object_name='zero', part=partition)
datafile_0 = df_0.manager.cleanup_ondisk_files( datafile_0 = df_0.manager.cleanup_ondisk_files(
@ -1480,7 +1495,7 @@ class TestGlobalSetupObjectReconstructor(unittest.TestCase):
actual_datafiles = [df for df in datafiles if os.path.exists(df)] actual_datafiles = [df for df in datafiles if os.path.exists(df)]
self.assertEqual(datafiles, actual_datafiles) self.assertEqual(datafiles, actual_datafiles)
# only two object will be sync'd and purged... # only two objects will be sync'd and purged...
ssync_calls = [] ssync_calls = []
conf = dict(self.conf, max_objects_per_revert=2, handoffs_only=True) conf = dict(self.conf, max_objects_per_revert=2, handoffs_only=True)
self.reconstructor = object_reconstructor.ObjectReconstructor( self.reconstructor = object_reconstructor.ObjectReconstructor(
@ -1493,18 +1508,25 @@ class TestGlobalSetupObjectReconstructor(unittest.TestCase):
self.assertEqual(2, context.get('max_objects')) self.assertEqual(2, context.get('max_objects'))
actual_datafiles = [df for df in datafiles if os.path.exists(df)] actual_datafiles = [df for df in datafiles if os.path.exists(df)]
self.assertEqual(1, len(actual_datafiles), actual_datafiles) self.assertEqual(1, len(actual_datafiles), actual_datafiles)
# handoff still reported as remaining
self.assertEqual(1, self.reconstructor.handoffs_remaining)
# ...until next reconstructor run which will sync and purge the last # ...until next reconstructor run which will sync and purge the last
# object # object; max_objects_per_revert == actual number of objects
ssync_calls = [] ssync_calls = []
conf = dict(self.conf, max_objects_per_revert=1, handoffs_only=True)
self.reconstructor = object_reconstructor.ObjectReconstructor(
conf, logger=self.logger)
with mock.patch('swift.obj.reconstructor.ssync_sender', with mock.patch('swift.obj.reconstructor.ssync_sender',
self._make_fake_ssync(ssync_calls)): self._make_fake_ssync(ssync_calls)):
self.reconstructor.reconstruct() self.reconstructor.reconstruct()
for context in ssync_calls: for context in ssync_calls:
self.assertEqual(REVERT, context['job']['job_type']) self.assertEqual(REVERT, context['job']['job_type'])
self.assertEqual(2, context.get('max_objects')) self.assertEqual(1, context.get('max_objects'))
actual_datafiles = [df for df in datafiles if os.path.exists(df)] actual_datafiles = [df for df in datafiles if os.path.exists(df)]
self.assertEqual([], actual_datafiles) self.assertEqual([], actual_datafiles)
# handoff is no longer remaining
self.assertEqual(0, self.reconstructor.handoffs_remaining)
def test_no_delete_failed_revert(self): def test_no_delete_failed_revert(self):
# test will only process revert jobs # test will only process revert jobs

View File

@ -794,11 +794,13 @@ class TestSender(BaseTest):
connection = FakeConnection() connection = FakeConnection()
response = FakeResponse() response = FakeResponse()
self.sender.daemon.node_timeout = 0.01 self.sender.daemon.node_timeout = 0.01
self.assertFalse(self.sender.limited_by_max_objects)
with mock.patch.object(connection, 'send', with mock.patch.object(connection, 'send',
side_effect=lambda *args: eventlet.sleep(1)): side_effect=lambda *args: eventlet.sleep(1)):
with self.assertRaises(exceptions.MessageTimeout) as cm: with self.assertRaises(exceptions.MessageTimeout) as cm:
self.sender.missing_check(connection, response) self.sender.missing_check(connection, response)
self.assertIn('0.01 seconds: missing_check start', str(cm.exception)) self.assertIn('0.01 seconds: missing_check start', str(cm.exception))
self.assertFalse(self.sender.limited_by_max_objects)
def test_missing_check_timeout_send_line(self): def test_missing_check_timeout_send_line(self):
def yield_hashes(device, partition, policy, suffixes=None, **kwargs): def yield_hashes(device, partition, policy, suffixes=None, **kwargs):
@ -815,6 +817,7 @@ class TestSender(BaseTest):
max_objects=0) max_objects=0)
self.sender.daemon.node_timeout = 0.01 self.sender.daemon.node_timeout = 0.01
self.sender.df_mgr.yield_hashes = yield_hashes self.sender.df_mgr.yield_hashes = yield_hashes
self.assertFalse(self.sender.limited_by_max_objects)
sleeps = [0, 0, 1] sleeps = [0, 0, 1]
with mock.patch.object( with mock.patch.object(
connection, 'send', connection, 'send',
@ -823,6 +826,7 @@ class TestSender(BaseTest):
self.sender.missing_check(connection, response) self.sender.missing_check(connection, response)
self.assertIn('0.01 seconds: missing_check send line: ' self.assertIn('0.01 seconds: missing_check send line: '
'1 lines (57 bytes) sent', str(cm.exception)) '1 lines (57 bytes) sent', str(cm.exception))
self.assertFalse(self.sender.limited_by_max_objects)
def test_missing_check_has_empty_suffixes(self): def test_missing_check_has_empty_suffixes(self):
def yield_hashes(device, partition, policy, suffixes=None, **kwargs): def yield_hashes(device, partition, policy, suffixes=None, **kwargs):
@ -846,6 +850,7 @@ class TestSender(BaseTest):
':MISSING_CHECK: START\r\n' ':MISSING_CHECK: START\r\n'
':MISSING_CHECK: END\r\n')) ':MISSING_CHECK: END\r\n'))
self.sender.df_mgr.yield_hashes = yield_hashes self.sender.df_mgr.yield_hashes = yield_hashes
self.assertFalse(self.sender.limited_by_max_objects)
available_map, send_map = self.sender.missing_check(connection, available_map, send_map = self.sender.missing_check(connection,
response) response)
self.assertEqual( self.assertEqual(
@ -854,6 +859,7 @@ class TestSender(BaseTest):
b'15\r\n:MISSING_CHECK: END\r\n\r\n') b'15\r\n:MISSING_CHECK: END\r\n\r\n')
self.assertEqual(send_map, {}) self.assertEqual(send_map, {})
self.assertEqual(available_map, {}) self.assertEqual(available_map, {})
self.assertFalse(self.sender.limited_by_max_objects)
def test_missing_check_has_suffixes(self): def test_missing_check_has_suffixes(self):
def yield_hashes(device, partition, policy, suffixes=None, **kwargs): def yield_hashes(device, partition, policy, suffixes=None, **kwargs):
@ -893,6 +899,7 @@ class TestSender(BaseTest):
':MISSING_CHECK: START\r\n' ':MISSING_CHECK: START\r\n'
':MISSING_CHECK: END\r\n')) ':MISSING_CHECK: END\r\n'))
self.sender.df_mgr.yield_hashes = yield_hashes self.sender.df_mgr.yield_hashes = yield_hashes
self.assertFalse(self.sender.limited_by_max_objects)
available_map, send_map = self.sender.missing_check(connection, available_map, send_map = self.sender.missing_check(connection,
response) response)
self.assertEqual( self.assertEqual(
@ -916,6 +923,7 @@ class TestSender(BaseTest):
ts_ctype=Timestamp(1380144474.44448)))] ts_ctype=Timestamp(1380144474.44448)))]
self.assertEqual(available_map, dict(candidates)) self.assertEqual(available_map, dict(candidates))
self.assertEqual([], self.daemon_logger.get_lines_for_level('info')) self.assertEqual([], self.daemon_logger.get_lines_for_level('info'))
self.assertFalse(self.sender.limited_by_max_objects)
def test_missing_check_max_objects_less_than_actual_objects(self): def test_missing_check_max_objects_less_than_actual_objects(self):
def yield_hashes(device, partition, policy, suffixes=None, **kwargs): def yield_hashes(device, partition, policy, suffixes=None, **kwargs):
@ -957,6 +965,7 @@ class TestSender(BaseTest):
':MISSING_CHECK: START\r\n' ':MISSING_CHECK: START\r\n'
':MISSING_CHECK: END\r\n')) ':MISSING_CHECK: END\r\n'))
self.sender.df_mgr.yield_hashes = yield_hashes self.sender.df_mgr.yield_hashes = yield_hashes
self.assertFalse(self.sender.limited_by_max_objects)
available_map, send_map = self.sender.missing_check(connection, available_map, send_map = self.sender.missing_check(connection,
response) response)
self.assertEqual( self.assertEqual(
@ -978,6 +987,7 @@ class TestSender(BaseTest):
'part: 9, policy: 0, last object hash: ' 'part: 9, policy: 0, last object hash: '
'9d41d8cd98f00b204e9800998ecf0def'], '9d41d8cd98f00b204e9800998ecf0def'],
self.daemon_logger.get_lines_for_level('info')) self.daemon_logger.get_lines_for_level('info'))
self.assertTrue(self.sender.limited_by_max_objects)
def test_missing_check_max_objects_exactly_actual_objects(self): def test_missing_check_max_objects_exactly_actual_objects(self):
def yield_hashes(device, partition, policy, suffixes=None, **kwargs): def yield_hashes(device, partition, policy, suffixes=None, **kwargs):
@ -1012,6 +1022,7 @@ class TestSender(BaseTest):
':MISSING_CHECK: START\r\n' ':MISSING_CHECK: START\r\n'
':MISSING_CHECK: END\r\n')) ':MISSING_CHECK: END\r\n'))
self.sender.df_mgr.yield_hashes = yield_hashes self.sender.df_mgr.yield_hashes = yield_hashes
self.assertFalse(self.sender.limited_by_max_objects)
available_map, send_map = self.sender.missing_check(connection, available_map, send_map = self.sender.missing_check(connection,
response) response)
self.assertEqual( self.assertEqual(
@ -1030,6 +1041,7 @@ class TestSender(BaseTest):
self.assertEqual(available_map, dict(candidates)) self.assertEqual(available_map, dict(candidates))
# nothing logged re: truncation # nothing logged re: truncation
self.assertEqual([], self.daemon_logger.get_lines_for_level('info')) self.assertEqual([], self.daemon_logger.get_lines_for_level('info'))
self.assertFalse(self.sender.limited_by_max_objects)
def test_missing_check_far_end_disconnect(self): def test_missing_check_far_end_disconnect(self):
def yield_hashes(device, partition, policy, suffixes=None, **kwargs): def yield_hashes(device, partition, policy, suffixes=None, **kwargs):
@ -1052,6 +1064,7 @@ class TestSender(BaseTest):
} }
self.sender.suffixes = ['abc'] self.sender.suffixes = ['abc']
self.sender.df_mgr.yield_hashes = yield_hashes self.sender.df_mgr.yield_hashes = yield_hashes
self.assertFalse(self.sender.limited_by_max_objects)
response = FakeResponse(chunk_body='\r\n') response = FakeResponse(chunk_body='\r\n')
exc = None exc = None
try: try:
@ -1064,6 +1077,7 @@ class TestSender(BaseTest):
b'17\r\n:MISSING_CHECK: START\r\n\r\n' b'17\r\n:MISSING_CHECK: START\r\n\r\n'
b'33\r\n9d41d8cd98f00b204e9800998ecf0abc 1380144470.00000\r\n\r\n' b'33\r\n9d41d8cd98f00b204e9800998ecf0abc 1380144470.00000\r\n\r\n'
b'15\r\n:MISSING_CHECK: END\r\n\r\n') b'15\r\n:MISSING_CHECK: END\r\n\r\n')
self.assertFalse(self.sender.limited_by_max_objects)
def test_missing_check_far_end_disconnect2(self): def test_missing_check_far_end_disconnect2(self):
def yield_hashes(device, partition, policy, suffixes=None, **kwargs): def yield_hashes(device, partition, policy, suffixes=None, **kwargs):
@ -1086,6 +1100,7 @@ class TestSender(BaseTest):
} }
self.sender.suffixes = ['abc'] self.sender.suffixes = ['abc']
self.sender.df_mgr.yield_hashes = yield_hashes self.sender.df_mgr.yield_hashes = yield_hashes
self.assertFalse(self.sender.limited_by_max_objects)
response = FakeResponse( response = FakeResponse(
chunk_body=':MISSING_CHECK: START\r\n') chunk_body=':MISSING_CHECK: START\r\n')
exc = None exc = None
@ -1099,6 +1114,7 @@ class TestSender(BaseTest):
b'17\r\n:MISSING_CHECK: START\r\n\r\n' b'17\r\n:MISSING_CHECK: START\r\n\r\n'
b'33\r\n9d41d8cd98f00b204e9800998ecf0abc 1380144470.00000\r\n\r\n' b'33\r\n9d41d8cd98f00b204e9800998ecf0abc 1380144470.00000\r\n\r\n'
b'15\r\n:MISSING_CHECK: END\r\n\r\n') b'15\r\n:MISSING_CHECK: END\r\n\r\n')
self.assertFalse(self.sender.limited_by_max_objects)
def test_missing_check_far_end_unexpected(self): def test_missing_check_far_end_unexpected(self):
def yield_hashes(device, partition, policy, suffixes=None, **kwargs): def yield_hashes(device, partition, policy, suffixes=None, **kwargs):
@ -1121,6 +1137,7 @@ class TestSender(BaseTest):
} }
self.sender.suffixes = ['abc'] self.sender.suffixes = ['abc']
self.sender.df_mgr.yield_hashes = yield_hashes self.sender.df_mgr.yield_hashes = yield_hashes
self.assertFalse(self.sender.limited_by_max_objects)
response = FakeResponse(chunk_body='OH HAI\r\n') response = FakeResponse(chunk_body='OH HAI\r\n')
exc = None exc = None
try: try:
@ -1133,6 +1150,7 @@ class TestSender(BaseTest):
b'17\r\n:MISSING_CHECK: START\r\n\r\n' b'17\r\n:MISSING_CHECK: START\r\n\r\n'
b'33\r\n9d41d8cd98f00b204e9800998ecf0abc 1380144470.00000\r\n\r\n' b'33\r\n9d41d8cd98f00b204e9800998ecf0abc 1380144470.00000\r\n\r\n'
b'15\r\n:MISSING_CHECK: END\r\n\r\n') b'15\r\n:MISSING_CHECK: END\r\n\r\n')
self.assertFalse(self.sender.limited_by_max_objects)
def test_missing_check_send_map(self): def test_missing_check_send_map(self):
def yield_hashes(device, partition, policy, suffixes=None, **kwargs): def yield_hashes(device, partition, policy, suffixes=None, **kwargs):
@ -1160,6 +1178,7 @@ class TestSender(BaseTest):
'0123abc dm\r\n' '0123abc dm\r\n'
':MISSING_CHECK: END\r\n')) ':MISSING_CHECK: END\r\n'))
self.sender.df_mgr.yield_hashes = yield_hashes self.sender.df_mgr.yield_hashes = yield_hashes
self.assertFalse(self.sender.limited_by_max_objects)
available_map, send_map = self.sender.missing_check(connection, available_map, send_map = self.sender.missing_check(connection,
response) response)
self.assertEqual( self.assertEqual(
@ -1171,6 +1190,7 @@ class TestSender(BaseTest):
self.assertEqual(available_map, self.assertEqual(available_map,
dict([('9d41d8cd98f00b204e9800998ecf0abc', dict([('9d41d8cd98f00b204e9800998ecf0abc',
{'ts_data': Timestamp(1380144470.00000)})])) {'ts_data': Timestamp(1380144470.00000)})]))
self.assertFalse(self.sender.limited_by_max_objects)
def test_missing_check_extra_line_parts(self): def test_missing_check_extra_line_parts(self):
# check that sender tolerates extra parts in missing check # check that sender tolerates extra parts in missing check
@ -1200,12 +1220,14 @@ class TestSender(BaseTest):
'0123abc d extra response parts\r\n' '0123abc d extra response parts\r\n'
':MISSING_CHECK: END\r\n')) ':MISSING_CHECK: END\r\n'))
self.sender.df_mgr.yield_hashes = yield_hashes self.sender.df_mgr.yield_hashes = yield_hashes
self.assertFalse(self.sender.limited_by_max_objects)
available_map, send_map = self.sender.missing_check(connection, available_map, send_map = self.sender.missing_check(connection,
response) response)
self.assertEqual(send_map, {'0123abc': {'data': True}}) self.assertEqual(send_map, {'0123abc': {'data': True}})
self.assertEqual(available_map, self.assertEqual(available_map,
dict([('9d41d8cd98f00b204e9800998ecf0abc', dict([('9d41d8cd98f00b204e9800998ecf0abc',
{'ts_data': Timestamp(1380144470.00000)})])) {'ts_data': Timestamp(1380144470.00000)})]))
self.assertFalse(self.sender.limited_by_max_objects)
def test_updates_timeout(self): def test_updates_timeout(self):
connection = FakeConnection() connection = FakeConnection()