2013-09-20 01:00:54 +08:00
|
|
|
# Copyright (c) 2010-2012 OpenStack Foundation
|
2012-05-16 21:08:34 +00:00
|
|
|
#
|
|
|
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
# you may not use this file except in compliance with the License.
|
|
|
|
# You may obtain a copy of the License at
|
|
|
|
#
|
|
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
#
|
|
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
|
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
|
|
|
|
# implied.
|
|
|
|
# See the License for the specific language governing permissions and
|
|
|
|
# limitations under the License.
|
|
|
|
|
|
|
|
import unittest
|
|
|
|
import time
|
|
|
|
import eventlet
|
2014-01-29 08:33:27 -08:00
|
|
|
import mock
|
2012-05-16 21:08:34 +00:00
|
|
|
from contextlib import contextmanager
|
|
|
|
|
|
|
|
from test.unit import FakeLogger
|
|
|
|
from swift.common.middleware import ratelimit
|
2016-04-27 13:31:11 -05:00
|
|
|
from swift.proxy.controllers.base import get_cache_key, \
|
2013-04-24 12:46:13 -07:00
|
|
|
headers_to_container_info
|
2012-05-16 21:08:34 +00:00
|
|
|
from swift.common.memcached import MemcacheConnectionError
|
2012-09-04 14:02:19 -07:00
|
|
|
from swift.common.swob import Request
|
Add Ratelimit parameters to /info
We previously registered with no parameters, added parms so they
are displayed as follows (example):
"ratelimit": {"max_sleep_time_seconds": 60.0,
"container_listing_ratelimits": [[0, 100.0], [10, 50.0], [50, 20.0]],
"container_ratelimits": [[0, 100.0], [10, 50.0], [50, 20.0]],
"account_ratelimit": 1.0}
Note that not all parameters are exposed (intentionally) via /info
Change-Id: I36c7ef15af17e3eb8ebb93429035bd06d089a945
Closes-Bug: 1308989
2014-04-18 10:16:11 -07:00
|
|
|
from swift.common import utils
|
2012-05-16 21:08:34 +00:00
|
|
|
|
2015-10-22 17:33:09 +02:00
|
|
|
threading = eventlet.patcher.original('threading')
|
|
|
|
|
2012-05-16 21:08:34 +00:00
|
|
|
|
|
|
|
class FakeMemcache(object):
|
|
|
|
|
|
|
|
def __init__(self):
|
|
|
|
self.store = {}
|
|
|
|
self.error_on_incr = False
|
|
|
|
self.init_incr_return_neg = False
|
|
|
|
|
|
|
|
def get(self, key):
|
|
|
|
return self.store.get(key)
|
|
|
|
|
Swift MemcacheRing (set) interface is incompatible fixes
This patch fixes the Swift MemcacheRing set and set_multi
interface incompatible problem with python memcache. The fix
added two extra named parameters to both set and set_multi
method. When only time or timeout parameter is present, then one
of the value will be used. When both time and timeout are present,
the time parameter will be used.
Named parameter min_compress_len is added for pure compatibility
purposes. The current implementation ignores this parameter.
To make swift memcached methods all consistent cross the board,
method incr and decr have also been changed to include a new
named parameter time.
In future OpenStack releases, the named parameter timeout will be
removed, keep the named parameter timeout around for now is
to make sure that mismatched releases between client and server
will still work.
From now on, when a call is made to set, set_multi, decr, incr
by using timeout parametner, a warning message will be logged to
indicate the deprecation of the parameter.
Fixes: bug #1095730
Change-Id: I07af784a54d7d79395fc3265e74145f92f38a893
2013-02-13 13:54:51 -05:00
|
|
|
def set(self, key, value, serialize=False, time=0):
|
2012-05-16 21:08:34 +00:00
|
|
|
self.store[key] = value
|
|
|
|
return True
|
|
|
|
|
Swift MemcacheRing (set) interface is incompatible fixes
This patch fixes the Swift MemcacheRing set and set_multi
interface incompatible problem with python memcache. The fix
added two extra named parameters to both set and set_multi
method. When only time or timeout parameter is present, then one
of the value will be used. When both time and timeout are present,
the time parameter will be used.
Named parameter min_compress_len is added for pure compatibility
purposes. The current implementation ignores this parameter.
To make swift memcached methods all consistent cross the board,
method incr and decr have also been changed to include a new
named parameter time.
In future OpenStack releases, the named parameter timeout will be
removed, keep the named parameter timeout around for now is
to make sure that mismatched releases between client and server
will still work.
From now on, when a call is made to set, set_multi, decr, incr
by using timeout parametner, a warning message will be logged to
indicate the deprecation of the parameter.
Fixes: bug #1095730
Change-Id: I07af784a54d7d79395fc3265e74145f92f38a893
2013-02-13 13:54:51 -05:00
|
|
|
def incr(self, key, delta=1, time=0):
|
2012-05-16 21:08:34 +00:00
|
|
|
if self.error_on_incr:
|
|
|
|
raise MemcacheConnectionError('Memcache restarting')
|
|
|
|
if self.init_incr_return_neg:
|
|
|
|
# simulate initial hit, force reset of memcache
|
|
|
|
self.init_incr_return_neg = False
|
|
|
|
return -10000000
|
|
|
|
self.store[key] = int(self.store.setdefault(key, 0)) + int(delta)
|
|
|
|
if self.store[key] < 0:
|
|
|
|
self.store[key] = 0
|
|
|
|
return int(self.store[key])
|
|
|
|
|
Swift MemcacheRing (set) interface is incompatible fixes
This patch fixes the Swift MemcacheRing set and set_multi
interface incompatible problem with python memcache. The fix
added two extra named parameters to both set and set_multi
method. When only time or timeout parameter is present, then one
of the value will be used. When both time and timeout are present,
the time parameter will be used.
Named parameter min_compress_len is added for pure compatibility
purposes. The current implementation ignores this parameter.
To make swift memcached methods all consistent cross the board,
method incr and decr have also been changed to include a new
named parameter time.
In future OpenStack releases, the named parameter timeout will be
removed, keep the named parameter timeout around for now is
to make sure that mismatched releases between client and server
will still work.
From now on, when a call is made to set, set_multi, decr, incr
by using timeout parametner, a warning message will be logged to
indicate the deprecation of the parameter.
Fixes: bug #1095730
Change-Id: I07af784a54d7d79395fc3265e74145f92f38a893
2013-02-13 13:54:51 -05:00
|
|
|
def decr(self, key, delta=1, time=0):
|
|
|
|
return self.incr(key, delta=-delta, time=time)
|
2012-05-16 21:08:34 +00:00
|
|
|
|
|
|
|
@contextmanager
|
|
|
|
def soft_lock(self, key, timeout=0, retries=5):
|
|
|
|
yield True
|
|
|
|
|
|
|
|
def delete(self, key):
|
|
|
|
try:
|
|
|
|
del self.store[key]
|
|
|
|
except Exception:
|
|
|
|
pass
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
|
|
class FakeApp(object):
|
|
|
|
|
|
|
|
def __call__(self, env, start_response):
|
|
|
|
return ['204 No Content']
|
|
|
|
|
|
|
|
|
|
|
|
def start_response(*args):
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
|
time_ticker = 0
|
|
|
|
time_override = []
|
|
|
|
|
|
|
|
|
|
|
|
def mock_sleep(x):
|
|
|
|
global time_ticker
|
|
|
|
time_ticker += x
|
|
|
|
|
|
|
|
|
|
|
|
def mock_time():
|
|
|
|
global time_override
|
|
|
|
global time_ticker
|
|
|
|
if time_override:
|
|
|
|
cur_time = time_override.pop(0)
|
|
|
|
if cur_time is None:
|
|
|
|
time_override = [None if i is None else i + time_ticker
|
|
|
|
for i in time_override]
|
|
|
|
return time_ticker
|
|
|
|
return cur_time
|
|
|
|
return time_ticker
|
|
|
|
|
|
|
|
|
|
|
|
class TestRateLimit(unittest.TestCase):
|
|
|
|
|
|
|
|
def _reset_time(self):
|
|
|
|
global time_ticker
|
|
|
|
time_ticker = 0
|
|
|
|
|
|
|
|
def setUp(self):
|
|
|
|
self.was_sleep = eventlet.sleep
|
|
|
|
eventlet.sleep = mock_sleep
|
|
|
|
self.was_time = time.time
|
|
|
|
time.time = mock_time
|
|
|
|
self._reset_time()
|
|
|
|
|
|
|
|
def tearDown(self):
|
|
|
|
eventlet.sleep = self.was_sleep
|
|
|
|
time.time = self.was_time
|
|
|
|
|
|
|
|
def _run(self, callable_func, num, rate, check_time=True):
|
|
|
|
global time_ticker
|
|
|
|
begin = time.time()
|
2013-12-10 16:16:44 -08:00
|
|
|
for x in range(num):
|
2013-03-26 20:42:26 +00:00
|
|
|
callable_func()
|
2012-05-16 21:08:34 +00:00
|
|
|
end = time.time()
|
2013-08-14 12:40:25 +00:00
|
|
|
total_time = float(num) / rate - 1.0 / rate # 1st request not limited
|
2012-05-16 21:08:34 +00:00
|
|
|
# Allow for one second of variation in the total time.
|
|
|
|
time_diff = abs(total_time - (end - begin))
|
|
|
|
if check_time:
|
2015-08-05 23:58:14 +05:30
|
|
|
self.assertEqual(round(total_time, 1), round(time_ticker, 1))
|
2012-05-16 21:08:34 +00:00
|
|
|
return time_diff
|
|
|
|
|
2013-08-14 12:40:25 +00:00
|
|
|
def test_get_maxrate(self):
|
2012-05-16 21:08:34 +00:00
|
|
|
conf_dict = {'container_ratelimit_10': 200,
|
|
|
|
'container_ratelimit_50': 100,
|
|
|
|
'container_ratelimit_75': 30}
|
Add Ratelimit parameters to /info
We previously registered with no parameters, added parms so they
are displayed as follows (example):
"ratelimit": {"max_sleep_time_seconds": 60.0,
"container_listing_ratelimits": [[0, 100.0], [10, 50.0], [50, 20.0]],
"container_ratelimits": [[0, 100.0], [10, 50.0], [50, 20.0]],
"account_ratelimit": 1.0}
Note that not all parameters are exposed (intentionally) via /info
Change-Id: I36c7ef15af17e3eb8ebb93429035bd06d089a945
Closes-Bug: 1308989
2014-04-18 10:16:11 -07:00
|
|
|
test_ratelimit = ratelimit.filter_factory(conf_dict)(FakeApp())
|
|
|
|
test_ratelimit.logger = FakeLogger()
|
2017-06-25 10:29:23 +08:00
|
|
|
self.assertIsNone(ratelimit.get_maxrate(
|
|
|
|
test_ratelimit.container_ratelimits, 0))
|
|
|
|
self.assertIsNone(ratelimit.get_maxrate(
|
|
|
|
test_ratelimit.container_ratelimits, 5))
|
2015-08-05 23:58:14 +05:30
|
|
|
self.assertEqual(ratelimit.get_maxrate(
|
2013-08-14 12:40:25 +00:00
|
|
|
test_ratelimit.container_ratelimits, 10), 200)
|
2015-08-05 23:58:14 +05:30
|
|
|
self.assertEqual(ratelimit.get_maxrate(
|
2013-08-14 12:40:25 +00:00
|
|
|
test_ratelimit.container_ratelimits, 60), 72)
|
2015-08-05 23:58:14 +05:30
|
|
|
self.assertEqual(ratelimit.get_maxrate(
|
2013-08-14 12:40:25 +00:00
|
|
|
test_ratelimit.container_ratelimits, 160), 30)
|
2012-05-16 21:08:34 +00:00
|
|
|
|
|
|
|
def test_get_ratelimitable_key_tuples(self):
|
|
|
|
current_rate = 13
|
|
|
|
conf_dict = {'account_ratelimit': current_rate,
|
|
|
|
'container_ratelimit_3': 200}
|
|
|
|
fake_memcache = FakeMemcache()
|
2016-04-27 13:31:11 -05:00
|
|
|
fake_memcache.store[get_cache_key('a', 'c')] = \
|
2013-04-24 12:46:13 -07:00
|
|
|
{'object_count': '5'}
|
Add Ratelimit parameters to /info
We previously registered with no parameters, added parms so they
are displayed as follows (example):
"ratelimit": {"max_sleep_time_seconds": 60.0,
"container_listing_ratelimits": [[0, 100.0], [10, 50.0], [50, 20.0]],
"container_ratelimits": [[0, 100.0], [10, 50.0], [50, 20.0]],
"account_ratelimit": 1.0}
Note that not all parameters are exposed (intentionally) via /info
Change-Id: I36c7ef15af17e3eb8ebb93429035bd06d089a945
Closes-Bug: 1308989
2014-04-18 10:16:11 -07:00
|
|
|
the_app = ratelimit.filter_factory(conf_dict)(FakeApp())
|
2012-05-16 21:08:34 +00:00
|
|
|
the_app.memcache_client = fake_memcache
|
2014-01-29 08:33:27 -08:00
|
|
|
req = lambda: None
|
2014-09-25 16:29:57 +00:00
|
|
|
req.environ = {'swift.cache': fake_memcache, 'PATH_INFO': '/v1/a/c/o'}
|
2014-01-29 08:33:27 -08:00
|
|
|
with mock.patch('swift.common.middleware.ratelimit.get_account_info',
|
|
|
|
lambda *args, **kwargs: {}):
|
|
|
|
req.method = 'DELETE'
|
2015-08-05 23:58:14 +05:30
|
|
|
self.assertEqual(len(the_app.get_ratelimitable_key_tuples(
|
2014-01-29 08:33:27 -08:00
|
|
|
req, 'a', None, None)), 0)
|
|
|
|
req.method = 'PUT'
|
2015-08-05 23:58:14 +05:30
|
|
|
self.assertEqual(len(the_app.get_ratelimitable_key_tuples(
|
2014-01-29 08:33:27 -08:00
|
|
|
req, 'a', 'c', None)), 1)
|
|
|
|
req.method = 'DELETE'
|
2015-08-05 23:58:14 +05:30
|
|
|
self.assertEqual(len(the_app.get_ratelimitable_key_tuples(
|
2014-01-29 08:33:27 -08:00
|
|
|
req, 'a', 'c', None)), 1)
|
|
|
|
req.method = 'GET'
|
2015-08-05 23:58:14 +05:30
|
|
|
self.assertEqual(len(the_app.get_ratelimitable_key_tuples(
|
2014-01-29 08:33:27 -08:00
|
|
|
req, 'a', 'c', 'o')), 0)
|
|
|
|
req.method = 'PUT'
|
2015-08-05 23:58:14 +05:30
|
|
|
self.assertEqual(len(the_app.get_ratelimitable_key_tuples(
|
2014-01-29 08:33:27 -08:00
|
|
|
req, 'a', 'c', 'o')), 1)
|
|
|
|
|
2014-10-03 12:11:06 -07:00
|
|
|
req.method = 'PUT'
|
2015-08-05 23:58:14 +05:30
|
|
|
self.assertEqual(len(the_app.get_ratelimitable_key_tuples(
|
2014-10-03 12:11:06 -07:00
|
|
|
req, 'a', 'c', None, global_ratelimit=10)), 2)
|
2015-08-05 23:58:14 +05:30
|
|
|
self.assertEqual(the_app.get_ratelimitable_key_tuples(
|
2014-10-03 12:11:06 -07:00
|
|
|
req, 'a', 'c', None, global_ratelimit=10)[1],
|
|
|
|
('ratelimit/global-write/a', 10))
|
2014-01-29 08:33:27 -08:00
|
|
|
|
2014-10-03 12:11:06 -07:00
|
|
|
req.method = 'PUT'
|
2015-08-05 23:58:14 +05:30
|
|
|
self.assertEqual(len(the_app.get_ratelimitable_key_tuples(
|
2014-10-03 12:11:06 -07:00
|
|
|
req, 'a', 'c', None, global_ratelimit='notafloat')), 1)
|
2012-05-16 21:08:34 +00:00
|
|
|
|
2013-04-24 12:46:13 -07:00
|
|
|
def test_memcached_container_info_dict(self):
|
|
|
|
mdict = headers_to_container_info({'x-container-object-count': '45'})
|
2015-08-05 23:58:14 +05:30
|
|
|
self.assertEqual(mdict['object_count'], '45')
|
2013-04-24 12:46:13 -07:00
|
|
|
|
2012-09-07 11:44:19 -07:00
|
|
|
def test_ratelimit_old_memcache_format(self):
|
|
|
|
current_rate = 13
|
|
|
|
conf_dict = {'account_ratelimit': current_rate,
|
|
|
|
'container_ratelimit_3': 200}
|
|
|
|
fake_memcache = FakeMemcache()
|
2016-04-27 13:31:11 -05:00
|
|
|
fake_memcache.store[get_cache_key('a', 'c')] = \
|
2012-09-07 11:44:19 -07:00
|
|
|
{'container_size': 5}
|
Add Ratelimit parameters to /info
We previously registered with no parameters, added parms so they
are displayed as follows (example):
"ratelimit": {"max_sleep_time_seconds": 60.0,
"container_listing_ratelimits": [[0, 100.0], [10, 50.0], [50, 20.0]],
"container_ratelimits": [[0, 100.0], [10, 50.0], [50, 20.0]],
"account_ratelimit": 1.0}
Note that not all parameters are exposed (intentionally) via /info
Change-Id: I36c7ef15af17e3eb8ebb93429035bd06d089a945
Closes-Bug: 1308989
2014-04-18 10:16:11 -07:00
|
|
|
the_app = ratelimit.filter_factory(conf_dict)(FakeApp())
|
2012-09-07 11:44:19 -07:00
|
|
|
the_app.memcache_client = fake_memcache
|
2014-01-29 08:33:27 -08:00
|
|
|
req = lambda: None
|
|
|
|
req.method = 'PUT'
|
2014-09-25 16:29:57 +00:00
|
|
|
req.environ = {'PATH_INFO': '/v1/a/c/o', 'swift.cache': fake_memcache}
|
2014-01-29 08:33:27 -08:00
|
|
|
with mock.patch('swift.common.middleware.ratelimit.get_account_info',
|
|
|
|
lambda *args, **kwargs: {}):
|
|
|
|
tuples = the_app.get_ratelimitable_key_tuples(req, 'a', 'c', 'o')
|
2015-08-05 23:58:14 +05:30
|
|
|
self.assertEqual(tuples, [('ratelimit/a/c', 200.0)])
|
2012-09-07 11:44:19 -07:00
|
|
|
|
2012-05-16 21:08:34 +00:00
|
|
|
def test_account_ratelimit(self):
|
|
|
|
current_rate = 5
|
|
|
|
num_calls = 50
|
|
|
|
conf_dict = {'account_ratelimit': current_rate}
|
|
|
|
self.test_ratelimit = ratelimit.filter_factory(conf_dict)(FakeApp())
|
2014-09-25 16:29:57 +00:00
|
|
|
with mock.patch('swift.common.middleware.ratelimit.get_container_info',
|
2014-01-29 08:33:27 -08:00
|
|
|
lambda *args, **kwargs: {}):
|
2014-09-25 16:29:57 +00:00
|
|
|
with mock.patch(
|
|
|
|
'swift.common.middleware.ratelimit.get_account_info',
|
|
|
|
lambda *args, **kwargs: {}):
|
|
|
|
for meth, exp_time in [('DELETE', 9.8), ('GET', 0),
|
|
|
|
('POST', 0), ('PUT', 9.8)]:
|
|
|
|
req = Request.blank('/v/a%s/c' % meth)
|
|
|
|
req.method = meth
|
|
|
|
req.environ['swift.cache'] = FakeMemcache()
|
|
|
|
make_app_call = lambda: self.test_ratelimit(req.environ,
|
|
|
|
start_response)
|
|
|
|
begin = time.time()
|
|
|
|
self._run(make_app_call, num_calls, current_rate,
|
|
|
|
check_time=bool(exp_time))
|
2015-08-05 23:58:14 +05:30
|
|
|
self.assertEqual(round(time.time() - begin, 1), exp_time)
|
2014-09-25 16:29:57 +00:00
|
|
|
self._reset_time()
|
2012-05-16 21:08:34 +00:00
|
|
|
|
|
|
|
def test_ratelimit_set_incr(self):
|
|
|
|
current_rate = 5
|
|
|
|
num_calls = 50
|
|
|
|
conf_dict = {'account_ratelimit': current_rate}
|
|
|
|
self.test_ratelimit = ratelimit.filter_factory(conf_dict)(FakeApp())
|
|
|
|
req = Request.blank('/v/a/c')
|
|
|
|
req.method = 'PUT'
|
|
|
|
req.environ['swift.cache'] = FakeMemcache()
|
|
|
|
req.environ['swift.cache'].init_incr_return_neg = True
|
|
|
|
make_app_call = lambda: self.test_ratelimit(req.environ,
|
|
|
|
start_response)
|
|
|
|
begin = time.time()
|
2014-01-29 08:33:27 -08:00
|
|
|
with mock.patch('swift.common.middleware.ratelimit.get_account_info',
|
|
|
|
lambda *args, **kwargs: {}):
|
|
|
|
self._run(make_app_call, num_calls, current_rate, check_time=False)
|
2015-08-05 23:58:14 +05:30
|
|
|
self.assertEqual(round(time.time() - begin, 1), 9.8)
|
2012-05-16 21:08:34 +00:00
|
|
|
|
2014-10-03 12:11:06 -07:00
|
|
|
def test_ratelimit_old_white_black_list(self):
|
|
|
|
global time_ticker
|
|
|
|
current_rate = 2
|
|
|
|
conf_dict = {'account_ratelimit': current_rate,
|
|
|
|
'max_sleep_time_seconds': 2,
|
|
|
|
'account_whitelist': 'a',
|
|
|
|
'account_blacklist': 'b'}
|
|
|
|
self.test_ratelimit = ratelimit.filter_factory(conf_dict)(FakeApp())
|
|
|
|
req = Request.blank('/')
|
|
|
|
with mock.patch.object(self.test_ratelimit,
|
|
|
|
'memcache_client', FakeMemcache()):
|
|
|
|
self.assertEqual(
|
|
|
|
self.test_ratelimit.handle_ratelimit(req, 'a', 'c', 'o'),
|
|
|
|
None)
|
|
|
|
self.assertEqual(
|
|
|
|
self.test_ratelimit.handle_ratelimit(
|
|
|
|
req, 'b', 'c', 'o').status_int,
|
|
|
|
497)
|
|
|
|
|
|
|
|
def test_ratelimit_whitelist_sysmeta(self):
|
2012-05-16 21:08:34 +00:00
|
|
|
global time_ticker
|
|
|
|
current_rate = 2
|
|
|
|
conf_dict = {'account_ratelimit': current_rate,
|
|
|
|
'max_sleep_time_seconds': 2,
|
|
|
|
'account_whitelist': 'a',
|
|
|
|
'account_blacklist': 'b'}
|
Add Ratelimit parameters to /info
We previously registered with no parameters, added parms so they
are displayed as follows (example):
"ratelimit": {"max_sleep_time_seconds": 60.0,
"container_listing_ratelimits": [[0, 100.0], [10, 50.0], [50, 20.0]],
"container_ratelimits": [[0, 100.0], [10, 50.0], [50, 20.0]],
"account_ratelimit": 1.0}
Note that not all parameters are exposed (intentionally) via /info
Change-Id: I36c7ef15af17e3eb8ebb93429035bd06d089a945
Closes-Bug: 1308989
2014-04-18 10:16:11 -07:00
|
|
|
self.test_ratelimit = ratelimit.filter_factory(conf_dict)(FakeApp())
|
2012-05-16 21:08:34 +00:00
|
|
|
req = Request.blank('/v/a/c')
|
|
|
|
req.environ['swift.cache'] = FakeMemcache()
|
|
|
|
|
2015-10-22 17:33:09 +02:00
|
|
|
class rate_caller(threading.Thread):
|
2012-05-16 21:08:34 +00:00
|
|
|
|
|
|
|
def __init__(self, parent):
|
2015-10-22 17:33:09 +02:00
|
|
|
threading.Thread.__init__(self)
|
2012-05-16 21:08:34 +00:00
|
|
|
self.parent = parent
|
|
|
|
|
|
|
|
def run(self):
|
|
|
|
self.result = self.parent.test_ratelimit(req.environ,
|
|
|
|
start_response)
|
2014-10-03 12:11:06 -07:00
|
|
|
|
|
|
|
def get_fake_ratelimit(*args, **kwargs):
|
|
|
|
return {'sysmeta': {'global-write-ratelimit': 'WHITELIST'}}
|
|
|
|
|
|
|
|
with mock.patch('swift.common.middleware.ratelimit.get_account_info',
|
|
|
|
get_fake_ratelimit):
|
|
|
|
nt = 5
|
|
|
|
threads = []
|
|
|
|
for i in range(nt):
|
|
|
|
rc = rate_caller(self)
|
|
|
|
rc.start()
|
|
|
|
threads.append(rc)
|
|
|
|
for thread in threads:
|
|
|
|
thread.join()
|
|
|
|
the_498s = [
|
|
|
|
t for t in threads
|
|
|
|
if ''.join(t.result).startswith('Slow down')]
|
2015-08-05 23:58:14 +05:30
|
|
|
self.assertEqual(len(the_498s), 0)
|
|
|
|
self.assertEqual(time_ticker, 0)
|
2012-05-16 21:08:34 +00:00
|
|
|
|
|
|
|
def test_ratelimit_blacklist(self):
|
|
|
|
global time_ticker
|
|
|
|
current_rate = 2
|
|
|
|
conf_dict = {'account_ratelimit': current_rate,
|
|
|
|
'max_sleep_time_seconds': 2,
|
|
|
|
'account_whitelist': 'a',
|
|
|
|
'account_blacklist': 'b'}
|
Add Ratelimit parameters to /info
We previously registered with no parameters, added parms so they
are displayed as follows (example):
"ratelimit": {"max_sleep_time_seconds": 60.0,
"container_listing_ratelimits": [[0, 100.0], [10, 50.0], [50, 20.0]],
"container_ratelimits": [[0, 100.0], [10, 50.0], [50, 20.0]],
"account_ratelimit": 1.0}
Note that not all parameters are exposed (intentionally) via /info
Change-Id: I36c7ef15af17e3eb8ebb93429035bd06d089a945
Closes-Bug: 1308989
2014-04-18 10:16:11 -07:00
|
|
|
self.test_ratelimit = ratelimit.filter_factory(conf_dict)(FakeApp())
|
|
|
|
self.test_ratelimit.logger = FakeLogger()
|
2012-05-16 21:08:34 +00:00
|
|
|
self.test_ratelimit.BLACK_LIST_SLEEP = 0
|
|
|
|
req = Request.blank('/v/b/c')
|
|
|
|
req.environ['swift.cache'] = FakeMemcache()
|
|
|
|
|
2015-10-22 17:33:09 +02:00
|
|
|
class rate_caller(threading.Thread):
|
2012-05-16 21:08:34 +00:00
|
|
|
|
|
|
|
def __init__(self, parent):
|
2015-10-22 17:33:09 +02:00
|
|
|
threading.Thread.__init__(self)
|
2012-05-16 21:08:34 +00:00
|
|
|
self.parent = parent
|
|
|
|
|
|
|
|
def run(self):
|
|
|
|
self.result = self.parent.test_ratelimit(req.environ,
|
|
|
|
start_response)
|
2014-10-03 12:11:06 -07:00
|
|
|
|
|
|
|
def get_fake_ratelimit(*args, **kwargs):
|
|
|
|
return {'sysmeta': {'global-write-ratelimit': 'BLACKLIST'}}
|
|
|
|
|
|
|
|
with mock.patch('swift.common.middleware.ratelimit.get_account_info',
|
|
|
|
get_fake_ratelimit):
|
|
|
|
nt = 5
|
|
|
|
threads = []
|
|
|
|
for i in range(nt):
|
|
|
|
rc = rate_caller(self)
|
|
|
|
rc.start()
|
|
|
|
threads.append(rc)
|
|
|
|
for thread in threads:
|
|
|
|
thread.join()
|
|
|
|
the_497s = [
|
|
|
|
t for t in threads
|
|
|
|
if ''.join(t.result).startswith('Your account')]
|
2015-08-05 23:58:14 +05:30
|
|
|
self.assertEqual(len(the_497s), 5)
|
|
|
|
self.assertEqual(time_ticker, 0)
|
2012-05-16 21:08:34 +00:00
|
|
|
|
|
|
|
def test_ratelimit_max_rate_double(self):
|
|
|
|
global time_ticker
|
|
|
|
global time_override
|
|
|
|
current_rate = 2
|
|
|
|
conf_dict = {'account_ratelimit': current_rate,
|
|
|
|
'clock_accuracy': 100,
|
|
|
|
'max_sleep_time_seconds': 1}
|
Add Ratelimit parameters to /info
We previously registered with no parameters, added parms so they
are displayed as follows (example):
"ratelimit": {"max_sleep_time_seconds": 60.0,
"container_listing_ratelimits": [[0, 100.0], [10, 50.0], [50, 20.0]],
"container_ratelimits": [[0, 100.0], [10, 50.0], [50, 20.0]],
"account_ratelimit": 1.0}
Note that not all parameters are exposed (intentionally) via /info
Change-Id: I36c7ef15af17e3eb8ebb93429035bd06d089a945
Closes-Bug: 1308989
2014-04-18 10:16:11 -07:00
|
|
|
self.test_ratelimit = ratelimit.filter_factory(conf_dict)(FakeApp())
|
2012-05-16 21:08:34 +00:00
|
|
|
self.test_ratelimit.log_sleep_time_seconds = .00001
|
|
|
|
req = Request.blank('/v/a/c')
|
|
|
|
req.method = 'PUT'
|
|
|
|
req.environ['swift.cache'] = FakeMemcache()
|
|
|
|
|
|
|
|
time_override = [0, 0, 0, 0, None]
|
|
|
|
# simulates 4 requests coming in at same time, then sleeping
|
2014-01-29 08:33:27 -08:00
|
|
|
with mock.patch('swift.common.middleware.ratelimit.get_account_info',
|
|
|
|
lambda *args, **kwargs: {}):
|
|
|
|
r = self.test_ratelimit(req.environ, start_response)
|
|
|
|
mock_sleep(.1)
|
|
|
|
r = self.test_ratelimit(req.environ, start_response)
|
|
|
|
mock_sleep(.1)
|
|
|
|
r = self.test_ratelimit(req.environ, start_response)
|
2015-08-05 23:58:14 +05:30
|
|
|
self.assertEqual(r[0], 'Slow down')
|
2014-01-29 08:33:27 -08:00
|
|
|
mock_sleep(.1)
|
|
|
|
r = self.test_ratelimit(req.environ, start_response)
|
2015-08-05 23:58:14 +05:30
|
|
|
self.assertEqual(r[0], 'Slow down')
|
2014-01-29 08:33:27 -08:00
|
|
|
mock_sleep(.1)
|
|
|
|
r = self.test_ratelimit(req.environ, start_response)
|
2015-08-05 23:58:14 +05:30
|
|
|
self.assertEqual(r[0], '204 No Content')
|
2012-05-16 21:08:34 +00:00
|
|
|
|
2013-08-14 12:40:25 +00:00
|
|
|
def test_ratelimit_max_rate_double_container(self):
|
|
|
|
global time_ticker
|
|
|
|
global time_override
|
|
|
|
current_rate = 2
|
|
|
|
conf_dict = {'container_ratelimit_0': current_rate,
|
|
|
|
'clock_accuracy': 100,
|
|
|
|
'max_sleep_time_seconds': 1}
|
Add Ratelimit parameters to /info
We previously registered with no parameters, added parms so they
are displayed as follows (example):
"ratelimit": {"max_sleep_time_seconds": 60.0,
"container_listing_ratelimits": [[0, 100.0], [10, 50.0], [50, 20.0]],
"container_ratelimits": [[0, 100.0], [10, 50.0], [50, 20.0]],
"account_ratelimit": 1.0}
Note that not all parameters are exposed (intentionally) via /info
Change-Id: I36c7ef15af17e3eb8ebb93429035bd06d089a945
Closes-Bug: 1308989
2014-04-18 10:16:11 -07:00
|
|
|
self.test_ratelimit = ratelimit.filter_factory(conf_dict)(FakeApp())
|
2013-08-14 12:40:25 +00:00
|
|
|
self.test_ratelimit.log_sleep_time_seconds = .00001
|
|
|
|
req = Request.blank('/v/a/c/o')
|
|
|
|
req.method = 'PUT'
|
|
|
|
req.environ['swift.cache'] = FakeMemcache()
|
|
|
|
req.environ['swift.cache'].set(
|
2016-04-27 13:31:11 -05:00
|
|
|
get_cache_key('a', 'c'),
|
Fix up get_account_info and get_container_info
get_account_info used to work like this:
* make an account HEAD request
* ignore the response
* get the account info by digging around in the request environment,
where it had been deposited by elves or something
Not actually elves, but the proxy's GETorHEAD_base method would take
the HEAD response and cache it in the response environment, which was
the same object as the request environment, thus enabling
get_account_info to find it.
This was extraordinarily brittle. If a WSGI middleware were to
shallow-copy the request environment, then any middlewares to its left
could not use get_account_info, as the left middleware's request
environment would no longer be identical to the response environment
down in GETorHEAD_base.
Now, get_account_info works like this:
* make an account HEAD request.
* if the account info is in the request environment, return it. This
is an optimization to avoid a double-set in memcached.
* else, compute the account info from the response headers, store it
in caches, and return it.
This is much easier to think about; get_account_info can get and cache
account info all on its own; the cache check and cache set are right
next to each other.
All the above is true for get_container_info as well.
get_info() is still around, but it's just a shim. It was trying to
unify get_account_info and get_container_info to exploit the
commonalities, but the number of times that "if container:" showed up
in get_info and its helpers really indicated that something was
wrong. I'd rather have two functions with some duplication than one
function with no duplication but a bunch of "if container:" branches.
Other things of note:
* a HEAD request to a deleted account returns 410, but
get_account_info would return 404 since the 410 came from the
account controller *after* GETorHEAD_base ran. Now
get_account_info returns 410 as well.
* cache validity period (recheck_account_existence and
recheck_container_existence) is now communicated to
get_account_info via an X-Backend header. This way,
get_account_info doesn't need a reference to the
swift.proxy.server.Application object.
* both logged swift_source values are now correct for
get_container_info calls; before, on a cold cache,
get_container_info would call get_account_info but not pass along
swift_source, resulting in get_account_info logging "GET_INFO" as
the source. Amusingly, there was a unit test asserting this bogus
behavior.
* callers that modify the return value of get_account_info or of
get_container_info don't modify what's stored in swift.infocache.
* get_account_info on an account that *can* be autocreated but has
not been will return a 200, same as a HEAD request. The old
behavior was a 404 from get_account_info but a 200 from
HEAD. Callers can tell the difference by looking at
info['account_really_exists'] if they need to know the difference
(there is one call site that needs to know, in container
PUT). Note: this is for all accounts when the proxy's
"account_autocreate" setting is on.
Change-Id: I5167714025ec7237f7e6dd4759c2c6eb959b3fca
2016-02-11 15:51:45 -08:00
|
|
|
{'object_count': 1})
|
2013-08-14 12:40:25 +00:00
|
|
|
|
|
|
|
time_override = [0, 0, 0, 0, None]
|
|
|
|
# simulates 4 requests coming in at same time, then sleeping
|
2014-01-29 08:33:27 -08:00
|
|
|
with mock.patch('swift.common.middleware.ratelimit.get_account_info',
|
|
|
|
lambda *args, **kwargs: {}):
|
|
|
|
r = self.test_ratelimit(req.environ, start_response)
|
|
|
|
mock_sleep(.1)
|
|
|
|
r = self.test_ratelimit(req.environ, start_response)
|
|
|
|
mock_sleep(.1)
|
|
|
|
r = self.test_ratelimit(req.environ, start_response)
|
2015-08-05 23:58:14 +05:30
|
|
|
self.assertEqual(r[0], 'Slow down')
|
2014-01-29 08:33:27 -08:00
|
|
|
mock_sleep(.1)
|
|
|
|
r = self.test_ratelimit(req.environ, start_response)
|
2015-08-05 23:58:14 +05:30
|
|
|
self.assertEqual(r[0], 'Slow down')
|
2014-01-29 08:33:27 -08:00
|
|
|
mock_sleep(.1)
|
|
|
|
r = self.test_ratelimit(req.environ, start_response)
|
2015-08-05 23:58:14 +05:30
|
|
|
self.assertEqual(r[0], '204 No Content')
|
2013-08-14 12:40:25 +00:00
|
|
|
|
|
|
|
def test_ratelimit_max_rate_double_container_listing(self):
|
|
|
|
global time_ticker
|
|
|
|
global time_override
|
|
|
|
current_rate = 2
|
|
|
|
conf_dict = {'container_listing_ratelimit_0': current_rate,
|
|
|
|
'clock_accuracy': 100,
|
|
|
|
'max_sleep_time_seconds': 1}
|
Add Ratelimit parameters to /info
We previously registered with no parameters, added parms so they
are displayed as follows (example):
"ratelimit": {"max_sleep_time_seconds": 60.0,
"container_listing_ratelimits": [[0, 100.0], [10, 50.0], [50, 20.0]],
"container_ratelimits": [[0, 100.0], [10, 50.0], [50, 20.0]],
"account_ratelimit": 1.0}
Note that not all parameters are exposed (intentionally) via /info
Change-Id: I36c7ef15af17e3eb8ebb93429035bd06d089a945
Closes-Bug: 1308989
2014-04-18 10:16:11 -07:00
|
|
|
self.test_ratelimit = ratelimit.filter_factory(conf_dict)(FakeApp())
|
2013-08-14 12:40:25 +00:00
|
|
|
self.test_ratelimit.log_sleep_time_seconds = .00001
|
|
|
|
req = Request.blank('/v/a/c')
|
|
|
|
req.method = 'GET'
|
|
|
|
req.environ['swift.cache'] = FakeMemcache()
|
|
|
|
req.environ['swift.cache'].set(
|
2016-04-27 13:31:11 -05:00
|
|
|
get_cache_key('a', 'c'),
|
Fix up get_account_info and get_container_info
get_account_info used to work like this:
* make an account HEAD request
* ignore the response
* get the account info by digging around in the request environment,
where it had been deposited by elves or something
Not actually elves, but the proxy's GETorHEAD_base method would take
the HEAD response and cache it in the response environment, which was
the same object as the request environment, thus enabling
get_account_info to find it.
This was extraordinarily brittle. If a WSGI middleware were to
shallow-copy the request environment, then any middlewares to its left
could not use get_account_info, as the left middleware's request
environment would no longer be identical to the response environment
down in GETorHEAD_base.
Now, get_account_info works like this:
* make an account HEAD request.
* if the account info is in the request environment, return it. This
is an optimization to avoid a double-set in memcached.
* else, compute the account info from the response headers, store it
in caches, and return it.
This is much easier to think about; get_account_info can get and cache
account info all on its own; the cache check and cache set are right
next to each other.
All the above is true for get_container_info as well.
get_info() is still around, but it's just a shim. It was trying to
unify get_account_info and get_container_info to exploit the
commonalities, but the number of times that "if container:" showed up
in get_info and its helpers really indicated that something was
wrong. I'd rather have two functions with some duplication than one
function with no duplication but a bunch of "if container:" branches.
Other things of note:
* a HEAD request to a deleted account returns 410, but
get_account_info would return 404 since the 410 came from the
account controller *after* GETorHEAD_base ran. Now
get_account_info returns 410 as well.
* cache validity period (recheck_account_existence and
recheck_container_existence) is now communicated to
get_account_info via an X-Backend header. This way,
get_account_info doesn't need a reference to the
swift.proxy.server.Application object.
* both logged swift_source values are now correct for
get_container_info calls; before, on a cold cache,
get_container_info would call get_account_info but not pass along
swift_source, resulting in get_account_info logging "GET_INFO" as
the source. Amusingly, there was a unit test asserting this bogus
behavior.
* callers that modify the return value of get_account_info or of
get_container_info don't modify what's stored in swift.infocache.
* get_account_info on an account that *can* be autocreated but has
not been will return a 200, same as a HEAD request. The old
behavior was a 404 from get_account_info but a 200 from
HEAD. Callers can tell the difference by looking at
info['account_really_exists'] if they need to know the difference
(there is one call site that needs to know, in container
PUT). Note: this is for all accounts when the proxy's
"account_autocreate" setting is on.
Change-Id: I5167714025ec7237f7e6dd4759c2c6eb959b3fca
2016-02-11 15:51:45 -08:00
|
|
|
{'object_count': 1})
|
2013-08-14 12:40:25 +00:00
|
|
|
|
2014-10-03 12:11:06 -07:00
|
|
|
with mock.patch('swift.common.middleware.ratelimit.get_account_info',
|
|
|
|
lambda *args, **kwargs: {}):
|
|
|
|
time_override = [0, 0, 0, 0, None]
|
|
|
|
# simulates 4 requests coming in at same time, then sleeping
|
|
|
|
r = self.test_ratelimit(req.environ, start_response)
|
|
|
|
mock_sleep(.1)
|
|
|
|
r = self.test_ratelimit(req.environ, start_response)
|
|
|
|
mock_sleep(.1)
|
|
|
|
r = self.test_ratelimit(req.environ, start_response)
|
2015-08-05 23:58:14 +05:30
|
|
|
self.assertEqual(r[0], 'Slow down')
|
2014-10-03 12:11:06 -07:00
|
|
|
mock_sleep(.1)
|
|
|
|
r = self.test_ratelimit(req.environ, start_response)
|
2015-08-05 23:58:14 +05:30
|
|
|
self.assertEqual(r[0], 'Slow down')
|
2014-10-03 12:11:06 -07:00
|
|
|
mock_sleep(.1)
|
|
|
|
r = self.test_ratelimit(req.environ, start_response)
|
2015-08-05 23:58:14 +05:30
|
|
|
self.assertEqual(r[0], '204 No Content')
|
2014-10-03 12:11:06 -07:00
|
|
|
mc = self.test_ratelimit.memcache_client
|
|
|
|
try:
|
|
|
|
self.test_ratelimit.memcache_client = None
|
2017-06-25 10:29:23 +08:00
|
|
|
self.assertIsNone(
|
|
|
|
self.test_ratelimit.handle_ratelimit(req, 'n', 'c', None))
|
2014-10-03 12:11:06 -07:00
|
|
|
finally:
|
|
|
|
self.test_ratelimit.memcache_client = mc
|
2013-08-14 12:40:25 +00:00
|
|
|
|
2012-05-16 21:08:34 +00:00
|
|
|
def test_ratelimit_max_rate_multiple_acc(self):
|
|
|
|
num_calls = 4
|
|
|
|
current_rate = 2
|
|
|
|
conf_dict = {'account_ratelimit': current_rate,
|
|
|
|
'max_sleep_time_seconds': 2}
|
|
|
|
fake_memcache = FakeMemcache()
|
|
|
|
|
Add Ratelimit parameters to /info
We previously registered with no parameters, added parms so they
are displayed as follows (example):
"ratelimit": {"max_sleep_time_seconds": 60.0,
"container_listing_ratelimits": [[0, 100.0], [10, 50.0], [50, 20.0]],
"container_ratelimits": [[0, 100.0], [10, 50.0], [50, 20.0]],
"account_ratelimit": 1.0}
Note that not all parameters are exposed (intentionally) via /info
Change-Id: I36c7ef15af17e3eb8ebb93429035bd06d089a945
Closes-Bug: 1308989
2014-04-18 10:16:11 -07:00
|
|
|
the_app = ratelimit.filter_factory(conf_dict)(FakeApp())
|
2012-05-16 21:08:34 +00:00
|
|
|
the_app.memcache_client = fake_memcache
|
|
|
|
req = lambda: None
|
|
|
|
req.method = 'PUT'
|
2014-01-29 08:33:27 -08:00
|
|
|
req.environ = {}
|
2012-05-16 21:08:34 +00:00
|
|
|
|
2015-10-22 17:33:09 +02:00
|
|
|
class rate_caller(threading.Thread):
|
2012-05-16 21:08:34 +00:00
|
|
|
|
|
|
|
def __init__(self, name):
|
|
|
|
self.myname = name
|
2015-10-22 17:33:09 +02:00
|
|
|
threading.Thread.__init__(self)
|
2012-05-16 21:08:34 +00:00
|
|
|
|
|
|
|
def run(self):
|
|
|
|
for j in range(num_calls):
|
|
|
|
self.result = the_app.handle_ratelimit(req, self.myname,
|
|
|
|
'c', None)
|
|
|
|
|
2014-01-29 08:33:27 -08:00
|
|
|
with mock.patch('swift.common.middleware.ratelimit.get_account_info',
|
|
|
|
lambda *args, **kwargs: {}):
|
|
|
|
nt = 15
|
|
|
|
begin = time.time()
|
|
|
|
threads = []
|
|
|
|
for i in range(nt):
|
|
|
|
rc = rate_caller('a%s' % i)
|
|
|
|
rc.start()
|
|
|
|
threads.append(rc)
|
|
|
|
for thread in threads:
|
|
|
|
thread.join()
|
2012-05-16 21:08:34 +00:00
|
|
|
|
2014-01-29 08:33:27 -08:00
|
|
|
time_took = time.time() - begin
|
2015-08-05 23:58:14 +05:30
|
|
|
self.assertEqual(1.5, round(time_took, 1))
|
2012-05-16 21:08:34 +00:00
|
|
|
|
|
|
|
def test_call_invalid_path(self):
|
|
|
|
env = {'REQUEST_METHOD': 'GET',
|
|
|
|
'SCRIPT_NAME': '',
|
|
|
|
'PATH_INFO': '//v1/AUTH_1234567890',
|
|
|
|
'SERVER_NAME': '127.0.0.1',
|
|
|
|
'SERVER_PORT': '80',
|
|
|
|
'swift.cache': FakeMemcache(),
|
|
|
|
'SERVER_PROTOCOL': 'HTTP/1.0'}
|
|
|
|
|
|
|
|
app = lambda *args, **kwargs: ['fake_app']
|
Add Ratelimit parameters to /info
We previously registered with no parameters, added parms so they
are displayed as follows (example):
"ratelimit": {"max_sleep_time_seconds": 60.0,
"container_listing_ratelimits": [[0, 100.0], [10, 50.0], [50, 20.0]],
"container_ratelimits": [[0, 100.0], [10, 50.0], [50, 20.0]],
"account_ratelimit": 1.0}
Note that not all parameters are exposed (intentionally) via /info
Change-Id: I36c7ef15af17e3eb8ebb93429035bd06d089a945
Closes-Bug: 1308989
2014-04-18 10:16:11 -07:00
|
|
|
rate_mid = ratelimit.filter_factory({})(app)
|
2012-05-16 21:08:34 +00:00
|
|
|
|
|
|
|
class a_callable(object):
|
|
|
|
|
|
|
|
def __call__(self, *args, **kwargs):
|
|
|
|
pass
|
|
|
|
resp = rate_mid.__call__(env, a_callable())
|
2015-07-21 19:23:00 +05:30
|
|
|
self.assertTrue('fake_app' == resp[0])
|
2012-05-16 21:08:34 +00:00
|
|
|
|
|
|
|
def test_no_memcache(self):
|
|
|
|
current_rate = 13
|
|
|
|
num_calls = 5
|
|
|
|
conf_dict = {'account_ratelimit': current_rate}
|
|
|
|
self.test_ratelimit = ratelimit.filter_factory(conf_dict)(FakeApp())
|
|
|
|
req = Request.blank('/v/a')
|
|
|
|
req.environ['swift.cache'] = None
|
|
|
|
make_app_call = lambda: self.test_ratelimit(req.environ,
|
|
|
|
start_response)
|
|
|
|
begin = time.time()
|
|
|
|
self._run(make_app_call, num_calls, current_rate, check_time=False)
|
|
|
|
time_took = time.time() - begin
|
2015-08-05 23:58:14 +05:30
|
|
|
self.assertEqual(round(time_took, 1), 0) # no memcache, no limiting
|
2012-05-16 21:08:34 +00:00
|
|
|
|
|
|
|
def test_restarting_memcache(self):
|
|
|
|
current_rate = 2
|
|
|
|
num_calls = 5
|
|
|
|
conf_dict = {'account_ratelimit': current_rate}
|
|
|
|
self.test_ratelimit = ratelimit.filter_factory(conf_dict)(FakeApp())
|
|
|
|
req = Request.blank('/v/a/c')
|
|
|
|
req.method = 'PUT'
|
|
|
|
req.environ['swift.cache'] = FakeMemcache()
|
|
|
|
req.environ['swift.cache'].error_on_incr = True
|
|
|
|
make_app_call = lambda: self.test_ratelimit(req.environ,
|
|
|
|
start_response)
|
|
|
|
begin = time.time()
|
2014-01-29 08:33:27 -08:00
|
|
|
with mock.patch('swift.common.middleware.ratelimit.get_account_info',
|
|
|
|
lambda *args, **kwargs: {}):
|
|
|
|
self._run(make_app_call, num_calls, current_rate, check_time=False)
|
|
|
|
time_took = time.time() - begin
|
2015-08-05 23:58:14 +05:30
|
|
|
self.assertEqual(round(time_took, 1), 0) # no memcache, no limit
|
2012-05-16 21:08:34 +00:00
|
|
|
|
Add Ratelimit parameters to /info
We previously registered with no parameters, added parms so they
are displayed as follows (example):
"ratelimit": {"max_sleep_time_seconds": 60.0,
"container_listing_ratelimits": [[0, 100.0], [10, 50.0], [50, 20.0]],
"container_ratelimits": [[0, 100.0], [10, 50.0], [50, 20.0]],
"account_ratelimit": 1.0}
Note that not all parameters are exposed (intentionally) via /info
Change-Id: I36c7ef15af17e3eb8ebb93429035bd06d089a945
Closes-Bug: 1308989
2014-04-18 10:16:11 -07:00
|
|
|
|
|
|
|
class TestSwiftInfo(unittest.TestCase):
|
|
|
|
def setUp(self):
|
|
|
|
utils._swift_info = {}
|
|
|
|
utils._swift_admin_info = {}
|
|
|
|
|
|
|
|
def test_registered_defaults(self):
|
|
|
|
|
|
|
|
def check_key_is_absnet(key):
|
|
|
|
try:
|
|
|
|
swift_info[key]
|
|
|
|
except KeyError as err:
|
|
|
|
if key not in err:
|
|
|
|
raise
|
|
|
|
|
|
|
|
test_limits = {'account_ratelimit': 1,
|
|
|
|
'max_sleep_time_seconds': 60,
|
|
|
|
'container_ratelimit_0': 0,
|
|
|
|
'container_ratelimit_10': 10,
|
|
|
|
'container_ratelimit_50': 50,
|
|
|
|
'container_listing_ratelimit_0': 0,
|
|
|
|
'container_listing_ratelimit_10': 10,
|
|
|
|
'container_listing_ratelimit_50': 50}
|
|
|
|
|
|
|
|
ratelimit.filter_factory(test_limits)('have to pass in an app')
|
|
|
|
swift_info = utils.get_swift_info()
|
|
|
|
self.assertTrue('ratelimit' in swift_info)
|
|
|
|
self.assertEqual(swift_info['ratelimit']
|
|
|
|
['account_ratelimit'], 1.0)
|
|
|
|
self.assertEqual(swift_info['ratelimit']
|
|
|
|
['max_sleep_time_seconds'], 60.0)
|
|
|
|
self.assertEqual(swift_info['ratelimit']
|
|
|
|
['container_ratelimits'][0][0], 0)
|
|
|
|
self.assertEqual(swift_info['ratelimit']
|
|
|
|
['container_ratelimits'][0][1], 0.0)
|
|
|
|
self.assertEqual(swift_info['ratelimit']
|
|
|
|
['container_ratelimits'][1][0], 10)
|
|
|
|
self.assertEqual(swift_info['ratelimit']
|
|
|
|
['container_ratelimits'][1][1], 10.0)
|
|
|
|
self.assertEqual(swift_info['ratelimit']
|
|
|
|
['container_ratelimits'][2][0], 50)
|
|
|
|
self.assertEqual(swift_info['ratelimit']
|
|
|
|
['container_ratelimits'][2][1], 50.0)
|
|
|
|
self.assertEqual(swift_info['ratelimit']
|
|
|
|
['container_listing_ratelimits'][0][0], 0)
|
|
|
|
self.assertEqual(swift_info['ratelimit']
|
|
|
|
['container_listing_ratelimits'][0][1], 0.0)
|
|
|
|
self.assertEqual(swift_info['ratelimit']
|
|
|
|
['container_listing_ratelimits'][1][0], 10)
|
|
|
|
self.assertEqual(swift_info['ratelimit']
|
|
|
|
['container_listing_ratelimits'][1][1], 10.0)
|
|
|
|
self.assertEqual(swift_info['ratelimit']
|
|
|
|
['container_listing_ratelimits'][2][0], 50)
|
|
|
|
self.assertEqual(swift_info['ratelimit']
|
|
|
|
['container_listing_ratelimits'][2][1], 50.0)
|
|
|
|
|
|
|
|
# these were left out on purpose
|
|
|
|
for key in ['log_sleep_time_seconds', 'clock_accuracy',
|
|
|
|
'rate_buffer_seconds', 'ratelimit_whitelis',
|
|
|
|
'ratelimit_blacklist']:
|
|
|
|
check_key_is_absnet(key)
|
|
|
|
|
|
|
|
|
2012-05-16 21:08:34 +00:00
|
|
|
if __name__ == '__main__':
|
|
|
|
unittest.main()
|