Refactor Bulk middleware to handle long running requests

Change-Id: I8ea0ff86518d453597faae44ec3918298e2d5147
This commit is contained in:
David Goetz 2013-04-30 14:45:46 -07:00
parent 407e08fa30
commit af2607c457
7 changed files with 327 additions and 224 deletions

View File

@ -362,8 +362,9 @@ use = egg:swift#proxy_logging
[filter:bulk] [filter:bulk]
use = egg:swift#bulk use = egg:swift#bulk
# max_containers_per_extraction = 10000 # max_containers_per_extraction = 10000
# max_failed_files = 1000 # max_failed_extractions = 1000
# max_deletes_per_request = 1000 # max_deletes_per_request = 10000
# yield_frequency = 60
# Note: Put after auth in the pipeline. # Note: Put after auth in the pipeline.
[filter:container-quotas] [filter:container-quotas]

View File

@ -16,11 +16,12 @@
import tarfile import tarfile
from urllib import quote, unquote from urllib import quote, unquote
from xml.sax import saxutils from xml.sax import saxutils
from time import time
from swift.common.swob import Request, HTTPBadGateway, \ from swift.common.swob import Request, HTTPBadGateway, \
HTTPCreated, HTTPBadRequest, HTTPNotFound, HTTPUnauthorized, HTTPOk, \ HTTPCreated, HTTPBadRequest, HTTPNotFound, HTTPUnauthorized, HTTPOk, \
HTTPPreconditionFailed, HTTPRequestEntityTooLarge, HTTPNotAcceptable, \ HTTPPreconditionFailed, HTTPRequestEntityTooLarge, HTTPNotAcceptable, \
HTTPLengthRequired, wsgify HTTPLengthRequired, HTTPException, HTTPServerError, wsgify
from swift.common.utils import json from swift.common.utils import json, get_logger
from swift.common.constraints import check_utf8, MAX_FILE_SIZE from swift.common.constraints import check_utf8, MAX_FILE_SIZE
from swift.common.http import HTTP_BAD_REQUEST, HTTP_UNAUTHORIZED, \ from swift.common.http import HTTP_BAD_REQUEST, HTTP_UNAUTHORIZED, \
HTTP_NOT_FOUND HTTP_NOT_FOUND
@ -44,25 +45,18 @@ ACCEPTABLE_FORMATS = ['text/plain', 'application/json', 'application/xml',
def get_response_body(data_format, data_dict, error_list): def get_response_body(data_format, data_dict, error_list):
""" """
Returns a properly formatted response body according to format. Returns a properly formatted response body according to format. Handles
json and xml, otherwise will return text/plain. Note: xml response does not
include xml declaration.
:params data_format: resulting format :params data_format: resulting format
:params data_dict: generated data about results. :params data_dict: generated data about results.
:params error_list: list of quoted filenames that failed :params error_list: list of quoted filenames that failed
""" """
if data_format == 'text/plain':
output = ''
for key in sorted(data_dict.keys()):
output += '%s: %s\n' % (key, data_dict[key])
output += 'Errors:\n'
output += '\n'.join(
['%s, %s' % (name, status)
for name, status in error_list])
return output
if data_format == 'application/json': if data_format == 'application/json':
data_dict['Errors'] = error_list data_dict['Errors'] = error_list
return json.dumps(data_dict) return json.dumps(data_dict)
if data_format.endswith('/xml'): if data_format and data_format.endswith('/xml'):
output = '<?xml version="1.0" encoding="UTF-8"?>\n<delete>\n' output = '<delete>\n'
for key in sorted(data_dict.keys()): for key in sorted(data_dict.keys()):
xml_key = key.replace(' ', '_').lower() xml_key = key.replace(' ', '_').lower()
output += '<%s>%s</%s>\n' % (xml_key, data_dict[key], xml_key) output += '<%s>%s</%s>\n' % (xml_key, data_dict[key], xml_key)
@ -74,7 +68,15 @@ def get_response_body(data_format, data_dict, error_list):
name, status in error_list]) name, status in error_list])
output += '</errors>\n</delete>\n' output += '</errors>\n</delete>\n'
return output return output
raise HTTPNotAcceptable('Invalid output type')
output = ''
for key in sorted(data_dict.keys()):
output += '%s: %s\n' % (key, data_dict[key])
output += 'Errors:\n'
output += '\n'.join(
['%s, %s' % (name, status)
for name, status in error_list])
return output
class Bulk(object): class Bulk(object):
@ -106,13 +108,31 @@ class Bulk(object):
Only regular files will be uploaded. Empty directories, symlinks, etc will Only regular files will be uploaded. Empty directories, symlinks, etc will
not be uploaded. not be uploaded.
If all valid files were uploaded successfully will return an HTTPCreated The response from bulk operations functions differently from other swift
response. If any files failed to be created will return an HTTPBadGateway responses. This is because a short request body sent from the client could
response. In both cases the response body will specify the number of files result in many operations on the proxy server and precautions need to be
successfully uploaded and a list of the files that failed. The return body made to prevent the request from timing out due to lack of activity. To
will be formatted in the way specified in the request's Accept header. this end, the client will always receive a 200 Ok response, regardless of
Acceptable formats are text/plain, application/json, application/xml, and the actual success of the call. The body of the response must be parsed to
text/xml. determine the actual success of the operation. In addition to this the
client may receive zero or more whitespace characters prepended to the
actual response body while the proxy server is completing the request.
The format of the response body defaults to text/plain but can be either
json or xml depending on the Accept header. Acceptable formats are
text/plain, application/json, application/xml, and text/xml. An example
body is as follows:
{"Response Code": "201 Created",
"Response Body": "",
"Errors": [],
"Number Files Created": 10}
If all valid files were uploaded successfully the Response Code will be a
201 Created. If any files failed to be created the response code
corresponds to the subrequest's error. Possible codes are 400, 401, 502 (on
server errors), etc. In both cases the response body will specify the
number of files successfully uploaded and a list of the files that failed.
There are proxy logs created for each file (which becomes a subrequest) in There are proxy logs created for each file (which becomes a subrequest) in
the tar. The subrequest's proxy log will have a swift.source set to "EA" the tar. The subrequest's proxy log will have a swift.source set to "EA"
@ -127,7 +147,7 @@ class Bulk(object):
single request. Responds to DELETE requests with query parameter single request. Responds to DELETE requests with query parameter
?bulk-delete set. The Content-Type should be set to text/plain. ?bulk-delete set. The Content-Type should be set to text/plain.
The body of the DELETE request will be a newline separated list of url The body of the DELETE request will be a newline separated list of url
encoded objects to delete. You can only delete 1000 (configurable) objects encoded objects to delete. You can delete 10,000 (configurable) objects
per request. The objects specified in the DELETE request body must be URL per request. The objects specified in the DELETE request body must be URL
encoded and in the form: encoded and in the form:
@ -137,10 +157,21 @@ class Bulk(object):
/container_name /container_name
If all items were successfully deleted (or did not exist), will return an The response is similar to bulk deletes as in every response will be a 200
HTTPOk. If any failed to delete, will return an HTTPBadGateway. In Ok and you must parse the response body for acutal results. An example
both cases the response body will specify the number of items response is:
successfully deleted, not found, and a list of those that failed.
{"Number Not Found": 0,
"Response Code": "200 OK",
"Response Body": "",
"Errors": [],
"Number Deleted": 6}
If all items were successfully deleted (or did not exist), the Response
Code will be a 200 Ok. If any failed to delete, the response code
corresponds to the subrequest's error. Possible codes are 400, 401, 502 (on
server errors), etc. In all cases the response body will specify the number
of items successfully deleted, not found, and a list of those that failed.
The return body will be formatted in the way specified in the request's The return body will be formatted in the way specified in the request's
Accept header. Acceptable formats are text/plain, application/json, Accept header. Acceptable formats are text/plain, application/json,
application/xml, and text/xml. application/xml, and text/xml.
@ -155,12 +186,14 @@ class Bulk(object):
def __init__(self, app, conf): def __init__(self, app, conf):
self.app = app self.app = app
self.logger = get_logger(conf, log_route='bulk')
self.max_containers = int( self.max_containers = int(
conf.get('max_containers_per_extraction', 10000)) conf.get('max_containers_per_extraction', 10000))
self.max_failed_extractions = int( self.max_failed_extractions = int(
conf.get('max_failed_extractions', 1000)) conf.get('max_failed_extractions', 1000))
self.max_deletes_per_request = int( self.max_deletes_per_request = int(
conf.get('max_deletes_per_request', 1000)) conf.get('max_deletes_per_request', 10000))
self.yield_frequency = int(conf.get('yield_frequency', 60))
def create_container(self, req, container_path): def create_container(self, req, container_path):
""" """
@ -213,99 +246,145 @@ class Bulk(object):
raise HTTPBadRequest('Invalid File Name') raise HTTPBadRequest('Invalid File Name')
return objs_to_delete return objs_to_delete
def handle_delete(self, req, objs_to_delete=None, user_agent='BulkDelete', def handle_delete_iter(self, req, objs_to_delete=None,
swift_source='BD'): user_agent='BulkDelete', swift_source='BD'):
""" """
A generator that can be assigned to a swob Response's app_iter which,
when iterated over, will delete the objects specified in request body.
Will occasionally yield whitespace while request is being processed.
When the request is completed will yield a response body that can be
parsed to determine success. See above documentation for details.
:params req: a swob Request :params req: a swob Request
:raises HTTPException: on unhandled errors :params objs_to_delete: a list of dictionaries that specifies the
:returns: a swob Response objects to be deleted. If None, uses
self.get_objs_to_delete to query request.
""" """
try: last_yield = time()
vrs, account, _junk = req.split_path(2, 3, True) separator = ''
except ValueError:
return HTTPNotFound(request=req)
incoming_format = req.headers.get('Content-Type')
if incoming_format and not incoming_format.startswith('text/plain'):
# For now only accept newline separated object names
return HTTPNotAcceptable(request=req)
out_content_type = req.accept.best_match(ACCEPTABLE_FORMATS)
if not out_content_type:
return HTTPNotAcceptable(request=req)
if objs_to_delete is None:
objs_to_delete = self.get_objs_to_delete(req)
failed_files = [] failed_files = []
success_count = not_found_count = 0 resp_dict = {'Response Status': HTTPOk().status,
failed_file_response_type = HTTPBadRequest 'Response Body': '',
for obj_to_delete in objs_to_delete: 'Number Deleted': 0,
obj_to_delete = obj_to_delete.strip().lstrip('/') 'Number Not Found': 0}
if not obj_to_delete: try:
continue out_content_type = req.accept.best_match(ACCEPTABLE_FORMATS)
delete_path = '/'.join(['', vrs, account, obj_to_delete]) if not out_content_type:
if not check_utf8(delete_path): raise HTTPNotAcceptable(request=req)
failed_files.append([quote(delete_path), if out_content_type.endswith('/xml'):
HTTPPreconditionFailed().status]) yield '<?xml version="1.0" encoding="UTF-8"?>\n'
continue
new_env = req.environ.copy()
new_env['PATH_INFO'] = delete_path
del(new_env['wsgi.input'])
new_env['CONTENT_LENGTH'] = 0
new_env['HTTP_USER_AGENT'] = \
'%s %s' % (req.environ.get('HTTP_USER_AGENT'), user_agent)
new_env['swift.source'] = swift_source
delete_obj_req = Request.blank(delete_path, new_env)
resp = delete_obj_req.get_response(self.app)
if resp.status_int // 100 == 2:
success_count += 1
elif resp.status_int == HTTP_NOT_FOUND:
not_found_count += 1
elif resp.status_int == HTTP_UNAUTHORIZED:
return HTTPUnauthorized(request=req)
else:
if resp.status_int // 100 == 5:
failed_file_response_type = HTTPBadGateway
failed_files.append([quote(delete_path), resp.status])
resp_body = get_response_body( try:
out_content_type, vrs, account, _junk = req.split_path(2, 3, True)
{'Number Deleted': success_count, except ValueError:
'Number Not Found': not_found_count}, raise HTTPNotFound(request=req)
failed_files)
if (success_count or not_found_count) and not failed_files:
return HTTPOk(resp_body, content_type=out_content_type)
if failed_files:
return failed_file_response_type(
resp_body, content_type=out_content_type)
return HTTPBadRequest('Invalid bulk delete.')
def handle_extract(self, req, compress_type): incoming_format = req.headers.get('Content-Type')
if incoming_format and \
not incoming_format.startswith('text/plain'):
# For now only accept newline separated object names
raise HTTPNotAcceptable(request=req)
if objs_to_delete is None:
objs_to_delete = self.get_objs_to_delete(req)
failed_file_response_type = HTTPBadRequest
req.environ['eventlet.minimum_write_chunk_size'] = 0
for obj_to_delete in objs_to_delete:
if last_yield + self.yield_frequency < time():
separator = '\r\n\r\n'
last_yield = time()
yield ' '
obj_to_delete = obj_to_delete.strip().lstrip('/')
if not obj_to_delete:
continue
delete_path = '/'.join(['', vrs, account, obj_to_delete])
if not check_utf8(delete_path):
failed_files.append([quote(delete_path),
HTTPPreconditionFailed().status])
continue
new_env = req.environ.copy()
new_env['PATH_INFO'] = delete_path
del(new_env['wsgi.input'])
new_env['CONTENT_LENGTH'] = 0
new_env['HTTP_USER_AGENT'] = \
'%s %s' % (req.environ.get('HTTP_USER_AGENT'), user_agent)
new_env['swift.source'] = swift_source
delete_obj_req = Request.blank(delete_path, new_env)
resp = delete_obj_req.get_response(self.app)
if resp.status_int // 100 == 2:
resp_dict['Number Deleted'] += 1
elif resp.status_int == HTTP_NOT_FOUND:
resp_dict['Number Not Found'] += 1
elif resp.status_int == HTTP_UNAUTHORIZED:
failed_files.append([quote(delete_path),
HTTP_UNAUTHORIZED])
raise HTTPUnauthorized(request=req)
else:
if resp.status_int // 100 == 5:
failed_file_response_type = HTTPBadGateway
failed_files.append([quote(delete_path), resp.status])
if failed_files:
resp_dict['Response Status'] = \
failed_file_response_type().status
elif not (resp_dict['Number Deleted'] or
resp_dict['Number Not Found']):
resp_dict['Response Status'] = HTTPBadRequest().status
resp_dict['Response Body'] = 'Invalid bulk delete.'
except HTTPException, err:
resp_dict['Response Status'] = err.status
resp_dict['Response Body'] = err.body
except Exception:
self.logger.exception('Error in bulk delete.')
resp_dict['Response Status'] = HTTPServerError().status
yield separator + get_response_body(out_content_type,
resp_dict, failed_files)
def handle_extract_iter(self, req, compress_type):
""" """
A generator that can be assigned to a swob Response's app_iter which,
when iterated over, will extract and PUT the objects pulled from the
request body. Will occasionally yield whitespace while request is being
processed. When the request is completed will yield a response body
that can be parsed to determine success. See above documentation for
details.
:params req: a swob Request :params req: a swob Request
:params compress_type: specifying the compression type of the tar. :params compress_type: specifying the compression type of the tar.
Accepts '', 'gz, or 'bz2' Accepts '', 'gz, or 'bz2'
:raises HTTPException: on unhandled errors
:returns: a swob response to request
""" """
success_count = 0 resp_dict = {'Response Status': HTTPCreated().status,
'Response Body': '', 'Number Files Created': 0}
failed_files = [] failed_files = []
last_yield = time()
separator = ''
existing_containers = set() existing_containers = set()
out_content_type = req.accept.best_match(ACCEPTABLE_FORMATS)
if not out_content_type:
return HTTPNotAcceptable(request=req)
if req.content_length is None and \
req.headers.get('transfer-encoding', '').lower() != 'chunked':
return HTTPLengthRequired(request=req)
try:
vrs, account, extract_base = req.split_path(2, 3, True)
except ValueError:
return HTTPNotFound(request=req)
extract_base = extract_base or ''
extract_base = extract_base.rstrip('/')
try: try:
out_content_type = req.accept.best_match(ACCEPTABLE_FORMATS)
if not out_content_type:
raise HTTPNotAcceptable(request=req)
if out_content_type.endswith('/xml'):
yield '<?xml version="1.0" encoding="UTF-8"?>\n'
if req.content_length is None and \
req.headers.get('transfer-encoding',
'').lower() != 'chunked':
raise HTTPLengthRequired(request=req)
try:
vrs, account, extract_base = req.split_path(2, 3, True)
except ValueError:
raise HTTPNotFound(request=req)
extract_base = extract_base or ''
extract_base = extract_base.rstrip('/')
tar = tarfile.open(mode='r|' + compress_type, tar = tarfile.open(mode='r|' + compress_type,
fileobj=req.body_file) fileobj=req.body_file)
failed_response_type = HTTPBadRequest
req.environ['eventlet.minimum_write_chunk_size'] = 0
while True: while True:
if last_yield + self.yield_frequency < time():
separator = '\r\n\r\n'
last_yield = time()
yield ' '
tar_info = tar.next() tar_info = tar.next()
if tar_info is None or \ if tar_info is None or \
len(failed_files) >= self.max_failed_extractions: len(failed_files) >= self.max_failed_extractions:
@ -340,7 +419,7 @@ class Bulk(object):
existing_containers.add(container) existing_containers.add(container)
except CreateContainerError, err: except CreateContainerError, err:
if err.status_int == HTTP_UNAUTHORIZED: if err.status_int == HTTP_UNAUTHORIZED:
return HTTPUnauthorized(request=req) raise HTTPUnauthorized(request=req)
failed_files.append([ failed_files.append([
quote(destination[:MAX_PATH_LENGTH]), quote(destination[:MAX_PATH_LENGTH]),
err.status]) err.status])
@ -351,7 +430,7 @@ class Bulk(object):
HTTP_BAD_REQUEST]) HTTP_BAD_REQUEST])
continue continue
if len(existing_containers) > self.max_containers: if len(existing_containers) > self.max_containers:
return HTTPBadRequest( raise HTTPBadRequest(
'More than %d base level containers in tar.' % 'More than %d base level containers in tar.' %
self.max_containers) self.max_containers)
@ -366,41 +445,55 @@ class Bulk(object):
create_obj_req = Request.blank(destination, new_env) create_obj_req = Request.blank(destination, new_env)
resp = create_obj_req.get_response(self.app) resp = create_obj_req.get_response(self.app)
if resp.status_int // 100 == 2: if resp.status_int // 100 == 2:
success_count += 1 resp_dict['Number Files Created'] += 1
else: else:
if resp.status_int == HTTP_UNAUTHORIZED: if resp.status_int == HTTP_UNAUTHORIZED:
return HTTPUnauthorized(request=req) failed_files.append([
quote(destination[:MAX_PATH_LENGTH]),
HTTP_UNAUTHORIZED])
raise HTTPUnauthorized(request=req)
if resp.status_int // 100 == 5:
failed_response_type = HTTPBadGateway
failed_files.append([ failed_files.append([
quote(destination[:MAX_PATH_LENGTH]), resp.status]) quote(destination[:MAX_PATH_LENGTH]), resp.status])
resp_body = get_response_body(
out_content_type,
{'Number Files Created': success_count},
failed_files)
if success_count and not failed_files:
return HTTPCreated(resp_body, content_type=out_content_type)
if failed_files: if failed_files:
return HTTPBadGateway(resp_body, content_type=out_content_type) resp_dict['Response Status'] = failed_response_type().status
return HTTPBadRequest('Invalid Tar File: No Valid Files') elif not resp_dict['Number Files Created']:
resp_dict['Response Status'] = HTTPBadRequest().status
resp_dict['Response Body'] = 'Invalid Tar File: No Valid Files'
except HTTPException, err:
resp_dict['Response Status'] = err.status
resp_dict['Response Body'] = err.body
except tarfile.TarError, tar_error: except tarfile.TarError, tar_error:
return HTTPBadRequest('Invalid Tar File: %s' % tar_error) resp_dict['Response Status'] = HTTPBadRequest().status,
resp_dict['Response Body'] = 'Invalid Tar File: %s' % tar_error
except Exception:
self.logger.exception('Error in extract archive.')
resp_dict['Response Status'] = HTTPServerError().status
yield separator + get_response_body(
out_content_type, resp_dict, failed_files)
@wsgify @wsgify
def __call__(self, req): def __call__(self, req):
extract_type = req.params.get('extract-archive') extract_type = req.params.get('extract-archive')
resp = None
if extract_type is not None and req.method == 'PUT': if extract_type is not None and req.method == 'PUT':
archive_type = { archive_type = {
'tar': '', 'tar.gz': 'gz', 'tar': '', 'tar.gz': 'gz',
'tar.bz2': 'bz2'}.get(extract_type.lower().strip('.')) 'tar.bz2': 'bz2'}.get(extract_type.lower().strip('.'))
if archive_type is not None: if archive_type is not None:
return self.handle_extract(req, archive_type) resp = HTTPOk(request=req)
resp.app_iter = self.handle_extract_iter(req, archive_type)
else: else:
return HTTPBadRequest("Unsupported archive format") resp = HTTPBadRequest("Unsupported archive format")
if 'bulk-delete' in req.params and req.method == 'DELETE': if 'bulk-delete' in req.params and req.method == 'DELETE':
return self.handle_delete(req) resp = HTTPOk(request=req)
resp.app_iter = self.handle_delete_iter(req)
return self.app return resp or self.app
def filter_factory(global_conf, **local_conf): def filter_factory(global_conf, **local_conf):

