2012-05-16 21:08:34 +00:00
|
|
|
# Copyright (c) 2010 OpenStack, LLC.
|
|
|
|
#
|
|
|
|
# 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.
|
|
|
|
|
|
|
|
try:
|
|
|
|
import simplejson as json
|
|
|
|
except ImportError:
|
|
|
|
import json
|
|
|
|
import unittest
|
|
|
|
from contextlib import contextmanager
|
|
|
|
|
2012-09-04 14:02:19 -07:00
|
|
|
from swift.common.swob import Request, Response
|
2012-05-16 21:08:34 +00:00
|
|
|
from swift.common.middleware import staticweb
|
|
|
|
|
|
|
|
|
|
|
|
class FakeMemcache(object):
|
|
|
|
|
|
|
|
def __init__(self):
|
|
|
|
self.store = {}
|
|
|
|
|
|
|
|
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, 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, time=0):
|
2012-05-16 21:08:34 +00:00
|
|
|
self.store[key] = self.store.setdefault(key, 0) + 1
|
|
|
|
return self.store[key]
|
|
|
|
|
|
|
|
@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 __init__(self, status_headers_body_iter=None):
|
|
|
|
self.calls = 0
|
|
|
|
self.get_c4_called = False
|
|
|
|
|
|
|
|
def __call__(self, env, start_response):
|
|
|
|
self.calls += 1
|
|
|
|
if env['PATH_INFO'] == '/':
|
|
|
|
return Response(status='404 Not Found')(env, start_response)
|
|
|
|
elif env['PATH_INFO'] == '/v1':
|
|
|
|
return Response(
|
|
|
|
status='412 Precondition Failed')(env, start_response)
|
|
|
|
elif env['PATH_INFO'] == '/v1/a':
|
|
|
|
return Response(status='401 Unauthorized')(env, start_response)
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c1':
|
|
|
|
return Response(status='401 Unauthorized')(env, start_response)
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c2':
|
|
|
|
return self.listing(env, start_response,
|
|
|
|
{'x-container-read': '.r:*'})
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c2/one.txt':
|
|
|
|
return Response(status='404 Not Found')(env, start_response)
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c3':
|
|
|
|
return self.listing(env, start_response,
|
|
|
|
{'x-container-read': '.r:*',
|
|
|
|
'x-container-meta-web-index': 'index.html',
|
|
|
|
'x-container-meta-web-listings': 't'})
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c3/index.html':
|
|
|
|
return Response(status='200 Ok', body='''
|
|
|
|
<html>
|
|
|
|
<body>
|
|
|
|
<h1>Test main index.html file.</h1>
|
|
|
|
<p>Visit <a href="subdir">subdir</a>.</p>
|
|
|
|
<p>Don't visit <a href="subdir2/">subdir2</a> because it doesn't really
|
|
|
|
exist.</p>
|
|
|
|
<p>Visit <a href="subdir3">subdir3</a>.</p>
|
|
|
|
<p>Visit <a href="subdir3/subsubdir">subdir3/subsubdir</a>.</p>
|
|
|
|
</body>
|
|
|
|
</html>
|
|
|
|
''')(env, start_response)
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c3b':
|
|
|
|
return self.listing(env, start_response,
|
|
|
|
{'x-container-read': '.r:*',
|
|
|
|
'x-container-meta-web-index': 'index.html',
|
|
|
|
'x-container-meta-web-listings': 't'})
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c3b/index.html':
|
|
|
|
resp = Response(status='204 No Content')
|
|
|
|
resp.app_iter = iter([])
|
|
|
|
return resp(env, start_response)
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c3/subdir':
|
|
|
|
return Response(status='404 Not Found')(env, start_response)
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c3/subdir/':
|
|
|
|
return Response(status='404 Not Found')(env, start_response)
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c3/subdir/index.html':
|
|
|
|
return Response(status='404 Not Found')(env, start_response)
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c3/subdir3/subsubdir':
|
|
|
|
return Response(status='404 Not Found')(env, start_response)
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c3/subdir3/subsubdir/':
|
|
|
|
return Response(status='404 Not Found')(env, start_response)
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c3/subdir3/subsubdir/index.html':
|
|
|
|
return Response(status='200 Ok', body='index file')(env,
|
|
|
|
start_response)
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c3/subdirx/':
|
|
|
|
return Response(status='404 Not Found')(env, start_response)
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c3/subdirx/index.html':
|
|
|
|
return Response(status='404 Not Found')(env, start_response)
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c3/subdiry/':
|
|
|
|
return Response(status='404 Not Found')(env, start_response)
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c3/subdiry/index.html':
|
|
|
|
return Response(status='404 Not Found')(env, start_response)
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c3/subdirz':
|
|
|
|
return Response(status='404 Not Found')(env, start_response)
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c3/subdirz/index.html':
|
|
|
|
return Response(status='404 Not Found')(env, start_response)
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c3/unknown':
|
|
|
|
return Response(status='404 Not Found')(env, start_response)
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c3/unknown/index.html':
|
|
|
|
return Response(status='404 Not Found')(env, start_response)
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c4':
|
|
|
|
self.get_c4_called = True
|
|
|
|
return self.listing(env, start_response,
|
|
|
|
{'x-container-read': '.r:*',
|
|
|
|
'x-container-meta-web-index': 'index.html',
|
|
|
|
'x-container-meta-web-error': 'error.html',
|
|
|
|
'x-container-meta-web-listings': 't',
|
|
|
|
'x-container-meta-web-listings-css': 'listing.css'})
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c4/one.txt':
|
|
|
|
return Response(status='200 Ok',
|
|
|
|
headers={'x-object-meta-test': 'value'},
|
|
|
|
body='1')(env, start_response)
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c4/two.txt':
|
|
|
|
return Response(status='503 Service Unavailable')(env,
|
|
|
|
start_response)
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c4/index.html':
|
|
|
|
return Response(status='404 Not Found')(env, start_response)
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c4/subdir/':
|
|
|
|
return Response(status='404 Not Found')(env, start_response)
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c4/subdir/index.html':
|
|
|
|
return Response(status='404 Not Found')(env, start_response)
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c4/unknown':
|
|
|
|
return Response(status='404 Not Found')(env, start_response)
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c4/unknown/index.html':
|
|
|
|
return Response(status='404 Not Found')(env, start_response)
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c4/404error.html':
|
|
|
|
return Response(status='200 Ok', body='''
|
|
|
|
<html>
|
|
|
|
<body style="background: #000000; color: #ffaaaa">
|
|
|
|
<p>Chrome's 404 fancy-page sucks.</p>
|
|
|
|
<body>
|
|
|
|
<html>
|
|
|
|
'''.strip())(env, start_response)
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c5':
|
|
|
|
return self.listing(env, start_response,
|
|
|
|
{'x-container-read': '.r:*',
|
|
|
|
'x-container-meta-web-index': 'index.html',
|
|
|
|
'x-container-meta-listings': 't',
|
|
|
|
'x-container-meta-web-error': 'error.html'})
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c5/index.html':
|
|
|
|
return Response(status='503 Service Unavailable')(env,
|
|
|
|
start_response)
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c5/503error.html':
|
|
|
|
return Response(status='404 Not Found')(env, start_response)
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c5/unknown':
|
|
|
|
return Response(status='404 Not Found')(env, start_response)
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c5/unknown/index.html':
|
|
|
|
return Response(status='404 Not Found')(env, start_response)
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c5/404error.html':
|
|
|
|
return Response(status='404 Not Found')(env, start_response)
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c6':
|
|
|
|
return self.listing(env, start_response,
|
|
|
|
{'x-container-read': '.r:*',
|
|
|
|
'x-container-meta-web-listings': 't'})
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c6/subdir':
|
|
|
|
return Response(status='404 Not Found')(env, start_response)
|
|
|
|
elif env['PATH_INFO'] in ('/v1/a/c7', '/v1/a/c7/'):
|
|
|
|
return self.listing(env, start_response,
|
|
|
|
{'x-container-read': '.r:*',
|
|
|
|
'x-container-meta-web-listings': 'f'})
|
|
|
|
elif env['PATH_INFO'] in ('/v1/a/c8', '/v1/a/c8/'):
|
|
|
|
return self.listing(env, start_response,
|
|
|
|
{'x-container-read': '.r:*',
|
|
|
|
'x-container-meta-web-error': 'error.html',
|
|
|
|
'x-container-meta-web-listings': 't',
|
|
|
|
'x-container-meta-web-listings-css': \
|
|
|
|
'http://localhost/stylesheets/listing.css'})
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c8/subdir/':
|
|
|
|
return Response(status='404 Not Found')(env, start_response)
|
|
|
|
elif env['PATH_INFO'] in ('/v1/a/c9', '/v1/a/c9/'):
|
|
|
|
return self.listing(env, start_response,
|
|
|
|
{'x-container-read': '.r:*',
|
|
|
|
'x-container-meta-web-error': 'error.html',
|
|
|
|
'x-container-meta-web-listings': 't',
|
|
|
|
'x-container-meta-web-listings-css': \
|
|
|
|
'/absolute/listing.css'})
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c9/subdir/':
|
|
|
|
return Response(status='404 Not Found')(env, start_response)
|
2013-03-21 11:35:11 +09:00
|
|
|
elif env['PATH_INFO'] in ('/v1/a/c10', '/v1/a/c10/'):
|
|
|
|
return self.listing(env, start_response,
|
|
|
|
{'x-container-read': '.r:*',
|
|
|
|
'x-container-meta-web-listings': 't'})
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c10/\xe2\x98\x83/':
|
|
|
|
return Response(status='404 Not Found')(env, start_response)
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c10/\xe2\x98\x83/\xe2\x98\x83/':
|
|
|
|
return Response(status='404 Not Found')(env, start_response)
|
2012-05-16 21:08:34 +00:00
|
|
|
else:
|
|
|
|
raise Exception('Unknown path %r' % env['PATH_INFO'])
|
|
|
|
|
|
|
|
def listing(self, env, start_response, headers):
|
|
|
|
if env['PATH_INFO'] in ('/v1/a/c3', '/v1/a/c4', '/v1/a/c8', \
|
|
|
|
'/v1/a/c9') and \
|
|
|
|
env['QUERY_STRING'] == 'delimiter=/&format=json&prefix=subdir/':
|
2013-03-21 11:35:11 +09:00
|
|
|
headers.update({'X-Container-Object-Count': '12',
|
|
|
|
'X-Container-Bytes-Used': '73763',
|
2012-05-16 21:08:34 +00:00
|
|
|
'X-Container-Read': '.r:*',
|
|
|
|
'Content-Type': 'application/json; charset=utf-8'})
|
|
|
|
body = '''
|
|
|
|
[{"name":"subdir/1.txt",
|
|
|
|
"hash":"5f595114a4b3077edfac792c61ca4fe4", "bytes":20,
|
|
|
|
"content_type":"text/plain",
|
|
|
|
"last_modified":"2011-03-24T04:27:52.709100"},
|
|
|
|
{"name":"subdir/2.txt",
|
|
|
|
"hash":"c85c1dcd19cf5cbac84e6043c31bb63e", "bytes":20,
|
|
|
|
"content_type":"text/plain",
|
|
|
|
"last_modified":"2011-03-24T04:27:52.734140"},
|
|
|
|
{"subdir":"subdir3/subsubdir/"}]
|
|
|
|
'''.strip()
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c3' and env['QUERY_STRING'] == \
|
|
|
|
'delimiter=/&format=json&prefix=subdiry/':
|
2013-03-21 11:35:11 +09:00
|
|
|
headers.update({'X-Container-Object-Count': '12',
|
|
|
|
'X-Container-Bytes-Used': '73763',
|
2012-05-16 21:08:34 +00:00
|
|
|
'X-Container-Read': '.r:*',
|
|
|
|
'Content-Type': 'application/json; charset=utf-8'})
|
|
|
|
body = '[]'
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c3' and env['QUERY_STRING'] == \
|
|
|
|
'limit=1&format=json&delimiter=/&limit=1&prefix=subdirz/':
|
2013-03-21 11:35:11 +09:00
|
|
|
headers.update({'X-Container-Object-Count': '12',
|
|
|
|
'X-Container-Bytes-Used': '73763',
|
2012-05-16 21:08:34 +00:00
|
|
|
'X-Container-Read': '.r:*',
|
|
|
|
'Content-Type': 'application/json; charset=utf-8'})
|
|
|
|
body = '''
|
|
|
|
[{"name":"subdirz/1.txt",
|
|
|
|
"hash":"5f595114a4b3077edfac792c61ca4fe4", "bytes":20,
|
|
|
|
"content_type":"text/plain",
|
|
|
|
"last_modified":"2011-03-24T04:27:52.709100"}]
|
|
|
|
'''.strip()
|
|
|
|
elif env['PATH_INFO'] == '/v1/a/c6' and env['QUERY_STRING'] == \
|
|
|
|
'limit=1&format=json&delimiter=/&limit=1&prefix=subdir/':
|
2013-03-21 11:35:11 +09:00
|
|
|
headers.update({'X-Container-Object-Count': '12',
|
|
|
|
'X-Container-Bytes-Used': '73763',
|
2012-05-16 21:08:34 +00:00
|
|
|
'X-Container-Read': '.r:*',
|
|
|
|
'X-Container-Web-Listings': 't',
|
|
|
|
'Content-Type': 'application/json; charset=utf-8'})
|
|
|
|
body = '''
|
|
|
|
[{"name":"subdir/1.txt",
|
|
|
|
"hash":"5f595114a4b3077edfac792c61ca4fe4", "bytes":20,
|
|
|
|
"content_type":"text/plain",
|
|
|
|
"last_modified":"2011-03-24T04:27:52.709100"}]
|
|
|
|
'''.strip()
|
2013-03-21 11:35:11 +09:00
|
|
|
elif env['PATH_INFO'] == '/v1/a/c10' and (env['QUERY_STRING'] == \
|
|
|
|
'delimiter=/&format=json&prefix=%E2%98%83/' or
|
|
|
|
env['QUERY_STRING'] == \
|
|
|
|
'delimiter=/&format=json&prefix=%E2%98%83/%E2%98%83/'):
|
|
|
|
headers.update({'X-Container-Object-Count': '12',
|
|
|
|
'X-Container-Bytes-Used': '73763',
|
|
|
|
'X-Container-Read': '.r:*',
|
|
|
|
'X-Container-Web-Listings': 't',
|
|
|
|
'Content-Type': 'application/json; charset=utf-8'})
|
|
|
|
body = '''
|
|
|
|
[{"name":"\u2603/\u2603/one.txt",
|
|
|
|
"hash":"73f1dd69bacbf0847cc9cffa3c6b23a1", "bytes":22,
|
|
|
|
"content_type":"text/plain",
|
|
|
|
"last_modified":"2011-03-24T04:27:52.709100"},
|
|
|
|
{"subdir":"\u2603/\u2603/"}]
|
|
|
|
'''.strip()
|
2012-05-16 21:08:34 +00:00
|
|
|
elif 'prefix=' in env['QUERY_STRING']:
|
|
|
|
return Response(status='204 No Content')(env, start_response)
|
|
|
|
elif 'format=json' in env['QUERY_STRING']:
|
2013-03-21 11:35:11 +09:00
|
|
|
headers.update({'X-Container-Object-Count': '12',
|
|
|
|
'X-Container-Bytes-Used': '73763',
|
2012-05-16 21:08:34 +00:00
|
|
|
'Content-Type': 'application/json; charset=utf-8'})
|
|
|
|
body = '''
|
|
|
|
[{"name":"401error.html",
|
|
|
|
"hash":"893f8d80692a4d3875b45be8f152ad18", "bytes":110,
|
|
|
|
"content_type":"text/html",
|
|
|
|
"last_modified":"2011-03-24T04:27:52.713710"},
|
|
|
|
{"name":"404error.html",
|
|
|
|
"hash":"62dcec9c34ed2b347d94e6ca707aff8c", "bytes":130,
|
|
|
|
"content_type":"text/html",
|
|
|
|
"last_modified":"2011-03-24T04:27:52.720850"},
|
|
|
|
{"name":"index.html",
|
|
|
|
"hash":"8b469f2ca117668a5131fe9ee0815421", "bytes":347,
|
|
|
|
"content_type":"text/html",
|
|
|
|
"last_modified":"2011-03-24T04:27:52.683590"},
|
|
|
|
{"name":"listing.css",
|
|
|
|
"hash":"7eab5d169f3fcd06a08c130fa10c5236", "bytes":17,
|
|
|
|
"content_type":"text/css",
|
|
|
|
"last_modified":"2011-03-24T04:27:52.721610"},
|
|
|
|
{"name":"one.txt", "hash":"73f1dd69bacbf0847cc9cffa3c6b23a1",
|
|
|
|
"bytes":22, "content_type":"text/plain",
|
|
|
|
"last_modified":"2011-03-24T04:27:52.722270"},
|
|
|
|
{"name":"subdir/1.txt",
|
|
|
|
"hash":"5f595114a4b3077edfac792c61ca4fe4", "bytes":20,
|
|
|
|
"content_type":"text/plain",
|
|
|
|
"last_modified":"2011-03-24T04:27:52.709100"},
|
|
|
|
{"name":"subdir/2.txt",
|
|
|
|
"hash":"c85c1dcd19cf5cbac84e6043c31bb63e", "bytes":20,
|
|
|
|
"content_type":"text/plain",
|
|
|
|
"last_modified":"2011-03-24T04:27:52.734140"},
|
|
|
|
{"name":"subdir/\u2603.txt",
|
|
|
|
"hash":"7337d028c093130898d937c319cc9865", "bytes":72981,
|
|
|
|
"content_type":"text/plain",
|
|
|
|
"last_modified":"2011-03-24T04:27:52.735460"},
|
|
|
|
{"name":"subdir2", "hash":"d41d8cd98f00b204e9800998ecf8427e",
|
|
|
|
"bytes":0, "content_type":"text/directory",
|
|
|
|
"last_modified":"2011-03-24T04:27:52.676690"},
|
|
|
|
{"name":"subdir3/subsubdir/index.html",
|
|
|
|
"hash":"04eea67110f883b1a5c97eb44ccad08c", "bytes":72,
|
|
|
|
"content_type":"text/html",
|
|
|
|
"last_modified":"2011-03-24T04:27:52.751260"},
|
|
|
|
{"name":"two.txt", "hash":"10abb84c63a5cff379fdfd6385918833",
|
|
|
|
"bytes":22, "content_type":"text/plain",
|
2013-03-21 11:35:11 +09:00
|
|
|
"last_modified":"2011-03-24T04:27:52.825110"},
|
|
|
|
{"name":"\u2603/\u2603/one.txt",
|
|
|
|
"hash":"73f1dd69bacbf0847cc9cffa3c6b23a1", "bytes":22,
|
|
|
|
"content_type":"text/plain",
|
|
|
|
"last_modified":"2011-03-24T04:27:52.935560"}]
|
2012-05-16 21:08:34 +00:00
|
|
|
'''.strip()
|
|
|
|
else:
|
2013-03-21 11:35:11 +09:00
|
|
|
headers.update({'X-Container-Object-Count': '12',
|
|
|
|
'X-Container-Bytes-Used': '73763',
|
2012-05-16 21:08:34 +00:00
|
|
|
'Content-Type': 'text/plain; charset=utf-8'})
|
|
|
|
body = '\n'.join(['401error.html', '404error.html', 'index.html',
|
|
|
|
'listing.css', 'one.txt', 'subdir/1.txt',
|
|
|
|
'subdir/2.txt', u'subdir/\u2603.txt', 'subdir2',
|
2013-03-21 11:35:11 +09:00
|
|
|
'subdir3/subsubdir/index.html', 'two.txt',
|
|
|
|
u'\u2603/\u2603/one.txt'])
|
2012-05-16 21:08:34 +00:00
|
|
|
return Response(status='200 Ok', headers=headers,
|
|
|
|
body=body)(env, start_response)
|
|
|
|
|
|
|
|
|
|
|
|
class TestStaticWeb(unittest.TestCase):
|
|
|
|
|
|
|
|
def setUp(self):
|
|
|
|
self.app = FakeApp()
|
|
|
|
self.test_staticweb = staticweb.filter_factory({})(self.app)
|
|
|
|
|
|
|
|
def test_app_set(self):
|
|
|
|
app = FakeApp()
|
|
|
|
sw = staticweb.filter_factory({})(app)
|
|
|
|
self.assertEquals(sw.app, app)
|
|
|
|
|
|
|
|
def test_conf_set(self):
|
|
|
|
conf = {'blah': 1}
|
|
|
|
sw = staticweb.filter_factory(conf)(FakeApp())
|
|
|
|
self.assertEquals(sw.conf, conf)
|
|
|
|
|
|
|
|
def test_cache_timeout_unset(self):
|
|
|
|
sw = staticweb.filter_factory({})(FakeApp())
|
|
|
|
self.assertEquals(sw.cache_timeout, 300)
|
|
|
|
|
|
|
|
def test_cache_timeout_set(self):
|
|
|
|
sw = staticweb.filter_factory({'cache_timeout': '1'})(FakeApp())
|
|
|
|
self.assertEquals(sw.cache_timeout, 1)
|
|
|
|
|
|
|
|
def test_root(self):
|
|
|
|
resp = Request.blank('/').get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 404)
|
|
|
|
|
|
|
|
def test_version(self):
|
|
|
|
resp = Request.blank('/v1').get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 412)
|
|
|
|
|
|
|
|
def test_account(self):
|
|
|
|
resp = Request.blank('/v1/a').get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 401)
|
|
|
|
|
|
|
|
def test_container1(self):
|
|
|
|
resp = Request.blank('/v1/a/c1').get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 401)
|
|
|
|
|
|
|
|
def test_container1_web_mode_explicitly_off(self):
|
|
|
|
resp = Request.blank('/v1/a/c1',
|
|
|
|
headers={'x-web-mode': 'false'}).get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 401)
|
|
|
|
|
|
|
|
def test_container1_web_mode_explicitly_on(self):
|
|
|
|
resp = Request.blank('/v1/a/c1',
|
|
|
|
headers={'x-web-mode': 'true'}).get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 404)
|
|
|
|
|
|
|
|
def test_container2(self):
|
|
|
|
resp = Request.blank('/v1/a/c2').get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 200)
|
|
|
|
self.assertEquals(resp.content_type, 'text/plain')
|
|
|
|
self.assertEquals(len(resp.body.split('\n')),
|
|
|
|
int(resp.headers['x-container-object-count']))
|
|
|
|
|
|
|
|
def test_container2_web_mode_explicitly_off(self):
|
|
|
|
resp = Request.blank('/v1/a/c2',
|
|
|
|
headers={'x-web-mode': 'false'}).get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 200)
|
|
|
|
self.assertEquals(resp.content_type, 'text/plain')
|
|
|
|
self.assertEquals(len(resp.body.split('\n')),
|
|
|
|
int(resp.headers['x-container-object-count']))
|
|
|
|
|
|
|
|
def test_container2_web_mode_explicitly_on(self):
|
|
|
|
resp = Request.blank('/v1/a/c2',
|
|
|
|
headers={'x-web-mode': 'true'}).get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 404)
|
|
|
|
|
|
|
|
def test_container2onetxt(self):
|
|
|
|
resp = Request.blank(
|
|
|
|
'/v1/a/c2/one.txt').get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 404)
|
|
|
|
|
|
|
|
def test_container2json(self):
|
|
|
|
resp = Request.blank(
|
|
|
|
'/v1/a/c2?format=json').get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 200)
|
|
|
|
self.assertEquals(resp.content_type, 'application/json')
|
|
|
|
self.assertEquals(len(json.loads(resp.body)),
|
|
|
|
int(resp.headers['x-container-object-count']))
|
|
|
|
|
|
|
|
def test_container2json_web_mode_explicitly_off(self):
|
|
|
|
resp = Request.blank('/v1/a/c2?format=json',
|
|
|
|
headers={'x-web-mode': 'false'}).get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 200)
|
|
|
|
self.assertEquals(resp.content_type, 'application/json')
|
|
|
|
self.assertEquals(len(json.loads(resp.body)),
|
|
|
|
int(resp.headers['x-container-object-count']))
|
|
|
|
|
|
|
|
def test_container2json_web_mode_explicitly_on(self):
|
|
|
|
resp = Request.blank('/v1/a/c2?format=json',
|
|
|
|
headers={'x-web-mode': 'true'}).get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 404)
|
|
|
|
|
|
|
|
def test_container3(self):
|
|
|
|
resp = Request.blank('/v1/a/c3').get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 301)
|
|
|
|
self.assertEquals(resp.headers['location'],
|
|
|
|
'http://localhost/v1/a/c3/')
|
|
|
|
|
|
|
|
def test_container3indexhtml(self):
|
|
|
|
resp = Request.blank('/v1/a/c3/').get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 200)
|
|
|
|
self.assert_('Test main index.html file.' in resp.body)
|
|
|
|
|
|
|
|
def test_container3subsubdir(self):
|
|
|
|
resp = Request.blank(
|
|
|
|
'/v1/a/c3/subdir3/subsubdir').get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 301)
|
|
|
|
|
|
|
|
def test_container3subsubdircontents(self):
|
|
|
|
resp = Request.blank(
|
|
|
|
'/v1/a/c3/subdir3/subsubdir/').get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 200)
|
|
|
|
self.assertEquals(resp.body, 'index file')
|
|
|
|
|
|
|
|
def test_container3subdir(self):
|
|
|
|
resp = Request.blank(
|
|
|
|
'/v1/a/c3/subdir/').get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 200)
|
|
|
|
self.assert_('Listing of /v1/a/c3/subdir/' in resp.body)
|
|
|
|
self.assert_('</style>' in resp.body)
|
|
|
|
self.assert_('<link' not in resp.body)
|
|
|
|
self.assert_('listing.css' not in resp.body)
|
|
|
|
|
|
|
|
def test_container3subdirx(self):
|
|
|
|
resp = Request.blank(
|
|
|
|
'/v1/a/c3/subdirx/').get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 404)
|
|
|
|
|
|
|
|
def test_container3subdiry(self):
|
|
|
|
resp = Request.blank(
|
|
|
|
'/v1/a/c3/subdiry/').get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 404)
|
|
|
|
|
|
|
|
def test_container3subdirz(self):
|
|
|
|
resp = Request.blank(
|
|
|
|
'/v1/a/c3/subdirz').get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 301)
|
|
|
|
|
|
|
|
def test_container3unknown(self):
|
|
|
|
resp = Request.blank(
|
|
|
|
'/v1/a/c3/unknown').get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 404)
|
|
|
|
self.assert_("Chrome's 404 fancy-page sucks." not in resp.body)
|
|
|
|
|
|
|
|
def test_container3bindexhtml(self):
|
|
|
|
resp = Request.blank('/v1/a/c3b/').get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 204)
|
|
|
|
self.assertEquals(resp.body, '')
|
|
|
|
|
|
|
|
def test_container4indexhtml(self):
|
|
|
|
resp = Request.blank('/v1/a/c4/').get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 200)
|
|
|
|
self.assert_('Listing of /v1/a/c4/' in resp.body)
|
|
|
|
self.assert_('href="listing.css"' in resp.body)
|
|
|
|
|
|
|
|
def test_container4indexhtmlauthed(self):
|
|
|
|
resp = Request.blank('/v1/a/c4').get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 301)
|
|
|
|
resp = Request.blank('/v1/a/c4',
|
|
|
|
environ={'REMOTE_USER': 'authed'}).get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 200)
|
|
|
|
resp = Request.blank('/v1/a/c4', headers={'x-web-mode': 't'},
|
|
|
|
environ={'REMOTE_USER': 'authed'}).get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 301)
|
|
|
|
|
|
|
|
def test_container4unknown(self):
|
|
|
|
resp = Request.blank(
|
|
|
|
'/v1/a/c4/unknown').get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 404)
|
|
|
|
self.assert_("Chrome's 404 fancy-page sucks." in resp.body)
|
|
|
|
|
|
|
|
def test_container4unknown_memcache(self):
|
|
|
|
fake_memcache = FakeMemcache()
|
|
|
|
self.assertEquals(fake_memcache.store, {})
|
|
|
|
resp = Request.blank('/v1/a/c4',
|
|
|
|
environ={'swift.cache': fake_memcache}
|
|
|
|
).get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 301)
|
|
|
|
self.assertEquals(fake_memcache.store,
|
|
|
|
{'/staticweb/v1/a/c4':
|
|
|
|
('index.html', 'error.html', 't', 'listing.css')})
|
|
|
|
self.assert_(self.test_staticweb.app.get_c4_called)
|
|
|
|
self.test_staticweb.app.get_c4_called = False
|
|
|
|
resp = Request.blank('/v1/a/c4',
|
|
|
|
environ={'swift.cache': fake_memcache}
|
|
|
|
).get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 301)
|
|
|
|
self.assert_(not self.test_staticweb.app.get_c4_called)
|
|
|
|
self.assertEquals(fake_memcache.store,
|
|
|
|
{'/staticweb/v1/a/c4':
|
|
|
|
('index.html', 'error.html', 't', 'listing.css')})
|
|
|
|
resp = Request.blank('/v1/a/c4',
|
|
|
|
environ={'swift.cache': fake_memcache, 'REQUEST_METHOD': 'PUT'}
|
|
|
|
).get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 200)
|
|
|
|
self.assertEquals(fake_memcache.store, {})
|
|
|
|
resp = Request.blank('/v1/a/c4',
|
|
|
|
environ={'swift.cache': fake_memcache}
|
|
|
|
).get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 301)
|
|
|
|
self.assertEquals(fake_memcache.store,
|
|
|
|
{'/staticweb/v1/a/c4':
|
|
|
|
('index.html', 'error.html', 't', 'listing.css')})
|
|
|
|
resp = Request.blank('/v1/a/c4',
|
|
|
|
environ={'swift.cache': fake_memcache, 'REQUEST_METHOD': 'POST'}
|
|
|
|
).get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 200)
|
|
|
|
self.assertEquals(fake_memcache.store, {})
|
|
|
|
|
|
|
|
def test_container4subdir(self):
|
|
|
|
resp = Request.blank(
|
|
|
|
'/v1/a/c4/subdir/').get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 200)
|
|
|
|
self.assert_('Listing of /v1/a/c4/subdir/' in resp.body)
|
|
|
|
self.assert_('</style>' not in resp.body)
|
|
|
|
self.assert_('<link' in resp.body)
|
|
|
|
self.assert_('href="../listing.css"' in resp.body)
|
|
|
|
self.assertEquals(resp.headers['content-type'],
|
|
|
|
'text/html; charset=UTF-8')
|
|
|
|
|
|
|
|
def test_container4onetxt(self):
|
|
|
|
resp = Request.blank(
|
|
|
|
'/v1/a/c4/one.txt').get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 200)
|
|
|
|
|
|
|
|
def test_container4twotxt(self):
|
|
|
|
resp = Request.blank(
|
|
|
|
'/v1/a/c4/two.txt').get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 503)
|
|
|
|
|
|
|
|
def test_container5indexhtml(self):
|
|
|
|
resp = Request.blank('/v1/a/c5/').get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 503)
|
|
|
|
|
|
|
|
def test_container5unknown(self):
|
|
|
|
resp = Request.blank(
|
|
|
|
'/v1/a/c5/unknown').get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 404)
|
|
|
|
self.assert_("Chrome's 404 fancy-page sucks." not in resp.body)
|
|
|
|
|
|
|
|
def test_container6subdir(self):
|
|
|
|
resp = Request.blank(
|
|
|
|
'/v1/a/c6/subdir').get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 301)
|
|
|
|
|
|
|
|
def test_container7listing(self):
|
|
|
|
resp = Request.blank('/v1/a/c7/').get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 404)
|
|
|
|
|
|
|
|
def test_container8listingcss(self):
|
|
|
|
resp = Request.blank(
|
|
|
|
'/v1/a/c8/').get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 200)
|
|
|
|
self.assert_('Listing of /v1/a/c8/' in resp.body)
|
|
|
|
self.assert_('<link' in resp.body)
|
|
|
|
self.assert_(
|
|
|
|
'href="http://localhost/stylesheets/listing.css"' in resp.body)
|
|
|
|
|
|
|
|
def test_container8subdirlistingcss(self):
|
|
|
|
resp = Request.blank(
|
|
|
|
'/v1/a/c8/subdir/').get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 200)
|
|
|
|
self.assert_('Listing of /v1/a/c8/subdir/' in resp.body)
|
|
|
|
self.assert_('<link' in resp.body)
|
|
|
|
self.assert_(
|
|
|
|
'href="http://localhost/stylesheets/listing.css"' in resp.body)
|
|
|
|
|
|
|
|
def test_container9listingcss(self):
|
|
|
|
resp = Request.blank(
|
|
|
|
'/v1/a/c9/').get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 200)
|
|
|
|
self.assert_('Listing of /v1/a/c9/' in resp.body)
|
|
|
|
self.assert_('<link' in resp.body)
|
|
|
|
self.assert_('href="/absolute/listing.css"' in resp.body)
|
|
|
|
|
|
|
|
def test_container9subdirlistingcss(self):
|
|
|
|
resp = Request.blank(
|
|
|
|
'/v1/a/c9/subdir/').get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 200)
|
|
|
|
self.assert_('Listing of /v1/a/c9/subdir/' in resp.body)
|
|
|
|
self.assert_('<link' in resp.body)
|
|
|
|
self.assert_('href="/absolute/listing.css"' in resp.body)
|
|
|
|
|
2013-03-21 11:35:11 +09:00
|
|
|
def test_container10unicodesubdirlisting(self):
|
|
|
|
resp = Request.blank(
|
|
|
|
'/v1/a/c10/').get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 200)
|
|
|
|
self.assert_('Listing of /v1/a/c10/' in resp.body)
|
|
|
|
resp = Request.blank(
|
|
|
|
'/v1/a/c10/\xe2\x98\x83/').get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 200)
|
|
|
|
self.assert_('Listing of /v1/a/c10/\xe2\x98\x83/' in resp.body)
|
|
|
|
resp = Request.blank(
|
|
|
|
'/v1/a/c10/\xe2\x98\x83/\xe2\x98\x83/'
|
|
|
|
).get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 200)
|
|
|
|
self.assert_(
|
|
|
|
'Listing of /v1/a/c10/\xe2\x98\x83/\xe2\x98\x83/' in resp.body)
|
|
|
|
|
2012-05-16 21:08:34 +00:00
|
|
|
def test_subrequest_once_if_possible(self):
|
|
|
|
resp = Request.blank(
|
|
|
|
'/v1/a/c4/one.txt').get_response(self.test_staticweb)
|
|
|
|
self.assertEquals(resp.status_int, 200)
|
|
|
|
self.assertEquals(resp.headers['x-object-meta-test'], 'value')
|
|
|
|
self.assertEquals(resp.body, '1')
|
|
|
|
self.assertEquals(self.app.calls, 1)
|
|
|
|
|
2012-07-29 18:17:14 +00:00
|
|
|
|
2012-05-16 21:08:34 +00:00
|
|
|
if __name__ == '__main__':
|
|
|
|
unittest.main()
|