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
|
|
|
|
from contextlib import contextmanager
|
|
|
|
from threading import Thread
|
|
|
|
|
|
|
|
from test.unit import FakeLogger
|
|
|
|
from swift.common.middleware import ratelimit
|
2013-04-24 12:46:13 -07:00
|
|
|
from swift.proxy.controllers.base import get_container_memcache_key, \
|
|
|
|
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
|
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
|
|
|
|
|
|
|
|
|
|
|
|
def mock_http_connect(response, headers=None, with_exc=False):
|
|
|
|
|
|
|
|
class FakeConn(object):
|
|
|
|
|
|
|
|
def __init__(self, status, headers, with_exc):
|
|
|
|
self.status = status
|
|
|
|
self.reason = 'Fake'
|
|
|
|
self.host = '1.2.3.4'
|
|
|
|
self.port = '1234'
|
|
|
|
self.with_exc = with_exc
|
|
|
|
self.headers = headers
|
|
|
|
if self.headers is None:
|
|
|
|
self.headers = {}
|
|
|
|
|
|
|
|
def getresponse(self):
|
|
|
|
if self.with_exc:
|
|
|
|
raise Exception('test')
|
|
|
|
return self
|
|
|
|
|
|
|
|
def getheader(self, header):
|
|
|
|
return self.headers[header]
|
|
|
|
|
|
|
|
def read(self, amt=None):
|
|
|
|
return ''
|
|
|
|
|
|
|
|
def close(self):
|
|
|
|
return
|
|
|
|
return lambda *args, **kwargs: FakeConn(response, headers, with_exc)
|
|
|
|
|
|
|
|
|
|
|
|
class FakeApp(object):
|
|
|
|
|
|
|
|
def __call__(self, env, start_response):
|
|
|
|
return ['204 No Content']
|
|
|
|
|
|
|
|
|
|
|
|
def start_response(*args):
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
|
def dummy_filter_factory(global_conf, **local_conf):
|
|
|
|
conf = global_conf.copy()
|
|
|
|
conf.update(local_conf)
|
|
|
|
|
|
|
|
def limit_filter(app):
|
|
|
|
return ratelimit.RateLimitMiddleware(app, conf, logger=FakeLogger())
|
|
|
|
return limit_filter
|
|
|
|
|
|
|
|
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()
|
|
|
|
for x in range(0, 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:
|
|
|
|
self.assertEquals(round(total_time, 1), round(time_ticker, 1))
|
|
|
|
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}
|
|
|
|
test_ratelimit = dummy_filter_factory(conf_dict)(FakeApp())
|
2013-08-14 12:40:25 +00:00
|
|
|
self.assertEquals(ratelimit.get_maxrate(
|
|
|
|
test_ratelimit.container_ratelimits, 0), None)
|
|
|
|
self.assertEquals(ratelimit.get_maxrate(
|
|
|
|
test_ratelimit.container_ratelimits, 5), None)
|
|
|
|
self.assertEquals(ratelimit.get_maxrate(
|
|
|
|
test_ratelimit.container_ratelimits, 10), 200)
|
|
|
|
self.assertEquals(ratelimit.get_maxrate(
|
|
|
|
test_ratelimit.container_ratelimits, 60), 72)
|
|
|
|
self.assertEquals(ratelimit.get_maxrate(
|
|
|
|
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()
|
|
|
|
fake_memcache.store[get_container_memcache_key('a', 'c')] = \
|
2013-04-24 12:46:13 -07:00
|
|
|
{'object_count': '5'}
|
2012-05-16 21:08:34 +00:00
|
|
|
the_app = ratelimit.RateLimitMiddleware(None, conf_dict,
|
|
|
|
logger=FakeLogger())
|
|
|
|
the_app.memcache_client = fake_memcache
|
|
|
|
self.assertEquals(len(the_app.get_ratelimitable_key_tuples(
|
2013-08-14 12:40:25 +00:00
|
|
|
'DELETE', 'a', None, None)), 0)
|
2012-05-16 21:08:34 +00:00
|
|
|
self.assertEquals(len(the_app.get_ratelimitable_key_tuples(
|
2013-08-14 12:40:25 +00:00
|
|
|
'PUT', 'a', 'c', None)), 1)
|
2012-05-16 21:08:34 +00:00
|
|
|
self.assertEquals(len(the_app.get_ratelimitable_key_tuples(
|
2013-08-14 12:40:25 +00:00
|
|
|
'DELETE', 'a', 'c', None)), 1)
|
2012-05-16 21:08:34 +00:00
|
|
|
self.assertEquals(len(the_app.get_ratelimitable_key_tuples(
|
2013-08-14 12:40:25 +00:00
|
|
|
'GET', 'a', 'c', 'o')), 0)
|
2012-05-16 21:08:34 +00:00
|
|
|
self.assertEquals(len(the_app.get_ratelimitable_key_tuples(
|
2013-08-14 12:40:25 +00:00
|
|
|
'PUT', 'a', 'c', 'o')), 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'})
|
|
|
|
self.assertEquals(mdict['object_count'], '45')
|
|
|
|
|
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()
|
|
|
|
fake_memcache.store[get_container_memcache_key('a', 'c')] = \
|
|
|
|
{'container_size': 5}
|
|
|
|
the_app = ratelimit.RateLimitMiddleware(None, conf_dict,
|
|
|
|
logger=FakeLogger())
|
|
|
|
the_app.memcache_client = fake_memcache
|
|
|
|
tuples = the_app.get_ratelimitable_key_tuples('PUT', 'a', 'c', 'o')
|
|
|
|
self.assertEquals(tuples, [('ratelimit/a/c', 200.0)])
|
|
|
|
|
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())
|
|
|
|
ratelimit.http_connect = mock_http_connect(204)
|
2013-08-14 12:40:25 +00:00
|
|
|
for meth, exp_time in [
|
|
|
|
('DELETE', 9.8), ('GET', 0), ('POST', 0), ('PUT', 9.8)]:
|
2012-05-16 21:08:34 +00:00
|
|
|
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))
|
|
|
|
self.assertEquals(round(time.time() - begin, 1), exp_time)
|
|
|
|
self._reset_time()
|
|
|
|
|
|
|
|
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())
|
|
|
|
ratelimit.http_connect = mock_http_connect(204)
|
|
|
|
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()
|
|
|
|
self._run(make_app_call, num_calls, current_rate, check_time=False)
|
|
|
|
self.assertEquals(round(time.time() - begin, 1), 9.8)
|
|
|
|
|
|
|
|
def test_ratelimit_whitelist(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 = dummy_filter_factory(conf_dict)(FakeApp())
|
|
|
|
ratelimit.http_connect = mock_http_connect(204)
|
|
|
|
req = Request.blank('/v/a/c')
|
|
|
|
req.environ['swift.cache'] = FakeMemcache()
|
|
|
|
|
|
|
|
class rate_caller(Thread):
|
|
|
|
|
|
|
|
def __init__(self, parent):
|
|
|
|
Thread.__init__(self)
|
|
|
|
self.parent = parent
|
|
|
|
|
|
|
|
def run(self):
|
|
|
|
self.result = self.parent.test_ratelimit(req.environ,
|
|
|
|
start_response)
|
|
|
|
nt = 5
|
|
|
|
threads = []
|
|
|
|
for i in range(nt):
|
|
|
|
rc = rate_caller(self)
|
|
|
|
rc.start()
|
|
|
|
threads.append(rc)
|
|
|
|
for thread in threads:
|
|
|
|
thread.join()
|
2013-08-14 12:40:25 +00:00
|
|
|
the_498s = [
|
|
|
|
t for t in threads if ''.join(t.result).startswith('Slow down')]
|
2012-05-16 21:08:34 +00:00
|
|
|
self.assertEquals(len(the_498s), 0)
|
|
|
|
self.assertEquals(time_ticker, 0)
|
|
|
|
|
|
|
|
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'}
|
|
|
|
self.test_ratelimit = dummy_filter_factory(conf_dict)(FakeApp())
|
|
|
|
self.test_ratelimit.BLACK_LIST_SLEEP = 0
|
|
|
|
ratelimit.http_connect = mock_http_connect(204)
|
|
|
|
req = Request.blank('/v/b/c')
|
|
|
|
req.environ['swift.cache'] = FakeMemcache()
|
|
|
|
|
|
|
|
class rate_caller(Thread):
|
|
|
|
|
|
|
|
def __init__(self, parent):
|
|
|
|
Thread.__init__(self)
|
|
|
|
self.parent = parent
|
|
|
|
|
|
|
|
def run(self):
|
|
|
|
self.result = self.parent.test_ratelimit(req.environ,
|
|
|
|
start_response)
|
|
|
|
nt = 5
|
|
|
|
threads = []
|
|
|
|
for i in range(nt):
|
|
|
|
rc = rate_caller(self)
|
|
|
|
rc.start()
|
|
|
|
threads.append(rc)
|
|
|
|
for thread in threads:
|
|
|
|
thread.join()
|
2013-08-14 12:40:25 +00:00
|
|
|
the_497s = [
|
|
|
|
t for t in threads if ''.join(t.result).startswith('Your account')]
|
2012-05-16 21:08:34 +00:00
|
|
|
self.assertEquals(len(the_497s), 5)
|
|
|
|
self.assertEquals(time_ticker, 0)
|
|
|
|
|
|
|
|
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}
|
|
|
|
self.test_ratelimit = dummy_filter_factory(conf_dict)(FakeApp())
|
|
|
|
ratelimit.http_connect = mock_http_connect(204)
|
|
|
|
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
|
|
|
|
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)
|
|
|
|
self.assertEquals(r[0], 'Slow down')
|
|
|
|
mock_sleep(.1)
|
|
|
|
r = self.test_ratelimit(req.environ, start_response)
|
|
|
|
self.assertEquals(r[0], 'Slow down')
|
|
|
|
mock_sleep(.1)
|
|
|
|
r = self.test_ratelimit(req.environ, start_response)
|
|
|
|
self.assertEquals(r[0], '204 No Content')
|
|
|
|
|
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}
|
|
|
|
self.test_ratelimit = dummy_filter_factory(conf_dict)(FakeApp())
|
|
|
|
ratelimit.http_connect = mock_http_connect(204)
|
|
|
|
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(
|
|
|
|
ratelimit.get_container_memcache_key('a', 'c'),
|
|
|
|
{'container_size': 1})
|
|
|
|
|
|
|
|
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)
|
|
|
|
self.assertEquals(r[0], 'Slow down')
|
|
|
|
mock_sleep(.1)
|
|
|
|
r = self.test_ratelimit(req.environ, start_response)
|
|
|
|
self.assertEquals(r[0], 'Slow down')
|
|
|
|
mock_sleep(.1)
|
|
|
|
r = self.test_ratelimit(req.environ, start_response)
|
|
|
|
self.assertEquals(r[0], '204 No Content')
|
|
|
|
|
|
|
|
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}
|
|
|
|
self.test_ratelimit = dummy_filter_factory(conf_dict)(FakeApp())
|
|
|
|
ratelimit.http_connect = mock_http_connect(204)
|
|
|
|
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(
|
|
|
|
ratelimit.get_container_memcache_key('a', 'c'),
|
|
|
|
{'container_size': 1})
|
|
|
|
|
|
|
|
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)
|
|
|
|
self.assertEquals(r[0], 'Slow down')
|
|
|
|
mock_sleep(.1)
|
|
|
|
r = self.test_ratelimit(req.environ, start_response)
|
|
|
|
self.assertEquals(r[0], 'Slow down')
|
|
|
|
mock_sleep(.1)
|
|
|
|
r = self.test_ratelimit(req.environ, start_response)
|
|
|
|
self.assertEquals(r[0], '204 No Content')
|
|
|
|
|
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()
|
|
|
|
|
|
|
|
the_app = ratelimit.RateLimitMiddleware(None, conf_dict,
|
|
|
|
logger=FakeLogger())
|
|
|
|
the_app.memcache_client = fake_memcache
|
|
|
|
req = lambda: None
|
|
|
|
req.method = 'PUT'
|
|
|
|
|
|
|
|
class rate_caller(Thread):
|
|
|
|
|
|
|
|
def __init__(self, name):
|
|
|
|
self.myname = name
|
|
|
|
Thread.__init__(self)
|
|
|
|
|
|
|
|
def run(self):
|
|
|
|
for j in range(num_calls):
|
|
|
|
self.result = the_app.handle_ratelimit(req, self.myname,
|
|
|
|
'c', None)
|
|
|
|
|
|
|
|
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()
|
|
|
|
|
|
|
|
time_took = time.time() - begin
|
|
|
|
self.assertEquals(1.5, round(time_took, 1))
|
|
|
|
|
|
|
|
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']
|
|
|
|
rate_mid = ratelimit.RateLimitMiddleware(app, {},
|
|
|
|
logger=FakeLogger())
|
|
|
|
|
|
|
|
class a_callable(object):
|
|
|
|
|
|
|
|
def __call__(self, *args, **kwargs):
|
|
|
|
pass
|
|
|
|
resp = rate_mid.__call__(env, a_callable())
|
|
|
|
self.assert_('fake_app' == resp[0])
|
|
|
|
|
|
|
|
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())
|
|
|
|
ratelimit.http_connect = mock_http_connect(204)
|
|
|
|
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
|
2013-08-14 12:40:25 +00:00
|
|
|
self.assertEquals(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())
|
|
|
|
ratelimit.http_connect = mock_http_connect(204)
|
|
|
|
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()
|
|
|
|
self._run(make_app_call, num_calls, current_rate, check_time=False)
|
|
|
|
time_took = time.time() - begin
|
2013-08-14 12:40:25 +00:00
|
|
|
self.assertEquals(round(time_took, 1), 0) # no memcache, no limiting
|
2012-05-16 21:08:34 +00:00
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
unittest.main()
|