Replace slowdown option with *_per_second option

container and object updaters sleeps "slowdown" (default 0.01) seconds
after every processed container/object. Because time.sleep call adds overhead,
use ratelimit_sleep from common.utils instead. Same as in auditor.

Change-Id: I362aa0f13c78ad03ce1f76ee0257b0646f981212
This commit is contained in:
Ondřej Nový 2017-03-21 20:10:12 +01:00 committed by Tim Burke
parent 55075ea8be
commit a8bc94c7e3
9 changed files with 152 additions and 15 deletions

View File

@ -358,8 +358,10 @@ Number of reaper workers to spawn. The default is 4.
Request timeout to external services. The default is 3 seconds. Request timeout to external services. The default is 3 seconds.
.IP \fBconn_timeout\fR .IP \fBconn_timeout\fR
Connection timeout to external services. The default is 0.5 seconds. Connection timeout to external services. The default is 0.5 seconds.
.IP \fBcontainers_per_second\fR
Maximum containers updated per second. Should be tuned according to individual system specs. 0 is unlimited. The default is 50.
.IP \fBslowdown\fR .IP \fBslowdown\fR
Slowdown will sleep that amount between containers. The default is 0.01 seconds. Slowdown will sleep that amount between containers. The default is 0.01 seconds. Deprecated in favor of containers_per_second
.IP \fBaccount_suppression_time\fR .IP \fBaccount_suppression_time\fR
Seconds to suppress updating an account that has generated an error. The default is 60 seconds. Seconds to suppress updating an account that has generated an error. The default is 60 seconds.
.IP \fBrecon_cache_path\fR .IP \fBrecon_cache_path\fR

View File

@ -500,8 +500,10 @@ Minimum time for a pass to take. The default is 300 seconds.
Number of reaper workers to spawn. The default is 1. Number of reaper workers to spawn. The default is 1.
.IP \fBnode_timeout\fR .IP \fBnode_timeout\fR
Request timeout to external services. The default is 10 seconds. Request timeout to external services. The default is 10 seconds.
.IP \fBobjects_per_second\fR
Maximum objects updated per second. Should be tuned according to individual system specs. 0 is unlimited. The default is 50.
.IP \fBslowdown\fR .IP \fBslowdown\fR
Slowdown will sleep that amount between objects. The default is 0.01 seconds. Slowdown will sleep that amount between objects. The default is 0.01 seconds. Deprecated in favor of objects_per_second.
.IP "\fBrecon_cache_path\fR" .IP "\fBrecon_cache_path\fR"
The recon_cache_path simply sets the directory where stats for a few items will be stored. The recon_cache_path simply sets the directory where stats for a few items will be stored.
Depending on the method of deployment you may need to create this directory manually Depending on the method of deployment you may need to create this directory manually

View File

