2012-03-19 13:45:34 -05:00
|
|
|
# Copyright (c) 2010-2012 OpenStack, LLC.
|
2010-07-12 17:03:45 -05: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
|
2012-01-06 21:27:44 +00:00
|
|
|
from ConfigParser import NoSectionError, NoOptionError
|
2010-07-12 17:03:45 -05:00
|
|
|
|
2010-08-24 13:58:32 +00:00
|
|
|
from swift.common.middleware import memcache
|
2010-08-20 00:42:38 +00:00
|
|
|
from swift.common.memcached import MemcacheRing
|
2012-09-04 14:02:19 -07:00
|
|
|
from swift.common.swob import Request
|
|
|
|
|
2010-07-12 17:03:45 -05:00
|
|
|
|
2010-08-20 00:42:38 +00:00
|
|
|
class FakeApp(object):
|
|
|
|
def __call__(self, env, start_response):
|
|
|
|
return env
|
2010-07-12 17:03:45 -05:00
|
|
|
|
2012-01-06 21:27:44 +00:00
|
|
|
|
|
|
|
class ExcConfigParser(object):
|
|
|
|
|
|
|
|
def read(self, path):
|
|
|
|
raise Exception('read called with %r' % path)
|
|
|
|
|
|
|
|
|
|
|
|
class EmptyConfigParser(object):
|
|
|
|
|
|
|
|
def read(self, path):
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
|
|
|
class SetConfigParser(object):
|
|
|
|
|
|
|
|
def read(self, path):
|
|
|
|
return True
|
|
|
|
|
|
|
|
def get(self, section, option):
|
|
|
|
if section == 'memcache':
|
|
|
|
if option == 'memcache_servers':
|
|
|
|
return '1.2.3.4:5'
|
Do not use pickle for serialization in memcache, but JSON
We don't want to use pickle as it can execute arbitrary code. JSON is
safer. However, note that it supports serialization for only some
specific subset of object types; this should be enough for what we need,
though.
To avoid issues on upgrades (unability to read pickled values, and cache
poisoning for old servers not understanding JSON), we add a
memcache_serialization_support configuration option, with the following
values:
0 = older, insecure pickle serialization
1 = json serialization but pickles can still be read (still insecure)
2 = json serialization only (secure and the default)
To avoid an instant full cache flush, existing installations should
upgrade with 0, then set to 1 and reload, then after some time (24
hours) set to 2 and reload. Support for 0 and 1 will be removed in
future versions.
Part of bug 1006414.
Change-Id: Id7d6d547b103b4f23ebf5be98b88f09ec6027ce4
2012-06-21 14:37:41 +02:00
|
|
|
elif option == 'memcache_serialization_support':
|
2012-09-18 18:24:47 +00:00
|
|
|
return '1'
|
2012-01-06 21:27:44 +00:00
|
|
|
else:
|
|
|
|
raise NoOptionError(option)
|
|
|
|
else:
|
|
|
|
raise NoSectionError(option)
|
|
|
|
|
|
|
|
|
2010-08-20 00:42:38 +00:00
|
|
|
def start_response(*args):
|
|
|
|
pass
|
2010-07-12 17:03:45 -05:00
|
|
|
|
2010-08-20 00:42:38 +00:00
|
|
|
class TestCacheMiddleware(unittest.TestCase):
|
2010-07-12 17:03:45 -05:00
|
|
|
|
2010-08-20 00:42:38 +00:00
|
|
|
def setUp(self):
|
2010-08-24 13:58:32 +00:00
|
|
|
self.app = memcache.MemcacheMiddleware(FakeApp(), {})
|
2010-08-20 00:42:38 +00:00
|
|
|
|
|
|
|
def test_cache_middleware(self):
|
|
|
|
req = Request.blank('/something', environ={'REQUEST_METHOD': 'GET'})
|
|
|
|
resp = self.app(req.environ, start_response)
|
|
|
|
self.assertTrue('swift.cache' in resp)
|
|
|
|
self.assertTrue(isinstance(resp['swift.cache'], MemcacheRing))
|
2010-07-12 17:03:45 -05:00
|
|
|
|
2012-01-06 21:27:44 +00:00
|
|
|
def test_conf_default_read(self):
|
|
|
|
orig_parser = memcache.ConfigParser
|
|
|
|
memcache.ConfigParser = ExcConfigParser
|
|
|
|
exc = None
|
|
|
|
try:
|
|
|
|
app = memcache.MemcacheMiddleware(FakeApp(), {})
|
|
|
|
except Exception, err:
|
|
|
|
exc = err
|
|
|
|
finally:
|
|
|
|
memcache.ConfigParser = orig_parser
|
|
|
|
self.assertEquals(str(exc),
|
|
|
|
"read called with '/etc/swift/memcache.conf'")
|
|
|
|
|
|
|
|
def test_conf_set_no_read(self):
|
|
|
|
orig_parser = memcache.ConfigParser
|
|
|
|
memcache.ConfigParser = ExcConfigParser
|
|
|
|
exc = None
|
|
|
|
try:
|
|
|
|
app = memcache.MemcacheMiddleware(
|
Do not use pickle for serialization in memcache, but JSON
We don't want to use pickle as it can execute arbitrary code. JSON is
safer. However, note that it supports serialization for only some
specific subset of object types; this should be enough for what we need,
though.
To avoid issues on upgrades (unability to read pickled values, and cache
poisoning for old servers not understanding JSON), we add a
memcache_serialization_support configuration option, with the following
values:
0 = older, insecure pickle serialization
1 = json serialization but pickles can still be read (still insecure)
2 = json serialization only (secure and the default)
To avoid an instant full cache flush, existing installations should
upgrade with 0, then set to 1 and reload, then after some time (24
hours) set to 2 and reload. Support for 0 and 1 will be removed in
future versions.
Part of bug 1006414.
Change-Id: Id7d6d547b103b4f23ebf5be98b88f09ec6027ce4
2012-06-21 14:37:41 +02:00
|
|
|
FakeApp(), {'memcache_servers': '1.2.3.4:5',
|
|
|
|
'memcache_serialization_support': '2'})
|
2012-01-06 21:27:44 +00:00
|
|
|
except Exception, err:
|
|
|
|
exc = err
|
|
|
|
finally:
|
|
|
|
memcache.ConfigParser = orig_parser
|
|
|
|
self.assertEquals(exc, None)
|
|
|
|
|
|
|
|
def test_conf_default(self):
|
|
|
|
orig_parser = memcache.ConfigParser
|
|
|
|
memcache.ConfigParser = EmptyConfigParser
|
|
|
|
try:
|
|
|
|
app = memcache.MemcacheMiddleware(FakeApp(), {})
|
|
|
|
finally:
|
|
|
|
memcache.ConfigParser = orig_parser
|
|
|
|
self.assertEquals(app.memcache_servers, '127.0.0.1:11211')
|
2012-09-18 18:24:47 +00:00
|
|
|
self.assertEquals(app.memcache._allow_pickle, False)
|
|
|
|
self.assertEquals(app.memcache._allow_unpickle, False)
|
2012-01-06 21:27:44 +00:00
|
|
|
|
|
|
|
def test_conf_from_extra_conf(self):
|
|
|
|
orig_parser = memcache.ConfigParser
|
|
|
|
memcache.ConfigParser = SetConfigParser
|
|
|
|
try:
|
|
|
|
app = memcache.MemcacheMiddleware(FakeApp(), {})
|
|
|
|
finally:
|
|
|
|
memcache.ConfigParser = orig_parser
|
|
|
|
self.assertEquals(app.memcache_servers, '1.2.3.4:5')
|
2012-09-18 18:24:47 +00:00
|
|
|
self.assertEquals(app.memcache._allow_pickle, False)
|
|
|
|
self.assertEquals(app.memcache._allow_unpickle, True)
|
2012-01-06 21:27:44 +00:00
|
|
|
|
|
|
|
def test_conf_from_inline_conf(self):
|
|
|
|
orig_parser = memcache.ConfigParser
|
|
|
|
memcache.ConfigParser = SetConfigParser
|
|
|
|
try:
|
|
|
|
app = memcache.MemcacheMiddleware(
|
2012-09-18 18:24:47 +00:00
|
|
|
FakeApp(),
|
|
|
|
{'memcache_servers': '6.7.8.9:10',
|
|
|
|
'serialization_format': '0'})
|
2012-01-06 21:27:44 +00:00
|
|
|
finally:
|
|
|
|
memcache.ConfigParser = orig_parser
|
|
|
|
self.assertEquals(app.memcache_servers, '6.7.8.9:10')
|
2012-09-18 18:24:47 +00:00
|
|
|
self.assertEquals(app.memcache._allow_pickle, False)
|
|
|
|
self.assertEquals(app.memcache._allow_unpickle, True)
|
2012-01-06 21:27:44 +00:00
|
|
|
|
|
|
|
|
2010-07-12 17:03:45 -05:00
|
|
|
if __name__ == '__main__':
|
|
|
|
unittest.main()
|