Merge pull request #733 from ashleysommer/file_stream
Add file_stream response handler
This commit is contained in:
commit
0858d3c544
|
@ -60,6 +60,16 @@ async def index(request):
|
||||||
return response.stream(streaming_fn, content_type='text/plain')
|
return response.stream(streaming_fn, content_type='text/plain')
|
||||||
```
|
```
|
||||||
|
|
||||||
|
## File Streaming
|
||||||
|
For large files, a combination of File and Streaming above
|
||||||
|
```python
|
||||||
|
from sanic import response
|
||||||
|
|
||||||
|
@app.route('/big_file.png')
|
||||||
|
async def handle_request(request):
|
||||||
|
return await response.file_stream('/srv/www/whatever.png')
|
||||||
|
```
|
||||||
|
|
||||||
## Redirect
|
## Redirect
|
||||||
|
|
||||||
```python
|
```python
|
||||||
|
|
|
@ -37,6 +37,10 @@ async def test_await(request):
|
||||||
async def test_file(request):
|
async def test_file(request):
|
||||||
return await response.file(os.path.abspath("setup.py"))
|
return await response.file(os.path.abspath("setup.py"))
|
||||||
|
|
||||||
|
@app.route("/file_stream")
|
||||||
|
async def test_file_stream(request):
|
||||||
|
return await response.file_stream(os.path.abspath("setup.py"),
|
||||||
|
chunk_size=1024)
|
||||||
|
|
||||||
# ----------------------------------------------- #
|
# ----------------------------------------------- #
|
||||||
# Exceptions
|
# Exceptions
|
||||||
|
|
|
@ -310,6 +310,53 @@ async def file(location, mime_type=None, headers=None, _range=None):
|
||||||
body_bytes=out_stream)
|
body_bytes=out_stream)
|
||||||
|
|
||||||
|
|
||||||
|
async def file_stream(location, chunk_size=4096, mime_type=None, headers=None,
|
||||||
|
_range=None):
|
||||||
|
"""Return a streaming response object with file data.
|
||||||
|
|
||||||
|
:param location: Location of file on system.
|
||||||
|
:param chunk_size: The size of each chunk in the stream (in bytes)
|
||||||
|
:param mime_type: Specific mime_type.
|
||||||
|
:param headers: Custom Headers.
|
||||||
|
:param _range:
|
||||||
|
"""
|
||||||
|
filename = path.split(location)[-1]
|
||||||
|
|
||||||
|
_file = await open_async(location, mode='rb')
|
||||||
|
|
||||||
|
async def _streaming_fn(response):
|
||||||
|
nonlocal _file, chunk_size
|
||||||
|
try:
|
||||||
|
if _range:
|
||||||
|
chunk_size = min((_range.size, chunk_size))
|
||||||
|
await _file.seek(_range.start)
|
||||||
|
to_send = _range.size
|
||||||
|
while to_send > 0:
|
||||||
|
content = await _file.read(chunk_size)
|
||||||
|
if len(content) < 1:
|
||||||
|
break
|
||||||
|
to_send -= len(content)
|
||||||
|
response.write(content)
|
||||||
|
else:
|
||||||
|
while True:
|
||||||
|
content = await _file.read(chunk_size)
|
||||||
|
if len(content) < 1:
|
||||||
|
break
|
||||||
|
response.write(content)
|
||||||
|
finally:
|
||||||
|
await _file.close()
|
||||||
|
return # Returning from this fn closes the stream
|
||||||
|
|
||||||
|
mime_type = mime_type or guess_type(filename)[0] or 'text/plain'
|
||||||
|
if _range:
|
||||||
|
headers['Content-Range'] = 'bytes %s-%s/%s' % (
|
||||||
|
_range.start, _range.end, _range.total)
|
||||||
|
return StreamingHTTPResponse(streaming_fn=_streaming_fn,
|
||||||
|
status=200,
|
||||||
|
headers=headers,
|
||||||
|
content_type=mime_type)
|
||||||
|
|
||||||
|
|
||||||
def stream(
|
def stream(
|
||||||
streaming_fn, status=200, headers=None,
|
streaming_fn, status=200, headers=None,
|
||||||
content_type="text/plain; charset=utf-8"):
|
content_type="text/plain; charset=utf-8"):
|
||||||
|
|
|
@ -1,13 +1,20 @@
|
||||||
import asyncio
|
import asyncio
|
||||||
|
import inspect
|
||||||
|
import os
|
||||||
|
from aiofiles import os as async_os
|
||||||
|
from mimetypes import guess_type
|
||||||
|
from urllib.parse import unquote
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from random import choice
|
from random import choice
|
||||||
|
|
||||||
from sanic import Sanic
|
from sanic import Sanic
|
||||||
from sanic.response import HTTPResponse, stream, StreamingHTTPResponse
|
from sanic.response import HTTPResponse, stream, StreamingHTTPResponse, file, file_stream
|
||||||
from sanic.testing import HOST, PORT
|
from sanic.testing import HOST, PORT
|
||||||
|
|
||||||
from unittest.mock import MagicMock
|
from unittest.mock import MagicMock
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
def test_response_body_not_a_string():
|
def test_response_body_not_a_string():
|
||||||
"""Test when a response body sent from the application is not a string"""
|
"""Test when a response body sent from the application is not a string"""
|
||||||
app = Sanic('response_body_not_a_string')
|
app = Sanic('response_body_not_a_string')
|
||||||
|
@ -94,3 +101,107 @@ def test_stream_response_writes_correct_content_to_transport(streaming_app):
|
||||||
app.stop()
|
app.stop()
|
||||||
|
|
||||||
streaming_app.run(host=HOST, port=PORT)
|
streaming_app.run(host=HOST, port=PORT)
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture
|
||||||
|
def static_file_directory():
|
||||||
|
"""The static directory to serve"""
|
||||||
|
current_file = inspect.getfile(inspect.currentframe())
|
||||||
|
current_directory = os.path.dirname(os.path.abspath(current_file))
|
||||||
|
static_directory = os.path.join(current_directory, 'static')
|
||||||
|
return static_directory
|
||||||
|
|
||||||
|
|
||||||
|
def get_file_content(static_file_directory, file_name):
|
||||||
|
"""The content of the static file to check"""
|
||||||
|
with open(os.path.join(static_file_directory, file_name), 'rb') as file:
|
||||||
|
return file.read()
|
||||||
|
|
||||||
|
@pytest.mark.parametrize('file_name', ['test.file', 'decode me.txt', 'python.png'])
|
||||||
|
def test_file_response(file_name, static_file_directory):
|
||||||
|
app = Sanic('test_file_helper')
|
||||||
|
@app.route('/files/<filename>', methods=['GET'])
|
||||||
|
def file_route(request, filename):
|
||||||
|
file_path = os.path.join(static_file_directory, filename)
|
||||||
|
file_path = os.path.abspath(unquote(file_path))
|
||||||
|
return file(file_path, mime_type=guess_type(file_path)[0] or 'text/plain')
|
||||||
|
|
||||||
|
request, response = app.test_client.get('/files/{}'.format(file_name))
|
||||||
|
assert response.status == 200
|
||||||
|
assert response.body == get_file_content(static_file_directory, file_name)
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.parametrize('file_name', ['test.file', 'decode me.txt'])
|
||||||
|
def test_file_head_response(file_name, static_file_directory):
|
||||||
|
app = Sanic('test_file_helper')
|
||||||
|
@app.route('/files/<filename>', methods=['GET', 'HEAD'])
|
||||||
|
async def file_route(request, filename):
|
||||||
|
file_path = os.path.join(static_file_directory, filename)
|
||||||
|
file_path = os.path.abspath(unquote(file_path))
|
||||||
|
stats = await async_os.stat(file_path)
|
||||||
|
headers = dict()
|
||||||
|
headers['Accept-Ranges'] = 'bytes'
|
||||||
|
headers['Content-Length'] = str(stats.st_size)
|
||||||
|
if request.method == "HEAD":
|
||||||
|
return HTTPResponse(
|
||||||
|
headers=headers,
|
||||||
|
content_type=guess_type(file_path)[0] or 'text/plain')
|
||||||
|
else:
|
||||||
|
return file(file_path, headers=headers,
|
||||||
|
mime_type=guess_type(file_path)[0] or 'text/plain')
|
||||||
|
|
||||||
|
request, response = app.test_client.head('/files/{}'.format(file_name))
|
||||||
|
assert response.status == 200
|
||||||
|
assert 'Accept-Ranges' in response.headers
|
||||||
|
assert 'Content-Length' in response.headers
|
||||||
|
assert int(response.headers[
|
||||||
|
'Content-Length']) == len(
|
||||||
|
get_file_content(static_file_directory, file_name))
|
||||||
|
|
||||||
|
@pytest.mark.parametrize('file_name', ['test.file', 'decode me.txt', 'python.png'])
|
||||||
|
def test_file_stream_response(file_name, static_file_directory):
|
||||||
|
app = Sanic('test_file_helper')
|
||||||
|
@app.route('/files/<filename>', methods=['GET'])
|
||||||
|
def file_route(request, filename):
|
||||||
|
file_path = os.path.join(static_file_directory, filename)
|
||||||
|
file_path = os.path.abspath(unquote(file_path))
|
||||||
|
return file_stream(file_path, chunk_size=32,
|
||||||
|
mime_type=guess_type(file_path)[0] or 'text/plain')
|
||||||
|
|
||||||
|
request, response = app.test_client.get('/files/{}'.format(file_name))
|
||||||
|
assert response.status == 200
|
||||||
|
assert response.body == get_file_content(static_file_directory, file_name)
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.parametrize('file_name', ['test.file', 'decode me.txt'])
|
||||||
|
def test_file_stream_head_response(file_name, static_file_directory):
|
||||||
|
app = Sanic('test_file_helper')
|
||||||
|
@app.route('/files/<filename>', methods=['GET', 'HEAD'])
|
||||||
|
async def file_route(request, filename):
|
||||||
|
file_path = os.path.join(static_file_directory, filename)
|
||||||
|
file_path = os.path.abspath(unquote(file_path))
|
||||||
|
headers = dict()
|
||||||
|
headers['Accept-Ranges'] = 'bytes'
|
||||||
|
if request.method == "HEAD":
|
||||||
|
# Return a normal HTTPResponse, not a
|
||||||
|
# StreamingHTTPResponse for a HEAD request
|
||||||
|
stats = await async_os.stat(file_path)
|
||||||
|
headers['Content-Length'] = str(stats.st_size)
|
||||||
|
return HTTPResponse(
|
||||||
|
headers=headers,
|
||||||
|
content_type=guess_type(file_path)[0] or 'text/plain')
|
||||||
|
else:
|
||||||
|
return file_stream(file_path, chunk_size=32, headers=headers,
|
||||||
|
mime_type=guess_type(file_path)[0] or 'text/plain')
|
||||||
|
|
||||||
|
request, response = app.test_client.head('/files/{}'.format(file_name))
|
||||||
|
assert response.status == 200
|
||||||
|
# A HEAD request should never be streamed/chunked.
|
||||||
|
if 'Transfer-Encoding' in response.headers:
|
||||||
|
assert response.headers['Transfer-Encoding'] != "chunked"
|
||||||
|
assert 'Accept-Ranges' in response.headers
|
||||||
|
# A HEAD request should get the Content-Length too
|
||||||
|
assert 'Content-Length' in response.headers
|
||||||
|
assert int(response.headers[
|
||||||
|
'Content-Length']) == len(
|
||||||
|
get_file_content(static_file_directory, file_name))
|
Loading…
Reference in New Issue
Block a user