mirror of
				https://github.com/django/django.git
				synced 2025-10-25 06:36:07 +00:00 
			
		
		
		
	Fixed #7651: uploading multiple files with the same name now work. Also, in order to test the problem the test client now handles uploading multiple files at once. Patch from Mike Axiak.
git-svn-id: http://code.djangoproject.com/svn/django/trunk@7858 bcc190cf-cafb-0310-a4f2-bffc1f526a37
This commit is contained in:
		| @@ -136,6 +136,7 @@ class MultiPartParser(object): | |||||||
|                     # since we cannot be sure a file is complete until |                     # since we cannot be sure a file is complete until | ||||||
|                     # we hit the next boundary/part of the multipart content. |                     # we hit the next boundary/part of the multipart content. | ||||||
|                     self.handle_file_complete(old_field_name, counters) |                     self.handle_file_complete(old_field_name, counters) | ||||||
|  |                     old_field_name = None | ||||||
|  |  | ||||||
|                 try: |                 try: | ||||||
|                     disposition = meta_data['content-disposition'][1] |                     disposition = meta_data['content-disposition'][1] | ||||||
|   | |||||||
| @@ -90,32 +90,34 @@ def encode_multipart(boundary, data): | |||||||
|     """ |     """ | ||||||
|     lines = [] |     lines = [] | ||||||
|     to_str = lambda s: smart_str(s, settings.DEFAULT_CHARSET) |     to_str = lambda s: smart_str(s, settings.DEFAULT_CHARSET) | ||||||
|  |  | ||||||
|  |     # Not by any means perfect, but good enough for our purposes. | ||||||
|  |     is_file = lambda thing: hasattr(thing, "read") and callable(thing.read) | ||||||
|  |  | ||||||
|  |     # Each bit of the multipart form data could be either a form value or a | ||||||
|  |     # file, or a *list* of form values and/or files. Remember that HTTP field | ||||||
|  |     # names can be duplicated! | ||||||
|     for (key, value) in data.items(): |     for (key, value) in data.items(): | ||||||
|         if isinstance(value, file): |         if is_file(value): | ||||||
|             lines.extend([ |             lines.extend(encode_file(boundary, key, value)) | ||||||
|                 '--' + boundary, |         elif not isinstance(value, basestring) and is_iterable(value): | ||||||
|                 'Content-Disposition: form-data; name="%s"; filename="%s"' \ |             for item in value: | ||||||
|                     % (to_str(key), to_str(os.path.basename(value.name))), |                 if is_file(item): | ||||||
|                 'Content-Type: application/octet-stream', |                     lines.extend(encode_file(boundary, key, item)) | ||||||
|                 '', |                 else: | ||||||
|                 value.read() |  | ||||||
|             ]) |  | ||||||
|         else: |  | ||||||
|             if not isinstance(value, basestring) and is_iterable(value): |  | ||||||
|                 for item in value: |  | ||||||
|                     lines.extend([ |                     lines.extend([ | ||||||
|                         '--' + boundary, |                         '--' + boundary, | ||||||
|                         'Content-Disposition: form-data; name="%s"' % to_str(key), |                         'Content-Disposition: form-data; name="%s"' % to_str(key), | ||||||
|                         '', |                         '', | ||||||
|                         to_str(item) |                         to_str(item) | ||||||
|                     ]) |                     ]) | ||||||
|             else: |         else: | ||||||
|                 lines.extend([ |             lines.extend([ | ||||||
|                     '--' + boundary, |                 '--' + boundary, | ||||||
|                     'Content-Disposition: form-data; name="%s"' % to_str(key), |                 'Content-Disposition: form-data; name="%s"' % to_str(key), | ||||||
|                     '', |                 '', | ||||||
|                     to_str(value) |                 to_str(value) | ||||||
|                 ]) |             ]) | ||||||
|  |  | ||||||
|     lines.extend([ |     lines.extend([ | ||||||
|         '--' + boundary + '--', |         '--' + boundary + '--', | ||||||
| @@ -123,6 +125,17 @@ def encode_multipart(boundary, data): | |||||||
|     ]) |     ]) | ||||||
|     return '\r\n'.join(lines) |     return '\r\n'.join(lines) | ||||||
|  |  | ||||||
|  | def encode_file(boundary, key, file): | ||||||
|  |     to_str = lambda s: smart_str(s, settings.DEFAULT_CHARSET) | ||||||
|  |     return [ | ||||||
|  |         '--' + boundary, | ||||||
|  |         'Content-Disposition: form-data; name="%s"; filename="%s"' \ | ||||||
|  |             % (to_str(key), to_str(os.path.basename(file.name))), | ||||||
|  |         'Content-Type: application/octet-stream', | ||||||
|  |         '', | ||||||
|  |         file.read() | ||||||
|  |     ] | ||||||
|  |      | ||||||
| class Client: | class Client: | ||||||
|     """ |     """ | ||||||
|     A class that can act as a client for testing purposes. |     A class that can act as a client for testing purposes. | ||||||
|   | |||||||
| @@ -156,3 +156,26 @@ class FileUploadTests(TestCase): | |||||||
|             {'f': open(f.name)} |             {'f': open(f.name)} | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|  |     def test_fileupload_getlist(self): | ||||||
|  |         file1 = tempfile.NamedTemporaryFile() | ||||||
|  |         file1.write('a' * (2 ** 23)) | ||||||
|  |  | ||||||
|  |         file2 = tempfile.NamedTemporaryFile() | ||||||
|  |         file2.write('a' * (2 * 2 ** 18)) | ||||||
|  |  | ||||||
|  |         file2a = tempfile.NamedTemporaryFile() | ||||||
|  |         file2a.write('a' * (5 * 2 ** 20)) | ||||||
|  |  | ||||||
|  |         response = self.client.post('/file_uploads/getlist_count/', { | ||||||
|  |             'file1': open(file1.name), | ||||||
|  |             'field1': u'test', | ||||||
|  |             'field2': u'test3', | ||||||
|  |             'field3': u'test5', | ||||||
|  |             'field4': u'test6', | ||||||
|  |             'field5': u'test7', | ||||||
|  |             'file2': (open(file2.name), open(file2a.name)) | ||||||
|  |         }) | ||||||
|  |         got = simplejson.loads(response.content) | ||||||
|  |  | ||||||
|  |         self.assertEqual(got.get('file1'), 1) | ||||||
|  |         self.assertEqual(got.get('file2'), 2) | ||||||
|   | |||||||
| @@ -7,4 +7,5 @@ urlpatterns = patterns('', | |||||||
|     (r'^echo/$',            views.file_upload_echo), |     (r'^echo/$',            views.file_upload_echo), | ||||||
|     (r'^quota/$',           views.file_upload_quota), |     (r'^quota/$',           views.file_upload_quota), | ||||||
|     (r'^quota/broken/$',    views.file_upload_quota_broken), |     (r'^quota/broken/$',    views.file_upload_quota_broken), | ||||||
|  |     (r'^getlist_count/$',   views.file_upload_getlist_count), | ||||||
| ) | ) | ||||||
|   | |||||||
| @@ -68,3 +68,13 @@ def file_upload_quota_broken(request): | |||||||
|     response = file_upload_echo(request) |     response = file_upload_echo(request) | ||||||
|     request.upload_handlers.insert(0, QuotaUploadHandler()) |     request.upload_handlers.insert(0, QuotaUploadHandler()) | ||||||
|     return response |     return response | ||||||
|  |  | ||||||
|  | def file_upload_getlist_count(request): | ||||||
|  |     """ | ||||||
|  |     Check the .getlist() function to ensure we receive the correct number of files. | ||||||
|  |     """ | ||||||
|  |     file_counts = {} | ||||||
|  |  | ||||||
|  |     for key in request.FILES.keys(): | ||||||
|  |         file_counts[key] = len(request.FILES.getlist(key)) | ||||||
|  |     return HttpResponse(simplejson.dumps(file_counts)) | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user