View File

@ -138,8 +138,10 @@ from cStringIO import StringIO
from datetime import datetime from datetime import datetime
import mimetypes import mimetypes
from swift.common.swob import Request, HTTPBadRequest, HTTPServerError, \ from swift.common.swob import Request, HTTPBadRequest, HTTPServerError, \
HTTPMethodNotAllowed, HTTPRequestEntityTooLarge, HTTPLengthRequired, wsgify HTTPMethodNotAllowed, HTTPRequestEntityTooLarge, HTTPLengthRequired, \
HTTPOk, HTTPPreconditionFailed, wsgify
from swift.common.utils import json, get_logger, config_true_value from swift.common.utils import json, get_logger, config_true_value
from swift.common.constraints import check_utf8
from swift.common.middleware.bulk import get_response_body, \ from swift.common.middleware.bulk import get_response_body, \
ACCEPTABLE_FORMATS, Bulk ACCEPTABLE_FORMATS, Bulk
@ -303,8 +305,15 @@ class StaticLargeObject(object):
successful, will delete the manifest file. successful, will delete the manifest file.
:params req: a swob.Request with an obj in path :params req: a swob.Request with an obj in path
:raises HTTPServerError: on invalid manifest :raises HTTPServerError: on invalid manifest
:returns: swob.Response on failure, otherwise self.app :returns: swob.Response whose app_iter set to Bulk.handle_delete_iter
""" """
if not check_utf8(req.path_info):
raise HTTPPreconditionFailed(
request=req, body='Invalid UTF8 or contains NULL')
try:
vrs, account, container, obj = req.split_path(4, 4, True)
except ValueError:
raise HTTPBadRequest('Not an SLO manifest')
new_env = req.environ.copy() new_env = req.environ.copy()
new_env['REQUEST_METHOD'] = 'GET' new_env['REQUEST_METHOD'] = 'GET'
del(new_env['wsgi.input']) del(new_env['wsgi.input'])
@ -321,17 +330,17 @@ class StaticLargeObject(object):
raise HTTPBadRequest('Not an SLO manifest') raise HTTPBadRequest('Not an SLO manifest')
try: try:
manifest = json.loads(get_man_resp.body) manifest = json.loads(get_man_resp.body)
# append the manifest file for deletion at the end
manifest.append(
{'name': '/'.join(['', container, obj]).decode('utf-8')})
except ValueError: except ValueError:
raise HTTPServerError('Invalid manifest file') raise HTTPServerError('Invalid manifest file')
delete_resp = self.bulk_deleter.handle_delete( resp = HTTPOk(request=req)
resp.app_iter = self.bulk_deleter.handle_delete_iter(
req, req,
objs_to_delete=[o['name'].encode('utf-8') for o in manifest], objs_to_delete=[o['name'].encode('utf-8') for o in manifest],
user_agent='MultipartDELETE', swift_source='SLO') user_agent='MultipartDELETE', swift_source='SLO')
if delete_resp.status_int // 100 == 2: return resp
# delete the manifest file itself
return self.app
else:
return delete_resp
return get_man_resp return get_man_resp
@wsgify @wsgify