@ -772,9 +772,9 @@ ionice_priority None I/O scheduling priority o
[object-updater] [object-updater]
================== =================== ========================================== =================== =================== ==========================================
Option Default Description Option Default Description
------------------ ------------------- ------------------------------------------ ------------------- ------------------- ------------------------------------------
log_name object-updater Label used when logging log_name object-updater Label used when logging
log_facility LOG_LOCAL0 Syslog log facility log_facility LOG_LOCAL0 Syslog log facility
log_level INFO Logging level log_level INFO Logging level
@ -785,7 +785,11 @@ node_timeout DEFAULT or 10 Request timeout to external services. Th
uses what's set here, or what's set in the uses what's set here, or what's set in the
DEFAULT section, or 10 (though other DEFAULT section, or 10 (though other
sections use 3 as the final default). sections use 3 as the final default).
slowdown 0.01 Time in seconds to wait between objects objects_per_second 50 Maximum objects updated per second.
Should be tuned according to individual
system specs. 0 is unlimited.
slowdown 0.01 Time in seconds to wait between objects.
Deprecated in favor of objects_per_second.
recon_cache_path /var/cache/swift Path to recon cache recon_cache_path /var/cache/swift Path to recon cache
nice_priority None Scheduling priority of server processes. nice_priority None Scheduling priority of server processes.
Niceness values range from -20 (most Niceness values range from -20 (most
@ -808,7 +812,7 @@ ionice_priority None I/O scheduling priority of server
priority of the process. Work only with priority of the process. Work only with
ionice_class. ionice_class.
Ignored if IOPRIO_CLASS_IDLE is set. Ignored if IOPRIO_CLASS_IDLE is set.
================== =================== ========================================== =================== =================== ==========================================
[object-auditor] [object-auditor]
@ -1122,8 +1126,13 @@ node_timeout 3 Request timeout to external
services services
conn_timeout 0.5 Connection timeout to external conn_timeout 0.5 Connection timeout to external
services services
containers_per_second 50 Maximum containers updated per second.
Should be tuned according to individual
system specs. 0 is unlimited.
slowdown 0.01 Time in seconds to wait between slowdown 0.01 Time in seconds to wait between
containers containers. Deprecated in favor of
containers_per_second.
account_suppression_time 60 Seconds to suppress updating an account_suppression_time 60 Seconds to suppress updating an
account that has generated an account that has generated an
error (timeout, not yet found, error (timeout, not yet found,

View File

@ -185,7 +185,11 @@ use = egg:swift#recon
# node_timeout = 3 # node_timeout = 3
# conn_timeout = 0.5 # conn_timeout = 0.5
# #
# slowdown will sleep that amount between containers # Send at most this many container updates per second
# containers_per_second = 50
#
# slowdown will sleep that amount between containers. Deprecated; use
# containers_per_second instead.
# slowdown = 0.01 # slowdown = 0.01
# #
# Seconds to suppress updating an account that has generated an error # Seconds to suppress updating an account that has generated an error

View File

@ -333,7 +333,12 @@ use = egg:swift#recon
# interval = 300 # interval = 300
# concurrency = 1 # concurrency = 1
# node_timeout = <whatever's in the DEFAULT section or 10> # node_timeout = <whatever's in the DEFAULT section or 10>
# slowdown will sleep that amount between objects #
# Send at most this many object updates per second
# objects_per_second = 50
#
# slowdown will sleep that amount between objects. Deprecated; use
# objects_per_second instead.
# slowdown = 0.01 # slowdown = 0.01
# #
# recon_cache_path = /var/cache/swift # recon_cache_path = /var/cache/swift

View File

@ -31,7 +31,7 @@ from swift.common.bufferedhttp import http_connect
from swift.common.exceptions import ConnectionTimeout from swift.common.exceptions import ConnectionTimeout
from swift.common.ring import Ring from swift.common.ring import Ring
from swift.common.utils import get_logger, config_true_value, ismount, \ from swift.common.utils import get_logger, config_true_value, ismount, \
dump_recon_cache, majority_size, Timestamp dump_recon_cache, majority_size, Timestamp, ratelimit_sleep
from swift.common.daemon import Daemon from swift.common.daemon import Daemon
from swift.common.http import is_success, HTTP_INTERNAL_SERVER_ERROR from swift.common.http import is_success, HTTP_INTERNAL_SERVER_ERROR
@ -48,7 +48,19 @@ class ContainerUpdater(Daemon):
self.interval = int(conf.get('interval', 300)) self.interval = int(conf.get('interval', 300))
self.account_ring = None self.account_ring = None
self.concurrency = int(conf.get('concurrency', 4)) self.concurrency = int(conf.get('concurrency', 4))
self.slowdown = float(conf.get('slowdown', 0.01)) if 'slowdown' in conf:
self.logger.warning(
'The slowdown option is deprecated in favor of '
'containers_per_second. This option may be ignored in a '
'future release.')
containers_per_second = 1 / (
float(conf.get('slowdown', '0.01')) + 0.01)
else:
containers_per_second = 50
self.containers_running_time = 0
self.max_containers_per_second = \
float(conf.get('containers_per_second',
containers_per_second))
self.node_timeout = float(conf.get('node_timeout', 3)) self.node_timeout = float(conf.get('node_timeout', 3))
self.conn_timeout = float(conf.get('conn_timeout', 0.5)) self.conn_timeout = float(conf.get('conn_timeout', 0.5))
self.no_changes = 0 self.no_changes = 0
@ -206,7 +218,10 @@ class ContainerUpdater(Daemon):
for file in files: for file in files:
if file.endswith('.db'): if file.endswith('.db'):
self.process_container(os.path.join(root, file)) self.process_container(os.path.join(root, file))
time.sleep(self.slowdown)
self.containers_running_time = ratelimit_sleep(
self.containers_running_time,
self.max_containers_per_second)
def process_container(self, dbfile): def process_container(self, dbfile):
""" """

View File

@ -27,7 +27,7 @@ from swift.common.bufferedhttp import http_connect
from swift.common.exceptions import ConnectionTimeout from swift.common.exceptions import ConnectionTimeout
from swift.common.ring import Ring from swift.common.ring import Ring
from swift.common.utils import get_logger, renamer, write_pickle, \ from swift.common.utils import get_logger, renamer, write_pickle, \
dump_recon_cache, config_true_value, ismount dump_recon_cache, config_true_value, ismount, ratelimit_sleep
from swift.common.daemon import Daemon from swift.common.daemon import Daemon
from swift.common.header_key_dict import HeaderKeyDict from swift.common.header_key_dict import HeaderKeyDict
from swift.common.storage_policy import split_policy_string, PolicyError from swift.common.storage_policy import split_policy_string, PolicyError
@ -47,7 +47,19 @@ class ObjectUpdater(Daemon):
self.interval = int(conf.get('interval', 300)) self.interval = int(conf.get('interval', 300))
self.container_ring = None self.container_ring = None
self.concurrency = int(conf.get('concurrency', 1)) self.concurrency = int(conf.get('concurrency', 1))
self.slowdown = float(conf.get('slowdown', 0.01)) if 'slowdown' in conf:
self.logger.warning(
'The slowdown option is deprecated in favor of '
'objects_per_second. This option may be ignored in a '
'future release.')
objects_per_second = 1 / (
float(conf.get('slowdown', '0.01')) + 0.01)
else:
objects_per_second = 50
self.objects_running_time = 0
self.max_objects_per_second = \
float(conf.get('objects_per_second',
objects_per_second))
self.node_timeout = float(conf.get('node_timeout', 10)) self.node_timeout = float(conf.get('node_timeout', 10))
self.conn_timeout = float(conf.get('conn_timeout', 0.5)) self.conn_timeout = float(conf.get('conn_timeout', 0.5))
self.successes = 0 self.successes = 0
@ -189,7 +201,10 @@ class ObjectUpdater(Daemon):
self.process_object_update(update_path, device, self.process_object_update(update_path, device,
policy) policy)
last_obj_hash = obj_hash last_obj_hash = obj_hash
time.sleep(self.slowdown)
self.objects_running_time = ratelimit_sleep(
self.objects_running_time,
self.max_objects_per_second)
try: try:
os.rmdir(prefix_path) os.rmdir(prefix_path)
except OSError: except OSError:

View File

@ -83,6 +83,48 @@ class TestContainerUpdater(unittest.TestCase):
self.assertEqual(cu.account_suppression_time, 0) self.assertEqual(cu.account_suppression_time, 0)
self.assertTrue(cu.get_account_ring() is not None) self.assertTrue(cu.get_account_ring() is not None)
def test_conf_params(self):
# defaults
daemon = container_updater.ContainerUpdater({})
self.assertEqual(daemon.devices, '/srv/node')
self.assertEqual(daemon.mount_check, True)
self.assertEqual(daemon.swift_dir, '/etc/swift')
self.assertEqual(daemon.interval, 300)
self.assertEqual(daemon.concurrency, 4)
self.assertEqual(daemon.max_containers_per_second, 50.0)
# non-defaults
conf = {
'devices': '/some/where/else',
'mount_check': 'huh?',
'swift_dir': '/not/here',
'interval': '600',
'concurrency': '2',
'containers_per_second': '10.5',
}
daemon = container_updater.ContainerUpdater(conf)
self.assertEqual(daemon.devices, '/some/where/else')
self.assertEqual(daemon.mount_check, False)
self.assertEqual(daemon.swift_dir, '/not/here')
self.assertEqual(daemon.interval, 600)
self.assertEqual(daemon.concurrency, 2)
self.assertEqual(daemon.max_containers_per_second, 10.5)
# check deprecated option
daemon = container_updater.ContainerUpdater({'slowdown': '0.04'})
self.assertEqual(daemon.max_containers_per_second, 20.0)
def check_bad(conf):
with self.assertRaises(ValueError):
container_updater.ContainerUpdater(conf)
check_bad({'interval': 'foo'})
check_bad({'interval': '300.0'})
check_bad({'concurrency': 'bar'})
check_bad({'concurrency': '1.0'})
check_bad({'slowdown': 'baz'})
check_bad({'containers_per_second': 'quux'})
@mock.patch.object(container_updater, 'ismount') @mock.patch.object(container_updater, 'ismount')
@mock.patch.object(container_updater.ContainerUpdater, 'container_sweep') @mock.patch.object(container_updater.ContainerUpdater, 'container_sweep')
def test_run_once_with_device_unmounted(self, mock_sweep, mock_ismount): def test_run_once_with_device_unmounted(self, mock_sweep, mock_ismount):

View File

@ -92,6 +92,49 @@ class TestObjectUpdater(unittest.TestCase):
self.assertEqual(ou.node_timeout, 5.5) self.assertEqual(ou.node_timeout, 5.5)
self.assertTrue(ou.get_container_ring() is not None) self.assertTrue(ou.get_container_ring() is not None)
def test_conf_params(self):
# defaults
daemon = object_updater.ObjectUpdater({}, logger=self.logger)
self.assertEqual(daemon.devices, '/srv/node')
self.assertEqual(daemon.mount_check, True)
self.assertEqual(daemon.swift_dir, '/etc/swift')
self.assertEqual(daemon.interval, 300)
self.assertEqual(daemon.concurrency, 1)
self.assertEqual(daemon.max_objects_per_second, 50.0)
# non-defaults
conf = {
'devices': '/some/where/else',
'mount_check': 'huh?',
'swift_dir': '/not/here',
'interval': '600',
'concurrency': '2',
'objects_per_second': '10.5',
}
daemon = object_updater.ObjectUpdater(conf, logger=self.logger)
self.assertEqual(daemon.devices, '/some/where/else')
self.assertEqual(daemon.mount_check, False)
self.assertEqual(daemon.swift_dir, '/not/here')
self.assertEqual(daemon.interval, 600)
self.assertEqual(daemon.concurrency, 2)
self.assertEqual(daemon.max_objects_per_second, 10.5)
# check deprecated option
daemon = object_updater.ObjectUpdater({'slowdown': '0.04'},
logger=self.logger)
self.assertEqual(daemon.max_objects_per_second, 20.0)
def check_bad(conf):
with self.assertRaises(ValueError):
object_updater.ObjectUpdater(conf, logger=self.logger)
check_bad({'interval': 'foo'})
check_bad({'interval': '300.0'})
check_bad({'concurrency': 'bar'})
check_bad({'concurrency': '1.0'})
check_bad({'slowdown': 'baz'})
check_bad({'objects_per_second': 'quux'})
@mock.patch('os.listdir') @mock.patch('os.listdir')
def test_listdir_with_exception(self, mock_listdir): def test_listdir_with_exception(self, mock_listdir):
e = OSError('permission_denied') e = OSError('permission_denied')