Skip already checked partitions when auditing objects after a restart
The object auditor will save a short status file on each device, containing a list of remaining partitions for auditing. If the auditor is restarted, it will only audit partitions not yet checked. If all partitions on the current device have been checked, it will simply skip this device. Once all partitions on all disks are successfully audited, all status files are removed. Closes-Bug: #1183656 Change-Id: Icf1d920d0942ce48f1d3d374ea4d63dbc29ea464
This commit is contained in:
parent
6ef66378c9
commit
fd86d5a95d
@ -95,7 +95,9 @@ class AuditorWorker(object):
|
|||||||
# can find all diskfile locations regardless of policy -- so for now
|
# can find all diskfile locations regardless of policy -- so for now
|
||||||
# just use Policy-0's manager.
|
# just use Policy-0's manager.
|
||||||
all_locs = (self.diskfile_router[POLICIES[0]]
|
all_locs = (self.diskfile_router[POLICIES[0]]
|
||||||
.object_audit_location_generator(device_dirs=device_dirs))
|
.object_audit_location_generator(
|
||||||
|
device_dirs=device_dirs,
|
||||||
|
auditor_type=self.auditor_type))
|
||||||
for location in all_locs:
|
for location in all_locs:
|
||||||
loop_time = time.time()
|
loop_time = time.time()
|
||||||
self.failsafe_object_audit(location)
|
self.failsafe_object_audit(location)
|
||||||
@ -156,6 +158,9 @@ class AuditorWorker(object):
|
|||||||
self.logger.info(
|
self.logger.info(
|
||||||
_('Object audit stats: %s') % json.dumps(self.stats_buckets))
|
_('Object audit stats: %s') % json.dumps(self.stats_buckets))
|
||||||
|
|
||||||
|
# Unset remaining partitions to not skip them in the next run
|
||||||
|
diskfile.clear_auditor_status(self.devices, self.auditor_type)
|
||||||
|
|
||||||
def record_stats(self, obj_size):
|
def record_stats(self, obj_size):
|
||||||
"""
|
"""
|
||||||
Based on config's object_size_stats will keep track of how many objects
|
Based on config's object_size_stats will keep track of how many objects
|
||||||
|
@ -33,6 +33,7 @@ are also not considered part of the backend API.
|
|||||||
import six.moves.cPickle as pickle
|
import six.moves.cPickle as pickle
|
||||||
import errno
|
import errno
|
||||||
import fcntl
|
import fcntl
|
||||||
|
import json
|
||||||
import os
|
import os
|
||||||
import time
|
import time
|
||||||
import uuid
|
import uuid
|
||||||
@ -263,7 +264,7 @@ class AuditLocation(object):
|
|||||||
|
|
||||||
|
|
||||||
def object_audit_location_generator(devices, mount_check=True, logger=None,
|
def object_audit_location_generator(devices, mount_check=True, logger=None,
|
||||||
device_dirs=None):
|
device_dirs=None, auditor_type="ALL"):
|
||||||
"""
|
"""
|
||||||
Given a devices path (e.g. "/srv/node"), yield an AuditLocation for all
|
Given a devices path (e.g. "/srv/node"), yield an AuditLocation for all
|
||||||
objects stored under that directory if device_dirs isn't set. If
|
objects stored under that directory if device_dirs isn't set. If
|
||||||
@ -277,7 +278,8 @@ def object_audit_location_generator(devices, mount_check=True, logger=None,
|
|||||||
:param mount_check: flag to check if a mount check should be performed
|
:param mount_check: flag to check if a mount check should be performed
|
||||||
on devices
|
on devices
|
||||||
:param logger: a logger object
|
:param logger: a logger object
|
||||||
:device_dirs: a list of directories under devices to traverse
|
:param device_dirs: a list of directories under devices to traverse
|
||||||
|
:param auditor_type: either ALL or ZBF
|
||||||
"""
|
"""
|
||||||
if not device_dirs:
|
if not device_dirs:
|
||||||
device_dirs = listdir(devices)
|
device_dirs = listdir(devices)
|
||||||
@ -307,8 +309,12 @@ def object_audit_location_generator(devices, mount_check=True, logger=None,
|
|||||||
'to a valid policy (%s)') % (dir_, e))
|
'to a valid policy (%s)') % (dir_, e))
|
||||||
continue
|
continue
|
||||||
datadir_path = os.path.join(devices, device, dir_)
|
datadir_path = os.path.join(devices, device, dir_)
|
||||||
partitions = listdir(datadir_path)
|
|
||||||
for partition in partitions:
|
partitions = get_auditor_status(datadir_path, logger, auditor_type)
|
||||||
|
|
||||||
|
for pos, partition in enumerate(partitions):
|
||||||
|
update_auditor_status(datadir_path, logger,
|
||||||
|
partitions[pos:], auditor_type)
|
||||||
part_path = os.path.join(datadir_path, partition)
|
part_path = os.path.join(datadir_path, partition)
|
||||||
try:
|
try:
|
||||||
suffixes = listdir(part_path)
|
suffixes = listdir(part_path)
|
||||||
@ -329,6 +335,51 @@ def object_audit_location_generator(devices, mount_check=True, logger=None,
|
|||||||
yield AuditLocation(hsh_path, device, partition,
|
yield AuditLocation(hsh_path, device, partition,
|
||||||
policy)
|
policy)
|
||||||
|
|
||||||
|
update_auditor_status(datadir_path, logger, [], auditor_type)
|
||||||
|
|
||||||
|
|
||||||
|
def get_auditor_status(datadir_path, logger, auditor_type):
|
||||||
|
auditor_status = os.path.join(
|
||||||
|
datadir_path, "auditor_status_%s.json" % auditor_type)
|
||||||
|
status = {}
|
||||||
|
try:
|
||||||
|
with open(auditor_status) as statusfile:
|
||||||
|
status = statusfile.read()
|
||||||
|
except (OSError, IOError) as e:
|
||||||
|
if e.errno != errno.ENOENT and logger:
|
||||||
|
logger.warning(_('Cannot read %s (%s)') % (auditor_status, e))
|
||||||
|
return listdir(datadir_path)
|
||||||
|
try:
|
||||||
|
status = json.loads(status)
|
||||||
|
except ValueError as e:
|
||||||
|
logger.warning(_('Loading JSON from %s failed (%s)') % (
|
||||||
|
auditor_status, e))
|
||||||
|
return listdir(datadir_path)
|
||||||
|
return status['partitions']
|
||||||
|
|
||||||
|
|
||||||
|
def update_auditor_status(datadir_path, logger, partitions, auditor_type):
|
||||||
|
status = json.dumps({'partitions': partitions})
|
||||||
|
auditor_status = os.path.join(
|
||||||
|
datadir_path, "auditor_status_%s.json" % auditor_type)
|
||||||
|
try:
|
||||||
|
with open(auditor_status, "wb") as statusfile:
|
||||||
|
statusfile.write(status)
|
||||||
|
except (OSError, IOError) as e:
|
||||||
|
if logger:
|
||||||
|
logger.warning(_('Cannot write %s (%s)') % (auditor_status, e))
|
||||||
|
|
||||||
|
|
||||||
|
def clear_auditor_status(devices, auditor_type="ALL"):
|
||||||
|
for device in os.listdir(devices):
|
||||||
|
for dir_ in os.listdir(os.path.join(devices, device)):
|
||||||
|
if not dir_.startswith("objects"):
|
||||||
|
continue
|
||||||
|
datadir_path = os.path.join(devices, device, dir_)
|
||||||
|
auditor_status = os.path.join(
|
||||||
|
datadir_path, "auditor_status_%s.json" % auditor_type)
|
||||||
|
remove_file(auditor_status)
|
||||||
|
|
||||||
|
|
||||||
def strip_self(f):
|
def strip_self(f):
|
||||||
"""
|
"""
|
||||||
@ -897,14 +948,17 @@ class BaseDiskFileManager(object):
|
|||||||
policy=policy, use_splice=self.use_splice,
|
policy=policy, use_splice=self.use_splice,
|
||||||
pipe_size=self.pipe_size, **kwargs)
|
pipe_size=self.pipe_size, **kwargs)
|
||||||
|
|
||||||
def object_audit_location_generator(self, device_dirs=None):
|
def object_audit_location_generator(self, device_dirs=None,
|
||||||
|
auditor_type="ALL"):
|
||||||
"""
|
"""
|
||||||
Yield an AuditLocation for all objects stored under device_dirs.
|
Yield an AuditLocation for all objects stored under device_dirs.
|
||||||
|
|
||||||
:param device_dirs: directory of target device
|
:param device_dirs: directory of target device
|
||||||
|
:param auditor_type: either ALL or ZBF
|
||||||
"""
|
"""
|
||||||
return object_audit_location_generator(self.devices, self.mount_check,
|
return object_audit_location_generator(self.devices, self.mount_check,
|
||||||
self.logger, device_dirs)
|
self.logger, device_dirs,
|
||||||
|
auditor_type)
|
||||||
|
|
||||||
def get_diskfile_from_audit_location(self, audit_location):
|
def get_diskfile_from_audit_location(self, audit_location):
|
||||||
"""
|
"""
|
||||||
|
@ -26,7 +26,8 @@ from test.unit import FakeLogger, patch_policies, make_timestamp_iter, \
|
|||||||
DEFAULT_TEST_EC_TYPE
|
DEFAULT_TEST_EC_TYPE
|
||||||
from swift.obj import auditor
|
from swift.obj import auditor
|
||||||
from swift.obj.diskfile import DiskFile, write_metadata, invalidate_hash, \
|
from swift.obj.diskfile import DiskFile, write_metadata, invalidate_hash, \
|
||||||
get_data_dir, DiskFileManager, ECDiskFileManager, AuditLocation
|
get_data_dir, DiskFileManager, ECDiskFileManager, AuditLocation, \
|
||||||
|
clear_auditor_status, get_auditor_status
|
||||||
from swift.common.utils import mkdirs, normalize_timestamp, Timestamp
|
from swift.common.utils import mkdirs, normalize_timestamp, Timestamp
|
||||||
from swift.common.storage_policy import ECStoragePolicy, StoragePolicy, \
|
from swift.common.storage_policy import ECStoragePolicy, StoragePolicy, \
|
||||||
POLICIES
|
POLICIES
|
||||||
@ -460,6 +461,7 @@ class TestAuditor(unittest.TestCase):
|
|||||||
self.auditor.run_audit(**kwargs)
|
self.auditor.run_audit(**kwargs)
|
||||||
self.assertFalse(os.path.isdir(quarantine_path))
|
self.assertFalse(os.path.isdir(quarantine_path))
|
||||||
del(kwargs['zero_byte_fps'])
|
del(kwargs['zero_byte_fps'])
|
||||||
|
clear_auditor_status(self.devices)
|
||||||
self.auditor.run_audit(**kwargs)
|
self.auditor.run_audit(**kwargs)
|
||||||
self.assertTrue(os.path.isdir(quarantine_path))
|
self.assertTrue(os.path.isdir(quarantine_path))
|
||||||
|
|
||||||
@ -495,10 +497,20 @@ class TestAuditor(unittest.TestCase):
|
|||||||
self.setup_bad_zero_byte()
|
self.setup_bad_zero_byte()
|
||||||
kwargs = {'mode': 'once'}
|
kwargs = {'mode': 'once'}
|
||||||
kwargs['zero_byte_fps'] = 50
|
kwargs['zero_byte_fps'] = 50
|
||||||
self.auditor.run_audit(**kwargs)
|
|
||||||
|
called_args = [0]
|
||||||
|
|
||||||
|
def mock_get_auditor_status(path, logger, audit_type):
|
||||||
|
called_args[0] = audit_type
|
||||||
|
return get_auditor_status(path, logger, audit_type)
|
||||||
|
|
||||||
|
with mock.patch('swift.obj.diskfile.get_auditor_status',
|
||||||
|
mock_get_auditor_status):
|
||||||
|
self.auditor.run_audit(**kwargs)
|
||||||
quarantine_path = os.path.join(self.devices,
|
quarantine_path = os.path.join(self.devices,
|
||||||
'sda', 'quarantined', 'objects')
|
'sda', 'quarantined', 'objects')
|
||||||
self.assertTrue(os.path.isdir(quarantine_path))
|
self.assertTrue(os.path.isdir(quarantine_path))
|
||||||
|
self.assertEqual('ZBF', called_args[0])
|
||||||
|
|
||||||
def test_object_run_fast_track_zero_check_closed(self):
|
def test_object_run_fast_track_zero_check_closed(self):
|
||||||
rat = [False]
|
rat = [False]
|
||||||
|
@ -359,6 +359,9 @@ class TestObjectAuditLocationGenerator(unittest.TestCase):
|
|||||||
]
|
]
|
||||||
self.assertEqual(locations, expected)
|
self.assertEqual(locations, expected)
|
||||||
|
|
||||||
|
# Reset status file for next run
|
||||||
|
diskfile.clear_auditor_status(tmpdir)
|
||||||
|
|
||||||
# now without a logger
|
# now without a logger
|
||||||
locations = [(loc.path, loc.device, loc.partition, loc.policy)
|
locations = [(loc.path, loc.device, loc.partition, loc.policy)
|
||||||
for loc in diskfile.object_audit_location_generator(
|
for loc in diskfile.object_audit_location_generator(
|
||||||
@ -433,6 +436,40 @@ class TestObjectAuditLocationGenerator(unittest.TestCase):
|
|||||||
with mock.patch('os.listdir', splode_if_endswith("b54")):
|
with mock.patch('os.listdir', splode_if_endswith("b54")):
|
||||||
self.assertRaises(OSError, list_locations, tmpdir)
|
self.assertRaises(OSError, list_locations, tmpdir)
|
||||||
|
|
||||||
|
def test_auditor_status(self):
|
||||||
|
with temptree([]) as tmpdir:
|
||||||
|
os.makedirs(os.path.join(tmpdir, "sdf", "objects", "1", "a", "b"))
|
||||||
|
os.makedirs(os.path.join(tmpdir, "sdf", "objects", "2", "a", "b"))
|
||||||
|
|
||||||
|
# Auditor starts, there are two partitions to check
|
||||||
|
gen = diskfile.object_audit_location_generator(tmpdir, False)
|
||||||
|
gen.next()
|
||||||
|
gen.next()
|
||||||
|
|
||||||
|
# Auditor stopped for some reason without raising StopIterator in
|
||||||
|
# the generator and restarts There is now only one remaining
|
||||||
|
# partition to check
|
||||||
|
gen = diskfile.object_audit_location_generator(tmpdir, False)
|
||||||
|
gen.next()
|
||||||
|
|
||||||
|
# There are no more remaining partitions
|
||||||
|
self.assertRaises(StopIteration, gen.next)
|
||||||
|
|
||||||
|
# There are no partitions to check if the auditor restarts another
|
||||||
|
# time and the status files have not been cleared
|
||||||
|
gen = diskfile.object_audit_location_generator(tmpdir, False)
|
||||||
|
self.assertRaises(StopIteration, gen.next)
|
||||||
|
|
||||||
|
# Reset status file
|
||||||
|
diskfile.clear_auditor_status(tmpdir)
|
||||||
|
|
||||||
|
# If the auditor restarts another time, we expect to
|
||||||
|
# check two partitions again, because the remaining
|
||||||
|
# partitions were empty and a new listdir was executed
|
||||||
|
gen = diskfile.object_audit_location_generator(tmpdir, False)
|
||||||
|
gen.next()
|
||||||
|
gen.next()
|
||||||
|
|
||||||
|
|
||||||
class TestDiskFileRouter(unittest.TestCase):
|
class TestDiskFileRouter(unittest.TestCase):
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user