View File

@ -189,7 +189,7 @@ class SegmentedIterable(object):
'obj': self.controller.object_name, 'err': err}) 'obj': self.controller.object_name, 'err': err})
err.swift_logged = True err.swift_logged = True
self.response.status_int = HTTP_CONFLICT self.response.status_int = HTTP_CONFLICT
raise StopIteration('Invalid manifiest segment') raise
except (Exception, Timeout), err: except (Exception, Timeout), err:
if not getattr(err, 'swift_logged', False): if not getattr(err, 'swift_logged', False):
self.controller.app.logger.exception(_( self.controller.app.logger.exception(_(

View File

@ -111,6 +111,11 @@ class TestUntar(unittest.TestCase):
self.app.calls = 0 self.app.calls = 0
rmtree(self.testdir) rmtree(self.testdir)
def handle_extract_and_iter(self, req, compress_format):
resp_body = ''.join(
self.bulk.handle_extract_iter(req, compress_format))
return resp_body
def test_create_container_for_path(self): def test_create_container_for_path(self):
req = Request.blank('/') req = Request.blank('/')
self.assertEquals( self.assertEquals(
@ -147,8 +152,8 @@ class TestUntar(unittest.TestCase):
req.environ['wsgi.input'] = open( req.environ['wsgi.input'] = open(
os.path.join(self.testdir, 'tar_works.tar' + extension)) os.path.join(self.testdir, 'tar_works.tar' + extension))
req.headers['transfer-encoding'] = 'chunked' req.headers['transfer-encoding'] = 'chunked'
resp = self.bulk.handle_extract(req, compress_format) resp_body = self.handle_extract_and_iter(req, compress_format)
resp_data = json.loads(resp.body) resp_data = json.loads(resp_body)
self.assertEquals(resp_data['Number Files Created'], 6) self.assertEquals(resp_data['Number Files Created'], 6)
# test out xml # test out xml
@ -157,18 +162,19 @@ class TestUntar(unittest.TestCase):
req.environ['wsgi.input'] = open( req.environ['wsgi.input'] = open(
os.path.join(self.testdir, 'tar_works.tar' + extension)) os.path.join(self.testdir, 'tar_works.tar' + extension))
req.headers['transfer-encoding'] = 'chunked' req.headers['transfer-encoding'] = 'chunked'
resp = self.bulk.handle_extract(req, compress_format) resp_body = self.handle_extract_and_iter(req, compress_format)
self.assertEquals(resp.status_int, 201) self.assert_('<response_status>201 Created</response_status>' in
resp_body)
self.assert_('<number_files_created>6</number_files_created>' in self.assert_('<number_files_created>6</number_files_created>' in
resp.body) resp_body)
# test out nonexistent format # test out nonexistent format
req = Request.blank('/tar_works/acc/cont/', req = Request.blank('/tar_works/acc/cont/',
headers={'Accept': 'good_xml'}) headers={'Accept': 'good_xml'})
req.environ['wsgi.input'] = open( req.environ['wsgi.input'] = open(
os.path.join(self.testdir, 'tar_works.tar' + extension)) os.path.join(self.testdir, 'tar_works.tar' + extension))
resp = self.bulk.handle_extract(req, compress_format) resp_body = self.handle_extract_and_iter(req, compress_format)
self.assertEquals(resp.status_int, 406) self.assert_('Response Status: 406' in resp_body)
def test_extract_call(self): def test_extract_call(self):
base_name = 'base_works_gz' base_name = 'base_works_gz'
@ -198,7 +204,8 @@ class TestUntar(unittest.TestCase):
os.path.join(self.testdir, 'tar_works.tar.gz')) os.path.join(self.testdir, 'tar_works.tar.gz'))
req.headers['transfer-encoding'] = 'Chunked' req.headers['transfer-encoding'] = 'Chunked'
req.method = 'PUT' req.method = 'PUT'
self.bulk(req.environ, fake_start_response) app_iter = self.bulk(req.environ, fake_start_response)
resp_body = ''.join([i for i in app_iter])
self.assertEquals(self.app.calls, 7) self.assertEquals(self.app.calls, 7)
self.app.calls = 0 self.app.calls = 0
@ -221,18 +228,19 @@ class TestUntar(unittest.TestCase):
req.headers['transfer-encoding'] = 'Chunked' req.headers['transfer-encoding'] = 'Chunked'
req.environ['wsgi.input'] = open( req.environ['wsgi.input'] = open(
os.path.join(self.testdir, 'tar_works.tar')) os.path.join(self.testdir, 'tar_works.tar'))
t = self.bulk(req.environ, fake_start_response) app_iter = self.bulk(req.environ, fake_start_response)
resp_body = ''.join([i for i in app_iter])
self.assertEquals(self.app.calls, 7) self.assertEquals(self.app.calls, 7)
def test_bad_container(self): def test_bad_container(self):
req = Request.blank('/invalid/', body='') req = Request.blank('/invalid/', body='')
resp = self.bulk.handle_extract(req, '') resp_body = self.handle_extract_and_iter(req, '')
self.assertEquals(resp.status_int, 404) self.assertTrue('404 Not Found' in resp_body)
def test_content_length_required(self): def test_content_length_required(self):
req = Request.blank('/create_cont_fail/acc/cont') req = Request.blank('/create_cont_fail/acc/cont')
resp = self.bulk.handle_extract(req, '') resp_body = self.handle_extract_and_iter(req, '')
self.assertEquals(resp.status_int, 411) self.assertTrue('411 Length Required' in resp_body)
def build_tar(self, dir_tree=None): def build_tar(self, dir_tree=None):
if not dir_tree: if not dir_tree:
@ -260,8 +268,8 @@ class TestUntar(unittest.TestCase):
req.environ['wsgi.input'] = open(os.path.join(self.testdir, req.environ['wsgi.input'] = open(os.path.join(self.testdir,
'tar_fails.tar')) 'tar_fails.tar'))
req.headers['transfer-encoding'] = 'chunked' req.headers['transfer-encoding'] = 'chunked'
resp = self.bulk.handle_extract(req, '') resp_body = self.handle_extract_and_iter(req, '')
resp_data = json.loads(resp.body) resp_data = json.loads(resp_body)
self.assertEquals(resp_data['Number Files Created'], 4) self.assertEquals(resp_data['Number Files Created'], 4)
def test_extract_tar_fail_cont_401(self): def test_extract_tar_fail_cont_401(self):
@ -270,8 +278,8 @@ class TestUntar(unittest.TestCase):
req.environ['wsgi.input'] = open(os.path.join(self.testdir, req.environ['wsgi.input'] = open(os.path.join(self.testdir,
'tar_fails.tar')) 'tar_fails.tar'))
req.headers['transfer-encoding'] = 'chunked' req.headers['transfer-encoding'] = 'chunked'
resp = self.bulk.handle_extract(req, '') resp_body = self.handle_extract_and_iter(req, '')
self.assertEquals(resp.status_int, 401) self.assertTrue('Response Status: 401 Unauthorized' in resp_body)
def test_extract_tar_fail_obj_401(self): def test_extract_tar_fail_obj_401(self):
self.build_tar() self.build_tar()
@ -279,8 +287,8 @@ class TestUntar(unittest.TestCase):
req.environ['wsgi.input'] = open(os.path.join(self.testdir, req.environ['wsgi.input'] = open(os.path.join(self.testdir,
'tar_fails.tar')) 'tar_fails.tar'))
req.headers['transfer-encoding'] = 'chunked' req.headers['transfer-encoding'] = 'chunked'
resp = self.bulk.handle_extract(req, '') resp_body = self.handle_extract_and_iter(req, '')
self.assertEquals(resp.status_int, 401) self.assertTrue('Response Status: 401 Unauthorized' in resp_body)
def test_extract_tar_fail_obj_name_len(self): def test_extract_tar_fail_obj_name_len(self):
self.build_tar() self.build_tar()
@ -289,8 +297,8 @@ class TestUntar(unittest.TestCase):
req.environ['wsgi.input'] = open(os.path.join(self.testdir, req.environ['wsgi.input'] = open(os.path.join(self.testdir,
'tar_fails.tar')) 'tar_fails.tar'))
req.headers['transfer-encoding'] = 'chunked' req.headers['transfer-encoding'] = 'chunked'
resp = self.bulk.handle_extract(req, '') resp_body = self.handle_extract_and_iter(req, '')
resp_data = json.loads(resp.body) resp_data = json.loads(resp_body)
self.assertEquals(resp_data['Number Files Created'], 4) self.assertEquals(resp_data['Number Files Created'], 4)
self.assertEquals(resp_data['Errors'][0][0], self.assertEquals(resp_data['Errors'][0][0],
'/tar_works/acc/cont/base_fails1/' + ('f' * 101)) '/tar_works/acc/cont/base_fails1/' + ('f' * 101))
@ -301,8 +309,8 @@ class TestUntar(unittest.TestCase):
req.environ['wsgi.input'] = open(os.path.join(self.testdir, req.environ['wsgi.input'] = open(os.path.join(self.testdir,
'tar_fails.tar')) 'tar_fails.tar'))
req.headers['transfer-encoding'] = 'chunked' req.headers['transfer-encoding'] = 'chunked'
resp = self.bulk.handle_extract(req, 'gz') resp_body = self.handle_extract_and_iter(req, 'gz')
self.assertEquals(resp.status_int, 400) self.assert_('400 Bad Request' in resp_body)
self.assertEquals(self.app.calls, 0) self.assertEquals(self.app.calls, 0)
def test_extract_tar_fail_max_file_name_length(self): def test_extract_tar_fail_max_file_name_length(self):
@ -314,8 +322,8 @@ class TestUntar(unittest.TestCase):
req.environ['wsgi.input'] = open(os.path.join(self.testdir, req.environ['wsgi.input'] = open(os.path.join(self.testdir,
'tar_fails.tar')) 'tar_fails.tar'))
req.headers['transfer-encoding'] = 'chunked' req.headers['transfer-encoding'] = 'chunked'
resp = self.bulk.handle_extract(req, '') resp_body = self.handle_extract_and_iter(req, '')
resp_data = json.loads(resp.body) resp_data = json.loads(resp_body)
self.assertEquals(self.app.calls, 5) self.assertEquals(self.app.calls, 5)
self.assertEquals(resp_data['Errors'][0][0], self.assertEquals(resp_data['Errors'][0][0],
'/tar_works/acc/cont/base_fails1/' + ('f' * 101)) '/tar_works/acc/cont/base_fails1/' + ('f' * 101))
@ -336,8 +344,8 @@ class TestUntar(unittest.TestCase):
req.environ['wsgi.input'] = open( req.environ['wsgi.input'] = open(
os.path.join(self.testdir, 'tar_works.tar')) os.path.join(self.testdir, 'tar_works.tar'))
req.headers['transfer-encoding'] = 'chunked' req.headers['transfer-encoding'] = 'chunked'
resp = self.bulk.handle_extract(req, '') resp_body = self.handle_extract_and_iter(req, '')
resp_data = json.loads(resp.body) resp_data = json.loads(resp_body)
self.assert_(resp_data['Errors'][0][1].startswith('413')) self.assert_(resp_data['Errors'][0][1].startswith('413'))
def test_extract_tar_fail_max_cont(self): def test_extract_tar_fail_max_cont(self):
@ -351,9 +359,9 @@ class TestUntar(unittest.TestCase):
body = open(os.path.join(self.testdir, 'tar_fails.tar')).read() body = open(os.path.join(self.testdir, 'tar_fails.tar')).read()
req = Request.blank('/tar_works/acc/', body=body) req = Request.blank('/tar_works/acc/', body=body)
req.headers['transfer-encoding'] = 'chunked' req.headers['transfer-encoding'] = 'chunked'
resp = self.bulk.handle_extract(req, '') resp_body = self.handle_extract_and_iter(req, '')
self.assertEquals(self.app.calls, 3) self.assertEquals(self.app.calls, 3)
self.assertEquals(resp.status_int, 400) self.assert_('400 Bad Request' in resp_body)
def test_extract_tar_fail_create_cont(self): def test_extract_tar_fail_create_cont(self):
dir_tree = [{'base_fails1': [ dir_tree = [{'base_fails1': [
@ -367,8 +375,8 @@ class TestUntar(unittest.TestCase):
req.environ['wsgi.input'] = open(os.path.join(self.testdir, req.environ['wsgi.input'] = open(os.path.join(self.testdir,
'tar_fails.tar')) 'tar_fails.tar'))
req.headers['transfer-encoding'] = 'chunked' req.headers['transfer-encoding'] = 'chunked'
resp = self.bulk.handle_extract(req, '') resp_body = self.handle_extract_and_iter(req, '')
resp_data = json.loads(resp.body) resp_data = json.loads(resp_body)
self.assertEquals(self.app.calls, 4) self.assertEquals(self.app.calls, 4)
self.assertEquals(len(resp_data['Errors']), 5) self.assertEquals(len(resp_data['Errors']), 5)
@ -384,14 +392,15 @@ class TestUntar(unittest.TestCase):
raise ValueError('Test') raise ValueError('Test')
with patch.object(self.bulk, 'create_container', bad_create): with patch.object(self.bulk, 'create_container', bad_create):
resp = self.bulk.handle_extract(req, '') resp_body = self.handle_extract_and_iter(req, '')
resp_data = json.loads(resp.body) resp_data = json.loads(resp_body)
self.assertEquals(self.app.calls, 0) self.assertEquals(self.app.calls, 0)
self.assertEquals(len(resp_data['Errors']), 5) self.assertEquals(len(resp_data['Errors']), 5)
def test_get_response_body(self): def test_get_response_body(self):
self.assertRaises( txt_body = bulk.get_response_body(
HTTPException, bulk.get_response_body, 'badformat', {}, []) 'bad_formay', {'hey': 'there'}, [['json > xml', '202 Accepted']])
self.assert_('hey: there' in txt_body)
xml_body = bulk.get_response_body( xml_body = bulk.get_response_body(
'text/xml', {'hey': 'there'}, [['json > xml', '202 Accepted']]) 'text/xml', {'hey': 'there'}, [['json > xml', '202 Accepted']])
self.assert_('&gt' in xml_body) self.assert_('&gt' in xml_body)
@ -407,35 +416,35 @@ class TestDelete(unittest.TestCase):
self.app.calls = 0 self.app.calls = 0
self.app.delete_paths = [] self.app.delete_paths = []
def handle_delete_and_iter(self, req):
resp_body = ''.join(self.bulk.handle_delete_iter(req))
return resp_body
def test_bulk_delete_works(self): def test_bulk_delete_works(self):
req = Request.blank('/delete_works/AUTH_Acc', body='/c/f\n/c/f404', req = Request.blank('/delete_works/AUTH_Acc', body='/c/f\n/c/f404',
headers={'Accept': 'application/json'}) headers={'Accept': 'application/json'})
req.method = 'DELETE' req.method = 'DELETE'
resp = self.bulk.handle_delete(req) resp_body = self.handle_delete_and_iter(req)
self.assertEquals( self.assertEquals(
self.app.delete_paths, self.app.delete_paths,
['/delete_works/AUTH_Acc/c/f', '/delete_works/AUTH_Acc/c/f404']) ['/delete_works/AUTH_Acc/c/f', '/delete_works/AUTH_Acc/c/f404'])
self.assertEquals(self.app.calls, 2) self.assertEquals(self.app.calls, 2)
resp_data = json.loads(resp.body) resp_data = json.loads(resp_body)
self.assertEquals(resp_data['Number Deleted'], 1) self.assertEquals(resp_data['Number Deleted'], 1)
self.assertEquals(resp_data['Number Not Found'], 1) self.assertEquals(resp_data['Number Not Found'], 1)
def test_bulk_delete_bad_accept_and_content_type(self): def test_bulk_delete_bad_content_type(self):
req = Request.blank('/delete_works/AUTH_Acc', req = Request.blank('/delete_works/AUTH_Acc',
headers={'Accept': 'badformat'}) headers={'Accept': 'badformat'})
req.method = 'DELETE'
req.environ['wsgi.input'] = StringIO('/c/f\n/c/f404')
resp = self.bulk.handle_delete(req)
self.assertEquals(resp.status_int, 406)
req = Request.blank('/delete_works/AUTH_Acc', req = Request.blank('/delete_works/AUTH_Acc',
headers={'Accept': 'application/json', headers={'Accept': 'application/json',
'Content-Type': 'text/xml'}) 'Content-Type': 'text/xml'})
req.method = 'DELETE' req.method = 'DELETE'
req.environ['wsgi.input'] = StringIO('/c/f\n/c/f404') req.environ['wsgi.input'] = StringIO('/c/f\n/c/f404')
resp = self.bulk.handle_delete(req) resp_body = self.handle_delete_and_iter(req)
self.assertEquals(resp.status_int, 406) resp_data = json.loads(resp_body)
self.assertEquals(resp_data['Response Status'], '406 Not Acceptable')
def test_bulk_delete_call(self): def test_bulk_delete_call(self):
def fake_start_response(*args, **kwargs): def fake_start_response(*args, **kwargs):
@ -444,7 +453,8 @@ class TestDelete(unittest.TestCase):
req.method = 'DELETE' req.method = 'DELETE'
req.headers['Transfer-Encoding'] = 'chunked' req.headers['Transfer-Encoding'] = 'chunked'
req.environ['wsgi.input'] = StringIO('/c/f') req.environ['wsgi.input'] = StringIO('/c/f')
self.bulk(req.environ, fake_start_response) list(self.bulk(req.environ,
fake_start_response)) # iterate over whole resp
self.assertEquals( self.assertEquals(
self.app.delete_paths, ['/delete_works/AUTH_Acc/c/f']) self.app.delete_paths, ['/delete_works/AUTH_Acc/c/f'])
self.assertEquals(self.app.calls, 1) self.assertEquals(self.app.calls, 1)
@ -474,14 +484,14 @@ class TestDelete(unittest.TestCase):
body='/c/f\n\n\n/c/f404\n\n\n/c/%2525', body='/c/f\n\n\n/c/f404\n\n\n/c/%2525',
headers={'Accept': 'application/json'}) headers={'Accept': 'application/json'})
req.method = 'DELETE' req.method = 'DELETE'
resp = self.bulk.handle_delete(req) resp_body = self.handle_delete_and_iter(req)
self.assertEquals( self.assertEquals(
self.app.delete_paths, self.app.delete_paths,
['/delete_works/AUTH_Acc/c/f', ['/delete_works/AUTH_Acc/c/f',
'/delete_works/AUTH_Acc/c/f404', '/delete_works/AUTH_Acc/c/f404',
'/delete_works/AUTH_Acc/c/%25']) '/delete_works/AUTH_Acc/c/%25'])
self.assertEquals(self.app.calls, 3) self.assertEquals(self.app.calls, 3)
resp_data = json.loads(resp.body) resp_data = json.loads(resp_body)
self.assertEquals(resp_data['Number Deleted'], 2) self.assertEquals(resp_data['Number Deleted'], 2)
self.assertEquals(resp_data['Number Not Found'], 1) self.assertEquals(resp_data['Number Not Found'], 1)
@ -491,23 +501,8 @@ class TestDelete(unittest.TestCase):
data = '\n\n' * self.bulk.max_deletes_per_request data = '\n\n' * self.bulk.max_deletes_per_request
req.environ['wsgi.input'] = StringIO(data) req.environ['wsgi.input'] = StringIO(data)
req.content_length = len(data) req.content_length = len(data)
try: resp_body = self.handle_delete_and_iter(req)
self.bulk.handle_delete(req) self.assertTrue('413 Request Entity Too Large' in resp_body)
except HTTPException, err:
self.assertEquals(err.status_int, 413)
else:
self.fail('413 not raised')
def test_bulk_delete_raised_error(self):
def fake_start_response(*args, **kwargs):
self.assertTrue(args[0].startswith('413'))
req = Request.blank('/delete_works/AUTH_Acc?bulk-delete')
req.method = 'DELETE'
data = '\n\n' * self.bulk.max_deletes_per_request
req.environ['wsgi.input'] = StringIO(data)
req.content_length = len(data)
self.bulk(req.environ, fake_start_response)
def test_bulk_delete_works_unicode(self): def test_bulk_delete_works_unicode(self):
body = (u'/c/ obj \u2661\r\n'.encode('utf8') + body = (u'/c/ obj \u2661\r\n'.encode('utf8') +
@ -516,14 +511,14 @@ class TestDelete(unittest.TestCase):
req = Request.blank('/delete_works/AUTH_Acc', body=body, req = Request.blank('/delete_works/AUTH_Acc', body=body,
headers={'Accept': 'application/json'}) headers={'Accept': 'application/json'})
req.method = 'DELETE' req.method = 'DELETE'
resp = self.bulk.handle_delete(req) resp_body = self.handle_delete_and_iter(req)
self.assertEquals( self.assertEquals(
self.app.delete_paths, self.app.delete_paths,
['/delete_works/AUTH_Acc/c/ obj \xe2\x99\xa1', ['/delete_works/AUTH_Acc/c/ obj \xe2\x99\xa1',
'/delete_works/AUTH_Acc/c/ objbadutf8']) '/delete_works/AUTH_Acc/c/ objbadutf8'])
self.assertEquals(self.app.calls, 2) self.assertEquals(self.app.calls, 2)
resp_data = json.loads(resp.body) resp_data = json.loads(resp_body)
self.assertEquals(resp_data['Number Deleted'], 1) self.assertEquals(resp_data['Number Deleted'], 1)
self.assertEquals(len(resp_data['Errors']), 2) self.assertEquals(len(resp_data['Errors']), 2)
self.assertEquals( self.assertEquals(
@ -535,36 +530,37 @@ class TestDelete(unittest.TestCase):
def test_bulk_delete_no_body(self): def test_bulk_delete_no_body(self):
req = Request.blank('/unauth/AUTH_acc/') req = Request.blank('/unauth/AUTH_acc/')
self.assertRaises(HTTPException, self.bulk.handle_delete, req) resp_body = self.handle_delete_and_iter(req)
self.assertTrue('411 Length Required' in resp_body)
def test_bulk_delete_no_files_in_body(self): def test_bulk_delete_no_files_in_body(self):
req = Request.blank('/unauth/AUTH_acc/', body=' ') req = Request.blank('/unauth/AUTH_acc/', body=' ')
resp = self.bulk.handle_delete(req) resp_body = self.handle_delete_and_iter(req)
self.assertEquals(resp.status_int, 400) self.assertTrue('400 Bad Request' in resp_body)
def test_bulk_delete_unauth(self): def test_bulk_delete_unauth(self):
req = Request.blank('/unauth/AUTH_acc/', body='/c/f\n') req = Request.blank('/unauth/AUTH_acc/', body='/c/f\n')
req.method = 'DELETE' req.method = 'DELETE'
resp = self.bulk.handle_delete(req) resp_body = self.handle_delete_and_iter(req)
self.assertEquals(resp.status_int, 401) self.assertTrue('401 Unauthorized' in resp_body)
def test_bulk_delete_500_resp(self): def test_bulk_delete_500_resp(self):
req = Request.blank('/broke/AUTH_acc/', body='/c/f\n') req = Request.blank('/broke/AUTH_acc/', body='/c/f\n')
req.method = 'DELETE' req.method = 'DELETE'
resp = self.bulk.handle_delete(req) resp_body = self.handle_delete_and_iter(req)
self.assertEquals(resp.status_int, 502) self.assertTrue('502 Bad Gateway' in resp_body)
def test_bulk_delete_bad_path(self): def test_bulk_delete_bad_path(self):
req = Request.blank('/delete_cont_fail/') req = Request.blank('/delete_cont_fail/')
resp = self.bulk.handle_delete(req) resp_body = self.handle_delete_and_iter(req)
self.assertEquals(resp.status_int, 404) self.assertTrue('404 Not Found' in resp_body)
def test_bulk_delete_container_delete(self): def test_bulk_delete_container_delete(self):
req = Request.blank('/delete_cont_fail/AUTH_Acc', body='c\n', req = Request.blank('/delete_cont_fail/AUTH_Acc', body='c\n',
headers={'Accept': 'application/json'}) headers={'Accept': 'application/json'})
req.method = 'DELETE' req.method = 'DELETE'
resp = self.bulk.handle_delete(req) resp_body = self.handle_delete_and_iter(req)
resp_data = json.loads(resp.body) resp_data = json.loads(resp_body)
self.assertEquals(resp_data['Number Deleted'], 0) self.assertEquals(resp_data['Number Deleted'], 0)
self.assertEquals(resp_data['Errors'][0][1], '409 Conflict') self.assertEquals(resp_data['Errors'][0][1], '409 Conflict')
@ -575,8 +571,8 @@ class TestDelete(unittest.TestCase):
data = '/c/f\nc/' + ('1' * bulk.MAX_PATH_LENGTH) + '\n/c/f' data = '/c/f\nc/' + ('1' * bulk.MAX_PATH_LENGTH) + '\n/c/f'
req.environ['wsgi.input'] = StringIO(data) req.environ['wsgi.input'] = StringIO(data)
req.headers['Transfer-Encoding'] = 'chunked' req.headers['Transfer-Encoding'] = 'chunked'
resp = self.bulk.handle_delete(req) resp_body = self.handle_delete_and_iter(req)
resp_data = json.loads(resp.body) resp_data = json.loads(resp_body)
self.assertEquals(resp_data['Number Deleted'], 2) self.assertEquals(resp_data['Number Deleted'], 2)
self.assertEquals(resp_data['Errors'][0][1], '400 Bad Request') self.assertEquals(resp_data['Errors'][0][1], '400 Bad Request')
@ -584,12 +580,8 @@ class TestDelete(unittest.TestCase):
body = '/c/f\nc/' + ('123456' * bulk.MAX_PATH_LENGTH) + '\n' body = '/c/f\nc/' + ('123456' * bulk.MAX_PATH_LENGTH) + '\n'
req = Request.blank('/delete_works/AUTH_Acc', body=body) req = Request.blank('/delete_works/AUTH_Acc', body=body)
req.method = 'DELETE' req.method = 'DELETE'
try: resp_body = self.handle_delete_and_iter(req)
self.bulk.handle_delete(req) self.assertTrue('400 Bad Request' in resp_body)
except HTTPException, err:
self.assertEquals(err.status_int, 400)
else:
self.fail('400 not raised')
if __name__ == '__main__': if __name__ == '__main__':
unittest.main() unittest.main()

View File

@ -351,7 +351,8 @@ class TestStaticLargeObject(unittest.TestCase):
req = Request.blank( req = Request.blank(
'/test_delete/A/c/man?multipart-manifest=delete', '/test_delete/A/c/man?multipart-manifest=delete',
environ={'REQUEST_METHOD': 'DELETE'}) environ={'REQUEST_METHOD': 'DELETE'})
self.slo(req.environ, fake_start_response) app_iter = self.slo(req.environ, fake_start_response)
list(app_iter) # iterate through whole response
self.assertEquals(self.app.calls, 4) self.assertEquals(self.app.calls, 4)
self.assertEquals(self.app.req_method_paths, self.assertEquals(self.app.req_method_paths,
[('GET', '/test_delete/A/c/man'), [('GET', '/test_delete/A/c/man'),
@ -383,7 +384,8 @@ class TestStaticLargeObject(unittest.TestCase):
req = Request.blank( req = Request.blank(
'/test_delete_bad/A/c/man?multipart-manifest=delete', '/test_delete_bad/A/c/man?multipart-manifest=delete',
environ={'REQUEST_METHOD': 'DELETE'}) environ={'REQUEST_METHOD': 'DELETE'})
self.slo(req.environ, fake_start_response) app_iter = self.slo(req.environ, fake_start_response)
list(app_iter) # iterate through whole response
self.assertEquals(self.app.calls, 2) self.assertEquals(self.app.calls, 2)
self.assertEquals(self.app.req_method_paths, self.assertEquals(self.app.req_method_paths,
[('GET', '/test_delete_bad/A/c/man'), [('GET', '/test_delete_bad/A/c/man'),

View File

@ -40,7 +40,7 @@ from swift.account import server as account_server
from swift.container import server as container_server from swift.container import server as container_server
from swift.obj import server as object_server from swift.obj import server as object_server
from swift.common import ring from swift.common import ring
from swift.common.exceptions import ChunkReadTimeout from swift.common.exceptions import ChunkReadTimeout, SloSegmentError
from swift.common.constraints import MAX_META_NAME_LENGTH, \ from swift.common.constraints import MAX_META_NAME_LENGTH, \
MAX_META_VALUE_LENGTH, MAX_META_COUNT, MAX_META_OVERALL_SIZE, \ MAX_META_VALUE_LENGTH, MAX_META_COUNT, MAX_META_OVERALL_SIZE, \
MAX_FILE_SIZE, MAX_ACCOUNT_NAME_LENGTH, MAX_CONTAINER_NAME_LENGTH MAX_FILE_SIZE, MAX_ACCOUNT_NAME_LENGTH, MAX_CONTAINER_NAME_LENGTH
@ -1204,9 +1204,11 @@ class TestObjectController(unittest.TestCase):
req = Request.blank('/a/c/manifest') req = Request.blank('/a/c/manifest')
resp = controller.GET(req) resp = controller.GET(req)
self.assertEqual(resp.status_int, 200) self.assertEqual(resp.status_int, 200)
self.assertEqual(resp.body, 'Aa') # dropped connection
self.assertEqual(resp.content_length, 4) # content incomplete self.assertEqual(resp.content_length, 4) # content incomplete
self.assertEqual(resp.content_type, 'text/html') self.assertEqual(resp.content_type, 'text/html')
self.assertRaises(SloSegmentError, lambda: resp.body)
# dropped connection, exception is caught by eventlet as it is
# iterating over response
self.assertEqual( self.assertEqual(
requested, requested,
@ -1260,9 +1262,11 @@ class TestObjectController(unittest.TestCase):
req = Request.blank('/a/c/manifest') req = Request.blank('/a/c/manifest')
resp = controller.GET(req) resp = controller.GET(req)
self.assertEqual(resp.status_int, 200) self.assertEqual(resp.status_int, 200)
self.assertEqual(resp.body, 'Aa') # dropped connection
self.assertEqual(resp.content_length, 4) # content incomplete self.assertEqual(resp.content_length, 4) # content incomplete
self.assertEqual(resp.content_type, 'text/html') self.assertEqual(resp.content_type, 'text/html')
self.assertRaises(SloSegmentError, lambda: resp.body)
# dropped connection, exception is caught by eventlet as it is
# iterating over response
self.assertEqual( self.assertEqual(
requested, requested,
@ -1320,9 +1324,11 @@ class TestObjectController(unittest.TestCase):
req = Request.blank('/a/c/manifest') req = Request.blank('/a/c/manifest')
resp = controller.GET(req) resp = controller.GET(req)
self.assertEqual(resp.status_int, 200) self.assertEqual(resp.status_int, 200)
self.assertEqual(resp.body, 'Aa') # dropped connection
self.assertEqual(resp.content_length, 6) # content incomplete self.assertEqual(resp.content_length, 6) # content incomplete
self.assertEqual(resp.content_type, 'text/html') self.assertEqual(resp.content_type, 'text/html')
self.assertRaises(SloSegmentError, lambda: resp.body)
# dropped connection, exception is caught by eventlet as it is
# iterating over response
self.assertEqual( self.assertEqual(
requested, requested,