mirror of
				https://github.com/django/django.git
				synced 2025-10-24 22:26:08 +00:00 
			
		
		
		
	Fixed #24072 -- Added FileResponse for streaming binary files.
This commit is contained in:
		
				
					committed by
					
						 Tim Graham
						Tim Graham
					
				
			
			
				
	
			
			
			
						parent
						
							05f702b94c
						
					
				
				
					commit
					3d2cae0896
				
			| @@ -197,6 +197,8 @@ class WSGIHandler(base.BaseHandler): | |||||||
|         for c in response.cookies.values(): |         for c in response.cookies.values(): | ||||||
|             response_headers.append((str('Set-Cookie'), str(c.output(header='')))) |             response_headers.append((str('Set-Cookie'), str(c.output(header='')))) | ||||||
|         start_response(force_str(status), response_headers) |         start_response(force_str(status), response_headers) | ||||||
|  |         if getattr(response, 'file_to_stream', None) is not None and environ.get('wsgi.file_wrapper'): | ||||||
|  |             response = environ['wsgi.file_wrapper'](response.file_to_stream) | ||||||
|         return response |         return response | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,11 +1,13 @@ | |||||||
| from django.http.cookie import SimpleCookie, parse_cookie | from django.http.cookie import SimpleCookie, parse_cookie | ||||||
| from django.http.request import (HttpRequest, QueryDict, | from django.http.request import (HttpRequest, QueryDict, | ||||||
|     RawPostDataException, UnreadablePostError, build_request_repr) |     RawPostDataException, UnreadablePostError, build_request_repr) | ||||||
| from django.http.response import (HttpResponse, StreamingHttpResponse, | from django.http.response import ( | ||||||
|  |     HttpResponse, StreamingHttpResponse, FileResponse, | ||||||
|     HttpResponseRedirect, HttpResponsePermanentRedirect, |     HttpResponseRedirect, HttpResponsePermanentRedirect, | ||||||
|     HttpResponseNotModified, HttpResponseBadRequest, HttpResponseForbidden, |     HttpResponseNotModified, HttpResponseBadRequest, HttpResponseForbidden, | ||||||
|     HttpResponseNotFound, HttpResponseNotAllowed, HttpResponseGone, |     HttpResponseNotFound, HttpResponseNotAllowed, HttpResponseGone, | ||||||
|     HttpResponseServerError, Http404, BadHeaderError, JsonResponse) |     HttpResponseServerError, Http404, BadHeaderError, JsonResponse, | ||||||
|  | ) | ||||||
| from django.http.utils import fix_location_header, conditional_content_removal | from django.http.utils import fix_location_header, conditional_content_removal | ||||||
|  |  | ||||||
| __all__ = [ | __all__ = [ | ||||||
| @@ -16,5 +18,5 @@ __all__ = [ | |||||||
|     'HttpResponseBadRequest', 'HttpResponseForbidden', 'HttpResponseNotFound', |     'HttpResponseBadRequest', 'HttpResponseForbidden', 'HttpResponseNotFound', | ||||||
|     'HttpResponseNotAllowed', 'HttpResponseGone', 'HttpResponseServerError', |     'HttpResponseNotAllowed', 'HttpResponseGone', 'HttpResponseServerError', | ||||||
|     'Http404', 'BadHeaderError', 'fix_location_header', 'JsonResponse', |     'Http404', 'BadHeaderError', 'fix_location_header', 'JsonResponse', | ||||||
|     'conditional_content_removal', |     'FileResponse', 'conditional_content_removal', | ||||||
| ] | ] | ||||||
|   | |||||||
| @@ -417,6 +417,9 @@ class StreamingHttpResponse(HttpResponseBase): | |||||||
|  |  | ||||||
|     @streaming_content.setter |     @streaming_content.setter | ||||||
|     def streaming_content(self, value): |     def streaming_content(self, value): | ||||||
|  |         self._set_streaming_content(value) | ||||||
|  |  | ||||||
|  |     def _set_streaming_content(self, value): | ||||||
|         # Ensure we can never iterate on "value" more than once. |         # Ensure we can never iterate on "value" more than once. | ||||||
|         self._iterator = iter(value) |         self._iterator = iter(value) | ||||||
|         if hasattr(value, 'close'): |         if hasattr(value, 'close'): | ||||||
| @@ -429,6 +432,22 @@ class StreamingHttpResponse(HttpResponseBase): | |||||||
|         return b''.join(self.streaming_content) |         return b''.join(self.streaming_content) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class FileResponse(StreamingHttpResponse): | ||||||
|  |     """ | ||||||
|  |     A streaming HTTP response class optimized for files. | ||||||
|  |     """ | ||||||
|  |     block_size = 4096 | ||||||
|  |  | ||||||
|  |     def _set_streaming_content(self, value): | ||||||
|  |         if hasattr(value, 'read'): | ||||||
|  |             self.file_to_stream = value | ||||||
|  |             filelike = value | ||||||
|  |             value = iter(lambda: filelike.read(self.block_size), b'') | ||||||
|  |         else: | ||||||
|  |             self.file_to_stream = None | ||||||
|  |         super(FileResponse, self)._set_streaming_content(value) | ||||||
|  |  | ||||||
|  |  | ||||||
| class HttpResponseRedirectBase(HttpResponse): | class HttpResponseRedirectBase(HttpResponse): | ||||||
|     allowed_schemes = ['http', 'https', 'ftp'] |     allowed_schemes = ['http', 'https', 'ftp'] | ||||||
|  |  | ||||||
|   | |||||||
| @@ -11,7 +11,7 @@ import posixpath | |||||||
| import re | import re | ||||||
|  |  | ||||||
| from django.http import (Http404, HttpResponse, HttpResponseRedirect, | from django.http import (Http404, HttpResponse, HttpResponseRedirect, | ||||||
|     HttpResponseNotModified, StreamingHttpResponse) |     HttpResponseNotModified, FileResponse) | ||||||
| from django.template import loader, Template, Context, TemplateDoesNotExist | from django.template import loader, Template, Context, TemplateDoesNotExist | ||||||
| from django.utils.http import http_date, parse_http_date | from django.utils.http import http_date, parse_http_date | ||||||
| from django.utils.six.moves.urllib.parse import unquote | from django.utils.six.moves.urllib.parse import unquote | ||||||
| @@ -63,8 +63,7 @@ def serve(request, path, document_root=None, show_indexes=False): | |||||||
|         return HttpResponseNotModified() |         return HttpResponseNotModified() | ||||||
|     content_type, encoding = mimetypes.guess_type(fullpath) |     content_type, encoding = mimetypes.guess_type(fullpath) | ||||||
|     content_type = content_type or 'application/octet-stream' |     content_type = content_type or 'application/octet-stream' | ||||||
|     response = StreamingHttpResponse(open(fullpath, 'rb'), |     response = FileResponse(open(fullpath, 'rb'), content_type=content_type) | ||||||
|                                      content_type=content_type) |  | ||||||
|     response["Last-Modified"] = http_date(statobj.st_mtime) |     response["Last-Modified"] = http_date(statobj.st_mtime) | ||||||
|     if stat.S_ISREG(statobj.st_mode): |     if stat.S_ISREG(statobj.st_mode): | ||||||
|         response["Content-Length"] = statobj.st_size |         response["Content-Length"] = statobj.st_size | ||||||
|   | |||||||
| @@ -998,3 +998,21 @@ Attributes | |||||||
| .. attribute:: StreamingHttpResponse.streaming | .. attribute:: StreamingHttpResponse.streaming | ||||||
|  |  | ||||||
|     This is always ``True``. |     This is always ``True``. | ||||||
|  |  | ||||||
|  | FileResponse objects | ||||||
|  | ==================== | ||||||
|  |  | ||||||
|  | .. versionadded:: 1.8 | ||||||
|  |  | ||||||
|  | .. class:: FileResponse | ||||||
|  |  | ||||||
|  | :class:`FileResponse` is a subclass of :class:`StreamingHttpResponse` optimized | ||||||
|  | for binary files. It uses `wsgi.file_wrapper`_ if provided by the wsgi server, | ||||||
|  | otherwise it streams the file out in small chunks. | ||||||
|  |  | ||||||
|  | .. _wsgi.file_wrapper: https://www.python.org/dev/peps/pep-3333/#optional-platform-specific-file-handling | ||||||
|  |  | ||||||
|  | ``FileResponse`` expects a file open in binary mode like so:: | ||||||
|  |  | ||||||
|  |     >>> from django.http import FileResponse | ||||||
|  |     >>> response = FileResponse(open('myfile.png', 'rb')) | ||||||
|   | |||||||
| @@ -559,6 +559,8 @@ Requests and Responses | |||||||
|   <django.http.HttpResponse.setdefault>` method allows setting a header unless |   <django.http.HttpResponse.setdefault>` method allows setting a header unless | ||||||
|   it has already been set. |   it has already been set. | ||||||
|  |  | ||||||
|  | * You can use the new :class:`~django.http.FileResponse` to stream files. | ||||||
|  |  | ||||||
| * The :func:`~django.views.decorators.http.condition` decorator for | * The :func:`~django.views.decorators.http.condition` decorator for | ||||||
|   conditional view processing now supports the ``If-unmodified-since`` header. |   conditional view processing now supports the ``If-unmodified-since`` header. | ||||||
|  |  | ||||||
|   | |||||||
| @@ -10,8 +10,8 @@ from unittest import skipIf | |||||||
| from django.conf import settings | from django.conf import settings | ||||||
| from django.core import mail | from django.core import mail | ||||||
| from django.http import ( | from django.http import ( | ||||||
|     HttpRequest, HttpResponse, StreamingHttpResponse, HttpResponsePermanentRedirect, |     HttpRequest, HttpResponse, StreamingHttpResponse, FileResponse, | ||||||
|     HttpResponseRedirect, |     HttpResponseRedirect, HttpResponsePermanentRedirect, | ||||||
| ) | ) | ||||||
| from django.middleware.clickjacking import XFrameOptionsMiddleware | from django.middleware.clickjacking import XFrameOptionsMiddleware | ||||||
| from django.middleware.common import CommonMiddleware, BrokenLinkEmailsMiddleware | from django.middleware.common import CommonMiddleware, BrokenLinkEmailsMiddleware | ||||||
| @@ -624,6 +624,20 @@ class GZipMiddlewareTest(TestCase): | |||||||
|         self.assertEqual(r.get('Content-Encoding'), 'gzip') |         self.assertEqual(r.get('Content-Encoding'), 'gzip') | ||||||
|         self.assertFalse(r.has_header('Content-Length')) |         self.assertFalse(r.has_header('Content-Length')) | ||||||
|  |  | ||||||
|  |     def test_compress_file_response(self): | ||||||
|  |         """ | ||||||
|  |         Tests that compression is performed on FileResponse. | ||||||
|  |         """ | ||||||
|  |         open_file = lambda: open(__file__, 'rb') | ||||||
|  |         with open_file() as file1: | ||||||
|  |             file_resp = FileResponse(file1) | ||||||
|  |             file_resp['Content-Type'] = 'text/html; charset=UTF-8' | ||||||
|  |             r = GZipMiddleware().process_response(self.req, file_resp) | ||||||
|  |             with open_file() as file2: | ||||||
|  |                 self.assertEqual(self.decompress(b''.join(r)), file2.read()) | ||||||
|  |             self.assertEqual(r.get('Content-Encoding'), 'gzip') | ||||||
|  |             self.assertIsNot(r.file_to_stream, file1) | ||||||
|  |  | ||||||
|     def test_compress_non_200_response(self): |     def test_compress_non_200_response(self): | ||||||
|         """ |         """ | ||||||
|         Tests that compression is performed on responses with a status other than 200. |         Tests that compression is performed on responses with a status other than 200. | ||||||
|   | |||||||
| @@ -51,6 +51,28 @@ class WSGITest(TestCase): | |||||||
|             bytes(response), |             bytes(response), | ||||||
|             b"Content-Type: text/html; charset=utf-8\r\n\r\nHello World!") |             b"Content-Type: text/html; charset=utf-8\r\n\r\nHello World!") | ||||||
|  |  | ||||||
|  |     def test_file_wrapper(self): | ||||||
|  |         """ | ||||||
|  |         Verify that FileResponse uses wsgi.file_wrapper. | ||||||
|  |         """ | ||||||
|  |         class FileWrapper(object): | ||||||
|  |             def __init__(self, filelike, blksize=8192): | ||||||
|  |                 filelike.close() | ||||||
|  |         application = get_wsgi_application() | ||||||
|  |         environ = RequestFactory()._base_environ( | ||||||
|  |             PATH_INFO='/file/', | ||||||
|  |             REQUEST_METHOD='GET', | ||||||
|  |             **{'wsgi.file_wrapper': FileWrapper} | ||||||
|  |         ) | ||||||
|  |         response_data = {} | ||||||
|  |  | ||||||
|  |         def start_response(status, headers): | ||||||
|  |             response_data['status'] = status | ||||||
|  |             response_data['headers'] = headers | ||||||
|  |         response = application(environ, start_response) | ||||||
|  |         self.assertEqual(response_data['status'], '200 OK') | ||||||
|  |         self.assertIsInstance(response, FileWrapper) | ||||||
|  |  | ||||||
|  |  | ||||||
| class GetInternalWSGIApplicationTest(unittest.TestCase): | class GetInternalWSGIApplicationTest(unittest.TestCase): | ||||||
|     @override_settings(WSGI_APPLICATION="wsgi.wsgi.application") |     @override_settings(WSGI_APPLICATION="wsgi.wsgi.application") | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| from django.conf.urls import url | from django.conf.urls import url | ||||||
| from django.http import HttpResponse | from django.http import HttpResponse, FileResponse | ||||||
|  |  | ||||||
|  |  | ||||||
| def helloworld(request): | def helloworld(request): | ||||||
| @@ -7,4 +7,5 @@ def helloworld(request): | |||||||
|  |  | ||||||
| urlpatterns = [ | urlpatterns = [ | ||||||
|     url("^$", helloworld), |     url("^$", helloworld), | ||||||
|  |     url(r'^file/$', lambda x: FileResponse(open(__file__, 'rb'))), | ||||||
| ] | ] | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user