Merge pull request #30 from huge-success/master
Merge Upstream Master Branch
This commit is contained in:
commit
34e51f01d1
|
@ -21,7 +21,7 @@ matrix:
|
||||||
python: 3.7
|
python: 3.7
|
||||||
dist: xenial
|
dist: xenial
|
||||||
sudo: true
|
sudo: true
|
||||||
- env: TOX_ENV=flake8
|
- env: TOX_ENV=lint
|
||||||
python: 3.6
|
python: 3.6
|
||||||
- env: TOX_ENV=check
|
- env: TOX_ENV=check
|
||||||
python: 3.6
|
python: 3.6
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
Sanic
|
Sanic
|
||||||
=====
|
=====
|
||||||
|
|
||||||
|Join the chat at https://gitter.im/sanic-python/Lobby| |Build Status| |Codecov| |PyPI| |PyPI version|
|
|Join the chat at https://gitter.im/sanic-python/Lobby| |Build Status| |Codecov| |PyPI| |PyPI version| |Code style black|
|
||||||
|
|
||||||
Sanic is a Flask-like Python 3.5+ web server that's written to go fast. It's based on the work done by the amazing folks at magicstack, and was inspired by `this article <https://magic.io/blog/uvloop-blazing-fast-python-networking/>`_.
|
Sanic is a Flask-like Python 3.5+ web server that's written to go fast. It's based on the work done by the amazing folks at magicstack, and was inspired by `this article <https://magic.io/blog/uvloop-blazing-fast-python-networking/>`_.
|
||||||
|
|
||||||
|
@ -57,7 +57,8 @@ Documentation
|
||||||
:target: https://pypi.python.org/pypi/sanic/
|
:target: https://pypi.python.org/pypi/sanic/
|
||||||
.. |PyPI version| image:: https://img.shields.io/pypi/pyversions/sanic.svg
|
.. |PyPI version| image:: https://img.shields.io/pypi/pyversions/sanic.svg
|
||||||
:target: https://pypi.python.org/pypi/sanic/
|
:target: https://pypi.python.org/pypi/sanic/
|
||||||
|
.. |Code style black| image:: https://img.shields.io/badge/code%20style-black-000000.svg
|
||||||
|
:target: https://github.com/ambv/black
|
||||||
|
|
||||||
Questions and Discussion
|
Questions and Discussion
|
||||||
------------------------
|
------------------------
|
||||||
|
|
2
pyproject.toml
Normal file
2
pyproject.toml
Normal file
|
@ -0,0 +1,2 @@
|
||||||
|
[tool.black]
|
||||||
|
line-length = 79
|
|
@ -1,6 +1,6 @@
|
||||||
from sanic.app import Sanic
|
from sanic.app import Sanic
|
||||||
from sanic.blueprints import Blueprint
|
from sanic.blueprints import Blueprint
|
||||||
|
|
||||||
__version__ = '0.8.3'
|
__version__ = "0.8.3"
|
||||||
|
|
||||||
__all__ = ['Sanic', 'Blueprint']
|
__all__ = ["Sanic", "Blueprint"]
|
||||||
|
|
|
@ -5,16 +5,18 @@ from sanic.log import logger
|
||||||
from sanic.app import Sanic
|
from sanic.app import Sanic
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
parser = ArgumentParser(prog='sanic')
|
parser = ArgumentParser(prog="sanic")
|
||||||
parser.add_argument('--host', dest='host', type=str, default='127.0.0.1')
|
parser.add_argument("--host", dest="host", type=str, default="127.0.0.1")
|
||||||
parser.add_argument('--port', dest='port', type=int, default=8000)
|
parser.add_argument("--port", dest="port", type=int, default=8000)
|
||||||
parser.add_argument('--cert', dest='cert', type=str,
|
parser.add_argument(
|
||||||
help='location of certificate for SSL')
|
"--cert", dest="cert", type=str, help="location of certificate for SSL"
|
||||||
parser.add_argument('--key', dest='key', type=str,
|
)
|
||||||
help='location of keyfile for SSL.')
|
parser.add_argument(
|
||||||
parser.add_argument('--workers', dest='workers', type=int, default=1, )
|
"--key", dest="key", type=str, help="location of keyfile for SSL."
|
||||||
parser.add_argument('--debug', dest='debug', action="store_true")
|
)
|
||||||
parser.add_argument('module')
|
parser.add_argument("--workers", dest="workers", type=int, default=1)
|
||||||
|
parser.add_argument("--debug", dest="debug", action="store_true")
|
||||||
|
parser.add_argument("module")
|
||||||
args = parser.parse_args()
|
args = parser.parse_args()
|
||||||
|
|
||||||
try:
|
try:
|
||||||
|
@ -25,20 +27,29 @@ if __name__ == "__main__":
|
||||||
module = import_module(module_name)
|
module = import_module(module_name)
|
||||||
app = getattr(module, app_name, None)
|
app = getattr(module, app_name, None)
|
||||||
if not isinstance(app, Sanic):
|
if not isinstance(app, Sanic):
|
||||||
raise ValueError("Module is not a Sanic app, it is a {}. "
|
raise ValueError(
|
||||||
"Perhaps you meant {}.app?"
|
"Module is not a Sanic app, it is a {}. "
|
||||||
.format(type(app).__name__, args.module))
|
"Perhaps you meant {}.app?".format(
|
||||||
|
type(app).__name__, args.module
|
||||||
|
)
|
||||||
|
)
|
||||||
if args.cert is not None or args.key is not None:
|
if args.cert is not None or args.key is not None:
|
||||||
ssl = {'cert': args.cert, 'key': args.key}
|
ssl = {"cert": args.cert, "key": args.key}
|
||||||
else:
|
else:
|
||||||
ssl = None
|
ssl = None
|
||||||
|
|
||||||
app.run(host=args.host, port=args.port,
|
app.run(
|
||||||
workers=args.workers, debug=args.debug, ssl=ssl)
|
host=args.host,
|
||||||
|
port=args.port,
|
||||||
|
workers=args.workers,
|
||||||
|
debug=args.debug,
|
||||||
|
ssl=ssl,
|
||||||
|
)
|
||||||
except ImportError as e:
|
except ImportError as e:
|
||||||
logger.error("No module named {} found.\n"
|
logger.error(
|
||||||
|
"No module named {} found.\n"
|
||||||
" Example File: project/sanic_server.py -> app\n"
|
" Example File: project/sanic_server.py -> app\n"
|
||||||
" Example Module: project.sanic_server.app"
|
" Example Module: project.sanic_server.app".format(e.name)
|
||||||
.format(e.name))
|
)
|
||||||
except ValueError as e:
|
except ValueError as e:
|
||||||
logger.exception("Failed to run app")
|
logger.exception("Failed to run app")
|
||||||
|
|
643
sanic/app.py
643
sanic/app.py
|
@ -27,10 +27,17 @@ import sanic.reloader_helpers as reloader_helpers
|
||||||
|
|
||||||
|
|
||||||
class Sanic:
|
class Sanic:
|
||||||
def __init__(self, name=None, router=None, error_handler=None,
|
def __init__(
|
||||||
load_env=True, request_class=None,
|
self,
|
||||||
strict_slashes=False, log_config=None,
|
name=None,
|
||||||
configure_logging=True):
|
router=None,
|
||||||
|
error_handler=None,
|
||||||
|
load_env=True,
|
||||||
|
request_class=None,
|
||||||
|
strict_slashes=False,
|
||||||
|
log_config=None,
|
||||||
|
configure_logging=True,
|
||||||
|
):
|
||||||
|
|
||||||
# Get name from previous stack frame
|
# Get name from previous stack frame
|
||||||
if name is None:
|
if name is None:
|
||||||
|
@ -71,8 +78,9 @@ class Sanic:
|
||||||
"""
|
"""
|
||||||
if not self.is_running:
|
if not self.is_running:
|
||||||
raise SanicException(
|
raise SanicException(
|
||||||
'Loop can only be retrieved after the app has started '
|
"Loop can only be retrieved after the app has started "
|
||||||
'running. Not supported with `create_server` function')
|
"running. Not supported with `create_server` function"
|
||||||
|
)
|
||||||
return get_event_loop()
|
return get_event_loop()
|
||||||
|
|
||||||
# -------------------------------------------------------------------- #
|
# -------------------------------------------------------------------- #
|
||||||
|
@ -96,7 +104,8 @@ class Sanic:
|
||||||
else:
|
else:
|
||||||
self.loop.create_task(task)
|
self.loop.create_task(task)
|
||||||
except SanicException:
|
except SanicException:
|
||||||
@self.listener('before_server_start')
|
|
||||||
|
@self.listener("before_server_start")
|
||||||
def run(app, loop):
|
def run(app, loop):
|
||||||
if callable(task):
|
if callable(task):
|
||||||
try:
|
try:
|
||||||
|
@ -133,8 +142,16 @@ class Sanic:
|
||||||
return self.listener(event)(listener)
|
return self.listener(event)(listener)
|
||||||
|
|
||||||
# Decorator
|
# Decorator
|
||||||
def route(self, uri, methods=frozenset({'GET'}), host=None,
|
def route(
|
||||||
strict_slashes=None, stream=False, version=None, name=None):
|
self,
|
||||||
|
uri,
|
||||||
|
methods=frozenset({"GET"}),
|
||||||
|
host=None,
|
||||||
|
strict_slashes=None,
|
||||||
|
stream=False,
|
||||||
|
version=None,
|
||||||
|
name=None,
|
||||||
|
):
|
||||||
"""Decorate a function to be registered as a route
|
"""Decorate a function to be registered as a route
|
||||||
|
|
||||||
:param uri: path of the URL
|
:param uri: path of the URL
|
||||||
|
@ -149,8 +166,8 @@ class Sanic:
|
||||||
|
|
||||||
# Fix case where the user did not prefix the URL with a /
|
# Fix case where the user did not prefix the URL with a /
|
||||||
# and will probably get confused as to why it's not working
|
# and will probably get confused as to why it's not working
|
||||||
if not uri.startswith('/'):
|
if not uri.startswith("/"):
|
||||||
uri = '/' + uri
|
uri = "/" + uri
|
||||||
|
|
||||||
if stream:
|
if stream:
|
||||||
self.is_request_stream = True
|
self.is_request_stream = True
|
||||||
|
@ -164,63 +181,141 @@ class Sanic:
|
||||||
if stream:
|
if stream:
|
||||||
handler.is_stream = stream
|
handler.is_stream = stream
|
||||||
|
|
||||||
self.router.add(uri=uri, methods=methods, handler=handler,
|
self.router.add(
|
||||||
host=host, strict_slashes=strict_slashes,
|
uri=uri,
|
||||||
version=version, name=name)
|
methods=methods,
|
||||||
|
handler=handler,
|
||||||
|
host=host,
|
||||||
|
strict_slashes=strict_slashes,
|
||||||
|
version=version,
|
||||||
|
name=name,
|
||||||
|
)
|
||||||
return handler
|
return handler
|
||||||
else:
|
else:
|
||||||
raise ValueError(
|
raise ValueError(
|
||||||
'Required parameter `request` missing '
|
"Required parameter `request` missing "
|
||||||
'in the {0}() route?'.format(
|
"in the {0}() route?".format(handler.__name__)
|
||||||
handler.__name__))
|
)
|
||||||
|
|
||||||
return response
|
return response
|
||||||
|
|
||||||
# Shorthand method decorators
|
# Shorthand method decorators
|
||||||
def get(self, uri, host=None, strict_slashes=None, version=None,
|
def get(
|
||||||
name=None):
|
self, uri, host=None, strict_slashes=None, version=None, name=None
|
||||||
return self.route(uri, methods=frozenset({"GET"}), host=host,
|
):
|
||||||
strict_slashes=strict_slashes, version=version,
|
return self.route(
|
||||||
name=name)
|
uri,
|
||||||
|
methods=frozenset({"GET"}),
|
||||||
|
host=host,
|
||||||
|
strict_slashes=strict_slashes,
|
||||||
|
version=version,
|
||||||
|
name=name,
|
||||||
|
)
|
||||||
|
|
||||||
def post(self, uri, host=None, strict_slashes=None, stream=False,
|
def post(
|
||||||
version=None, name=None):
|
self,
|
||||||
return self.route(uri, methods=frozenset({"POST"}), host=host,
|
uri,
|
||||||
strict_slashes=strict_slashes, stream=stream,
|
host=None,
|
||||||
version=version, name=name)
|
strict_slashes=None,
|
||||||
|
stream=False,
|
||||||
|
version=None,
|
||||||
|
name=None,
|
||||||
|
):
|
||||||
|
return self.route(
|
||||||
|
uri,
|
||||||
|
methods=frozenset({"POST"}),
|
||||||
|
host=host,
|
||||||
|
strict_slashes=strict_slashes,
|
||||||
|
stream=stream,
|
||||||
|
version=version,
|
||||||
|
name=name,
|
||||||
|
)
|
||||||
|
|
||||||
def put(self, uri, host=None, strict_slashes=None, stream=False,
|
def put(
|
||||||
version=None, name=None):
|
self,
|
||||||
return self.route(uri, methods=frozenset({"PUT"}), host=host,
|
uri,
|
||||||
strict_slashes=strict_slashes, stream=stream,
|
host=None,
|
||||||
version=version, name=name)
|
strict_slashes=None,
|
||||||
|
stream=False,
|
||||||
|
version=None,
|
||||||
|
name=None,
|
||||||
|
):
|
||||||
|
return self.route(
|
||||||
|
uri,
|
||||||
|
methods=frozenset({"PUT"}),
|
||||||
|
host=host,
|
||||||
|
strict_slashes=strict_slashes,
|
||||||
|
stream=stream,
|
||||||
|
version=version,
|
||||||
|
name=name,
|
||||||
|
)
|
||||||
|
|
||||||
def head(self, uri, host=None, strict_slashes=None, version=None,
|
def head(
|
||||||
name=None):
|
self, uri, host=None, strict_slashes=None, version=None, name=None
|
||||||
return self.route(uri, methods=frozenset({"HEAD"}), host=host,
|
):
|
||||||
strict_slashes=strict_slashes, version=version,
|
return self.route(
|
||||||
name=name)
|
uri,
|
||||||
|
methods=frozenset({"HEAD"}),
|
||||||
|
host=host,
|
||||||
|
strict_slashes=strict_slashes,
|
||||||
|
version=version,
|
||||||
|
name=name,
|
||||||
|
)
|
||||||
|
|
||||||
def options(self, uri, host=None, strict_slashes=None, version=None,
|
def options(
|
||||||
name=None):
|
self, uri, host=None, strict_slashes=None, version=None, name=None
|
||||||
return self.route(uri, methods=frozenset({"OPTIONS"}), host=host,
|
):
|
||||||
strict_slashes=strict_slashes, version=version,
|
return self.route(
|
||||||
name=name)
|
uri,
|
||||||
|
methods=frozenset({"OPTIONS"}),
|
||||||
|
host=host,
|
||||||
|
strict_slashes=strict_slashes,
|
||||||
|
version=version,
|
||||||
|
name=name,
|
||||||
|
)
|
||||||
|
|
||||||
def patch(self, uri, host=None, strict_slashes=None, stream=False,
|
def patch(
|
||||||
version=None, name=None):
|
self,
|
||||||
return self.route(uri, methods=frozenset({"PATCH"}), host=host,
|
uri,
|
||||||
strict_slashes=strict_slashes, stream=stream,
|
host=None,
|
||||||
version=version, name=name)
|
strict_slashes=None,
|
||||||
|
stream=False,
|
||||||
|
version=None,
|
||||||
|
name=None,
|
||||||
|
):
|
||||||
|
return self.route(
|
||||||
|
uri,
|
||||||
|
methods=frozenset({"PATCH"}),
|
||||||
|
host=host,
|
||||||
|
strict_slashes=strict_slashes,
|
||||||
|
stream=stream,
|
||||||
|
version=version,
|
||||||
|
name=name,
|
||||||
|
)
|
||||||
|
|
||||||
def delete(self, uri, host=None, strict_slashes=None, version=None,
|
def delete(
|
||||||
name=None):
|
self, uri, host=None, strict_slashes=None, version=None, name=None
|
||||||
return self.route(uri, methods=frozenset({"DELETE"}), host=host,
|
):
|
||||||
strict_slashes=strict_slashes, version=version,
|
return self.route(
|
||||||
name=name)
|
uri,
|
||||||
|
methods=frozenset({"DELETE"}),
|
||||||
|
host=host,
|
||||||
|
strict_slashes=strict_slashes,
|
||||||
|
version=version,
|
||||||
|
name=name,
|
||||||
|
)
|
||||||
|
|
||||||
def add_route(self, handler, uri, methods=frozenset({'GET'}), host=None,
|
def add_route(
|
||||||
strict_slashes=None, version=None, name=None, stream=False):
|
self,
|
||||||
|
handler,
|
||||||
|
uri,
|
||||||
|
methods=frozenset({"GET"}),
|
||||||
|
host=None,
|
||||||
|
strict_slashes=None,
|
||||||
|
version=None,
|
||||||
|
name=None,
|
||||||
|
stream=False,
|
||||||
|
):
|
||||||
"""A helper method to register class instance or
|
"""A helper method to register class instance or
|
||||||
functions as a handler to the application url
|
functions as a handler to the application url
|
||||||
routes.
|
routes.
|
||||||
|
@ -237,35 +332,42 @@ class Sanic:
|
||||||
:return: function or class instance
|
:return: function or class instance
|
||||||
"""
|
"""
|
||||||
# Handle HTTPMethodView differently
|
# Handle HTTPMethodView differently
|
||||||
if hasattr(handler, 'view_class'):
|
if hasattr(handler, "view_class"):
|
||||||
methods = set()
|
methods = set()
|
||||||
|
|
||||||
for method in HTTP_METHODS:
|
for method in HTTP_METHODS:
|
||||||
_handler = getattr(handler.view_class, method.lower(), None)
|
_handler = getattr(handler.view_class, method.lower(), None)
|
||||||
if _handler:
|
if _handler:
|
||||||
methods.add(method)
|
methods.add(method)
|
||||||
if hasattr(_handler, 'is_stream'):
|
if hasattr(_handler, "is_stream"):
|
||||||
stream = True
|
stream = True
|
||||||
|
|
||||||
# handle composition view differently
|
# handle composition view differently
|
||||||
if isinstance(handler, CompositionView):
|
if isinstance(handler, CompositionView):
|
||||||
methods = handler.handlers.keys()
|
methods = handler.handlers.keys()
|
||||||
for _handler in handler.handlers.values():
|
for _handler in handler.handlers.values():
|
||||||
if hasattr(_handler, 'is_stream'):
|
if hasattr(_handler, "is_stream"):
|
||||||
stream = True
|
stream = True
|
||||||
break
|
break
|
||||||
|
|
||||||
if strict_slashes is None:
|
if strict_slashes is None:
|
||||||
strict_slashes = self.strict_slashes
|
strict_slashes = self.strict_slashes
|
||||||
|
|
||||||
self.route(uri=uri, methods=methods, host=host,
|
self.route(
|
||||||
strict_slashes=strict_slashes, stream=stream,
|
uri=uri,
|
||||||
version=version, name=name)(handler)
|
methods=methods,
|
||||||
|
host=host,
|
||||||
|
strict_slashes=strict_slashes,
|
||||||
|
stream=stream,
|
||||||
|
version=version,
|
||||||
|
name=name,
|
||||||
|
)(handler)
|
||||||
return handler
|
return handler
|
||||||
|
|
||||||
# Decorator
|
# Decorator
|
||||||
def websocket(self, uri, host=None, strict_slashes=None,
|
def websocket(
|
||||||
subprotocols=None, name=None):
|
self, uri, host=None, strict_slashes=None, subprotocols=None, name=None
|
||||||
|
):
|
||||||
"""Decorate a function to be registered as a websocket route
|
"""Decorate a function to be registered as a websocket route
|
||||||
:param uri: path of the URL
|
:param uri: path of the URL
|
||||||
:param subprotocols: optional list of strings with the supported
|
:param subprotocols: optional list of strings with the supported
|
||||||
|
@ -277,8 +379,8 @@ class Sanic:
|
||||||
|
|
||||||
# Fix case where the user did not prefix the URL with a /
|
# Fix case where the user did not prefix the URL with a /
|
||||||
# and will probably get confused as to why it's not working
|
# and will probably get confused as to why it's not working
|
||||||
if not uri.startswith('/'):
|
if not uri.startswith("/"):
|
||||||
uri = '/' + uri
|
uri = "/" + uri
|
||||||
|
|
||||||
if strict_slashes is None:
|
if strict_slashes is None:
|
||||||
strict_slashes = self.strict_slashes
|
strict_slashes = self.strict_slashes
|
||||||
|
@ -307,21 +409,38 @@ class Sanic:
|
||||||
self.websocket_tasks.remove(fut)
|
self.websocket_tasks.remove(fut)
|
||||||
await ws.close()
|
await ws.close()
|
||||||
|
|
||||||
self.router.add(uri=uri, handler=websocket_handler,
|
self.router.add(
|
||||||
methods=frozenset({'GET'}), host=host,
|
uri=uri,
|
||||||
strict_slashes=strict_slashes, name=name)
|
handler=websocket_handler,
|
||||||
|
methods=frozenset({"GET"}),
|
||||||
|
host=host,
|
||||||
|
strict_slashes=strict_slashes,
|
||||||
|
name=name,
|
||||||
|
)
|
||||||
return handler
|
return handler
|
||||||
|
|
||||||
return response
|
return response
|
||||||
|
|
||||||
def add_websocket_route(self, handler, uri, host=None,
|
def add_websocket_route(
|
||||||
strict_slashes=None, subprotocols=None, name=None):
|
self,
|
||||||
|
handler,
|
||||||
|
uri,
|
||||||
|
host=None,
|
||||||
|
strict_slashes=None,
|
||||||
|
subprotocols=None,
|
||||||
|
name=None,
|
||||||
|
):
|
||||||
"""A helper method to register a function as a websocket route."""
|
"""A helper method to register a function as a websocket route."""
|
||||||
if strict_slashes is None:
|
if strict_slashes is None:
|
||||||
strict_slashes = self.strict_slashes
|
strict_slashes = self.strict_slashes
|
||||||
|
|
||||||
return self.websocket(uri, host=host, strict_slashes=strict_slashes,
|
return self.websocket(
|
||||||
subprotocols=subprotocols, name=name)(handler)
|
uri,
|
||||||
|
host=host,
|
||||||
|
strict_slashes=strict_slashes,
|
||||||
|
subprotocols=subprotocols,
|
||||||
|
name=name,
|
||||||
|
)(handler)
|
||||||
|
|
||||||
def enable_websocket(self, enable=True):
|
def enable_websocket(self, enable=True):
|
||||||
"""Enable or disable the support for websocket.
|
"""Enable or disable the support for websocket.
|
||||||
|
@ -332,7 +451,7 @@ class Sanic:
|
||||||
if not self.websocket_enabled:
|
if not self.websocket_enabled:
|
||||||
# if the server is stopped, we want to cancel any ongoing
|
# if the server is stopped, we want to cancel any ongoing
|
||||||
# websocket tasks, to allow the server to exit promptly
|
# websocket tasks, to allow the server to exit promptly
|
||||||
@self.listener('before_server_stop')
|
@self.listener("before_server_stop")
|
||||||
def cancel_websocket_tasks(app, loop):
|
def cancel_websocket_tasks(app, loop):
|
||||||
for task in self.websocket_tasks:
|
for task in self.websocket_tasks:
|
||||||
task.cancel()
|
task.cancel()
|
||||||
|
@ -361,10 +480,10 @@ class Sanic:
|
||||||
|
|
||||||
return response
|
return response
|
||||||
|
|
||||||
def register_middleware(self, middleware, attach_to='request'):
|
def register_middleware(self, middleware, attach_to="request"):
|
||||||
if attach_to == 'request':
|
if attach_to == "request":
|
||||||
self.request_middleware.append(middleware)
|
self.request_middleware.append(middleware)
|
||||||
if attach_to == 'response':
|
if attach_to == "response":
|
||||||
self.response_middleware.appendleft(middleware)
|
self.response_middleware.appendleft(middleware)
|
||||||
return middleware
|
return middleware
|
||||||
|
|
||||||
|
@ -379,21 +498,40 @@ class Sanic:
|
||||||
return self.register_middleware(middleware_or_request)
|
return self.register_middleware(middleware_or_request)
|
||||||
|
|
||||||
else:
|
else:
|
||||||
return partial(self.register_middleware,
|
return partial(
|
||||||
attach_to=middleware_or_request)
|
self.register_middleware, attach_to=middleware_or_request
|
||||||
|
)
|
||||||
|
|
||||||
# Static Files
|
# Static Files
|
||||||
def static(self, uri, file_or_directory, pattern=r'/?.+',
|
def static(
|
||||||
use_modified_since=True, use_content_range=False,
|
self,
|
||||||
stream_large_files=False, name='static', host=None,
|
uri,
|
||||||
strict_slashes=None, content_type=None):
|
file_or_directory,
|
||||||
|
pattern=r"/?.+",
|
||||||
|
use_modified_since=True,
|
||||||
|
use_content_range=False,
|
||||||
|
stream_large_files=False,
|
||||||
|
name="static",
|
||||||
|
host=None,
|
||||||
|
strict_slashes=None,
|
||||||
|
content_type=None,
|
||||||
|
):
|
||||||
"""Register a root to serve files from. The input can either be a
|
"""Register a root to serve files from. The input can either be a
|
||||||
file or a directory. See
|
file or a directory. See
|
||||||
"""
|
"""
|
||||||
static_register(self, uri, file_or_directory, pattern,
|
static_register(
|
||||||
use_modified_since, use_content_range,
|
self,
|
||||||
stream_large_files, name, host, strict_slashes,
|
uri,
|
||||||
content_type)
|
file_or_directory,
|
||||||
|
pattern,
|
||||||
|
use_modified_since,
|
||||||
|
use_content_range,
|
||||||
|
stream_large_files,
|
||||||
|
name,
|
||||||
|
host,
|
||||||
|
strict_slashes,
|
||||||
|
content_type,
|
||||||
|
)
|
||||||
|
|
||||||
def blueprint(self, blueprint, **options):
|
def blueprint(self, blueprint, **options):
|
||||||
"""Register a blueprint on the application.
|
"""Register a blueprint on the application.
|
||||||
|
@ -407,10 +545,10 @@ class Sanic:
|
||||||
self.blueprint(item, **options)
|
self.blueprint(item, **options)
|
||||||
return
|
return
|
||||||
if blueprint.name in self.blueprints:
|
if blueprint.name in self.blueprints:
|
||||||
assert self.blueprints[blueprint.name] is blueprint, \
|
assert self.blueprints[blueprint.name] is blueprint, (
|
||||||
'A blueprint with the name "%s" is already registered. ' \
|
'A blueprint with the name "%s" is already registered. '
|
||||||
'Blueprint names must be unique.' % \
|
"Blueprint names must be unique." % (blueprint.name,)
|
||||||
(blueprint.name,)
|
)
|
||||||
else:
|
else:
|
||||||
self.blueprints[blueprint.name] = blueprint
|
self.blueprints[blueprint.name] = blueprint
|
||||||
self._blueprint_order.append(blueprint)
|
self._blueprint_order.append(blueprint)
|
||||||
|
@ -419,11 +557,13 @@ class Sanic:
|
||||||
def register_blueprint(self, *args, **kwargs):
|
def register_blueprint(self, *args, **kwargs):
|
||||||
# TODO: deprecate 1.0
|
# TODO: deprecate 1.0
|
||||||
if self.debug:
|
if self.debug:
|
||||||
warnings.simplefilter('default')
|
warnings.simplefilter("default")
|
||||||
warnings.warn("Use of register_blueprint will be deprecated in "
|
warnings.warn(
|
||||||
|
"Use of register_blueprint will be deprecated in "
|
||||||
"version 1.0. Please use the blueprint method"
|
"version 1.0. Please use the blueprint method"
|
||||||
" instead",
|
" instead",
|
||||||
DeprecationWarning)
|
DeprecationWarning,
|
||||||
|
)
|
||||||
return self.blueprint(*args, **kwargs)
|
return self.blueprint(*args, **kwargs)
|
||||||
|
|
||||||
def url_for(self, view_name: str, **kwargs):
|
def url_for(self, view_name: str, **kwargs):
|
||||||
|
@ -449,67 +589,66 @@ class Sanic:
|
||||||
# find the route by the supplied view name
|
# find the route by the supplied view name
|
||||||
kw = {}
|
kw = {}
|
||||||
# special static files url_for
|
# special static files url_for
|
||||||
if view_name == 'static':
|
if view_name == "static":
|
||||||
kw.update(name=kwargs.pop('name', 'static'))
|
kw.update(name=kwargs.pop("name", "static"))
|
||||||
elif view_name.endswith('.static'): # blueprint.static
|
elif view_name.endswith(".static"): # blueprint.static
|
||||||
kwargs.pop('name', None)
|
kwargs.pop("name", None)
|
||||||
kw.update(name=view_name)
|
kw.update(name=view_name)
|
||||||
|
|
||||||
uri, route = self.router.find_route_by_view_name(view_name, **kw)
|
uri, route = self.router.find_route_by_view_name(view_name, **kw)
|
||||||
if not (uri and route):
|
if not (uri and route):
|
||||||
raise URLBuildError('Endpoint with name `{}` was not found'.format(
|
raise URLBuildError(
|
||||||
view_name))
|
"Endpoint with name `{}` was not found".format(view_name)
|
||||||
|
)
|
||||||
|
|
||||||
if view_name == 'static' or view_name.endswith('.static'):
|
if view_name == "static" or view_name.endswith(".static"):
|
||||||
filename = kwargs.pop('filename', None)
|
filename = kwargs.pop("filename", None)
|
||||||
# it's static folder
|
# it's static folder
|
||||||
if '<file_uri:' in uri:
|
if "<file_uri:" in uri:
|
||||||
folder_ = uri.split('<file_uri:', 1)[0]
|
folder_ = uri.split("<file_uri:", 1)[0]
|
||||||
if folder_.endswith('/'):
|
if folder_.endswith("/"):
|
||||||
folder_ = folder_[:-1]
|
folder_ = folder_[:-1]
|
||||||
|
|
||||||
if filename.startswith('/'):
|
if filename.startswith("/"):
|
||||||
filename = filename[1:]
|
filename = filename[1:]
|
||||||
|
|
||||||
uri = '{}/{}'.format(folder_, filename)
|
uri = "{}/{}".format(folder_, filename)
|
||||||
|
|
||||||
if uri != '/' and uri.endswith('/'):
|
if uri != "/" and uri.endswith("/"):
|
||||||
uri = uri[:-1]
|
uri = uri[:-1]
|
||||||
|
|
||||||
out = uri
|
out = uri
|
||||||
|
|
||||||
# find all the parameters we will need to build in the URL
|
# find all the parameters we will need to build in the URL
|
||||||
matched_params = re.findall(
|
matched_params = re.findall(self.router.parameter_pattern, uri)
|
||||||
self.router.parameter_pattern, uri)
|
|
||||||
|
|
||||||
# _method is only a placeholder now, don't know how to support it
|
# _method is only a placeholder now, don't know how to support it
|
||||||
kwargs.pop('_method', None)
|
kwargs.pop("_method", None)
|
||||||
anchor = kwargs.pop('_anchor', '')
|
anchor = kwargs.pop("_anchor", "")
|
||||||
# _external need SERVER_NAME in config or pass _server arg
|
# _external need SERVER_NAME in config or pass _server arg
|
||||||
external = kwargs.pop('_external', False)
|
external = kwargs.pop("_external", False)
|
||||||
scheme = kwargs.pop('_scheme', '')
|
scheme = kwargs.pop("_scheme", "")
|
||||||
if scheme and not external:
|
if scheme and not external:
|
||||||
raise ValueError('When specifying _scheme, _external must be True')
|
raise ValueError("When specifying _scheme, _external must be True")
|
||||||
|
|
||||||
netloc = kwargs.pop('_server', None)
|
netloc = kwargs.pop("_server", None)
|
||||||
if netloc is None and external:
|
if netloc is None and external:
|
||||||
netloc = self.config.get('SERVER_NAME', '')
|
netloc = self.config.get("SERVER_NAME", "")
|
||||||
|
|
||||||
if external:
|
if external:
|
||||||
if not scheme:
|
if not scheme:
|
||||||
if ':' in netloc[:8]:
|
if ":" in netloc[:8]:
|
||||||
scheme = netloc[:8].split(':', 1)[0]
|
scheme = netloc[:8].split(":", 1)[0]
|
||||||
else:
|
else:
|
||||||
scheme = 'http'
|
scheme = "http"
|
||||||
|
|
||||||
if '://' in netloc[:8]:
|
if "://" in netloc[:8]:
|
||||||
netloc = netloc.split('://', 1)[-1]
|
netloc = netloc.split("://", 1)[-1]
|
||||||
|
|
||||||
for match in matched_params:
|
for match in matched_params:
|
||||||
name, _type, pattern = self.router.parse_parameter_string(
|
name, _type, pattern = self.router.parse_parameter_string(match)
|
||||||
match)
|
|
||||||
# we only want to match against each individual parameter
|
# we only want to match against each individual parameter
|
||||||
specific_pattern = '^{}$'.format(pattern)
|
specific_pattern = "^{}$".format(pattern)
|
||||||
supplied_param = None
|
supplied_param = None
|
||||||
|
|
||||||
if name in kwargs:
|
if name in kwargs:
|
||||||
|
@ -517,8 +656,10 @@ class Sanic:
|
||||||
del kwargs[name]
|
del kwargs[name]
|
||||||
else:
|
else:
|
||||||
raise URLBuildError(
|
raise URLBuildError(
|
||||||
'Required parameter `{}` was not passed to url_for'.format(
|
"Required parameter `{}` was not passed to url_for".format(
|
||||||
name))
|
name
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
supplied_param = str(supplied_param)
|
supplied_param = str(supplied_param)
|
||||||
# determine if the parameter supplied by the caller passes the test
|
# determine if the parameter supplied by the caller passes the test
|
||||||
|
@ -529,25 +670,28 @@ class Sanic:
|
||||||
if _type != str:
|
if _type != str:
|
||||||
msg = (
|
msg = (
|
||||||
'Value "{}" for parameter `{}` does not '
|
'Value "{}" for parameter `{}` does not '
|
||||||
'match pattern for type `{}`: {}'.format(
|
"match pattern for type `{}`: {}".format(
|
||||||
supplied_param, name, _type.__name__, pattern))
|
supplied_param, name, _type.__name__, pattern
|
||||||
|
)
|
||||||
|
)
|
||||||
else:
|
else:
|
||||||
msg = (
|
msg = (
|
||||||
'Value "{}" for parameter `{}` '
|
'Value "{}" for parameter `{}` '
|
||||||
'does not satisfy pattern {}'.format(
|
"does not satisfy pattern {}".format(
|
||||||
supplied_param, name, pattern))
|
supplied_param, name, pattern
|
||||||
|
)
|
||||||
|
)
|
||||||
raise URLBuildError(msg)
|
raise URLBuildError(msg)
|
||||||
|
|
||||||
# replace the parameter in the URL with the supplied value
|
# replace the parameter in the URL with the supplied value
|
||||||
replacement_regex = '(<{}.*?>)'.format(name)
|
replacement_regex = "(<{}.*?>)".format(name)
|
||||||
|
|
||||||
out = re.sub(
|
out = re.sub(replacement_regex, supplied_param, out)
|
||||||
replacement_regex, supplied_param, out)
|
|
||||||
|
|
||||||
# parse the remainder of the keyword arguments into a querystring
|
# parse the remainder of the keyword arguments into a querystring
|
||||||
query_string = urlencode(kwargs, doseq=True) if kwargs else ''
|
query_string = urlencode(kwargs, doseq=True) if kwargs else ""
|
||||||
# scheme://netloc/path;parameters?query#fragment
|
# scheme://netloc/path;parameters?query#fragment
|
||||||
out = urlunparse((scheme, netloc, out, '', query_string, anchor))
|
out = urlunparse((scheme, netloc, out, "", query_string, anchor))
|
||||||
|
|
||||||
return out
|
return out
|
||||||
|
|
||||||
|
@ -594,8 +738,11 @@ class Sanic:
|
||||||
request.uri_template = uri
|
request.uri_template = uri
|
||||||
if handler is None:
|
if handler is None:
|
||||||
raise ServerError(
|
raise ServerError(
|
||||||
("'None' was returned while requesting a "
|
(
|
||||||
"handler from the router"))
|
"'None' was returned while requesting a "
|
||||||
|
"handler from the router"
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
# Run response handler
|
# Run response handler
|
||||||
response = handler(request, *args, **kwargs)
|
response = handler(request, *args, **kwargs)
|
||||||
|
@ -619,16 +766,20 @@ class Sanic:
|
||||||
response = await response
|
response = await response
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
if isinstance(e, SanicException):
|
if isinstance(e, SanicException):
|
||||||
response = self.error_handler.default(request=request,
|
response = self.error_handler.default(
|
||||||
exception=e)
|
request=request, exception=e
|
||||||
|
)
|
||||||
elif self.debug:
|
elif self.debug:
|
||||||
response = HTTPResponse(
|
response = HTTPResponse(
|
||||||
"Error while handling error: {}\nStack: {}".format(
|
"Error while handling error: {}\nStack: {}".format(
|
||||||
e, format_exc()), status=500)
|
e, format_exc()
|
||||||
|
),
|
||||||
|
status=500,
|
||||||
|
)
|
||||||
else:
|
else:
|
||||||
response = HTTPResponse(
|
response = HTTPResponse(
|
||||||
"An error occurred while handling an error",
|
"An error occurred while handling an error", status=500
|
||||||
status=500)
|
)
|
||||||
finally:
|
finally:
|
||||||
# -------------------------------------------- #
|
# -------------------------------------------- #
|
||||||
# Response Middleware
|
# Response Middleware
|
||||||
|
@ -636,16 +787,17 @@ class Sanic:
|
||||||
# Don't run response middleware if response is None
|
# Don't run response middleware if response is None
|
||||||
if response is not None:
|
if response is not None:
|
||||||
try:
|
try:
|
||||||
response = await self._run_response_middleware(request,
|
response = await self._run_response_middleware(
|
||||||
response)
|
request, response
|
||||||
|
)
|
||||||
except CancelledError:
|
except CancelledError:
|
||||||
# Response middleware can timeout too, as above.
|
# Response middleware can timeout too, as above.
|
||||||
response = None
|
response = None
|
||||||
cancelled = True
|
cancelled = True
|
||||||
except BaseException:
|
except BaseException:
|
||||||
error_logger.exception(
|
error_logger.exception(
|
||||||
'Exception occurred in one of response '
|
"Exception occurred in one of response "
|
||||||
'middleware handlers'
|
"middleware handlers"
|
||||||
)
|
)
|
||||||
if cancelled:
|
if cancelled:
|
||||||
raise CancelledError()
|
raise CancelledError()
|
||||||
|
@ -668,10 +820,21 @@ class Sanic:
|
||||||
# Execution
|
# Execution
|
||||||
# -------------------------------------------------------------------- #
|
# -------------------------------------------------------------------- #
|
||||||
|
|
||||||
def run(self, host=None, port=None, debug=False, ssl=None,
|
def run(
|
||||||
sock=None, workers=1, protocol=None,
|
self,
|
||||||
backlog=100, stop_event=None, register_sys_signals=True,
|
host=None,
|
||||||
access_log=True, **kwargs):
|
port=None,
|
||||||
|
debug=False,
|
||||||
|
ssl=None,
|
||||||
|
sock=None,
|
||||||
|
workers=1,
|
||||||
|
protocol=None,
|
||||||
|
backlog=100,
|
||||||
|
stop_event=None,
|
||||||
|
register_sys_signals=True,
|
||||||
|
access_log=True,
|
||||||
|
**kwargs
|
||||||
|
):
|
||||||
"""Run the HTTP Server and listen until keyboard interrupt or term
|
"""Run the HTTP Server and listen until keyboard interrupt or term
|
||||||
signal. On termination, drain connections before closing.
|
signal. On termination, drain connections before closing.
|
||||||
|
|
||||||
|
@ -692,7 +855,7 @@ class Sanic:
|
||||||
# Default auto_reload to false
|
# Default auto_reload to false
|
||||||
auto_reload = False
|
auto_reload = False
|
||||||
# If debug is set, default it to true (unless on windows)
|
# If debug is set, default it to true (unless on windows)
|
||||||
if debug and os.name == 'posix':
|
if debug and os.name == "posix":
|
||||||
auto_reload = True
|
auto_reload = True
|
||||||
# Allow for overriding either of the defaults
|
# Allow for overriding either of the defaults
|
||||||
auto_reload = kwargs.get("auto_reload", auto_reload)
|
auto_reload = kwargs.get("auto_reload", auto_reload)
|
||||||
|
@ -701,30 +864,43 @@ class Sanic:
|
||||||
host, port = host or "127.0.0.1", port or 8000
|
host, port = host or "127.0.0.1", port or 8000
|
||||||
|
|
||||||
if protocol is None:
|
if protocol is None:
|
||||||
protocol = (WebSocketProtocol if self.websocket_enabled
|
protocol = (
|
||||||
else HttpProtocol)
|
WebSocketProtocol if self.websocket_enabled else HttpProtocol
|
||||||
|
)
|
||||||
if stop_event is not None:
|
if stop_event is not None:
|
||||||
if debug:
|
if debug:
|
||||||
warnings.simplefilter('default')
|
warnings.simplefilter("default")
|
||||||
warnings.warn("stop_event will be removed from future versions.",
|
warnings.warn(
|
||||||
DeprecationWarning)
|
"stop_event will be removed from future versions.",
|
||||||
|
DeprecationWarning,
|
||||||
|
)
|
||||||
# compatibility old access_log params
|
# compatibility old access_log params
|
||||||
self.config.ACCESS_LOG = access_log
|
self.config.ACCESS_LOG = access_log
|
||||||
server_settings = self._helper(
|
server_settings = self._helper(
|
||||||
host=host, port=port, debug=debug, ssl=ssl, sock=sock,
|
host=host,
|
||||||
workers=workers, protocol=protocol, backlog=backlog,
|
port=port,
|
||||||
register_sys_signals=register_sys_signals, auto_reload=auto_reload)
|
debug=debug,
|
||||||
|
ssl=ssl,
|
||||||
|
sock=sock,
|
||||||
|
workers=workers,
|
||||||
|
protocol=protocol,
|
||||||
|
backlog=backlog,
|
||||||
|
register_sys_signals=register_sys_signals,
|
||||||
|
auto_reload=auto_reload,
|
||||||
|
)
|
||||||
|
|
||||||
try:
|
try:
|
||||||
self.is_running = True
|
self.is_running = True
|
||||||
if workers == 1:
|
if workers == 1:
|
||||||
if auto_reload and os.name != 'posix':
|
if auto_reload and os.name != "posix":
|
||||||
# This condition must be removed after implementing
|
# This condition must be removed after implementing
|
||||||
# auto reloader for other operating systems.
|
# auto reloader for other operating systems.
|
||||||
raise NotImplementedError
|
raise NotImplementedError
|
||||||
|
|
||||||
if auto_reload and \
|
if (
|
||||||
os.environ.get('SANIC_SERVER_RUNNING') != 'true':
|
auto_reload
|
||||||
|
and os.environ.get("SANIC_SERVER_RUNNING") != "true"
|
||||||
|
):
|
||||||
reloader_helpers.watchdog(2)
|
reloader_helpers.watchdog(2)
|
||||||
else:
|
else:
|
||||||
serve(**server_settings)
|
serve(**server_settings)
|
||||||
|
@ -732,7 +908,8 @@ class Sanic:
|
||||||
serve_multiple(server_settings, workers)
|
serve_multiple(server_settings, workers)
|
||||||
except BaseException:
|
except BaseException:
|
||||||
error_logger.exception(
|
error_logger.exception(
|
||||||
'Experienced exception while trying to serve')
|
"Experienced exception while trying to serve"
|
||||||
|
)
|
||||||
raise
|
raise
|
||||||
finally:
|
finally:
|
||||||
self.is_running = False
|
self.is_running = False
|
||||||
|
@ -746,10 +923,18 @@ class Sanic:
|
||||||
"""gunicorn compatibility"""
|
"""gunicorn compatibility"""
|
||||||
return self
|
return self
|
||||||
|
|
||||||
async def create_server(self, host=None, port=None, debug=False,
|
async def create_server(
|
||||||
ssl=None, sock=None, protocol=None,
|
self,
|
||||||
backlog=100, stop_event=None,
|
host=None,
|
||||||
access_log=True):
|
port=None,
|
||||||
|
debug=False,
|
||||||
|
ssl=None,
|
||||||
|
sock=None,
|
||||||
|
protocol=None,
|
||||||
|
backlog=100,
|
||||||
|
stop_event=None,
|
||||||
|
access_log=True,
|
||||||
|
):
|
||||||
"""Asynchronous version of `run`.
|
"""Asynchronous version of `run`.
|
||||||
|
|
||||||
NOTE: This does not support multiprocessing and is not the preferred
|
NOTE: This does not support multiprocessing and is not the preferred
|
||||||
|
@ -760,24 +945,34 @@ class Sanic:
|
||||||
host, port = host or "127.0.0.1", port or 8000
|
host, port = host or "127.0.0.1", port or 8000
|
||||||
|
|
||||||
if protocol is None:
|
if protocol is None:
|
||||||
protocol = (WebSocketProtocol if self.websocket_enabled
|
protocol = (
|
||||||
else HttpProtocol)
|
WebSocketProtocol if self.websocket_enabled else HttpProtocol
|
||||||
|
)
|
||||||
if stop_event is not None:
|
if stop_event is not None:
|
||||||
if debug:
|
if debug:
|
||||||
warnings.simplefilter('default')
|
warnings.simplefilter("default")
|
||||||
warnings.warn("stop_event will be removed from future versions.",
|
warnings.warn(
|
||||||
DeprecationWarning)
|
"stop_event will be removed from future versions.",
|
||||||
|
DeprecationWarning,
|
||||||
|
)
|
||||||
# compatibility old access_log params
|
# compatibility old access_log params
|
||||||
self.config.ACCESS_LOG = access_log
|
self.config.ACCESS_LOG = access_log
|
||||||
server_settings = self._helper(
|
server_settings = self._helper(
|
||||||
host=host, port=port, debug=debug, ssl=ssl, sock=sock,
|
host=host,
|
||||||
loop=get_event_loop(), protocol=protocol,
|
port=port,
|
||||||
backlog=backlog, run_async=True)
|
debug=debug,
|
||||||
|
ssl=ssl,
|
||||||
|
sock=sock,
|
||||||
|
loop=get_event_loop(),
|
||||||
|
protocol=protocol,
|
||||||
|
backlog=backlog,
|
||||||
|
run_async=True,
|
||||||
|
)
|
||||||
|
|
||||||
# Trigger before_start events
|
# Trigger before_start events
|
||||||
await self.trigger_events(
|
await self.trigger_events(
|
||||||
server_settings.get('before_start', []),
|
server_settings.get("before_start", []),
|
||||||
server_settings.get('loop')
|
server_settings.get("loop"),
|
||||||
)
|
)
|
||||||
|
|
||||||
return await serve(**server_settings)
|
return await serve(**server_settings)
|
||||||
|
@ -814,15 +1009,27 @@ class Sanic:
|
||||||
break
|
break
|
||||||
return response
|
return response
|
||||||
|
|
||||||
def _helper(self, host=None, port=None, debug=False,
|
def _helper(
|
||||||
ssl=None, sock=None, workers=1, loop=None,
|
self,
|
||||||
protocol=HttpProtocol, backlog=100, stop_event=None,
|
host=None,
|
||||||
register_sys_signals=True, run_async=False, auto_reload=False):
|
port=None,
|
||||||
|
debug=False,
|
||||||
|
ssl=None,
|
||||||
|
sock=None,
|
||||||
|
workers=1,
|
||||||
|
loop=None,
|
||||||
|
protocol=HttpProtocol,
|
||||||
|
backlog=100,
|
||||||
|
stop_event=None,
|
||||||
|
register_sys_signals=True,
|
||||||
|
run_async=False,
|
||||||
|
auto_reload=False,
|
||||||
|
):
|
||||||
"""Helper function used by `run` and `create_server`."""
|
"""Helper function used by `run` and `create_server`."""
|
||||||
if isinstance(ssl, dict):
|
if isinstance(ssl, dict):
|
||||||
# try common aliaseses
|
# try common aliaseses
|
||||||
cert = ssl.get('cert') or ssl.get('certificate')
|
cert = ssl.get("cert") or ssl.get("certificate")
|
||||||
key = ssl.get('key') or ssl.get('keyfile')
|
key = ssl.get("key") or ssl.get("keyfile")
|
||||||
if cert is None or key is None:
|
if cert is None or key is None:
|
||||||
raise ValueError("SSLContext or certificate and key required.")
|
raise ValueError("SSLContext or certificate and key required.")
|
||||||
context = create_default_context(purpose=Purpose.CLIENT_AUTH)
|
context = create_default_context(purpose=Purpose.CLIENT_AUTH)
|
||||||
|
@ -830,40 +1037,42 @@ class Sanic:
|
||||||
ssl = context
|
ssl = context
|
||||||
if stop_event is not None:
|
if stop_event is not None:
|
||||||
if debug:
|
if debug:
|
||||||
warnings.simplefilter('default')
|
warnings.simplefilter("default")
|
||||||
warnings.warn("stop_event will be removed from future versions.",
|
warnings.warn(
|
||||||
DeprecationWarning)
|
"stop_event will be removed from future versions.",
|
||||||
|
DeprecationWarning,
|
||||||
|
)
|
||||||
|
|
||||||
self.error_handler.debug = debug
|
self.error_handler.debug = debug
|
||||||
self.debug = debug
|
self.debug = debug
|
||||||
|
|
||||||
server_settings = {
|
server_settings = {
|
||||||
'protocol': protocol,
|
"protocol": protocol,
|
||||||
'request_class': self.request_class,
|
"request_class": self.request_class,
|
||||||
'is_request_stream': self.is_request_stream,
|
"is_request_stream": self.is_request_stream,
|
||||||
'router': self.router,
|
"router": self.router,
|
||||||
'host': host,
|
"host": host,
|
||||||
'port': port,
|
"port": port,
|
||||||
'sock': sock,
|
"sock": sock,
|
||||||
'ssl': ssl,
|
"ssl": ssl,
|
||||||
'signal': Signal(),
|
"signal": Signal(),
|
||||||
'debug': debug,
|
"debug": debug,
|
||||||
'request_handler': self.handle_request,
|
"request_handler": self.handle_request,
|
||||||
'error_handler': self.error_handler,
|
"error_handler": self.error_handler,
|
||||||
'request_timeout': self.config.REQUEST_TIMEOUT,
|
"request_timeout": self.config.REQUEST_TIMEOUT,
|
||||||
'response_timeout': self.config.RESPONSE_TIMEOUT,
|
"response_timeout": self.config.RESPONSE_TIMEOUT,
|
||||||
'keep_alive_timeout': self.config.KEEP_ALIVE_TIMEOUT,
|
"keep_alive_timeout": self.config.KEEP_ALIVE_TIMEOUT,
|
||||||
'request_max_size': self.config.REQUEST_MAX_SIZE,
|
"request_max_size": self.config.REQUEST_MAX_SIZE,
|
||||||
'keep_alive': self.config.KEEP_ALIVE,
|
"keep_alive": self.config.KEEP_ALIVE,
|
||||||
'loop': loop,
|
"loop": loop,
|
||||||
'register_sys_signals': register_sys_signals,
|
"register_sys_signals": register_sys_signals,
|
||||||
'backlog': backlog,
|
"backlog": backlog,
|
||||||
'access_log': self.config.ACCESS_LOG,
|
"access_log": self.config.ACCESS_LOG,
|
||||||
'websocket_max_size': self.config.WEBSOCKET_MAX_SIZE,
|
"websocket_max_size": self.config.WEBSOCKET_MAX_SIZE,
|
||||||
'websocket_max_queue': self.config.WEBSOCKET_MAX_QUEUE,
|
"websocket_max_queue": self.config.WEBSOCKET_MAX_QUEUE,
|
||||||
'websocket_read_limit': self.config.WEBSOCKET_READ_LIMIT,
|
"websocket_read_limit": self.config.WEBSOCKET_READ_LIMIT,
|
||||||
'websocket_write_limit': self.config.WEBSOCKET_WRITE_LIMIT,
|
"websocket_write_limit": self.config.WEBSOCKET_WRITE_LIMIT,
|
||||||
'graceful_shutdown_timeout': self.config.GRACEFUL_SHUTDOWN_TIMEOUT
|
"graceful_shutdown_timeout": self.config.GRACEFUL_SHUTDOWN_TIMEOUT,
|
||||||
}
|
}
|
||||||
|
|
||||||
# -------------------------------------------- #
|
# -------------------------------------------- #
|
||||||
|
@ -886,18 +1095,20 @@ class Sanic:
|
||||||
if self.configure_logging and debug:
|
if self.configure_logging and debug:
|
||||||
logger.setLevel(logging.DEBUG)
|
logger.setLevel(logging.DEBUG)
|
||||||
|
|
||||||
if self.config.LOGO is not None and \
|
if (
|
||||||
os.environ.get('SANIC_SERVER_RUNNING') != 'true':
|
self.config.LOGO is not None
|
||||||
|
and os.environ.get("SANIC_SERVER_RUNNING") != "true"
|
||||||
|
):
|
||||||
logger.debug(self.config.LOGO)
|
logger.debug(self.config.LOGO)
|
||||||
|
|
||||||
if run_async:
|
if run_async:
|
||||||
server_settings['run_async'] = True
|
server_settings["run_async"] = True
|
||||||
|
|
||||||
# Serve
|
# Serve
|
||||||
if host and port and os.environ.get('SANIC_SERVER_RUNNING') != 'true':
|
if host and port and os.environ.get("SANIC_SERVER_RUNNING") != "true":
|
||||||
proto = "http"
|
proto = "http"
|
||||||
if ssl is not None:
|
if ssl is not None:
|
||||||
proto = "https"
|
proto = "https"
|
||||||
logger.info('Goin\' Fast @ {}://{}:{}'.format(proto, host, port))
|
logger.info("Goin' Fast @ {}://{}:{}".format(proto, host, port))
|
||||||
|
|
||||||
return server_settings
|
return server_settings
|
||||||
|
|
|
@ -3,21 +3,36 @@ from collections import defaultdict, namedtuple
|
||||||
from sanic.constants import HTTP_METHODS
|
from sanic.constants import HTTP_METHODS
|
||||||
from sanic.views import CompositionView
|
from sanic.views import CompositionView
|
||||||
|
|
||||||
FutureRoute = namedtuple('Route',
|
FutureRoute = namedtuple(
|
||||||
['handler', 'uri', 'methods', 'host',
|
"Route",
|
||||||
'strict_slashes', 'stream', 'version', 'name'])
|
[
|
||||||
FutureListener = namedtuple('Listener', ['handler', 'uri', 'methods', 'host'])
|
"handler",
|
||||||
FutureMiddleware = namedtuple('Route', ['middleware', 'args', 'kwargs'])
|
"uri",
|
||||||
FutureException = namedtuple('Route', ['handler', 'args', 'kwargs'])
|
"methods",
|
||||||
FutureStatic = namedtuple('Route',
|
"host",
|
||||||
['uri', 'file_or_directory', 'args', 'kwargs'])
|
"strict_slashes",
|
||||||
|
"stream",
|
||||||
|
"version",
|
||||||
|
"name",
|
||||||
|
],
|
||||||
|
)
|
||||||
|
FutureListener = namedtuple("Listener", ["handler", "uri", "methods", "host"])
|
||||||
|
FutureMiddleware = namedtuple("Route", ["middleware", "args", "kwargs"])
|
||||||
|
FutureException = namedtuple("Route", ["handler", "args", "kwargs"])
|
||||||
|
FutureStatic = namedtuple(
|
||||||
|
"Route", ["uri", "file_or_directory", "args", "kwargs"]
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
class Blueprint:
|
class Blueprint:
|
||||||
def __init__(self, name,
|
def __init__(
|
||||||
|
self,
|
||||||
|
name,
|
||||||
url_prefix=None,
|
url_prefix=None,
|
||||||
host=None, version=None,
|
host=None,
|
||||||
strict_slashes=False):
|
version=None,
|
||||||
|
strict_slashes=False,
|
||||||
|
):
|
||||||
"""Create a new blueprint
|
"""Create a new blueprint
|
||||||
|
|
||||||
:param name: unique name of the blueprint
|
:param name: unique name of the blueprint
|
||||||
|
@ -38,13 +53,14 @@ class Blueprint:
|
||||||
self.strict_slashes = strict_slashes
|
self.strict_slashes = strict_slashes
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def group(*blueprints, url_prefix=''):
|
def group(*blueprints, url_prefix=""):
|
||||||
"""Create a list of blueprints, optionally
|
"""Create a list of blueprints, optionally
|
||||||
grouping them under a general URL prefix.
|
grouping them under a general URL prefix.
|
||||||
|
|
||||||
:param blueprints: blueprints to be registered as a group
|
:param blueprints: blueprints to be registered as a group
|
||||||
:param url_prefix: URL route to be prepended to all sub-prefixes
|
:param url_prefix: URL route to be prepended to all sub-prefixes
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def chain(nested):
|
def chain(nested):
|
||||||
"""itertools.chain() but leaves strings untouched"""
|
"""itertools.chain() but leaves strings untouched"""
|
||||||
for i in nested:
|
for i in nested:
|
||||||
|
@ -52,10 +68,11 @@ class Blueprint:
|
||||||
yield from chain(i)
|
yield from chain(i)
|
||||||
else:
|
else:
|
||||||
yield i
|
yield i
|
||||||
|
|
||||||
bps = []
|
bps = []
|
||||||
for bp in chain(blueprints):
|
for bp in chain(blueprints):
|
||||||
if bp.url_prefix is None:
|
if bp.url_prefix is None:
|
||||||
bp.url_prefix = ''
|
bp.url_prefix = ""
|
||||||
bp.url_prefix = url_prefix + bp.url_prefix
|
bp.url_prefix = url_prefix + bp.url_prefix
|
||||||
bps.append(bp)
|
bps.append(bp)
|
||||||
return bps
|
return bps
|
||||||
|
@ -63,7 +80,7 @@ class Blueprint:
|
||||||
def register(self, app, options):
|
def register(self, app, options):
|
||||||
"""Register the blueprint to the sanic app."""
|
"""Register the blueprint to the sanic app."""
|
||||||
|
|
||||||
url_prefix = options.get('url_prefix', self.url_prefix)
|
url_prefix = options.get("url_prefix", self.url_prefix)
|
||||||
|
|
||||||
# Routes
|
# Routes
|
||||||
for future in self.routes:
|
for future in self.routes:
|
||||||
|
@ -75,7 +92,8 @@ class Blueprint:
|
||||||
|
|
||||||
version = future.version or self.version
|
version = future.version or self.version
|
||||||
|
|
||||||
app.route(uri=uri[1:] if uri.startswith('//') else uri,
|
app.route(
|
||||||
|
uri=uri[1:] if uri.startswith("//") else uri,
|
||||||
methods=future.methods,
|
methods=future.methods,
|
||||||
host=future.host or self.host,
|
host=future.host or self.host,
|
||||||
strict_slashes=future.strict_slashes,
|
strict_slashes=future.strict_slashes,
|
||||||
|
@ -90,7 +108,8 @@ class Blueprint:
|
||||||
future.handler.__blueprintname__ = self.name
|
future.handler.__blueprintname__ = self.name
|
||||||
# Prepend the blueprint URI prefix if available
|
# Prepend the blueprint URI prefix if available
|
||||||
uri = url_prefix + future.uri if url_prefix else future.uri
|
uri = url_prefix + future.uri if url_prefix else future.uri
|
||||||
app.websocket(uri=uri,
|
app.websocket(
|
||||||
|
uri=uri,
|
||||||
host=future.host or self.host,
|
host=future.host or self.host,
|
||||||
strict_slashes=future.strict_slashes,
|
strict_slashes=future.strict_slashes,
|
||||||
name=future.name,
|
name=future.name,
|
||||||
|
@ -99,9 +118,9 @@ class Blueprint:
|
||||||
# Middleware
|
# Middleware
|
||||||
for future in self.middlewares:
|
for future in self.middlewares:
|
||||||
if future.args or future.kwargs:
|
if future.args or future.kwargs:
|
||||||
app.register_middleware(future.middleware,
|
app.register_middleware(
|
||||||
*future.args,
|
future.middleware, *future.args, **future.kwargs
|
||||||
**future.kwargs)
|
)
|
||||||
else:
|
else:
|
||||||
app.register_middleware(future.middleware)
|
app.register_middleware(future.middleware)
|
||||||
|
|
||||||
|
@ -113,16 +132,25 @@ class Blueprint:
|
||||||
for future in self.statics:
|
for future in self.statics:
|
||||||
# Prepend the blueprint URI prefix if available
|
# Prepend the blueprint URI prefix if available
|
||||||
uri = url_prefix + future.uri if url_prefix else future.uri
|
uri = url_prefix + future.uri if url_prefix else future.uri
|
||||||
app.static(uri, future.file_or_directory,
|
app.static(
|
||||||
*future.args, **future.kwargs)
|
uri, future.file_or_directory, *future.args, **future.kwargs
|
||||||
|
)
|
||||||
|
|
||||||
# Event listeners
|
# Event listeners
|
||||||
for event, listeners in self.listeners.items():
|
for event, listeners in self.listeners.items():
|
||||||
for listener in listeners:
|
for listener in listeners:
|
||||||
app.listener(event)(listener)
|
app.listener(event)(listener)
|
||||||
|
|
||||||
def route(self, uri, methods=frozenset({'GET'}), host=None,
|
def route(
|
||||||
strict_slashes=None, stream=False, version=None, name=None):
|
self,
|
||||||
|
uri,
|
||||||
|
methods=frozenset({"GET"}),
|
||||||
|
host=None,
|
||||||
|
strict_slashes=None,
|
||||||
|
stream=False,
|
||||||
|
version=None,
|
||||||
|
name=None,
|
||||||
|
):
|
||||||
"""Create a blueprint route from a decorated function.
|
"""Create a blueprint route from a decorated function.
|
||||||
|
|
||||||
:param uri: endpoint at which the route will be accessible.
|
:param uri: endpoint at which the route will be accessible.
|
||||||
|
@ -133,14 +161,30 @@ class Blueprint:
|
||||||
|
|
||||||
def decorator(handler):
|
def decorator(handler):
|
||||||
route = FutureRoute(
|
route = FutureRoute(
|
||||||
handler, uri, methods, host, strict_slashes, stream, version,
|
handler,
|
||||||
name)
|
uri,
|
||||||
|
methods,
|
||||||
|
host,
|
||||||
|
strict_slashes,
|
||||||
|
stream,
|
||||||
|
version,
|
||||||
|
name,
|
||||||
|
)
|
||||||
self.routes.append(route)
|
self.routes.append(route)
|
||||||
return handler
|
return handler
|
||||||
|
|
||||||
return decorator
|
return decorator
|
||||||
|
|
||||||
def add_route(self, handler, uri, methods=frozenset({'GET'}), host=None,
|
def add_route(
|
||||||
strict_slashes=None, version=None, name=None):
|
self,
|
||||||
|
handler,
|
||||||
|
uri,
|
||||||
|
methods=frozenset({"GET"}),
|
||||||
|
host=None,
|
||||||
|
strict_slashes=None,
|
||||||
|
version=None,
|
||||||
|
name=None,
|
||||||
|
):
|
||||||
"""Create a blueprint route from a function.
|
"""Create a blueprint route from a function.
|
||||||
|
|
||||||
:param handler: function for handling uri requests. Accepts function,
|
:param handler: function for handling uri requests. Accepts function,
|
||||||
|
@ -154,7 +198,7 @@ class Blueprint:
|
||||||
:return: function or class instance
|
:return: function or class instance
|
||||||
"""
|
"""
|
||||||
# Handle HTTPMethodView differently
|
# Handle HTTPMethodView differently
|
||||||
if hasattr(handler, 'view_class'):
|
if hasattr(handler, "view_class"):
|
||||||
methods = set()
|
methods = set()
|
||||||
|
|
||||||
for method in HTTP_METHODS:
|
for method in HTTP_METHODS:
|
||||||
|
@ -168,13 +212,19 @@ class Blueprint:
|
||||||
if isinstance(handler, CompositionView):
|
if isinstance(handler, CompositionView):
|
||||||
methods = handler.handlers.keys()
|
methods = handler.handlers.keys()
|
||||||
|
|
||||||
self.route(uri=uri, methods=methods, host=host,
|
self.route(
|
||||||
strict_slashes=strict_slashes, version=version,
|
uri=uri,
|
||||||
name=name)(handler)
|
methods=methods,
|
||||||
|
host=host,
|
||||||
|
strict_slashes=strict_slashes,
|
||||||
|
version=version,
|
||||||
|
name=name,
|
||||||
|
)(handler)
|
||||||
return handler
|
return handler
|
||||||
|
|
||||||
def websocket(self, uri, host=None, strict_slashes=None, version=None,
|
def websocket(
|
||||||
name=None):
|
self, uri, host=None, strict_slashes=None, version=None, name=None
|
||||||
|
):
|
||||||
"""Create a blueprint websocket route from a decorated function.
|
"""Create a blueprint websocket route from a decorated function.
|
||||||
|
|
||||||
:param uri: endpoint at which the route will be accessible.
|
:param uri: endpoint at which the route will be accessible.
|
||||||
|
@ -183,14 +233,17 @@ class Blueprint:
|
||||||
strict_slashes = self.strict_slashes
|
strict_slashes = self.strict_slashes
|
||||||
|
|
||||||
def decorator(handler):
|
def decorator(handler):
|
||||||
route = FutureRoute(handler, uri, [], host, strict_slashes,
|
route = FutureRoute(
|
||||||
False, version, name)
|
handler, uri, [], host, strict_slashes, False, version, name
|
||||||
|
)
|
||||||
self.websocket_routes.append(route)
|
self.websocket_routes.append(route)
|
||||||
return handler
|
return handler
|
||||||
|
|
||||||
return decorator
|
return decorator
|
||||||
|
|
||||||
def add_websocket_route(self, handler, uri, host=None, version=None,
|
def add_websocket_route(
|
||||||
name=None):
|
self, handler, uri, host=None, version=None, name=None
|
||||||
|
):
|
||||||
"""Create a blueprint websocket route from a function.
|
"""Create a blueprint websocket route from a function.
|
||||||
|
|
||||||
:param handler: function for handling uri requests. Accepts function,
|
:param handler: function for handling uri requests. Accepts function,
|
||||||
|
@ -206,13 +259,16 @@ class Blueprint:
|
||||||
|
|
||||||
:param event: Event to listen to.
|
:param event: Event to listen to.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def decorator(listener):
|
def decorator(listener):
|
||||||
self.listeners[event].append(listener)
|
self.listeners[event].append(listener)
|
||||||
return listener
|
return listener
|
||||||
|
|
||||||
return decorator
|
return decorator
|
||||||
|
|
||||||
def middleware(self, *args, **kwargs):
|
def middleware(self, *args, **kwargs):
|
||||||
"""Create a blueprint middleware from a decorated function."""
|
"""Create a blueprint middleware from a decorated function."""
|
||||||
|
|
||||||
def register_middleware(_middleware):
|
def register_middleware(_middleware):
|
||||||
future_middleware = FutureMiddleware(_middleware, args, kwargs)
|
future_middleware = FutureMiddleware(_middleware, args, kwargs)
|
||||||
self.middlewares.append(future_middleware)
|
self.middlewares.append(future_middleware)
|
||||||
|
@ -228,10 +284,12 @@ class Blueprint:
|
||||||
|
|
||||||
def exception(self, *args, **kwargs):
|
def exception(self, *args, **kwargs):
|
||||||
"""Create a blueprint exception from a decorated function."""
|
"""Create a blueprint exception from a decorated function."""
|
||||||
|
|
||||||
def decorator(handler):
|
def decorator(handler):
|
||||||
exception = FutureException(handler, args, kwargs)
|
exception = FutureException(handler, args, kwargs)
|
||||||
self.exceptions.append(exception)
|
self.exceptions.append(exception)
|
||||||
return handler
|
return handler
|
||||||
|
|
||||||
return decorator
|
return decorator
|
||||||
|
|
||||||
def static(self, uri, file_or_directory, *args, **kwargs):
|
def static(self, uri, file_or_directory, *args, **kwargs):
|
||||||
|
@ -240,12 +298,12 @@ class Blueprint:
|
||||||
:param uri: endpoint at which the route will be accessible.
|
:param uri: endpoint at which the route will be accessible.
|
||||||
:param file_or_directory: Static asset.
|
:param file_or_directory: Static asset.
|
||||||
"""
|
"""
|
||||||
name = kwargs.pop('name', 'static')
|
name = kwargs.pop("name", "static")
|
||||||
if not name.startswith(self.name + '.'):
|
if not name.startswith(self.name + "."):
|
||||||
name = '{}.{}'.format(self.name, name)
|
name = "{}.{}".format(self.name, name)
|
||||||
kwargs.update(name=name)
|
kwargs.update(name=name)
|
||||||
|
|
||||||
strict_slashes = kwargs.get('strict_slashes')
|
strict_slashes = kwargs.get("strict_slashes")
|
||||||
if strict_slashes is None and self.strict_slashes is not None:
|
if strict_slashes is None and self.strict_slashes is not None:
|
||||||
kwargs.update(strict_slashes=self.strict_slashes)
|
kwargs.update(strict_slashes=self.strict_slashes)
|
||||||
|
|
||||||
|
@ -253,44 +311,107 @@ class Blueprint:
|
||||||
self.statics.append(static)
|
self.statics.append(static)
|
||||||
|
|
||||||
# Shorthand method decorators
|
# Shorthand method decorators
|
||||||
def get(self, uri, host=None, strict_slashes=None, version=None,
|
def get(
|
||||||
name=None):
|
self, uri, host=None, strict_slashes=None, version=None, name=None
|
||||||
return self.route(uri, methods=["GET"], host=host,
|
):
|
||||||
strict_slashes=strict_slashes, version=version,
|
return self.route(
|
||||||
name=name)
|
uri,
|
||||||
|
methods=["GET"],
|
||||||
|
host=host,
|
||||||
|
strict_slashes=strict_slashes,
|
||||||
|
version=version,
|
||||||
|
name=name,
|
||||||
|
)
|
||||||
|
|
||||||
def post(self, uri, host=None, strict_slashes=None, stream=False,
|
def post(
|
||||||
version=None, name=None):
|
self,
|
||||||
return self.route(uri, methods=["POST"], host=host,
|
uri,
|
||||||
strict_slashes=strict_slashes, stream=stream,
|
host=None,
|
||||||
version=version, name=name)
|
strict_slashes=None,
|
||||||
|
stream=False,
|
||||||
|
version=None,
|
||||||
|
name=None,
|
||||||
|
):
|
||||||
|
return self.route(
|
||||||
|
uri,
|
||||||
|
methods=["POST"],
|
||||||
|
host=host,
|
||||||
|
strict_slashes=strict_slashes,
|
||||||
|
stream=stream,
|
||||||
|
version=version,
|
||||||
|
name=name,
|
||||||
|
)
|
||||||
|
|
||||||
def put(self, uri, host=None, strict_slashes=None, stream=False,
|
def put(
|
||||||
version=None, name=None):
|
self,
|
||||||
return self.route(uri, methods=["PUT"], host=host,
|
uri,
|
||||||
strict_slashes=strict_slashes, stream=stream,
|
host=None,
|
||||||
version=version, name=name)
|
strict_slashes=None,
|
||||||
|
stream=False,
|
||||||
|
version=None,
|
||||||
|
name=None,
|
||||||
|
):
|
||||||
|
return self.route(
|
||||||
|
uri,
|
||||||
|
methods=["PUT"],
|
||||||
|
host=host,
|
||||||
|
strict_slashes=strict_slashes,
|
||||||
|
stream=stream,
|
||||||
|
version=version,
|
||||||
|
name=name,
|
||||||
|
)
|
||||||
|
|
||||||
def head(self, uri, host=None, strict_slashes=None, version=None,
|
def head(
|
||||||
name=None):
|
self, uri, host=None, strict_slashes=None, version=None, name=None
|
||||||
return self.route(uri, methods=["HEAD"], host=host,
|
):
|
||||||
strict_slashes=strict_slashes, version=version,
|
return self.route(
|
||||||
name=name)
|
uri,
|
||||||
|
methods=["HEAD"],
|
||||||
|
host=host,
|
||||||
|
strict_slashes=strict_slashes,
|
||||||
|
version=version,
|
||||||
|
name=name,
|
||||||
|
)
|
||||||
|
|
||||||
def options(self, uri, host=None, strict_slashes=None, version=None,
|
def options(
|
||||||
name=None):
|
self, uri, host=None, strict_slashes=None, version=None, name=None
|
||||||
return self.route(uri, methods=["OPTIONS"], host=host,
|
):
|
||||||
strict_slashes=strict_slashes, version=version,
|
return self.route(
|
||||||
name=name)
|
uri,
|
||||||
|
methods=["OPTIONS"],
|
||||||
|
host=host,
|
||||||
|
strict_slashes=strict_slashes,
|
||||||
|
version=version,
|
||||||
|
name=name,
|
||||||
|
)
|
||||||
|
|
||||||
def patch(self, uri, host=None, strict_slashes=None, stream=False,
|
def patch(
|
||||||
version=None, name=None):
|
self,
|
||||||
return self.route(uri, methods=["PATCH"], host=host,
|
uri,
|
||||||
strict_slashes=strict_slashes, stream=stream,
|
host=None,
|
||||||
version=version, name=name)
|
strict_slashes=None,
|
||||||
|
stream=False,
|
||||||
|
version=None,
|
||||||
|
name=None,
|
||||||
|
):
|
||||||
|
return self.route(
|
||||||
|
uri,
|
||||||
|
methods=["PATCH"],
|
||||||
|
host=host,
|
||||||
|
strict_slashes=strict_slashes,
|
||||||
|
stream=stream,
|
||||||
|
version=version,
|
||||||
|
name=name,
|
||||||
|
)
|
||||||
|
|
||||||
def delete(self, uri, host=None, strict_slashes=None, version=None,
|
def delete(
|
||||||
name=None):
|
self, uri, host=None, strict_slashes=None, version=None, name=None
|
||||||
return self.route(uri, methods=["DELETE"], host=host,
|
):
|
||||||
strict_slashes=strict_slashes, version=version,
|
return self.route(
|
||||||
name=name)
|
uri,
|
||||||
|
methods=["DELETE"],
|
||||||
|
host=host,
|
||||||
|
strict_slashes=strict_slashes,
|
||||||
|
version=version,
|
||||||
|
name=name,
|
||||||
|
)
|
||||||
|
|
|
@ -4,7 +4,7 @@ import types
|
||||||
from sanic.exceptions import PyFileError
|
from sanic.exceptions import PyFileError
|
||||||
|
|
||||||
|
|
||||||
SANIC_PREFIX = 'SANIC_'
|
SANIC_PREFIX = "SANIC_"
|
||||||
|
|
||||||
|
|
||||||
class Config(dict):
|
class Config(dict):
|
||||||
|
@ -65,9 +65,10 @@ class Config(dict):
|
||||||
"""
|
"""
|
||||||
config_file = os.environ.get(variable_name)
|
config_file = os.environ.get(variable_name)
|
||||||
if not config_file:
|
if not config_file:
|
||||||
raise RuntimeError('The environment variable %r is not set and '
|
raise RuntimeError(
|
||||||
'thus configuration could not be loaded.' %
|
"The environment variable %r is not set and "
|
||||||
variable_name)
|
"thus configuration could not be loaded." % variable_name
|
||||||
|
)
|
||||||
return self.from_pyfile(config_file)
|
return self.from_pyfile(config_file)
|
||||||
|
|
||||||
def from_pyfile(self, filename):
|
def from_pyfile(self, filename):
|
||||||
|
@ -76,14 +77,16 @@ class Config(dict):
|
||||||
|
|
||||||
:param filename: an absolute path to the config file
|
:param filename: an absolute path to the config file
|
||||||
"""
|
"""
|
||||||
module = types.ModuleType('config')
|
module = types.ModuleType("config")
|
||||||
module.__file__ = filename
|
module.__file__ = filename
|
||||||
try:
|
try:
|
||||||
with open(filename) as config_file:
|
with open(filename) as config_file:
|
||||||
exec(compile(config_file.read(), filename, 'exec'),
|
exec(
|
||||||
module.__dict__)
|
compile(config_file.read(), filename, "exec"),
|
||||||
|
module.__dict__,
|
||||||
|
)
|
||||||
except IOError as e:
|
except IOError as e:
|
||||||
e.strerror = 'Unable to load configuration file (%s)' % e.strerror
|
e.strerror = "Unable to load configuration file (%s)" % e.strerror
|
||||||
raise
|
raise
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
raise PyFileError(filename) from e
|
raise PyFileError(filename) from e
|
||||||
|
|
|
@ -1 +1 @@
|
||||||
HTTP_METHODS = ('GET', 'POST', 'PUT', 'HEAD', 'OPTIONS', 'PATCH', 'DELETE')
|
HTTP_METHODS = ("GET", "POST", "PUT", "HEAD", "OPTIONS", "PATCH", "DELETE")
|
||||||
|
|
|
@ -8,14 +8,12 @@ import string
|
||||||
# Straight up copied this section of dark magic from SimpleCookie
|
# Straight up copied this section of dark magic from SimpleCookie
|
||||||
|
|
||||||
_LegalChars = string.ascii_letters + string.digits + "!#$%&'*+-.^_`|~:"
|
_LegalChars = string.ascii_letters + string.digits + "!#$%&'*+-.^_`|~:"
|
||||||
_UnescapedChars = _LegalChars + ' ()/<=>?@[]{}'
|
_UnescapedChars = _LegalChars + " ()/<=>?@[]{}"
|
||||||
|
|
||||||
_Translator = {n: '\\%03o' % n
|
_Translator = {
|
||||||
for n in set(range(256)) - set(map(ord, _UnescapedChars))}
|
n: "\\%03o" % n for n in set(range(256)) - set(map(ord, _UnescapedChars))
|
||||||
_Translator.update({
|
}
|
||||||
ord('"'): '\\"',
|
_Translator.update({ord('"'): '\\"', ord("\\"): "\\\\"})
|
||||||
ord('\\'): '\\\\',
|
|
||||||
})
|
|
||||||
|
|
||||||
|
|
||||||
def _quote(str):
|
def _quote(str):
|
||||||
|
@ -30,7 +28,7 @@ def _quote(str):
|
||||||
return '"' + str.translate(_Translator) + '"'
|
return '"' + str.translate(_Translator) + '"'
|
||||||
|
|
||||||
|
|
||||||
_is_legal_key = re.compile('[%s]+' % re.escape(_LegalChars)).fullmatch
|
_is_legal_key = re.compile("[%s]+" % re.escape(_LegalChars)).fullmatch
|
||||||
|
|
||||||
# ------------------------------------------------------------ #
|
# ------------------------------------------------------------ #
|
||||||
# Custom SimpleCookie
|
# Custom SimpleCookie
|
||||||
|
@ -53,7 +51,7 @@ class CookieJar(dict):
|
||||||
# If this cookie doesn't exist, add it to the header keys
|
# If this cookie doesn't exist, add it to the header keys
|
||||||
if not self.cookie_headers.get(key):
|
if not self.cookie_headers.get(key):
|
||||||
cookie = Cookie(key, value)
|
cookie = Cookie(key, value)
|
||||||
cookie['path'] = '/'
|
cookie["path"] = "/"
|
||||||
self.cookie_headers[key] = self.header_key
|
self.cookie_headers[key] = self.header_key
|
||||||
self.headers.add(self.header_key, cookie)
|
self.headers.add(self.header_key, cookie)
|
||||||
return super().__setitem__(key, cookie)
|
return super().__setitem__(key, cookie)
|
||||||
|
@ -62,8 +60,8 @@ class CookieJar(dict):
|
||||||
|
|
||||||
def __delitem__(self, key):
|
def __delitem__(self, key):
|
||||||
if key not in self.cookie_headers:
|
if key not in self.cookie_headers:
|
||||||
self[key] = ''
|
self[key] = ""
|
||||||
self[key]['max-age'] = 0
|
self[key]["max-age"] = 0
|
||||||
else:
|
else:
|
||||||
cookie_header = self.cookie_headers[key]
|
cookie_header = self.cookie_headers[key]
|
||||||
# remove it from header
|
# remove it from header
|
||||||
|
@ -77,6 +75,7 @@ class CookieJar(dict):
|
||||||
|
|
||||||
class Cookie(dict):
|
class Cookie(dict):
|
||||||
"""A stripped down version of Morsel from SimpleCookie #gottagofast"""
|
"""A stripped down version of Morsel from SimpleCookie #gottagofast"""
|
||||||
|
|
||||||
_keys = {
|
_keys = {
|
||||||
"expires": "expires",
|
"expires": "expires",
|
||||||
"path": "Path",
|
"path": "Path",
|
||||||
|
@ -88,7 +87,7 @@ class Cookie(dict):
|
||||||
"version": "Version",
|
"version": "Version",
|
||||||
"samesite": "SameSite",
|
"samesite": "SameSite",
|
||||||
}
|
}
|
||||||
_flags = {'secure', 'httponly'}
|
_flags = {"secure", "httponly"}
|
||||||
|
|
||||||
def __init__(self, key, value):
|
def __init__(self, key, value):
|
||||||
if key in self._keys:
|
if key in self._keys:
|
||||||
|
@ -106,24 +105,27 @@ class Cookie(dict):
|
||||||
return super().__setitem__(key, value)
|
return super().__setitem__(key, value)
|
||||||
|
|
||||||
def encode(self, encoding):
|
def encode(self, encoding):
|
||||||
output = ['%s=%s' % (self.key, _quote(self.value))]
|
output = ["%s=%s" % (self.key, _quote(self.value))]
|
||||||
for key, value in self.items():
|
for key, value in self.items():
|
||||||
if key == 'max-age':
|
if key == "max-age":
|
||||||
try:
|
try:
|
||||||
output.append('%s=%d' % (self._keys[key], value))
|
output.append("%s=%d" % (self._keys[key], value))
|
||||||
except TypeError:
|
except TypeError:
|
||||||
output.append('%s=%s' % (self._keys[key], value))
|
output.append("%s=%s" % (self._keys[key], value))
|
||||||
elif key == 'expires':
|
elif key == "expires":
|
||||||
try:
|
try:
|
||||||
output.append('%s=%s' % (
|
output.append(
|
||||||
|
"%s=%s"
|
||||||
|
% (
|
||||||
self._keys[key],
|
self._keys[key],
|
||||||
value.strftime("%a, %d-%b-%Y %T GMT")
|
value.strftime("%a, %d-%b-%Y %T GMT"),
|
||||||
))
|
)
|
||||||
|
)
|
||||||
except AttributeError:
|
except AttributeError:
|
||||||
output.append('%s=%s' % (self._keys[key], value))
|
output.append("%s=%s" % (self._keys[key], value))
|
||||||
elif key in self._flags and self[key]:
|
elif key in self._flags and self[key]:
|
||||||
output.append(self._keys[key])
|
output.append(self._keys[key])
|
||||||
else:
|
else:
|
||||||
output.append('%s=%s' % (self._keys[key], value))
|
output.append("%s=%s" % (self._keys[key], value))
|
||||||
|
|
||||||
return "; ".join(output).encode(encoding)
|
return "; ".join(output).encode(encoding)
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
from sanic.helpers import STATUS_CODES
|
from sanic.helpers import STATUS_CODES
|
||||||
|
|
||||||
TRACEBACK_STYLE = '''
|
TRACEBACK_STYLE = """
|
||||||
<style>
|
<style>
|
||||||
body {
|
body {
|
||||||
padding: 20px;
|
padding: 20px;
|
||||||
|
@ -61,9 +61,9 @@ TRACEBACK_STYLE = '''
|
||||||
font-size: 14px;
|
font-size: 14px;
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
'''
|
"""
|
||||||
|
|
||||||
TRACEBACK_WRAPPER_HTML = '''
|
TRACEBACK_WRAPPER_HTML = """
|
||||||
<html>
|
<html>
|
||||||
<head>
|
<head>
|
||||||
{style}
|
{style}
|
||||||
|
@ -78,27 +78,27 @@ TRACEBACK_WRAPPER_HTML = '''
|
||||||
</div>
|
</div>
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
||||||
'''
|
"""
|
||||||
|
|
||||||
TRACEBACK_WRAPPER_INNER_HTML = '''
|
TRACEBACK_WRAPPER_INNER_HTML = """
|
||||||
<h1>{exc_name}</h1>
|
<h1>{exc_name}</h1>
|
||||||
<h3><code>{exc_value}</code></h3>
|
<h3><code>{exc_value}</code></h3>
|
||||||
<div class="tb-wrapper">
|
<div class="tb-wrapper">
|
||||||
<p class="tb-header">Traceback (most recent call last):</p>
|
<p class="tb-header">Traceback (most recent call last):</p>
|
||||||
{frame_html}
|
{frame_html}
|
||||||
</div>
|
</div>
|
||||||
'''
|
"""
|
||||||
|
|
||||||
TRACEBACK_BORDER = '''
|
TRACEBACK_BORDER = """
|
||||||
<div class="tb-border">
|
<div class="tb-border">
|
||||||
<b><i>
|
<b><i>
|
||||||
The above exception was the direct cause of the
|
The above exception was the direct cause of the
|
||||||
following exception:
|
following exception:
|
||||||
</i></b>
|
</i></b>
|
||||||
</div>
|
</div>
|
||||||
'''
|
"""
|
||||||
|
|
||||||
TRACEBACK_LINE_HTML = '''
|
TRACEBACK_LINE_HTML = """
|
||||||
<div class="frame-line">
|
<div class="frame-line">
|
||||||
<p class="frame-descriptor">
|
<p class="frame-descriptor">
|
||||||
File {0.filename}, line <i>{0.lineno}</i>,
|
File {0.filename}, line <i>{0.lineno}</i>,
|
||||||
|
@ -106,15 +106,15 @@ TRACEBACK_LINE_HTML = '''
|
||||||
</p>
|
</p>
|
||||||
<p class="frame-code"><code>{0.line}</code></p>
|
<p class="frame-code"><code>{0.line}</code></p>
|
||||||
</div>
|
</div>
|
||||||
'''
|
"""
|
||||||
|
|
||||||
INTERNAL_SERVER_ERROR_HTML = '''
|
INTERNAL_SERVER_ERROR_HTML = """
|
||||||
<h1>Internal Server Error</h1>
|
<h1>Internal Server Error</h1>
|
||||||
<p>
|
<p>
|
||||||
The server encountered an internal error and cannot complete
|
The server encountered an internal error and cannot complete
|
||||||
your request.
|
your request.
|
||||||
</p>
|
</p>
|
||||||
'''
|
"""
|
||||||
|
|
||||||
|
|
||||||
_sanic_exceptions = {}
|
_sanic_exceptions = {}
|
||||||
|
@ -124,15 +124,16 @@ def add_status_code(code):
|
||||||
"""
|
"""
|
||||||
Decorator used for adding exceptions to _sanic_exceptions.
|
Decorator used for adding exceptions to _sanic_exceptions.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def class_decorator(cls):
|
def class_decorator(cls):
|
||||||
cls.status_code = code
|
cls.status_code = code
|
||||||
_sanic_exceptions[code] = cls
|
_sanic_exceptions[code] = cls
|
||||||
return cls
|
return cls
|
||||||
|
|
||||||
return class_decorator
|
return class_decorator
|
||||||
|
|
||||||
|
|
||||||
class SanicException(Exception):
|
class SanicException(Exception):
|
||||||
|
|
||||||
def __init__(self, message, status_code=None):
|
def __init__(self, message, status_code=None):
|
||||||
super().__init__(message)
|
super().__init__(message)
|
||||||
|
|
||||||
|
@ -156,8 +157,8 @@ class MethodNotSupported(SanicException):
|
||||||
super().__init__(message)
|
super().__init__(message)
|
||||||
self.headers = dict()
|
self.headers = dict()
|
||||||
self.headers["Allow"] = ", ".join(allowed_methods)
|
self.headers["Allow"] = ", ".join(allowed_methods)
|
||||||
if method in ['HEAD', 'PATCH', 'PUT', 'DELETE']:
|
if method in ["HEAD", "PATCH", "PUT", "DELETE"]:
|
||||||
self.headers['Content-Length'] = 0
|
self.headers["Content-Length"] = 0
|
||||||
|
|
||||||
|
|
||||||
@add_status_code(500)
|
@add_status_code(500)
|
||||||
|
@ -169,6 +170,7 @@ class ServerError(SanicException):
|
||||||
class ServiceUnavailable(SanicException):
|
class ServiceUnavailable(SanicException):
|
||||||
"""The server is currently unavailable (because it is overloaded or
|
"""The server is currently unavailable (because it is overloaded or
|
||||||
down for maintenance). Generally, this is a temporary state."""
|
down for maintenance). Generally, this is a temporary state."""
|
||||||
|
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
@ -192,6 +194,7 @@ class RequestTimeout(SanicException):
|
||||||
the connection. The socket connection has actually been lost - the Web
|
the connection. The socket connection has actually been lost - the Web
|
||||||
server has 'timed out' on that particular socket connection.
|
server has 'timed out' on that particular socket connection.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
@ -209,8 +212,8 @@ class ContentRangeError(SanicException):
|
||||||
def __init__(self, message, content_range):
|
def __init__(self, message, content_range):
|
||||||
super().__init__(message)
|
super().__init__(message)
|
||||||
self.headers = {
|
self.headers = {
|
||||||
'Content-Type': 'text/plain',
|
"Content-Type": "text/plain",
|
||||||
"Content-Range": "bytes */%s" % (content_range.total,)
|
"Content-Range": "bytes */%s" % (content_range.total,),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -225,7 +228,7 @@ class InvalidRangeType(ContentRangeError):
|
||||||
|
|
||||||
class PyFileError(Exception):
|
class PyFileError(Exception):
|
||||||
def __init__(self, file):
|
def __init__(self, file):
|
||||||
super().__init__('could not execute config file %s', file)
|
super().__init__("could not execute config file %s", file)
|
||||||
|
|
||||||
|
|
||||||
@add_status_code(401)
|
@add_status_code(401)
|
||||||
|
@ -263,13 +266,14 @@ class Unauthorized(SanicException):
|
||||||
scheme="Bearer",
|
scheme="Bearer",
|
||||||
realm="Restricted Area")
|
realm="Restricted Area")
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def __init__(self, message, status_code=None, scheme=None, **kwargs):
|
def __init__(self, message, status_code=None, scheme=None, **kwargs):
|
||||||
super().__init__(message, status_code)
|
super().__init__(message, status_code)
|
||||||
|
|
||||||
# if auth-scheme is specified, set "WWW-Authenticate" header
|
# if auth-scheme is specified, set "WWW-Authenticate" header
|
||||||
if scheme is not None:
|
if scheme is not None:
|
||||||
values = ['{!s}="{!s}"'.format(k, v) for k, v in kwargs.items()]
|
values = ['{!s}="{!s}"'.format(k, v) for k, v in kwargs.items()]
|
||||||
challenge = ', '.join(values)
|
challenge = ", ".join(values)
|
||||||
|
|
||||||
self.headers = {
|
self.headers = {
|
||||||
"WWW-Authenticate": "{} {}".format(scheme, challenge).rstrip()
|
"WWW-Authenticate": "{} {}".format(scheme, challenge).rstrip()
|
||||||
|
@ -288,6 +292,6 @@ def abort(status_code, message=None):
|
||||||
if message is None:
|
if message is None:
|
||||||
message = STATUS_CODES.get(status_code)
|
message = STATUS_CODES.get(status_code)
|
||||||
# These are stored as bytes in the STATUS_CODES dict
|
# These are stored as bytes in the STATUS_CODES dict
|
||||||
message = message.decode('utf8')
|
message = message.decode("utf8")
|
||||||
sanic_exception = _sanic_exceptions.get(status_code, SanicException)
|
sanic_exception = _sanic_exceptions.get(status_code, SanicException)
|
||||||
raise sanic_exception(message=message, status_code=status_code)
|
raise sanic_exception(message=message, status_code=status_code)
|
||||||
|
|
|
@ -11,7 +11,8 @@ from sanic.exceptions import (
|
||||||
TRACEBACK_STYLE,
|
TRACEBACK_STYLE,
|
||||||
TRACEBACK_WRAPPER_HTML,
|
TRACEBACK_WRAPPER_HTML,
|
||||||
TRACEBACK_WRAPPER_INNER_HTML,
|
TRACEBACK_WRAPPER_INNER_HTML,
|
||||||
TRACEBACK_BORDER)
|
TRACEBACK_BORDER,
|
||||||
|
)
|
||||||
from sanic.log import logger
|
from sanic.log import logger
|
||||||
from sanic.response import text, html
|
from sanic.response import text, html
|
||||||
|
|
||||||
|
@ -36,7 +37,8 @@ class ErrorHandler:
|
||||||
return TRACEBACK_WRAPPER_INNER_HTML.format(
|
return TRACEBACK_WRAPPER_INNER_HTML.format(
|
||||||
exc_name=exception.__class__.__name__,
|
exc_name=exception.__class__.__name__,
|
||||||
exc_value=exception,
|
exc_value=exception,
|
||||||
frame_html=''.join(frame_html))
|
frame_html="".join(frame_html),
|
||||||
|
)
|
||||||
|
|
||||||
def _render_traceback_html(self, exception, request):
|
def _render_traceback_html(self, exception, request):
|
||||||
exc_type, exc_value, tb = sys.exc_info()
|
exc_type, exc_value, tb = sys.exc_info()
|
||||||
|
@ -51,7 +53,8 @@ class ErrorHandler:
|
||||||
exc_name=exception.__class__.__name__,
|
exc_name=exception.__class__.__name__,
|
||||||
exc_value=exception,
|
exc_value=exception,
|
||||||
inner_html=TRACEBACK_BORDER.join(reversed(exceptions)),
|
inner_html=TRACEBACK_BORDER.join(reversed(exceptions)),
|
||||||
path=request.path)
|
path=request.path,
|
||||||
|
)
|
||||||
|
|
||||||
def add(self, exception, handler):
|
def add(self, exception, handler):
|
||||||
self.handlers.append((exception, handler))
|
self.handlers.append((exception, handler))
|
||||||
|
@ -88,17 +91,18 @@ class ErrorHandler:
|
||||||
url = repr(request.url)
|
url = repr(request.url)
|
||||||
except AttributeError:
|
except AttributeError:
|
||||||
url = "unknown"
|
url = "unknown"
|
||||||
response_message = ('Exception raised in exception handler '
|
response_message = (
|
||||||
'"%s" for uri: %s')
|
"Exception raised in exception handler " '"%s" for uri: %s'
|
||||||
|
)
|
||||||
logger.exception(response_message, handler.__name__, url)
|
logger.exception(response_message, handler.__name__, url)
|
||||||
|
|
||||||
if self.debug:
|
if self.debug:
|
||||||
return text(response_message % (handler.__name__, url), 500)
|
return text(response_message % (handler.__name__, url), 500)
|
||||||
else:
|
else:
|
||||||
return text('An error occurred while handling an error', 500)
|
return text("An error occurred while handling an error", 500)
|
||||||
return response
|
return response
|
||||||
|
|
||||||
def log(self, message, level='error'):
|
def log(self, message, level="error"):
|
||||||
"""
|
"""
|
||||||
Deprecated, do not use.
|
Deprecated, do not use.
|
||||||
"""
|
"""
|
||||||
|
@ -110,14 +114,14 @@ class ErrorHandler:
|
||||||
except AttributeError:
|
except AttributeError:
|
||||||
url = "unknown"
|
url = "unknown"
|
||||||
|
|
||||||
response_message = ('Exception occurred while handling uri: %s')
|
response_message = "Exception occurred while handling uri: %s"
|
||||||
logger.exception(response_message, url)
|
logger.exception(response_message, url)
|
||||||
|
|
||||||
if issubclass(type(exception), SanicException):
|
if issubclass(type(exception), SanicException):
|
||||||
return text(
|
return text(
|
||||||
'Error: {}'.format(exception),
|
"Error: {}".format(exception),
|
||||||
status=getattr(exception, 'status_code', 500),
|
status=getattr(exception, "status_code", 500),
|
||||||
headers=getattr(exception, 'headers', dict())
|
headers=getattr(exception, "headers", dict()),
|
||||||
)
|
)
|
||||||
elif self.debug:
|
elif self.debug:
|
||||||
html_output = self._render_traceback_html(exception, request)
|
html_output = self._render_traceback_html(exception, request)
|
||||||
|
@ -129,32 +133,37 @@ class ErrorHandler:
|
||||||
|
|
||||||
class ContentRangeHandler:
|
class ContentRangeHandler:
|
||||||
"""Class responsible for parsing request header"""
|
"""Class responsible for parsing request header"""
|
||||||
__slots__ = ('start', 'end', 'size', 'total', 'headers')
|
|
||||||
|
__slots__ = ("start", "end", "size", "total", "headers")
|
||||||
|
|
||||||
def __init__(self, request, stats):
|
def __init__(self, request, stats):
|
||||||
self.total = stats.st_size
|
self.total = stats.st_size
|
||||||
_range = request.headers.get('Range')
|
_range = request.headers.get("Range")
|
||||||
if _range is None:
|
if _range is None:
|
||||||
raise HeaderNotFound('Range Header Not Found')
|
raise HeaderNotFound("Range Header Not Found")
|
||||||
unit, _, value = tuple(map(str.strip, _range.partition('=')))
|
unit, _, value = tuple(map(str.strip, _range.partition("=")))
|
||||||
if unit != 'bytes':
|
if unit != "bytes":
|
||||||
raise InvalidRangeType(
|
raise InvalidRangeType(
|
||||||
'%s is not a valid Range Type' % (unit,), self)
|
"%s is not a valid Range Type" % (unit,), self
|
||||||
start_b, _, end_b = tuple(map(str.strip, value.partition('-')))
|
)
|
||||||
|
start_b, _, end_b = tuple(map(str.strip, value.partition("-")))
|
||||||
try:
|
try:
|
||||||
self.start = int(start_b) if start_b else None
|
self.start = int(start_b) if start_b else None
|
||||||
except ValueError:
|
except ValueError:
|
||||||
raise ContentRangeError(
|
raise ContentRangeError(
|
||||||
'\'%s\' is invalid for Content Range' % (start_b,), self)
|
"'%s' is invalid for Content Range" % (start_b,), self
|
||||||
|
)
|
||||||
try:
|
try:
|
||||||
self.end = int(end_b) if end_b else None
|
self.end = int(end_b) if end_b else None
|
||||||
except ValueError:
|
except ValueError:
|
||||||
raise ContentRangeError(
|
raise ContentRangeError(
|
||||||
'\'%s\' is invalid for Content Range' % (end_b,), self)
|
"'%s' is invalid for Content Range" % (end_b,), self
|
||||||
|
)
|
||||||
if self.end is None:
|
if self.end is None:
|
||||||
if self.start is None:
|
if self.start is None:
|
||||||
raise ContentRangeError(
|
raise ContentRangeError(
|
||||||
'Invalid for Content Range parameters', self)
|
"Invalid for Content Range parameters", self
|
||||||
|
)
|
||||||
else:
|
else:
|
||||||
# this case represents `Content-Range: bytes 5-`
|
# this case represents `Content-Range: bytes 5-`
|
||||||
self.end = self.total
|
self.end = self.total
|
||||||
|
@ -165,11 +174,13 @@ class ContentRangeHandler:
|
||||||
self.end = self.total
|
self.end = self.total
|
||||||
if self.start >= self.end:
|
if self.start >= self.end:
|
||||||
raise ContentRangeError(
|
raise ContentRangeError(
|
||||||
'Invalid for Content Range parameters', self)
|
"Invalid for Content Range parameters", self
|
||||||
|
)
|
||||||
self.size = self.end - self.start
|
self.size = self.end - self.start
|
||||||
self.headers = {
|
self.headers = {
|
||||||
'Content-Range': "bytes %s-%s/%s" % (
|
"Content-Range": "bytes %s-%s/%s"
|
||||||
self.start, self.end, self.total)}
|
% (self.start, self.end, self.total)
|
||||||
|
}
|
||||||
|
|
||||||
def __bool__(self):
|
def __bool__(self):
|
||||||
return self.size > 0
|
return self.size > 0
|
||||||
|
|
179
sanic/helpers.py
179
sanic/helpers.py
|
@ -1,93 +1,97 @@
|
||||||
"""Defines basics of HTTP standard."""
|
"""Defines basics of HTTP standard."""
|
||||||
|
|
||||||
STATUS_CODES = {
|
STATUS_CODES = {
|
||||||
100: b'Continue',
|
100: b"Continue",
|
||||||
101: b'Switching Protocols',
|
101: b"Switching Protocols",
|
||||||
102: b'Processing',
|
102: b"Processing",
|
||||||
200: b'OK',
|
200: b"OK",
|
||||||
201: b'Created',
|
201: b"Created",
|
||||||
202: b'Accepted',
|
202: b"Accepted",
|
||||||
203: b'Non-Authoritative Information',
|
203: b"Non-Authoritative Information",
|
||||||
204: b'No Content',
|
204: b"No Content",
|
||||||
205: b'Reset Content',
|
205: b"Reset Content",
|
||||||
206: b'Partial Content',
|
206: b"Partial Content",
|
||||||
207: b'Multi-Status',
|
207: b"Multi-Status",
|
||||||
208: b'Already Reported',
|
208: b"Already Reported",
|
||||||
226: b'IM Used',
|
226: b"IM Used",
|
||||||
300: b'Multiple Choices',
|
300: b"Multiple Choices",
|
||||||
301: b'Moved Permanently',
|
301: b"Moved Permanently",
|
||||||
302: b'Found',
|
302: b"Found",
|
||||||
303: b'See Other',
|
303: b"See Other",
|
||||||
304: b'Not Modified',
|
304: b"Not Modified",
|
||||||
305: b'Use Proxy',
|
305: b"Use Proxy",
|
||||||
307: b'Temporary Redirect',
|
307: b"Temporary Redirect",
|
||||||
308: b'Permanent Redirect',
|
308: b"Permanent Redirect",
|
||||||
400: b'Bad Request',
|
400: b"Bad Request",
|
||||||
401: b'Unauthorized',
|
401: b"Unauthorized",
|
||||||
402: b'Payment Required',
|
402: b"Payment Required",
|
||||||
403: b'Forbidden',
|
403: b"Forbidden",
|
||||||
404: b'Not Found',
|
404: b"Not Found",
|
||||||
405: b'Method Not Allowed',
|
405: b"Method Not Allowed",
|
||||||
406: b'Not Acceptable',
|
406: b"Not Acceptable",
|
||||||
407: b'Proxy Authentication Required',
|
407: b"Proxy Authentication Required",
|
||||||
408: b'Request Timeout',
|
408: b"Request Timeout",
|
||||||
409: b'Conflict',
|
409: b"Conflict",
|
||||||
410: b'Gone',
|
410: b"Gone",
|
||||||
411: b'Length Required',
|
411: b"Length Required",
|
||||||
412: b'Precondition Failed',
|
412: b"Precondition Failed",
|
||||||
413: b'Request Entity Too Large',
|
413: b"Request Entity Too Large",
|
||||||
414: b'Request-URI Too Long',
|
414: b"Request-URI Too Long",
|
||||||
415: b'Unsupported Media Type',
|
415: b"Unsupported Media Type",
|
||||||
416: b'Requested Range Not Satisfiable',
|
416: b"Requested Range Not Satisfiable",
|
||||||
417: b'Expectation Failed',
|
417: b"Expectation Failed",
|
||||||
418: b'I\'m a teapot',
|
418: b"I'm a teapot",
|
||||||
422: b'Unprocessable Entity',
|
422: b"Unprocessable Entity",
|
||||||
423: b'Locked',
|
423: b"Locked",
|
||||||
424: b'Failed Dependency',
|
424: b"Failed Dependency",
|
||||||
426: b'Upgrade Required',
|
426: b"Upgrade Required",
|
||||||
428: b'Precondition Required',
|
428: b"Precondition Required",
|
||||||
429: b'Too Many Requests',
|
429: b"Too Many Requests",
|
||||||
431: b'Request Header Fields Too Large',
|
431: b"Request Header Fields Too Large",
|
||||||
451: b'Unavailable For Legal Reasons',
|
451: b"Unavailable For Legal Reasons",
|
||||||
500: b'Internal Server Error',
|
500: b"Internal Server Error",
|
||||||
501: b'Not Implemented',
|
501: b"Not Implemented",
|
||||||
502: b'Bad Gateway',
|
502: b"Bad Gateway",
|
||||||
503: b'Service Unavailable',
|
503: b"Service Unavailable",
|
||||||
504: b'Gateway Timeout',
|
504: b"Gateway Timeout",
|
||||||
505: b'HTTP Version Not Supported',
|
505: b"HTTP Version Not Supported",
|
||||||
506: b'Variant Also Negotiates',
|
506: b"Variant Also Negotiates",
|
||||||
507: b'Insufficient Storage',
|
507: b"Insufficient Storage",
|
||||||
508: b'Loop Detected',
|
508: b"Loop Detected",
|
||||||
510: b'Not Extended',
|
510: b"Not Extended",
|
||||||
511: b'Network Authentication Required'
|
511: b"Network Authentication Required",
|
||||||
}
|
}
|
||||||
|
|
||||||
# According to https://tools.ietf.org/html/rfc2616#section-7.1
|
# According to https://tools.ietf.org/html/rfc2616#section-7.1
|
||||||
_ENTITY_HEADERS = frozenset([
|
_ENTITY_HEADERS = frozenset(
|
||||||
'allow',
|
[
|
||||||
'content-encoding',
|
"allow",
|
||||||
'content-language',
|
"content-encoding",
|
||||||
'content-length',
|
"content-language",
|
||||||
'content-location',
|
"content-length",
|
||||||
'content-md5',
|
"content-location",
|
||||||
'content-range',
|
"content-md5",
|
||||||
'content-type',
|
"content-range",
|
||||||
'expires',
|
"content-type",
|
||||||
'last-modified',
|
"expires",
|
||||||
'extension-header'
|
"last-modified",
|
||||||
])
|
"extension-header",
|
||||||
|
]
|
||||||
|
)
|
||||||
|
|
||||||
# According to https://tools.ietf.org/html/rfc2616#section-13.5.1
|
# According to https://tools.ietf.org/html/rfc2616#section-13.5.1
|
||||||
_HOP_BY_HOP_HEADERS = frozenset([
|
_HOP_BY_HOP_HEADERS = frozenset(
|
||||||
'connection',
|
[
|
||||||
'keep-alive',
|
"connection",
|
||||||
'proxy-authenticate',
|
"keep-alive",
|
||||||
'proxy-authorization',
|
"proxy-authenticate",
|
||||||
'te',
|
"proxy-authorization",
|
||||||
'trailers',
|
"te",
|
||||||
'transfer-encoding',
|
"trailers",
|
||||||
'upgrade'
|
"transfer-encoding",
|
||||||
])
|
"upgrade",
|
||||||
|
]
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
def has_message_body(status):
|
def has_message_body(status):
|
||||||
|
@ -110,8 +114,7 @@ def is_hop_by_hop_header(header):
|
||||||
return header.lower() in _HOP_BY_HOP_HEADERS
|
return header.lower() in _HOP_BY_HOP_HEADERS
|
||||||
|
|
||||||
|
|
||||||
def remove_entity_headers(headers,
|
def remove_entity_headers(headers, allowed=("content-location", "expires")):
|
||||||
allowed=('content-location', 'expires')):
|
|
||||||
"""
|
"""
|
||||||
Removes all the entity headers present in the headers given.
|
Removes all the entity headers present in the headers given.
|
||||||
According to RFC 2616 Section 10.3.5,
|
According to RFC 2616 Section 10.3.5,
|
||||||
|
@ -122,7 +125,9 @@ def remove_entity_headers(headers,
|
||||||
returns the headers without the entity headers
|
returns the headers without the entity headers
|
||||||
"""
|
"""
|
||||||
allowed = set([h.lower() for h in allowed])
|
allowed = set([h.lower() for h in allowed])
|
||||||
headers = {header: value for header, value in headers.items()
|
headers = {
|
||||||
if not is_entity_header(header)
|
header: value
|
||||||
and header.lower() not in allowed}
|
for header, value in headers.items()
|
||||||
|
if not is_entity_header(header) and header.lower() not in allowed
|
||||||
|
}
|
||||||
return headers
|
return headers
|
||||||
|
|
35
sanic/log.py
35
sanic/log.py
|
@ -5,59 +5,54 @@ import sys
|
||||||
LOGGING_CONFIG_DEFAULTS = dict(
|
LOGGING_CONFIG_DEFAULTS = dict(
|
||||||
version=1,
|
version=1,
|
||||||
disable_existing_loggers=False,
|
disable_existing_loggers=False,
|
||||||
|
|
||||||
loggers={
|
loggers={
|
||||||
"root": {
|
"root": {"level": "INFO", "handlers": ["console"]},
|
||||||
"level": "INFO",
|
|
||||||
"handlers": ["console"]
|
|
||||||
},
|
|
||||||
"sanic.error": {
|
"sanic.error": {
|
||||||
"level": "INFO",
|
"level": "INFO",
|
||||||
"handlers": ["error_console"],
|
"handlers": ["error_console"],
|
||||||
"propagate": True,
|
"propagate": True,
|
||||||
"qualname": "sanic.error"
|
"qualname": "sanic.error",
|
||||||
},
|
},
|
||||||
|
|
||||||
"sanic.access": {
|
"sanic.access": {
|
||||||
"level": "INFO",
|
"level": "INFO",
|
||||||
"handlers": ["access_console"],
|
"handlers": ["access_console"],
|
||||||
"propagate": True,
|
"propagate": True,
|
||||||
"qualname": "sanic.access"
|
"qualname": "sanic.access",
|
||||||
}
|
},
|
||||||
},
|
},
|
||||||
handlers={
|
handlers={
|
||||||
"console": {
|
"console": {
|
||||||
"class": "logging.StreamHandler",
|
"class": "logging.StreamHandler",
|
||||||
"formatter": "generic",
|
"formatter": "generic",
|
||||||
"stream": sys.stdout
|
"stream": sys.stdout,
|
||||||
},
|
},
|
||||||
"error_console": {
|
"error_console": {
|
||||||
"class": "logging.StreamHandler",
|
"class": "logging.StreamHandler",
|
||||||
"formatter": "generic",
|
"formatter": "generic",
|
||||||
"stream": sys.stderr
|
"stream": sys.stderr,
|
||||||
},
|
},
|
||||||
"access_console": {
|
"access_console": {
|
||||||
"class": "logging.StreamHandler",
|
"class": "logging.StreamHandler",
|
||||||
"formatter": "access",
|
"formatter": "access",
|
||||||
"stream": sys.stdout
|
"stream": sys.stdout,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
formatters={
|
formatters={
|
||||||
"generic": {
|
"generic": {
|
||||||
"format": "%(asctime)s [%(process)d] [%(levelname)s] %(message)s",
|
"format": "%(asctime)s [%(process)d] [%(levelname)s] %(message)s",
|
||||||
"datefmt": "[%Y-%m-%d %H:%M:%S %z]",
|
"datefmt": "[%Y-%m-%d %H:%M:%S %z]",
|
||||||
"class": "logging.Formatter"
|
"class": "logging.Formatter",
|
||||||
},
|
},
|
||||||
"access": {
|
"access": {
|
||||||
"format": "%(asctime)s - (%(name)s)[%(levelname)s][%(host)s]: " +
|
"format": "%(asctime)s - (%(name)s)[%(levelname)s][%(host)s]: "
|
||||||
"%(request)s %(message)s %(status)d %(byte)d",
|
+ "%(request)s %(message)s %(status)d %(byte)d",
|
||||||
"datefmt": "[%Y-%m-%d %H:%M:%S %z]",
|
"datefmt": "[%Y-%m-%d %H:%M:%S %z]",
|
||||||
"class": "logging.Formatter"
|
"class": "logging.Formatter",
|
||||||
|
},
|
||||||
},
|
},
|
||||||
}
|
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
logger = logging.getLogger('sanic.root')
|
logger = logging.getLogger("sanic.root")
|
||||||
error_logger = logging.getLogger('sanic.error')
|
error_logger = logging.getLogger("sanic.error")
|
||||||
access_logger = logging.getLogger('sanic.access')
|
access_logger = logging.getLogger("sanic.access")
|
||||||
|
|
|
@ -18,7 +18,7 @@ def _iter_module_files():
|
||||||
for module in list(sys.modules.values()):
|
for module in list(sys.modules.values()):
|
||||||
if module is None:
|
if module is None:
|
||||||
continue
|
continue
|
||||||
filename = getattr(module, '__file__', None)
|
filename = getattr(module, "__file__", None)
|
||||||
if filename:
|
if filename:
|
||||||
old = None
|
old = None
|
||||||
while not os.path.isfile(filename):
|
while not os.path.isfile(filename):
|
||||||
|
@ -27,7 +27,7 @@ def _iter_module_files():
|
||||||
if filename == old:
|
if filename == old:
|
||||||
break
|
break
|
||||||
else:
|
else:
|
||||||
if filename[-4:] in ('.pyc', '.pyo'):
|
if filename[-4:] in (".pyc", ".pyo"):
|
||||||
filename = filename[:-1]
|
filename = filename[:-1]
|
||||||
yield filename
|
yield filename
|
||||||
|
|
||||||
|
@ -45,11 +45,13 @@ def restart_with_reloader():
|
||||||
"""
|
"""
|
||||||
args = _get_args_for_reloading()
|
args = _get_args_for_reloading()
|
||||||
new_environ = os.environ.copy()
|
new_environ = os.environ.copy()
|
||||||
new_environ['SANIC_SERVER_RUNNING'] = 'true'
|
new_environ["SANIC_SERVER_RUNNING"] = "true"
|
||||||
cmd = ' '.join(args)
|
cmd = " ".join(args)
|
||||||
worker_process = Process(
|
worker_process = Process(
|
||||||
target=subprocess.call, args=(cmd,),
|
target=subprocess.call,
|
||||||
kwargs=dict(shell=True, env=new_environ))
|
args=(cmd,),
|
||||||
|
kwargs=dict(shell=True, env=new_environ),
|
||||||
|
)
|
||||||
worker_process.start()
|
worker_process.start()
|
||||||
return worker_process
|
return worker_process
|
||||||
|
|
||||||
|
@ -67,8 +69,10 @@ def kill_process_children_unix(pid):
|
||||||
children_list_pid = children_list_file.read().split()
|
children_list_pid = children_list_file.read().split()
|
||||||
|
|
||||||
for child_pid in children_list_pid:
|
for child_pid in children_list_pid:
|
||||||
children_proc_path = "/proc/%s/task/%s/children" % \
|
children_proc_path = "/proc/%s/task/%s/children" % (
|
||||||
(child_pid, child_pid)
|
child_pid,
|
||||||
|
child_pid,
|
||||||
|
)
|
||||||
if not os.path.isfile(children_proc_path):
|
if not os.path.isfile(children_proc_path):
|
||||||
continue
|
continue
|
||||||
with open(children_proc_path) as children_list_file_2:
|
with open(children_proc_path) as children_list_file_2:
|
||||||
|
@ -90,7 +94,7 @@ def kill_process_children_osx(pid):
|
||||||
:param pid: PID of parent process (process ID)
|
:param pid: PID of parent process (process ID)
|
||||||
:return: Nothing
|
:return: Nothing
|
||||||
"""
|
"""
|
||||||
subprocess.run(['pkill', '-P', str(pid)])
|
subprocess.run(["pkill", "-P", str(pid)])
|
||||||
|
|
||||||
|
|
||||||
def kill_process_children(pid):
|
def kill_process_children(pid):
|
||||||
|
@ -99,9 +103,9 @@ def kill_process_children(pid):
|
||||||
:param pid: PID of parent process (process ID)
|
:param pid: PID of parent process (process ID)
|
||||||
:return: Nothing
|
:return: Nothing
|
||||||
"""
|
"""
|
||||||
if sys.platform == 'darwin':
|
if sys.platform == "darwin":
|
||||||
kill_process_children_osx(pid)
|
kill_process_children_osx(pid)
|
||||||
elif sys.platform == 'linux':
|
elif sys.platform == "linux":
|
||||||
kill_process_children_unix(pid)
|
kill_process_children_unix(pid)
|
||||||
else:
|
else:
|
||||||
pass # should signal error here
|
pass # should signal error here
|
||||||
|
@ -127,9 +131,11 @@ def watchdog(sleep_interval):
|
||||||
mtimes = {}
|
mtimes = {}
|
||||||
worker_process = restart_with_reloader()
|
worker_process = restart_with_reloader()
|
||||||
signal.signal(
|
signal.signal(
|
||||||
signal.SIGTERM, lambda *args: kill_program_completly(worker_process))
|
signal.SIGTERM, lambda *args: kill_program_completly(worker_process)
|
||||||
|
)
|
||||||
signal.signal(
|
signal.signal(
|
||||||
signal.SIGINT, lambda *args: kill_program_completly(worker_process))
|
signal.SIGINT, lambda *args: kill_program_completly(worker_process)
|
||||||
|
)
|
||||||
while True:
|
while True:
|
||||||
for filename in _iter_module_files():
|
for filename in _iter_module_files():
|
||||||
try:
|
try:
|
||||||
|
|
161
sanic/request.py
161
sanic/request.py
|
@ -10,9 +10,11 @@ try:
|
||||||
from ujson import loads as json_loads
|
from ujson import loads as json_loads
|
||||||
except ImportError:
|
except ImportError:
|
||||||
if sys.version_info[:2] == (3, 5):
|
if sys.version_info[:2] == (3, 5):
|
||||||
|
|
||||||
def json_loads(data):
|
def json_loads(data):
|
||||||
# on Python 3.5 json.loads only supports str not bytes
|
# on Python 3.5 json.loads only supports str not bytes
|
||||||
return json.loads(data.decode())
|
return json.loads(data.decode())
|
||||||
|
|
||||||
else:
|
else:
|
||||||
json_loads = json.loads
|
json_loads = json.loads
|
||||||
|
|
||||||
|
@ -43,11 +45,28 @@ class RequestParameters(dict):
|
||||||
|
|
||||||
class Request(dict):
|
class Request(dict):
|
||||||
"""Properties of an HTTP request such as URL, headers, etc."""
|
"""Properties of an HTTP request such as URL, headers, etc."""
|
||||||
|
|
||||||
__slots__ = (
|
__slots__ = (
|
||||||
'app', 'headers', 'version', 'method', '_cookies', 'transport',
|
"app",
|
||||||
'body', 'parsed_json', 'parsed_args', 'parsed_form', 'parsed_files',
|
"headers",
|
||||||
'_ip', '_parsed_url', 'uri_template', 'stream', '_remote_addr',
|
"version",
|
||||||
'_socket', '_port', '__weakref__', 'raw_url'
|
"method",
|
||||||
|
"_cookies",
|
||||||
|
"transport",
|
||||||
|
"body",
|
||||||
|
"parsed_json",
|
||||||
|
"parsed_args",
|
||||||
|
"parsed_form",
|
||||||
|
"parsed_files",
|
||||||
|
"_ip",
|
||||||
|
"_parsed_url",
|
||||||
|
"uri_template",
|
||||||
|
"stream",
|
||||||
|
"_remote_addr",
|
||||||
|
"_socket",
|
||||||
|
"_port",
|
||||||
|
"__weakref__",
|
||||||
|
"raw_url",
|
||||||
)
|
)
|
||||||
|
|
||||||
def __init__(self, url_bytes, headers, version, method, transport):
|
def __init__(self, url_bytes, headers, version, method, transport):
|
||||||
|
@ -73,10 +92,10 @@ class Request(dict):
|
||||||
|
|
||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
if self.method is None or not self.path:
|
if self.method is None or not self.path:
|
||||||
return '<{0}>'.format(self.__class__.__name__)
|
return "<{0}>".format(self.__class__.__name__)
|
||||||
return '<{0}: {1} {2}>'.format(self.__class__.__name__,
|
return "<{0}: {1} {2}>".format(
|
||||||
self.method,
|
self.__class__.__name__, self.method, self.path
|
||||||
self.path)
|
)
|
||||||
|
|
||||||
def __bool__(self):
|
def __bool__(self):
|
||||||
if self.transport:
|
if self.transport:
|
||||||
|
@ -106,8 +125,8 @@ class Request(dict):
|
||||||
|
|
||||||
:return: token related to request
|
:return: token related to request
|
||||||
"""
|
"""
|
||||||
prefixes = ('Bearer', 'Token')
|
prefixes = ("Bearer", "Token")
|
||||||
auth_header = self.headers.get('Authorization')
|
auth_header = self.headers.get("Authorization")
|
||||||
|
|
||||||
if auth_header is not None:
|
if auth_header is not None:
|
||||||
for prefix in prefixes:
|
for prefix in prefixes:
|
||||||
|
@ -122,17 +141,20 @@ class Request(dict):
|
||||||
self.parsed_form = RequestParameters()
|
self.parsed_form = RequestParameters()
|
||||||
self.parsed_files = RequestParameters()
|
self.parsed_files = RequestParameters()
|
||||||
content_type = self.headers.get(
|
content_type = self.headers.get(
|
||||||
'Content-Type', DEFAULT_HTTP_CONTENT_TYPE)
|
"Content-Type", DEFAULT_HTTP_CONTENT_TYPE
|
||||||
|
)
|
||||||
content_type, parameters = parse_header(content_type)
|
content_type, parameters = parse_header(content_type)
|
||||||
try:
|
try:
|
||||||
if content_type == 'application/x-www-form-urlencoded':
|
if content_type == "application/x-www-form-urlencoded":
|
||||||
self.parsed_form = RequestParameters(
|
self.parsed_form = RequestParameters(
|
||||||
parse_qs(self.body.decode('utf-8')))
|
parse_qs(self.body.decode("utf-8"))
|
||||||
elif content_type == 'multipart/form-data':
|
)
|
||||||
|
elif content_type == "multipart/form-data":
|
||||||
# TODO: Stream this instead of reading to/from memory
|
# TODO: Stream this instead of reading to/from memory
|
||||||
boundary = parameters['boundary'].encode('utf-8')
|
boundary = parameters["boundary"].encode("utf-8")
|
||||||
self.parsed_form, self.parsed_files = (
|
self.parsed_form, self.parsed_files = parse_multipart_form(
|
||||||
parse_multipart_form(self.body, boundary))
|
self.body, boundary
|
||||||
|
)
|
||||||
except Exception:
|
except Exception:
|
||||||
error_logger.exception("Failed when parsing form")
|
error_logger.exception("Failed when parsing form")
|
||||||
|
|
||||||
|
@ -150,7 +172,8 @@ class Request(dict):
|
||||||
if self.parsed_args is None:
|
if self.parsed_args is None:
|
||||||
if self.query_string:
|
if self.query_string:
|
||||||
self.parsed_args = RequestParameters(
|
self.parsed_args = RequestParameters(
|
||||||
parse_qs(self.query_string))
|
parse_qs(self.query_string)
|
||||||
|
)
|
||||||
else:
|
else:
|
||||||
self.parsed_args = RequestParameters()
|
self.parsed_args = RequestParameters()
|
||||||
return self.parsed_args
|
return self.parsed_args
|
||||||
|
@ -162,37 +185,40 @@ class Request(dict):
|
||||||
@property
|
@property
|
||||||
def cookies(self):
|
def cookies(self):
|
||||||
if self._cookies is None:
|
if self._cookies is None:
|
||||||
cookie = self.headers.get('Cookie')
|
cookie = self.headers.get("Cookie")
|
||||||
if cookie is not None:
|
if cookie is not None:
|
||||||
cookies = SimpleCookie()
|
cookies = SimpleCookie()
|
||||||
cookies.load(cookie)
|
cookies.load(cookie)
|
||||||
self._cookies = {name: cookie.value
|
self._cookies = {
|
||||||
for name, cookie in cookies.items()}
|
name: cookie.value for name, cookie in cookies.items()
|
||||||
|
}
|
||||||
else:
|
else:
|
||||||
self._cookies = {}
|
self._cookies = {}
|
||||||
return self._cookies
|
return self._cookies
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def ip(self):
|
def ip(self):
|
||||||
if not hasattr(self, '_socket'):
|
if not hasattr(self, "_socket"):
|
||||||
self._get_address()
|
self._get_address()
|
||||||
return self._ip
|
return self._ip
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def port(self):
|
def port(self):
|
||||||
if not hasattr(self, '_socket'):
|
if not hasattr(self, "_socket"):
|
||||||
self._get_address()
|
self._get_address()
|
||||||
return self._port
|
return self._port
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def socket(self):
|
def socket(self):
|
||||||
if not hasattr(self, '_socket'):
|
if not hasattr(self, "_socket"):
|
||||||
self._get_address()
|
self._get_address()
|
||||||
return self._socket
|
return self._socket
|
||||||
|
|
||||||
def _get_address(self):
|
def _get_address(self):
|
||||||
self._socket = self.transport.get_extra_info('peername') or \
|
self._socket = self.transport.get_extra_info("peername") or (
|
||||||
(None, None)
|
None,
|
||||||
|
None,
|
||||||
|
)
|
||||||
self._ip = self._socket[0]
|
self._ip = self._socket[0]
|
||||||
self._port = self._socket[1]
|
self._port = self._socket[1]
|
||||||
|
|
||||||
|
@ -202,29 +228,31 @@ class Request(dict):
|
||||||
|
|
||||||
:return: original client ip.
|
:return: original client ip.
|
||||||
"""
|
"""
|
||||||
if not hasattr(self, '_remote_addr'):
|
if not hasattr(self, "_remote_addr"):
|
||||||
forwarded_for = self.headers.get('X-Forwarded-For', '').split(',')
|
forwarded_for = self.headers.get("X-Forwarded-For", "").split(",")
|
||||||
remote_addrs = [
|
remote_addrs = [
|
||||||
addr for addr in [
|
addr
|
||||||
addr.strip() for addr in forwarded_for
|
for addr in [addr.strip() for addr in forwarded_for]
|
||||||
] if addr
|
if addr
|
||||||
]
|
]
|
||||||
if len(remote_addrs) > 0:
|
if len(remote_addrs) > 0:
|
||||||
self._remote_addr = remote_addrs[0]
|
self._remote_addr = remote_addrs[0]
|
||||||
else:
|
else:
|
||||||
self._remote_addr = ''
|
self._remote_addr = ""
|
||||||
return self._remote_addr
|
return self._remote_addr
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def scheme(self):
|
def scheme(self):
|
||||||
if self.app.websocket_enabled \
|
if (
|
||||||
and self.headers.get('upgrade') == 'websocket':
|
self.app.websocket_enabled
|
||||||
scheme = 'ws'
|
and self.headers.get("upgrade") == "websocket"
|
||||||
|
):
|
||||||
|
scheme = "ws"
|
||||||
else:
|
else:
|
||||||
scheme = 'http'
|
scheme = "http"
|
||||||
|
|
||||||
if self.transport.get_extra_info('sslcontext'):
|
if self.transport.get_extra_info("sslcontext"):
|
||||||
scheme += 's'
|
scheme += "s"
|
||||||
|
|
||||||
return scheme
|
return scheme
|
||||||
|
|
||||||
|
@ -232,11 +260,11 @@ class Request(dict):
|
||||||
def host(self):
|
def host(self):
|
||||||
# it appears that httptools doesn't return the host
|
# it appears that httptools doesn't return the host
|
||||||
# so pull it from the headers
|
# so pull it from the headers
|
||||||
return self.headers.get('Host', '')
|
return self.headers.get("Host", "")
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def content_type(self):
|
def content_type(self):
|
||||||
return self.headers.get('Content-Type', DEFAULT_HTTP_CONTENT_TYPE)
|
return self.headers.get("Content-Type", DEFAULT_HTTP_CONTENT_TYPE)
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def match_info(self):
|
def match_info(self):
|
||||||
|
@ -245,27 +273,23 @@ class Request(dict):
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def path(self):
|
def path(self):
|
||||||
return self._parsed_url.path.decode('utf-8')
|
return self._parsed_url.path.decode("utf-8")
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def query_string(self):
|
def query_string(self):
|
||||||
if self._parsed_url.query:
|
if self._parsed_url.query:
|
||||||
return self._parsed_url.query.decode('utf-8')
|
return self._parsed_url.query.decode("utf-8")
|
||||||
else:
|
else:
|
||||||
return ''
|
return ""
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def url(self):
|
def url(self):
|
||||||
return urlunparse((
|
return urlunparse(
|
||||||
self.scheme,
|
(self.scheme, self.host, self.path, None, self.query_string, None)
|
||||||
self.host,
|
)
|
||||||
self.path,
|
|
||||||
None,
|
|
||||||
self.query_string,
|
|
||||||
None))
|
|
||||||
|
|
||||||
|
|
||||||
File = namedtuple('File', ['type', 'body', 'name'])
|
File = namedtuple("File", ["type", "body", "name"])
|
||||||
|
|
||||||
|
|
||||||
def parse_multipart_form(body, boundary):
|
def parse_multipart_form(body, boundary):
|
||||||
|
@ -281,37 +305,38 @@ def parse_multipart_form(body, boundary):
|
||||||
form_parts = body.split(boundary)
|
form_parts = body.split(boundary)
|
||||||
for form_part in form_parts[1:-1]:
|
for form_part in form_parts[1:-1]:
|
||||||
file_name = None
|
file_name = None
|
||||||
content_type = 'text/plain'
|
content_type = "text/plain"
|
||||||
content_charset = 'utf-8'
|
content_charset = "utf-8"
|
||||||
field_name = None
|
field_name = None
|
||||||
line_index = 2
|
line_index = 2
|
||||||
line_end_index = 0
|
line_end_index = 0
|
||||||
while not line_end_index == -1:
|
while not line_end_index == -1:
|
||||||
line_end_index = form_part.find(b'\r\n', line_index)
|
line_end_index = form_part.find(b"\r\n", line_index)
|
||||||
form_line = form_part[line_index:line_end_index].decode('utf-8')
|
form_line = form_part[line_index:line_end_index].decode("utf-8")
|
||||||
line_index = line_end_index + 2
|
line_index = line_end_index + 2
|
||||||
|
|
||||||
if not form_line:
|
if not form_line:
|
||||||
break
|
break
|
||||||
|
|
||||||
colon_index = form_line.index(':')
|
colon_index = form_line.index(":")
|
||||||
form_header_field = form_line[0:colon_index].lower()
|
form_header_field = form_line[0:colon_index].lower()
|
||||||
form_header_value, form_parameters = parse_header(
|
form_header_value, form_parameters = parse_header(
|
||||||
form_line[colon_index + 2:])
|
form_line[colon_index + 2 :]
|
||||||
|
)
|
||||||
|
|
||||||
if form_header_field == 'content-disposition':
|
if form_header_field == "content-disposition":
|
||||||
file_name = form_parameters.get('filename')
|
file_name = form_parameters.get("filename")
|
||||||
field_name = form_parameters.get('name')
|
field_name = form_parameters.get("name")
|
||||||
elif form_header_field == 'content-type':
|
elif form_header_field == "content-type":
|
||||||
content_type = form_header_value
|
content_type = form_header_value
|
||||||
content_charset = form_parameters.get('charset', 'utf-8')
|
content_charset = form_parameters.get("charset", "utf-8")
|
||||||
|
|
||||||
if field_name:
|
if field_name:
|
||||||
post_data = form_part[line_index:-4]
|
post_data = form_part[line_index:-4]
|
||||||
if file_name:
|
if file_name:
|
||||||
form_file = File(type=content_type,
|
form_file = File(
|
||||||
name=file_name,
|
type=content_type, name=file_name, body=post_data
|
||||||
body=post_data)
|
)
|
||||||
if field_name in files:
|
if field_name in files:
|
||||||
files[field_name].append(form_file)
|
files[field_name].append(form_file)
|
||||||
else:
|
else:
|
||||||
|
@ -323,7 +348,9 @@ def parse_multipart_form(body, boundary):
|
||||||
else:
|
else:
|
||||||
fields[field_name] = [value]
|
fields[field_name] = [value]
|
||||||
else:
|
else:
|
||||||
logger.debug('Form-data field does not have a \'name\' parameter \
|
logger.debug(
|
||||||
in the Content-Disposition header')
|
"Form-data field does not have a 'name' parameter \
|
||||||
|
in the Content-Disposition header"
|
||||||
|
)
|
||||||
|
|
||||||
return fields, files
|
return fields, files
|
||||||
|
|
|
@ -24,16 +24,18 @@ class BaseHTTPResponse:
|
||||||
return str(data).encode()
|
return str(data).encode()
|
||||||
|
|
||||||
def _parse_headers(self):
|
def _parse_headers(self):
|
||||||
headers = b''
|
headers = b""
|
||||||
for name, value in self.headers.items():
|
for name, value in self.headers.items():
|
||||||
try:
|
try:
|
||||||
headers += (
|
headers += b"%b: %b\r\n" % (
|
||||||
b'%b: %b\r\n' % (
|
name.encode(),
|
||||||
name.encode(), value.encode('utf-8')))
|
value.encode("utf-8"),
|
||||||
|
)
|
||||||
except AttributeError:
|
except AttributeError:
|
||||||
headers += (
|
headers += b"%b: %b\r\n" % (
|
||||||
b'%b: %b\r\n' % (
|
str(name).encode(),
|
||||||
str(name).encode(), str(value).encode('utf-8')))
|
str(value).encode("utf-8"),
|
||||||
|
)
|
||||||
|
|
||||||
return headers
|
return headers
|
||||||
|
|
||||||
|
@ -46,12 +48,17 @@ class BaseHTTPResponse:
|
||||||
|
|
||||||
class StreamingHTTPResponse(BaseHTTPResponse):
|
class StreamingHTTPResponse(BaseHTTPResponse):
|
||||||
__slots__ = (
|
__slots__ = (
|
||||||
'protocol', 'streaming_fn', 'status',
|
"protocol",
|
||||||
'content_type', 'headers', '_cookies'
|
"streaming_fn",
|
||||||
|
"status",
|
||||||
|
"content_type",
|
||||||
|
"headers",
|
||||||
|
"_cookies",
|
||||||
)
|
)
|
||||||
|
|
||||||
def __init__(self, streaming_fn, status=200, headers=None,
|
def __init__(
|
||||||
content_type='text/plain'):
|
self, streaming_fn, status=200, headers=None, content_type="text/plain"
|
||||||
|
):
|
||||||
self.content_type = content_type
|
self.content_type = content_type
|
||||||
self.streaming_fn = streaming_fn
|
self.streaming_fn = streaming_fn
|
||||||
self.status = status
|
self.status = status
|
||||||
|
@ -66,61 +73,69 @@ class StreamingHTTPResponse(BaseHTTPResponse):
|
||||||
if type(data) != bytes:
|
if type(data) != bytes:
|
||||||
data = self._encode_body(data)
|
data = self._encode_body(data)
|
||||||
|
|
||||||
self.protocol.push_data(
|
self.protocol.push_data(b"%x\r\n%b\r\n" % (len(data), data))
|
||||||
b"%x\r\n%b\r\n" % (len(data), data))
|
|
||||||
await self.protocol.drain()
|
await self.protocol.drain()
|
||||||
|
|
||||||
async def stream(
|
async def stream(
|
||||||
self, version="1.1", keep_alive=False, keep_alive_timeout=None):
|
self, version="1.1", keep_alive=False, keep_alive_timeout=None
|
||||||
|
):
|
||||||
"""Streams headers, runs the `streaming_fn` callback that writes
|
"""Streams headers, runs the `streaming_fn` callback that writes
|
||||||
content to the response body, then finalizes the response body.
|
content to the response body, then finalizes the response body.
|
||||||
"""
|
"""
|
||||||
headers = self.get_headers(
|
headers = self.get_headers(
|
||||||
version, keep_alive=keep_alive,
|
version,
|
||||||
keep_alive_timeout=keep_alive_timeout)
|
keep_alive=keep_alive,
|
||||||
|
keep_alive_timeout=keep_alive_timeout,
|
||||||
|
)
|
||||||
self.protocol.push_data(headers)
|
self.protocol.push_data(headers)
|
||||||
await self.protocol.drain()
|
await self.protocol.drain()
|
||||||
await self.streaming_fn(self)
|
await self.streaming_fn(self)
|
||||||
self.protocol.push_data(b'0\r\n\r\n')
|
self.protocol.push_data(b"0\r\n\r\n")
|
||||||
# no need to await drain here after this write, because it is the
|
# no need to await drain here after this write, because it is the
|
||||||
# very last thing we write and nothing needs to wait for it.
|
# very last thing we write and nothing needs to wait for it.
|
||||||
|
|
||||||
def get_headers(
|
def get_headers(
|
||||||
self, version="1.1", keep_alive=False, keep_alive_timeout=None):
|
self, version="1.1", keep_alive=False, keep_alive_timeout=None
|
||||||
|
):
|
||||||
# This is all returned in a kind-of funky way
|
# This is all returned in a kind-of funky way
|
||||||
# We tried to make this as fast as possible in pure python
|
# We tried to make this as fast as possible in pure python
|
||||||
timeout_header = b''
|
timeout_header = b""
|
||||||
if keep_alive and keep_alive_timeout is not None:
|
if keep_alive and keep_alive_timeout is not None:
|
||||||
timeout_header = b'Keep-Alive: %d\r\n' % keep_alive_timeout
|
timeout_header = b"Keep-Alive: %d\r\n" % keep_alive_timeout
|
||||||
|
|
||||||
self.headers['Transfer-Encoding'] = 'chunked'
|
self.headers["Transfer-Encoding"] = "chunked"
|
||||||
self.headers.pop('Content-Length', None)
|
self.headers.pop("Content-Length", None)
|
||||||
self.headers['Content-Type'] = self.headers.get(
|
self.headers["Content-Type"] = self.headers.get(
|
||||||
'Content-Type', self.content_type)
|
"Content-Type", self.content_type
|
||||||
|
)
|
||||||
|
|
||||||
headers = self._parse_headers()
|
headers = self._parse_headers()
|
||||||
|
|
||||||
if self.status is 200:
|
if self.status is 200:
|
||||||
status = b'OK'
|
status = b"OK"
|
||||||
else:
|
else:
|
||||||
status = STATUS_CODES.get(self.status)
|
status = STATUS_CODES.get(self.status)
|
||||||
|
|
||||||
return (b'HTTP/%b %d %b\r\n'
|
return (b"HTTP/%b %d %b\r\n" b"%b" b"%b\r\n") % (
|
||||||
b'%b'
|
|
||||||
b'%b\r\n') % (
|
|
||||||
version.encode(),
|
version.encode(),
|
||||||
self.status,
|
self.status,
|
||||||
status,
|
status,
|
||||||
timeout_header,
|
timeout_header,
|
||||||
headers
|
headers,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
class HTTPResponse(BaseHTTPResponse):
|
class HTTPResponse(BaseHTTPResponse):
|
||||||
__slots__ = ('body', 'status', 'content_type', 'headers', '_cookies')
|
__slots__ = ("body", "status", "content_type", "headers", "_cookies")
|
||||||
|
|
||||||
def __init__(self, body=None, status=200, headers=None,
|
def __init__(
|
||||||
content_type='text/plain', body_bytes=b''):
|
self,
|
||||||
|
body=None,
|
||||||
|
status=200,
|
||||||
|
headers=None,
|
||||||
|
content_type="text/plain",
|
||||||
|
body_bytes=b"",
|
||||||
|
):
|
||||||
self.content_type = content_type
|
self.content_type = content_type
|
||||||
|
|
||||||
if body is not None:
|
if body is not None:
|
||||||
|
@ -132,22 +147,23 @@ class HTTPResponse(BaseHTTPResponse):
|
||||||
self.headers = CIMultiDict(headers or {})
|
self.headers = CIMultiDict(headers or {})
|
||||||
self._cookies = None
|
self._cookies = None
|
||||||
|
|
||||||
def output(
|
def output(self, version="1.1", keep_alive=False, keep_alive_timeout=None):
|
||||||
self, version="1.1", keep_alive=False, keep_alive_timeout=None):
|
|
||||||
# This is all returned in a kind-of funky way
|
# This is all returned in a kind-of funky way
|
||||||
# We tried to make this as fast as possible in pure python
|
# We tried to make this as fast as possible in pure python
|
||||||
timeout_header = b''
|
timeout_header = b""
|
||||||
if keep_alive and keep_alive_timeout is not None:
|
if keep_alive and keep_alive_timeout is not None:
|
||||||
timeout_header = b'Keep-Alive: %d\r\n' % keep_alive_timeout
|
timeout_header = b"Keep-Alive: %d\r\n" % keep_alive_timeout
|
||||||
|
|
||||||
body = b''
|
body = b""
|
||||||
if has_message_body(self.status):
|
if has_message_body(self.status):
|
||||||
body = self.body
|
body = self.body
|
||||||
self.headers['Content-Length'] = self.headers.get(
|
self.headers["Content-Length"] = self.headers.get(
|
||||||
'Content-Length', len(self.body))
|
"Content-Length", len(self.body)
|
||||||
|
)
|
||||||
|
|
||||||
self.headers['Content-Type'] = self.headers.get(
|
self.headers["Content-Type"] = self.headers.get(
|
||||||
'Content-Type', self.content_type)
|
"Content-Type", self.content_type
|
||||||
|
)
|
||||||
|
|
||||||
if self.status in (304, 412):
|
if self.status in (304, 412):
|
||||||
self.headers = remove_entity_headers(self.headers)
|
self.headers = remove_entity_headers(self.headers)
|
||||||
|
@ -155,22 +171,20 @@ class HTTPResponse(BaseHTTPResponse):
|
||||||
headers = self._parse_headers()
|
headers = self._parse_headers()
|
||||||
|
|
||||||
if self.status is 200:
|
if self.status is 200:
|
||||||
status = b'OK'
|
status = b"OK"
|
||||||
else:
|
else:
|
||||||
status = STATUS_CODES.get(self.status, b'UNKNOWN RESPONSE')
|
status = STATUS_CODES.get(self.status, b"UNKNOWN RESPONSE")
|
||||||
|
|
||||||
return (b'HTTP/%b %d %b\r\n'
|
return (
|
||||||
b'Connection: %b\r\n'
|
b"HTTP/%b %d %b\r\n" b"Connection: %b\r\n" b"%b" b"%b\r\n" b"%b"
|
||||||
b'%b'
|
) % (
|
||||||
b'%b\r\n'
|
|
||||||
b'%b') % (
|
|
||||||
version.encode(),
|
version.encode(),
|
||||||
self.status,
|
self.status,
|
||||||
status,
|
status,
|
||||||
b'keep-alive' if keep_alive else b'close',
|
b"keep-alive" if keep_alive else b"close",
|
||||||
timeout_header,
|
timeout_header,
|
||||||
headers,
|
headers,
|
||||||
body
|
body,
|
||||||
)
|
)
|
||||||
|
|
||||||
@property
|
@property
|
||||||
|
@ -180,9 +194,14 @@ class HTTPResponse(BaseHTTPResponse):
|
||||||
return self._cookies
|
return self._cookies
|
||||||
|
|
||||||
|
|
||||||
def json(body, status=200, headers=None,
|
def json(
|
||||||
content_type="application/json", dumps=json_dumps,
|
body,
|
||||||
**kwargs):
|
status=200,
|
||||||
|
headers=None,
|
||||||
|
content_type="application/json",
|
||||||
|
dumps=json_dumps,
|
||||||
|
**kwargs
|
||||||
|
):
|
||||||
"""
|
"""
|
||||||
Returns response object with body in json format.
|
Returns response object with body in json format.
|
||||||
|
|
||||||
|
@ -191,12 +210,17 @@ def json(body, status=200, headers=None,
|
||||||
:param headers: Custom Headers.
|
:param headers: Custom Headers.
|
||||||
:param kwargs: Remaining arguments that are passed to the json encoder.
|
:param kwargs: Remaining arguments that are passed to the json encoder.
|
||||||
"""
|
"""
|
||||||
return HTTPResponse(dumps(body, **kwargs), headers=headers,
|
return HTTPResponse(
|
||||||
status=status, content_type=content_type)
|
dumps(body, **kwargs),
|
||||||
|
headers=headers,
|
||||||
|
status=status,
|
||||||
|
content_type=content_type,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
def text(body, status=200, headers=None,
|
def text(
|
||||||
content_type="text/plain; charset=utf-8"):
|
body, status=200, headers=None, content_type="text/plain; charset=utf-8"
|
||||||
|
):
|
||||||
"""
|
"""
|
||||||
Returns response object with body in text format.
|
Returns response object with body in text format.
|
||||||
|
|
||||||
|
@ -206,12 +230,13 @@ def text(body, status=200, headers=None,
|
||||||
:param content_type: the content type (string) of the response
|
:param content_type: the content type (string) of the response
|
||||||
"""
|
"""
|
||||||
return HTTPResponse(
|
return HTTPResponse(
|
||||||
body, status=status, headers=headers,
|
body, status=status, headers=headers, content_type=content_type
|
||||||
content_type=content_type)
|
)
|
||||||
|
|
||||||
|
|
||||||
def raw(body, status=200, headers=None,
|
def raw(
|
||||||
content_type="application/octet-stream"):
|
body, status=200, headers=None, content_type="application/octet-stream"
|
||||||
|
):
|
||||||
"""
|
"""
|
||||||
Returns response object without encoding the body.
|
Returns response object without encoding the body.
|
||||||
|
|
||||||
|
@ -220,8 +245,12 @@ def raw(body, status=200, headers=None,
|
||||||
:param headers: Custom Headers.
|
:param headers: Custom Headers.
|
||||||
:param content_type: the content type (string) of the response.
|
:param content_type: the content type (string) of the response.
|
||||||
"""
|
"""
|
||||||
return HTTPResponse(body_bytes=body, status=status, headers=headers,
|
return HTTPResponse(
|
||||||
content_type=content_type)
|
body_bytes=body,
|
||||||
|
status=status,
|
||||||
|
headers=headers,
|
||||||
|
content_type=content_type,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
def html(body, status=200, headers=None):
|
def html(body, status=200, headers=None):
|
||||||
|
@ -232,12 +261,22 @@ def html(body, status=200, headers=None):
|
||||||
:param status: Response code.
|
:param status: Response code.
|
||||||
:param headers: Custom Headers.
|
:param headers: Custom Headers.
|
||||||
"""
|
"""
|
||||||
return HTTPResponse(body, status=status, headers=headers,
|
return HTTPResponse(
|
||||||
content_type="text/html; charset=utf-8")
|
body,
|
||||||
|
status=status,
|
||||||
|
headers=headers,
|
||||||
|
content_type="text/html; charset=utf-8",
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
async def file(location, status=200, mime_type=None, headers=None,
|
async def file(
|
||||||
filename=None, _range=None):
|
location,
|
||||||
|
status=200,
|
||||||
|
mime_type=None,
|
||||||
|
headers=None,
|
||||||
|
filename=None,
|
||||||
|
_range=None,
|
||||||
|
):
|
||||||
"""Return a response object with file data.
|
"""Return a response object with file data.
|
||||||
|
|
||||||
:param location: Location of file on system.
|
:param location: Location of file on system.
|
||||||
|
@ -249,28 +288,40 @@ async def file(location, status=200, mime_type=None, headers=None,
|
||||||
headers = headers or {}
|
headers = headers or {}
|
||||||
if filename:
|
if filename:
|
||||||
headers.setdefault(
|
headers.setdefault(
|
||||||
'Content-Disposition',
|
"Content-Disposition", 'attachment; filename="{}"'.format(filename)
|
||||||
'attachment; filename="{}"'.format(filename))
|
)
|
||||||
filename = filename or path.split(location)[-1]
|
filename = filename or path.split(location)[-1]
|
||||||
|
|
||||||
async with open_async(location, mode='rb') as _file:
|
async with open_async(location, mode="rb") as _file:
|
||||||
if _range:
|
if _range:
|
||||||
await _file.seek(_range.start)
|
await _file.seek(_range.start)
|
||||||
out_stream = await _file.read(_range.size)
|
out_stream = await _file.read(_range.size)
|
||||||
headers['Content-Range'] = 'bytes %s-%s/%s' % (
|
headers["Content-Range"] = "bytes %s-%s/%s" % (
|
||||||
_range.start, _range.end, _range.total)
|
_range.start,
|
||||||
|
_range.end,
|
||||||
|
_range.total,
|
||||||
|
)
|
||||||
else:
|
else:
|
||||||
out_stream = await _file.read()
|
out_stream = await _file.read()
|
||||||
|
|
||||||
mime_type = mime_type or guess_type(filename)[0] or 'text/plain'
|
mime_type = mime_type or guess_type(filename)[0] or "text/plain"
|
||||||
return HTTPResponse(status=status,
|
return HTTPResponse(
|
||||||
|
status=status,
|
||||||
headers=headers,
|
headers=headers,
|
||||||
content_type=mime_type,
|
content_type=mime_type,
|
||||||
body_bytes=out_stream)
|
body_bytes=out_stream,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
async def file_stream(location, status=200, chunk_size=4096, mime_type=None,
|
async def file_stream(
|
||||||
headers=None, filename=None, _range=None):
|
location,
|
||||||
|
status=200,
|
||||||
|
chunk_size=4096,
|
||||||
|
mime_type=None,
|
||||||
|
headers=None,
|
||||||
|
filename=None,
|
||||||
|
_range=None,
|
||||||
|
):
|
||||||
"""Return a streaming response object with file data.
|
"""Return a streaming response object with file data.
|
||||||
|
|
||||||
:param location: Location of file on system.
|
:param location: Location of file on system.
|
||||||
|
@ -283,11 +334,11 @@ async def file_stream(location, status=200, chunk_size=4096, mime_type=None,
|
||||||
headers = headers or {}
|
headers = headers or {}
|
||||||
if filename:
|
if filename:
|
||||||
headers.setdefault(
|
headers.setdefault(
|
||||||
'Content-Disposition',
|
"Content-Disposition", 'attachment; filename="{}"'.format(filename)
|
||||||
'attachment; filename="{}"'.format(filename))
|
)
|
||||||
filename = filename or path.split(location)[-1]
|
filename = filename or path.split(location)[-1]
|
||||||
|
|
||||||
_file = await open_async(location, mode='rb')
|
_file = await open_async(location, mode="rb")
|
||||||
|
|
||||||
async def _streaming_fn(response):
|
async def _streaming_fn(response):
|
||||||
nonlocal _file, chunk_size
|
nonlocal _file, chunk_size
|
||||||
|
@ -312,19 +363,27 @@ async def file_stream(location, status=200, chunk_size=4096, mime_type=None,
|
||||||
await _file.close()
|
await _file.close()
|
||||||
return # Returning from this fn closes the stream
|
return # Returning from this fn closes the stream
|
||||||
|
|
||||||
mime_type = mime_type or guess_type(filename)[0] or 'text/plain'
|
mime_type = mime_type or guess_type(filename)[0] or "text/plain"
|
||||||
if _range:
|
if _range:
|
||||||
headers['Content-Range'] = 'bytes %s-%s/%s' % (
|
headers["Content-Range"] = "bytes %s-%s/%s" % (
|
||||||
_range.start, _range.end, _range.total)
|
_range.start,
|
||||||
return StreamingHTTPResponse(streaming_fn=_streaming_fn,
|
_range.end,
|
||||||
|
_range.total,
|
||||||
|
)
|
||||||
|
return StreamingHTTPResponse(
|
||||||
|
streaming_fn=_streaming_fn,
|
||||||
status=status,
|
status=status,
|
||||||
headers=headers,
|
headers=headers,
|
||||||
content_type=mime_type)
|
content_type=mime_type,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
def stream(
|
def stream(
|
||||||
streaming_fn, status=200, headers=None,
|
streaming_fn,
|
||||||
content_type="text/plain; charset=utf-8"):
|
status=200,
|
||||||
|
headers=None,
|
||||||
|
content_type="text/plain; charset=utf-8",
|
||||||
|
):
|
||||||
"""Accepts an coroutine `streaming_fn` which can be used to
|
"""Accepts an coroutine `streaming_fn` which can be used to
|
||||||
write chunks to a streaming response. Returns a `StreamingHTTPResponse`.
|
write chunks to a streaming response. Returns a `StreamingHTTPResponse`.
|
||||||
|
|
||||||
|
@ -344,15 +403,13 @@ def stream(
|
||||||
:param headers: Custom Headers.
|
:param headers: Custom Headers.
|
||||||
"""
|
"""
|
||||||
return StreamingHTTPResponse(
|
return StreamingHTTPResponse(
|
||||||
streaming_fn,
|
streaming_fn, headers=headers, content_type=content_type, status=status
|
||||||
headers=headers,
|
|
||||||
content_type=content_type,
|
|
||||||
status=status
|
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
def redirect(to, headers=None, status=302,
|
def redirect(
|
||||||
content_type="text/html; charset=utf-8"):
|
to, headers=None, status=302, content_type="text/html; charset=utf-8"
|
||||||
|
):
|
||||||
"""Abort execution and cause a 302 redirect (by default).
|
"""Abort execution and cause a 302 redirect (by default).
|
||||||
|
|
||||||
:param to: path or fully qualified URL to redirect to
|
:param to: path or fully qualified URL to redirect to
|
||||||
|
@ -367,9 +424,8 @@ def redirect(to, headers=None, status=302,
|
||||||
safe_to = quote_plus(to, safe=":/#?&=@[]!$&'()*+,;")
|
safe_to = quote_plus(to, safe=":/#?&=@[]!$&'()*+,;")
|
||||||
|
|
||||||
# According to RFC 7231, a relative URI is now permitted.
|
# According to RFC 7231, a relative URI is now permitted.
|
||||||
headers['Location'] = safe_to
|
headers["Location"] = safe_to
|
||||||
|
|
||||||
return HTTPResponse(
|
return HTTPResponse(
|
||||||
status=status,
|
status=status, headers=headers, content_type=content_type
|
||||||
headers=headers,
|
)
|
||||||
content_type=content_type)
|
|
||||||
|
|
184
sanic/router.py
184
sanic/router.py
|
@ -9,25 +9,28 @@ from sanic.exceptions import NotFound, MethodNotSupported
|
||||||
from sanic.views import CompositionView
|
from sanic.views import CompositionView
|
||||||
|
|
||||||
Route = namedtuple(
|
Route = namedtuple(
|
||||||
'Route',
|
"Route", ["handler", "methods", "pattern", "parameters", "name", "uri"]
|
||||||
['handler', 'methods', 'pattern', 'parameters', 'name', 'uri'])
|
)
|
||||||
Parameter = namedtuple('Parameter', ['name', 'cast'])
|
Parameter = namedtuple("Parameter", ["name", "cast"])
|
||||||
|
|
||||||
REGEX_TYPES = {
|
REGEX_TYPES = {
|
||||||
'string': (str, r'[^/]+'),
|
"string": (str, r"[^/]+"),
|
||||||
'int': (int, r'\d+'),
|
"int": (int, r"\d+"),
|
||||||
'number': (float, r'[0-9\\.]+'),
|
"number": (float, r"[0-9\\.]+"),
|
||||||
'alpha': (str, r'[A-Za-z]+'),
|
"alpha": (str, r"[A-Za-z]+"),
|
||||||
'path': (str, r'[^/].*?'),
|
"path": (str, r"[^/].*?"),
|
||||||
'uuid': (uuid.UUID, r'[A-Fa-f0-9]{8}-[A-Fa-f0-9]{4}-'
|
"uuid": (
|
||||||
r'[A-Fa-f0-9]{4}-[A-Fa-f0-9]{4}-[A-Fa-f0-9]{12}')
|
uuid.UUID,
|
||||||
|
r"[A-Fa-f0-9]{8}-[A-Fa-f0-9]{4}-"
|
||||||
|
r"[A-Fa-f0-9]{4}-[A-Fa-f0-9]{4}-[A-Fa-f0-9]{12}",
|
||||||
|
),
|
||||||
}
|
}
|
||||||
|
|
||||||
ROUTER_CACHE_SIZE = 1024
|
ROUTER_CACHE_SIZE = 1024
|
||||||
|
|
||||||
|
|
||||||
def url_hash(url):
|
def url_hash(url):
|
||||||
return url.count('/')
|
return url.count("/")
|
||||||
|
|
||||||
|
|
||||||
class RouteExists(Exception):
|
class RouteExists(Exception):
|
||||||
|
@ -68,10 +71,11 @@ class Router:
|
||||||
also be passed in as the type. The argument given to the function will
|
also be passed in as the type. The argument given to the function will
|
||||||
always be a string, independent of the type.
|
always be a string, independent of the type.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
routes_static = None
|
routes_static = None
|
||||||
routes_dynamic = None
|
routes_dynamic = None
|
||||||
routes_always_check = None
|
routes_always_check = None
|
||||||
parameter_pattern = re.compile(r'<(.+?)>')
|
parameter_pattern = re.compile(r"<(.+?)>")
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
self.routes_all = {}
|
self.routes_all = {}
|
||||||
|
@ -98,9 +102,9 @@ class Router:
|
||||||
"""
|
"""
|
||||||
# We could receive NAME or NAME:PATTERN
|
# We could receive NAME or NAME:PATTERN
|
||||||
name = parameter_string
|
name = parameter_string
|
||||||
pattern = 'string'
|
pattern = "string"
|
||||||
if ':' in parameter_string:
|
if ":" in parameter_string:
|
||||||
name, pattern = parameter_string.split(':', 1)
|
name, pattern = parameter_string.split(":", 1)
|
||||||
if not name:
|
if not name:
|
||||||
raise ValueError(
|
raise ValueError(
|
||||||
"Invalid parameter syntax: {}".format(parameter_string)
|
"Invalid parameter syntax: {}".format(parameter_string)
|
||||||
|
@ -112,8 +116,16 @@ class Router:
|
||||||
|
|
||||||
return name, _type, pattern
|
return name, _type, pattern
|
||||||
|
|
||||||
def add(self, uri, methods, handler, host=None, strict_slashes=False,
|
def add(
|
||||||
version=None, name=None):
|
self,
|
||||||
|
uri,
|
||||||
|
methods,
|
||||||
|
handler,
|
||||||
|
host=None,
|
||||||
|
strict_slashes=False,
|
||||||
|
version=None,
|
||||||
|
name=None,
|
||||||
|
):
|
||||||
"""Add a handler to the route list
|
"""Add a handler to the route list
|
||||||
|
|
||||||
:param uri: path to match
|
:param uri: path to match
|
||||||
|
@ -127,8 +139,8 @@ class Router:
|
||||||
:return: Nothing
|
:return: Nothing
|
||||||
"""
|
"""
|
||||||
if version is not None:
|
if version is not None:
|
||||||
version = re.escape(str(version).strip('/').lstrip('v'))
|
version = re.escape(str(version).strip("/").lstrip("v"))
|
||||||
uri = "/".join(["/v{}".format(version), uri.lstrip('/')])
|
uri = "/".join(["/v{}".format(version), uri.lstrip("/")])
|
||||||
# add regular version
|
# add regular version
|
||||||
self._add(uri, methods, handler, host, name)
|
self._add(uri, methods, handler, host, name)
|
||||||
|
|
||||||
|
@ -143,28 +155,26 @@ class Router:
|
||||||
return
|
return
|
||||||
|
|
||||||
# Add versions with and without trailing /
|
# Add versions with and without trailing /
|
||||||
slashed_methods = self.routes_all.get(uri + '/', frozenset({}))
|
slashed_methods = self.routes_all.get(uri + "/", frozenset({}))
|
||||||
unslashed_methods = self.routes_all.get(uri[:-1], frozenset({}))
|
unslashed_methods = self.routes_all.get(uri[:-1], frozenset({}))
|
||||||
if isinstance(methods, Iterable):
|
if isinstance(methods, Iterable):
|
||||||
_slash_is_missing = all(method in slashed_methods for
|
_slash_is_missing = all(
|
||||||
method in methods)
|
method in slashed_methods for method in methods
|
||||||
_without_slash_is_missing = all(method in unslashed_methods for
|
)
|
||||||
method in methods)
|
_without_slash_is_missing = all(
|
||||||
|
method in unslashed_methods for method in methods
|
||||||
|
)
|
||||||
else:
|
else:
|
||||||
_slash_is_missing = methods in slashed_methods
|
_slash_is_missing = methods in slashed_methods
|
||||||
_without_slash_is_missing = methods in unslashed_methods
|
_without_slash_is_missing = methods in unslashed_methods
|
||||||
|
|
||||||
slash_is_missing = (
|
slash_is_missing = not uri[-1] == "/" and not _slash_is_missing
|
||||||
not uri[-1] == '/' and not _slash_is_missing
|
|
||||||
)
|
|
||||||
without_slash_is_missing = (
|
without_slash_is_missing = (
|
||||||
uri[-1] == '/' and not
|
uri[-1] == "/" and not _without_slash_is_missing and not uri == "/"
|
||||||
_without_slash_is_missing and not
|
|
||||||
uri == '/'
|
|
||||||
)
|
)
|
||||||
# add version with trailing slash
|
# add version with trailing slash
|
||||||
if slash_is_missing:
|
if slash_is_missing:
|
||||||
self._add(uri + '/', methods, handler, host, name)
|
self._add(uri + "/", methods, handler, host, name)
|
||||||
# add version without trailing slash
|
# add version without trailing slash
|
||||||
elif without_slash_is_missing:
|
elif without_slash_is_missing:
|
||||||
self._add(uri[:-1], methods, handler, host, name)
|
self._add(uri[:-1], methods, handler, host, name)
|
||||||
|
@ -187,8 +197,10 @@ class Router:
|
||||||
|
|
||||||
else:
|
else:
|
||||||
if not isinstance(host, Iterable):
|
if not isinstance(host, Iterable):
|
||||||
raise ValueError("Expected either string or Iterable of "
|
raise ValueError(
|
||||||
"host strings, not {!r}".format(host))
|
"Expected either string or Iterable of "
|
||||||
|
"host strings, not {!r}".format(host)
|
||||||
|
)
|
||||||
|
|
||||||
for host_ in host:
|
for host_ in host:
|
||||||
self.add(uri, methods, handler, host_, name)
|
self.add(uri, methods, handler, host_, name)
|
||||||
|
@ -209,37 +221,38 @@ class Router:
|
||||||
if name in parameter_names:
|
if name in parameter_names:
|
||||||
raise ParameterNameConflicts(
|
raise ParameterNameConflicts(
|
||||||
"Multiple parameter named <{name}> "
|
"Multiple parameter named <{name}> "
|
||||||
"in route uri {uri}".format(name=name, uri=uri))
|
"in route uri {uri}".format(name=name, uri=uri)
|
||||||
|
)
|
||||||
parameter_names.add(name)
|
parameter_names.add(name)
|
||||||
|
|
||||||
parameter = Parameter(
|
parameter = Parameter(name=name, cast=_type)
|
||||||
name=name, cast=_type)
|
|
||||||
parameters.append(parameter)
|
parameters.append(parameter)
|
||||||
|
|
||||||
# Mark the whole route as unhashable if it has the hash key in it
|
# Mark the whole route as unhashable if it has the hash key in it
|
||||||
if re.search(r'(^|[^^]){1}/', pattern):
|
if re.search(r"(^|[^^]){1}/", pattern):
|
||||||
properties['unhashable'] = True
|
properties["unhashable"] = True
|
||||||
# Mark the route as unhashable if it matches the hash key
|
# Mark the route as unhashable if it matches the hash key
|
||||||
elif re.search(r'/', pattern):
|
elif re.search(r"/", pattern):
|
||||||
properties['unhashable'] = True
|
properties["unhashable"] = True
|
||||||
|
|
||||||
return '({})'.format(pattern)
|
return "({})".format(pattern)
|
||||||
|
|
||||||
pattern_string = re.sub(self.parameter_pattern, add_parameter, uri)
|
pattern_string = re.sub(self.parameter_pattern, add_parameter, uri)
|
||||||
pattern = re.compile(r'^{}$'.format(pattern_string))
|
pattern = re.compile(r"^{}$".format(pattern_string))
|
||||||
|
|
||||||
def merge_route(route, methods, handler):
|
def merge_route(route, methods, handler):
|
||||||
# merge to the existing route when possible.
|
# merge to the existing route when possible.
|
||||||
if not route.methods or not methods:
|
if not route.methods or not methods:
|
||||||
# method-unspecified routes are not mergeable.
|
# method-unspecified routes are not mergeable.
|
||||||
raise RouteExists(
|
raise RouteExists("Route already registered: {}".format(uri))
|
||||||
"Route already registered: {}".format(uri))
|
|
||||||
elif route.methods.intersection(methods):
|
elif route.methods.intersection(methods):
|
||||||
# already existing method is not overloadable.
|
# already existing method is not overloadable.
|
||||||
duplicated = methods.intersection(route.methods)
|
duplicated = methods.intersection(route.methods)
|
||||||
raise RouteExists(
|
raise RouteExists(
|
||||||
"Route already registered: {} [{}]".format(
|
"Route already registered: {} [{}]".format(
|
||||||
uri, ','.join(list(duplicated))))
|
uri, ",".join(list(duplicated))
|
||||||
|
)
|
||||||
|
)
|
||||||
if isinstance(route.handler, CompositionView):
|
if isinstance(route.handler, CompositionView):
|
||||||
view = route.handler
|
view = route.handler
|
||||||
else:
|
else:
|
||||||
|
@ -247,19 +260,22 @@ class Router:
|
||||||
view.add(route.methods, route.handler)
|
view.add(route.methods, route.handler)
|
||||||
view.add(methods, handler)
|
view.add(methods, handler)
|
||||||
route = route._replace(
|
route = route._replace(
|
||||||
handler=view, methods=methods.union(route.methods))
|
handler=view, methods=methods.union(route.methods)
|
||||||
|
)
|
||||||
return route
|
return route
|
||||||
|
|
||||||
if parameters:
|
if parameters:
|
||||||
# TODO: This is too complex, we need to reduce the complexity
|
# TODO: This is too complex, we need to reduce the complexity
|
||||||
if properties['unhashable']:
|
if properties["unhashable"]:
|
||||||
routes_to_check = self.routes_always_check
|
routes_to_check = self.routes_always_check
|
||||||
ndx, route = self.check_dynamic_route_exists(
|
ndx, route = self.check_dynamic_route_exists(
|
||||||
pattern, routes_to_check, parameters)
|
pattern, routes_to_check, parameters
|
||||||
|
)
|
||||||
else:
|
else:
|
||||||
routes_to_check = self.routes_dynamic[url_hash(uri)]
|
routes_to_check = self.routes_dynamic[url_hash(uri)]
|
||||||
ndx, route = self.check_dynamic_route_exists(
|
ndx, route = self.check_dynamic_route_exists(
|
||||||
pattern, routes_to_check, parameters)
|
pattern, routes_to_check, parameters
|
||||||
|
)
|
||||||
if ndx != -1:
|
if ndx != -1:
|
||||||
# Pop the ndx of the route, no dups of the same route
|
# Pop the ndx of the route, no dups of the same route
|
||||||
routes_to_check.pop(ndx)
|
routes_to_check.pop(ndx)
|
||||||
|
@ -270,35 +286,41 @@ class Router:
|
||||||
# if available
|
# if available
|
||||||
# special prefix for static files
|
# special prefix for static files
|
||||||
is_static = False
|
is_static = False
|
||||||
if name and name.startswith('_static_'):
|
if name and name.startswith("_static_"):
|
||||||
is_static = True
|
is_static = True
|
||||||
name = name.split('_static_', 1)[-1]
|
name = name.split("_static_", 1)[-1]
|
||||||
|
|
||||||
if hasattr(handler, '__blueprintname__'):
|
if hasattr(handler, "__blueprintname__"):
|
||||||
handler_name = '{}.{}'.format(
|
handler_name = "{}.{}".format(
|
||||||
handler.__blueprintname__, name or handler.__name__)
|
handler.__blueprintname__, name or handler.__name__
|
||||||
|
)
|
||||||
else:
|
else:
|
||||||
handler_name = name or getattr(handler, '__name__', None)
|
handler_name = name or getattr(handler, "__name__", None)
|
||||||
|
|
||||||
if route:
|
if route:
|
||||||
route = merge_route(route, methods, handler)
|
route = merge_route(route, methods, handler)
|
||||||
else:
|
else:
|
||||||
route = Route(
|
route = Route(
|
||||||
handler=handler, methods=methods, pattern=pattern,
|
handler=handler,
|
||||||
parameters=parameters, name=handler_name, uri=uri)
|
methods=methods,
|
||||||
|
pattern=pattern,
|
||||||
|
parameters=parameters,
|
||||||
|
name=handler_name,
|
||||||
|
uri=uri,
|
||||||
|
)
|
||||||
|
|
||||||
self.routes_all[uri] = route
|
self.routes_all[uri] = route
|
||||||
if is_static:
|
if is_static:
|
||||||
pair = self.routes_static_files.get(handler_name)
|
pair = self.routes_static_files.get(handler_name)
|
||||||
if not (pair and (pair[0] + '/' == uri or uri + '/' == pair[0])):
|
if not (pair and (pair[0] + "/" == uri or uri + "/" == pair[0])):
|
||||||
self.routes_static_files[handler_name] = (uri, route)
|
self.routes_static_files[handler_name] = (uri, route)
|
||||||
|
|
||||||
else:
|
else:
|
||||||
pair = self.routes_names.get(handler_name)
|
pair = self.routes_names.get(handler_name)
|
||||||
if not (pair and (pair[0] + '/' == uri or uri + '/' == pair[0])):
|
if not (pair and (pair[0] + "/" == uri or uri + "/" == pair[0])):
|
||||||
self.routes_names[handler_name] = (uri, route)
|
self.routes_names[handler_name] = (uri, route)
|
||||||
|
|
||||||
if properties['unhashable']:
|
if properties["unhashable"]:
|
||||||
self.routes_always_check.append(route)
|
self.routes_always_check.append(route)
|
||||||
elif parameters:
|
elif parameters:
|
||||||
self.routes_dynamic[url_hash(uri)].append(route)
|
self.routes_dynamic[url_hash(uri)].append(route)
|
||||||
|
@ -333,8 +355,10 @@ class Router:
|
||||||
|
|
||||||
if route in self.routes_always_check:
|
if route in self.routes_always_check:
|
||||||
self.routes_always_check.remove(route)
|
self.routes_always_check.remove(route)
|
||||||
elif url_hash(uri) in self.routes_dynamic \
|
elif (
|
||||||
and route in self.routes_dynamic[url_hash(uri)]:
|
url_hash(uri) in self.routes_dynamic
|
||||||
|
and route in self.routes_dynamic[url_hash(uri)]
|
||||||
|
):
|
||||||
self.routes_dynamic[url_hash(uri)].remove(route)
|
self.routes_dynamic[url_hash(uri)].remove(route)
|
||||||
else:
|
else:
|
||||||
self.routes_static.pop(uri)
|
self.routes_static.pop(uri)
|
||||||
|
@ -353,7 +377,7 @@ class Router:
|
||||||
if not view_name:
|
if not view_name:
|
||||||
return (None, None)
|
return (None, None)
|
||||||
|
|
||||||
if view_name == 'static' or view_name.endswith('.static'):
|
if view_name == "static" or view_name.endswith(".static"):
|
||||||
return self.routes_static_files.get(name, (None, None))
|
return self.routes_static_files.get(name, (None, None))
|
||||||
|
|
||||||
return self.routes_names.get(view_name, (None, None))
|
return self.routes_names.get(view_name, (None, None))
|
||||||
|
@ -367,14 +391,15 @@ class Router:
|
||||||
"""
|
"""
|
||||||
# No virtual hosts specified; default behavior
|
# No virtual hosts specified; default behavior
|
||||||
if not self.hosts:
|
if not self.hosts:
|
||||||
return self._get(request.path, request.method, '')
|
return self._get(request.path, request.method, "")
|
||||||
# virtual hosts specified; try to match route to the host header
|
# virtual hosts specified; try to match route to the host header
|
||||||
try:
|
try:
|
||||||
return self._get(request.path, request.method,
|
return self._get(
|
||||||
request.headers.get("Host", ''))
|
request.path, request.method, request.headers.get("Host", "")
|
||||||
|
)
|
||||||
# try default hosts
|
# try default hosts
|
||||||
except NotFound:
|
except NotFound:
|
||||||
return self._get(request.path, request.method, '')
|
return self._get(request.path, request.method, "")
|
||||||
|
|
||||||
def get_supported_methods(self, url):
|
def get_supported_methods(self, url):
|
||||||
"""Get a list of supported methods for a url and optional host.
|
"""Get a list of supported methods for a url and optional host.
|
||||||
|
@ -384,7 +409,7 @@ class Router:
|
||||||
"""
|
"""
|
||||||
route = self.routes_all.get(url)
|
route = self.routes_all.get(url)
|
||||||
# if methods are None then this logic will prevent an error
|
# if methods are None then this logic will prevent an error
|
||||||
return getattr(route, 'methods', None) or frozenset()
|
return getattr(route, "methods", None) or frozenset()
|
||||||
|
|
||||||
@lru_cache(maxsize=ROUTER_CACHE_SIZE)
|
@lru_cache(maxsize=ROUTER_CACHE_SIZE)
|
||||||
def _get(self, url, method, host):
|
def _get(self, url, method, host):
|
||||||
|
@ -399,9 +424,10 @@ class Router:
|
||||||
# Check against known static routes
|
# Check against known static routes
|
||||||
route = self.routes_static.get(url)
|
route = self.routes_static.get(url)
|
||||||
method_not_supported = MethodNotSupported(
|
method_not_supported = MethodNotSupported(
|
||||||
'Method {} not allowed for URL {}'.format(method, url),
|
"Method {} not allowed for URL {}".format(method, url),
|
||||||
method=method,
|
method=method,
|
||||||
allowed_methods=self.get_supported_methods(url))
|
allowed_methods=self.get_supported_methods(url),
|
||||||
|
)
|
||||||
if route:
|
if route:
|
||||||
if route.methods and method not in route.methods:
|
if route.methods and method not in route.methods:
|
||||||
raise method_not_supported
|
raise method_not_supported
|
||||||
|
@ -427,13 +453,14 @@ class Router:
|
||||||
# Route was found but the methods didn't match
|
# Route was found but the methods didn't match
|
||||||
if route_found:
|
if route_found:
|
||||||
raise method_not_supported
|
raise method_not_supported
|
||||||
raise NotFound('Requested URL {} not found'.format(url))
|
raise NotFound("Requested URL {} not found".format(url))
|
||||||
|
|
||||||
kwargs = {p.name: p.cast(value)
|
kwargs = {
|
||||||
for value, p
|
p.name: p.cast(value)
|
||||||
in zip(match.groups(1), route.parameters)}
|
for value, p in zip(match.groups(1), route.parameters)
|
||||||
|
}
|
||||||
route_handler = route.handler
|
route_handler = route.handler
|
||||||
if hasattr(route_handler, 'handlers'):
|
if hasattr(route_handler, "handlers"):
|
||||||
route_handler = route_handler.handlers[method]
|
route_handler = route_handler.handlers[method]
|
||||||
return route_handler, [], kwargs, route.uri
|
return route_handler, [], kwargs, route.uri
|
||||||
|
|
||||||
|
@ -446,7 +473,8 @@ class Router:
|
||||||
handler = self.get(request)[0]
|
handler = self.get(request)[0]
|
||||||
except (NotFound, MethodNotSupported):
|
except (NotFound, MethodNotSupported):
|
||||||
return False
|
return False
|
||||||
if (hasattr(handler, 'view_class') and
|
if hasattr(handler, "view_class") and hasattr(
|
||||||
hasattr(handler.view_class, request.method.lower())):
|
handler.view_class, request.method.lower()
|
||||||
|
):
|
||||||
handler = getattr(handler.view_class, request.method.lower())
|
handler = getattr(handler.view_class, request.method.lower())
|
||||||
return hasattr(handler, 'is_stream')
|
return hasattr(handler, "is_stream")
|
||||||
|
|
353
sanic/server.py
353
sanic/server.py
|
@ -4,16 +4,8 @@ import traceback
|
||||||
from functools import partial
|
from functools import partial
|
||||||
from inspect import isawaitable
|
from inspect import isawaitable
|
||||||
from multiprocessing import Process
|
from multiprocessing import Process
|
||||||
from signal import (
|
from signal import SIGTERM, SIGINT, SIG_IGN, signal as signal_func, Signals
|
||||||
SIGTERM, SIGINT, SIG_IGN,
|
from socket import socket, SOL_SOCKET, SO_REUSEADDR
|
||||||
signal as signal_func,
|
|
||||||
Signals
|
|
||||||
)
|
|
||||||
from socket import (
|
|
||||||
socket,
|
|
||||||
SOL_SOCKET,
|
|
||||||
SO_REUSEADDR,
|
|
||||||
)
|
|
||||||
from time import time
|
from time import time
|
||||||
|
|
||||||
from httptools import HttpRequestParser
|
from httptools import HttpRequestParser
|
||||||
|
@ -22,6 +14,7 @@ from multidict import CIMultiDict
|
||||||
|
|
||||||
try:
|
try:
|
||||||
import uvloop
|
import uvloop
|
||||||
|
|
||||||
asyncio.set_event_loop_policy(uvloop.EventLoopPolicy())
|
asyncio.set_event_loop_policy(uvloop.EventLoopPolicy())
|
||||||
except ImportError:
|
except ImportError:
|
||||||
pass
|
pass
|
||||||
|
@ -30,8 +23,12 @@ from sanic.log import logger, access_logger
|
||||||
from sanic.response import HTTPResponse
|
from sanic.response import HTTPResponse
|
||||||
from sanic.request import Request
|
from sanic.request import Request
|
||||||
from sanic.exceptions import (
|
from sanic.exceptions import (
|
||||||
RequestTimeout, PayloadTooLarge, InvalidUsage, ServerError,
|
RequestTimeout,
|
||||||
ServiceUnavailable)
|
PayloadTooLarge,
|
||||||
|
InvalidUsage,
|
||||||
|
ServerError,
|
||||||
|
ServiceUnavailable,
|
||||||
|
)
|
||||||
|
|
||||||
current_time = None
|
current_time = None
|
||||||
|
|
||||||
|
@ -43,27 +40,58 @@ class Signal:
|
||||||
class HttpProtocol(asyncio.Protocol):
|
class HttpProtocol(asyncio.Protocol):
|
||||||
__slots__ = (
|
__slots__ = (
|
||||||
# event loop, connection
|
# event loop, connection
|
||||||
'loop', 'transport', 'connections', 'signal',
|
"loop",
|
||||||
|
"transport",
|
||||||
|
"connections",
|
||||||
|
"signal",
|
||||||
# request params
|
# request params
|
||||||
'parser', 'request', 'url', 'headers',
|
"parser",
|
||||||
|
"request",
|
||||||
|
"url",
|
||||||
|
"headers",
|
||||||
# request config
|
# request config
|
||||||
'request_handler', 'request_timeout', 'response_timeout',
|
"request_handler",
|
||||||
'keep_alive_timeout', 'request_max_size', 'request_class',
|
"request_timeout",
|
||||||
'is_request_stream', 'router',
|
"response_timeout",
|
||||||
|
"keep_alive_timeout",
|
||||||
|
"request_max_size",
|
||||||
|
"request_class",
|
||||||
|
"is_request_stream",
|
||||||
|
"router",
|
||||||
# enable or disable access log purpose
|
# enable or disable access log purpose
|
||||||
'access_log',
|
"access_log",
|
||||||
# connection management
|
# connection management
|
||||||
'_total_request_size', '_request_timeout_handler',
|
"_total_request_size",
|
||||||
'_response_timeout_handler', '_keep_alive_timeout_handler',
|
"_request_timeout_handler",
|
||||||
'_last_request_time', '_last_response_time', '_is_stream_handler',
|
"_response_timeout_handler",
|
||||||
'_not_paused')
|
"_keep_alive_timeout_handler",
|
||||||
|
"_last_request_time",
|
||||||
|
"_last_response_time",
|
||||||
|
"_is_stream_handler",
|
||||||
|
"_not_paused",
|
||||||
|
)
|
||||||
|
|
||||||
def __init__(self, *, loop, request_handler, error_handler,
|
def __init__(
|
||||||
signal=Signal(), connections=set(), request_timeout=60,
|
self,
|
||||||
response_timeout=60, keep_alive_timeout=5,
|
*,
|
||||||
request_max_size=None, request_class=None, access_log=True,
|
loop,
|
||||||
keep_alive=True, is_request_stream=False, router=None,
|
request_handler,
|
||||||
state=None, debug=False, **kwargs):
|
error_handler,
|
||||||
|
signal=Signal(),
|
||||||
|
connections=set(),
|
||||||
|
request_timeout=60,
|
||||||
|
response_timeout=60,
|
||||||
|
keep_alive_timeout=5,
|
||||||
|
request_max_size=None,
|
||||||
|
request_class=None,
|
||||||
|
access_log=True,
|
||||||
|
keep_alive=True,
|
||||||
|
is_request_stream=False,
|
||||||
|
router=None,
|
||||||
|
state=None,
|
||||||
|
debug=False,
|
||||||
|
**kwargs
|
||||||
|
):
|
||||||
self.loop = loop
|
self.loop = loop
|
||||||
self.transport = None
|
self.transport = None
|
||||||
self.request = None
|
self.request = None
|
||||||
|
@ -93,19 +121,20 @@ class HttpProtocol(asyncio.Protocol):
|
||||||
self._request_handler_task = None
|
self._request_handler_task = None
|
||||||
self._request_stream_task = None
|
self._request_stream_task = None
|
||||||
self._keep_alive = keep_alive
|
self._keep_alive = keep_alive
|
||||||
self._header_fragment = b''
|
self._header_fragment = b""
|
||||||
self.state = state if state else {}
|
self.state = state if state else {}
|
||||||
if 'requests_count' not in self.state:
|
if "requests_count" not in self.state:
|
||||||
self.state['requests_count'] = 0
|
self.state["requests_count"] = 0
|
||||||
self._debug = debug
|
self._debug = debug
|
||||||
self._not_paused.set()
|
self._not_paused.set()
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def keep_alive(self):
|
def keep_alive(self):
|
||||||
return (
|
return (
|
||||||
self._keep_alive and
|
self._keep_alive
|
||||||
not self.signal.stopped and
|
and not self.signal.stopped
|
||||||
self.parser.should_keep_alive())
|
and self.parser.should_keep_alive()
|
||||||
|
)
|
||||||
|
|
||||||
# -------------------------------------------- #
|
# -------------------------------------------- #
|
||||||
# Connection
|
# Connection
|
||||||
|
@ -114,7 +143,8 @@ class HttpProtocol(asyncio.Protocol):
|
||||||
def connection_made(self, transport):
|
def connection_made(self, transport):
|
||||||
self.connections.add(self)
|
self.connections.add(self)
|
||||||
self._request_timeout_handler = self.loop.call_later(
|
self._request_timeout_handler = self.loop.call_later(
|
||||||
self.request_timeout, self.request_timeout_callback)
|
self.request_timeout, self.request_timeout_callback
|
||||||
|
)
|
||||||
self.transport = transport
|
self.transport = transport
|
||||||
self._last_request_time = current_time
|
self._last_request_time = current_time
|
||||||
|
|
||||||
|
@ -145,16 +175,15 @@ class HttpProtocol(asyncio.Protocol):
|
||||||
time_elapsed = current_time - self._last_request_time
|
time_elapsed = current_time - self._last_request_time
|
||||||
if time_elapsed < self.request_timeout:
|
if time_elapsed < self.request_timeout:
|
||||||
time_left = self.request_timeout - time_elapsed
|
time_left = self.request_timeout - time_elapsed
|
||||||
self._request_timeout_handler = (
|
self._request_timeout_handler = self.loop.call_later(
|
||||||
self.loop.call_later(time_left,
|
time_left, self.request_timeout_callback
|
||||||
self.request_timeout_callback)
|
|
||||||
)
|
)
|
||||||
else:
|
else:
|
||||||
if self._request_stream_task:
|
if self._request_stream_task:
|
||||||
self._request_stream_task.cancel()
|
self._request_stream_task.cancel()
|
||||||
if self._request_handler_task:
|
if self._request_handler_task:
|
||||||
self._request_handler_task.cancel()
|
self._request_handler_task.cancel()
|
||||||
self.write_error(RequestTimeout('Request Timeout'))
|
self.write_error(RequestTimeout("Request Timeout"))
|
||||||
|
|
||||||
def response_timeout_callback(self):
|
def response_timeout_callback(self):
|
||||||
# Check if elapsed time since response was initiated exceeds our
|
# Check if elapsed time since response was initiated exceeds our
|
||||||
|
@ -162,16 +191,15 @@ class HttpProtocol(asyncio.Protocol):
|
||||||
time_elapsed = current_time - self._last_request_time
|
time_elapsed = current_time - self._last_request_time
|
||||||
if time_elapsed < self.response_timeout:
|
if time_elapsed < self.response_timeout:
|
||||||
time_left = self.response_timeout - time_elapsed
|
time_left = self.response_timeout - time_elapsed
|
||||||
self._response_timeout_handler = (
|
self._response_timeout_handler = self.loop.call_later(
|
||||||
self.loop.call_later(time_left,
|
time_left, self.response_timeout_callback
|
||||||
self.response_timeout_callback)
|
|
||||||
)
|
)
|
||||||
else:
|
else:
|
||||||
if self._request_stream_task:
|
if self._request_stream_task:
|
||||||
self._request_stream_task.cancel()
|
self._request_stream_task.cancel()
|
||||||
if self._request_handler_task:
|
if self._request_handler_task:
|
||||||
self._request_handler_task.cancel()
|
self._request_handler_task.cancel()
|
||||||
self.write_error(ServiceUnavailable('Response Timeout'))
|
self.write_error(ServiceUnavailable("Response Timeout"))
|
||||||
|
|
||||||
def keep_alive_timeout_callback(self):
|
def keep_alive_timeout_callback(self):
|
||||||
# Check if elapsed time since last response exceeds our configured
|
# Check if elapsed time since last response exceeds our configured
|
||||||
|
@ -179,12 +207,11 @@ class HttpProtocol(asyncio.Protocol):
|
||||||
time_elapsed = current_time - self._last_response_time
|
time_elapsed = current_time - self._last_response_time
|
||||||
if time_elapsed < self.keep_alive_timeout:
|
if time_elapsed < self.keep_alive_timeout:
|
||||||
time_left = self.keep_alive_timeout - time_elapsed
|
time_left = self.keep_alive_timeout - time_elapsed
|
||||||
self._keep_alive_timeout_handler = (
|
self._keep_alive_timeout_handler = self.loop.call_later(
|
||||||
self.loop.call_later(time_left,
|
time_left, self.keep_alive_timeout_callback
|
||||||
self.keep_alive_timeout_callback)
|
|
||||||
)
|
)
|
||||||
else:
|
else:
|
||||||
logger.debug('KeepAlive Timeout. Closing connection.')
|
logger.debug("KeepAlive Timeout. Closing connection.")
|
||||||
self.transport.close()
|
self.transport.close()
|
||||||
self.transport = None
|
self.transport = None
|
||||||
|
|
||||||
|
@ -197,7 +224,7 @@ class HttpProtocol(asyncio.Protocol):
|
||||||
# memory limits
|
# memory limits
|
||||||
self._total_request_size += len(data)
|
self._total_request_size += len(data)
|
||||||
if self._total_request_size > self.request_max_size:
|
if self._total_request_size > self.request_max_size:
|
||||||
self.write_error(PayloadTooLarge('Payload Too Large'))
|
self.write_error(PayloadTooLarge("Payload Too Large"))
|
||||||
|
|
||||||
# Create parser if this is the first time we're receiving data
|
# Create parser if this is the first time we're receiving data
|
||||||
if self.parser is None:
|
if self.parser is None:
|
||||||
|
@ -206,15 +233,15 @@ class HttpProtocol(asyncio.Protocol):
|
||||||
self.parser = HttpRequestParser(self)
|
self.parser = HttpRequestParser(self)
|
||||||
|
|
||||||
# requests count
|
# requests count
|
||||||
self.state['requests_count'] = self.state['requests_count'] + 1
|
self.state["requests_count"] = self.state["requests_count"] + 1
|
||||||
|
|
||||||
# Parse request chunk or close connection
|
# Parse request chunk or close connection
|
||||||
try:
|
try:
|
||||||
self.parser.feed_data(data)
|
self.parser.feed_data(data)
|
||||||
except HttpParserError:
|
except HttpParserError:
|
||||||
message = 'Bad Request'
|
message = "Bad Request"
|
||||||
if self._debug:
|
if self._debug:
|
||||||
message += '\n' + traceback.format_exc()
|
message += "\n" + traceback.format_exc()
|
||||||
self.write_error(InvalidUsage(message))
|
self.write_error(InvalidUsage(message))
|
||||||
|
|
||||||
def on_url(self, url):
|
def on_url(self, url):
|
||||||
|
@ -227,17 +254,20 @@ class HttpProtocol(asyncio.Protocol):
|
||||||
self._header_fragment += name
|
self._header_fragment += name
|
||||||
|
|
||||||
if value is not None:
|
if value is not None:
|
||||||
if self._header_fragment == b'Content-Length' \
|
if (
|
||||||
and int(value) > self.request_max_size:
|
self._header_fragment == b"Content-Length"
|
||||||
self.write_error(PayloadTooLarge('Payload Too Large'))
|
and int(value) > self.request_max_size
|
||||||
|
):
|
||||||
|
self.write_error(PayloadTooLarge("Payload Too Large"))
|
||||||
try:
|
try:
|
||||||
value = value.decode()
|
value = value.decode()
|
||||||
except UnicodeDecodeError:
|
except UnicodeDecodeError:
|
||||||
value = value.decode('latin_1')
|
value = value.decode("latin_1")
|
||||||
self.headers.append(
|
self.headers.append(
|
||||||
(self._header_fragment.decode().casefold(), value))
|
(self._header_fragment.decode().casefold(), value)
|
||||||
|
)
|
||||||
|
|
||||||
self._header_fragment = b''
|
self._header_fragment = b""
|
||||||
|
|
||||||
def on_headers_complete(self):
|
def on_headers_complete(self):
|
||||||
self.request = self.request_class(
|
self.request = self.request_class(
|
||||||
|
@ -245,7 +275,7 @@ class HttpProtocol(asyncio.Protocol):
|
||||||
headers=CIMultiDict(self.headers),
|
headers=CIMultiDict(self.headers),
|
||||||
version=self.parser.get_http_version(),
|
version=self.parser.get_http_version(),
|
||||||
method=self.parser.get_method().decode(),
|
method=self.parser.get_method().decode(),
|
||||||
transport=self.transport
|
transport=self.transport,
|
||||||
)
|
)
|
||||||
# Remove any existing KeepAlive handler here,
|
# Remove any existing KeepAlive handler here,
|
||||||
# It will be recreated if required on the new request.
|
# It will be recreated if required on the new request.
|
||||||
|
@ -254,7 +284,8 @@ class HttpProtocol(asyncio.Protocol):
|
||||||
self._keep_alive_timeout_handler = None
|
self._keep_alive_timeout_handler = None
|
||||||
if self.is_request_stream:
|
if self.is_request_stream:
|
||||||
self._is_stream_handler = self.router.is_stream_handler(
|
self._is_stream_handler = self.router.is_stream_handler(
|
||||||
self.request)
|
self.request
|
||||||
|
)
|
||||||
if self._is_stream_handler:
|
if self._is_stream_handler:
|
||||||
self.request.stream = asyncio.Queue()
|
self.request.stream = asyncio.Queue()
|
||||||
self.execute_request_handler()
|
self.execute_request_handler()
|
||||||
|
@ -262,7 +293,8 @@ class HttpProtocol(asyncio.Protocol):
|
||||||
def on_body(self, body):
|
def on_body(self, body):
|
||||||
if self.is_request_stream and self._is_stream_handler:
|
if self.is_request_stream and self._is_stream_handler:
|
||||||
self._request_stream_task = self.loop.create_task(
|
self._request_stream_task = self.loop.create_task(
|
||||||
self.request.stream.put(body))
|
self.request.stream.put(body)
|
||||||
|
)
|
||||||
return
|
return
|
||||||
self.request.body.append(body)
|
self.request.body.append(body)
|
||||||
|
|
||||||
|
@ -274,47 +306,49 @@ class HttpProtocol(asyncio.Protocol):
|
||||||
self._request_timeout_handler = None
|
self._request_timeout_handler = None
|
||||||
if self.is_request_stream and self._is_stream_handler:
|
if self.is_request_stream and self._is_stream_handler:
|
||||||
self._request_stream_task = self.loop.create_task(
|
self._request_stream_task = self.loop.create_task(
|
||||||
self.request.stream.put(None))
|
self.request.stream.put(None)
|
||||||
|
)
|
||||||
return
|
return
|
||||||
self.request.body = b''.join(self.request.body)
|
self.request.body = b"".join(self.request.body)
|
||||||
self.execute_request_handler()
|
self.execute_request_handler()
|
||||||
|
|
||||||
def execute_request_handler(self):
|
def execute_request_handler(self):
|
||||||
self._response_timeout_handler = self.loop.call_later(
|
self._response_timeout_handler = self.loop.call_later(
|
||||||
self.response_timeout, self.response_timeout_callback)
|
self.response_timeout, self.response_timeout_callback
|
||||||
|
)
|
||||||
self._last_request_time = current_time
|
self._last_request_time = current_time
|
||||||
self._request_handler_task = self.loop.create_task(
|
self._request_handler_task = self.loop.create_task(
|
||||||
self.request_handler(
|
self.request_handler(
|
||||||
self.request,
|
self.request, self.write_response, self.stream_response
|
||||||
self.write_response,
|
)
|
||||||
self.stream_response))
|
)
|
||||||
|
|
||||||
# -------------------------------------------- #
|
# -------------------------------------------- #
|
||||||
# Responding
|
# Responding
|
||||||
# -------------------------------------------- #
|
# -------------------------------------------- #
|
||||||
def log_response(self, response):
|
def log_response(self, response):
|
||||||
if self.access_log:
|
if self.access_log:
|
||||||
extra = {
|
extra = {"status": getattr(response, "status", 0)}
|
||||||
'status': getattr(response, 'status', 0),
|
|
||||||
}
|
|
||||||
|
|
||||||
if isinstance(response, HTTPResponse):
|
if isinstance(response, HTTPResponse):
|
||||||
extra['byte'] = len(response.body)
|
extra["byte"] = len(response.body)
|
||||||
else:
|
else:
|
||||||
extra['byte'] = -1
|
extra["byte"] = -1
|
||||||
|
|
||||||
extra['host'] = 'UNKNOWN'
|
extra["host"] = "UNKNOWN"
|
||||||
if self.request is not None:
|
if self.request is not None:
|
||||||
if self.request.ip:
|
if self.request.ip:
|
||||||
extra['host'] = '{0}:{1}'.format(self.request.ip,
|
extra["host"] = "{0}:{1}".format(
|
||||||
self.request.port)
|
self.request.ip, self.request.port
|
||||||
|
)
|
||||||
|
|
||||||
extra['request'] = '{0} {1}'.format(self.request.method,
|
extra["request"] = "{0} {1}".format(
|
||||||
self.request.url)
|
self.request.method, self.request.url
|
||||||
|
)
|
||||||
else:
|
else:
|
||||||
extra['request'] = 'nil'
|
extra["request"] = "nil"
|
||||||
|
|
||||||
access_logger.info('', extra=extra)
|
access_logger.info("", extra=extra)
|
||||||
|
|
||||||
def write_response(self, response):
|
def write_response(self, response):
|
||||||
"""
|
"""
|
||||||
|
@ -327,31 +361,37 @@ class HttpProtocol(asyncio.Protocol):
|
||||||
keep_alive = self.keep_alive
|
keep_alive = self.keep_alive
|
||||||
self.transport.write(
|
self.transport.write(
|
||||||
response.output(
|
response.output(
|
||||||
self.request.version, keep_alive,
|
self.request.version, keep_alive, self.keep_alive_timeout
|
||||||
self.keep_alive_timeout))
|
)
|
||||||
|
)
|
||||||
self.log_response(response)
|
self.log_response(response)
|
||||||
except AttributeError:
|
except AttributeError:
|
||||||
logger.error('Invalid response object for url %s, '
|
logger.error(
|
||||||
'Expected Type: HTTPResponse, Actual Type: %s',
|
"Invalid response object for url %s, "
|
||||||
self.url, type(response))
|
"Expected Type: HTTPResponse, Actual Type: %s",
|
||||||
self.write_error(ServerError('Invalid response type'))
|
self.url,
|
||||||
|
type(response),
|
||||||
|
)
|
||||||
|
self.write_error(ServerError("Invalid response type"))
|
||||||
except RuntimeError:
|
except RuntimeError:
|
||||||
if self._debug:
|
if self._debug:
|
||||||
logger.error('Connection lost before response written @ %s',
|
logger.error(
|
||||||
self.request.ip)
|
"Connection lost before response written @ %s",
|
||||||
|
self.request.ip,
|
||||||
|
)
|
||||||
keep_alive = False
|
keep_alive = False
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
self.bail_out(
|
self.bail_out(
|
||||||
"Writing response failed, connection closed {}".format(
|
"Writing response failed, connection closed {}".format(repr(e))
|
||||||
repr(e)))
|
)
|
||||||
finally:
|
finally:
|
||||||
if not keep_alive:
|
if not keep_alive:
|
||||||
self.transport.close()
|
self.transport.close()
|
||||||
self.transport = None
|
self.transport = None
|
||||||
else:
|
else:
|
||||||
self._keep_alive_timeout_handler = self.loop.call_later(
|
self._keep_alive_timeout_handler = self.loop.call_later(
|
||||||
self.keep_alive_timeout,
|
self.keep_alive_timeout, self.keep_alive_timeout_callback
|
||||||
self.keep_alive_timeout_callback)
|
)
|
||||||
self._last_response_time = current_time
|
self._last_response_time = current_time
|
||||||
self.cleanup()
|
self.cleanup()
|
||||||
|
|
||||||
|
@ -375,30 +415,36 @@ class HttpProtocol(asyncio.Protocol):
|
||||||
keep_alive = self.keep_alive
|
keep_alive = self.keep_alive
|
||||||
response.protocol = self
|
response.protocol = self
|
||||||
await response.stream(
|
await response.stream(
|
||||||
self.request.version, keep_alive, self.keep_alive_timeout)
|
self.request.version, keep_alive, self.keep_alive_timeout
|
||||||
|
)
|
||||||
self.log_response(response)
|
self.log_response(response)
|
||||||
except AttributeError:
|
except AttributeError:
|
||||||
logger.error('Invalid response object for url %s, '
|
logger.error(
|
||||||
'Expected Type: HTTPResponse, Actual Type: %s',
|
"Invalid response object for url %s, "
|
||||||
self.url, type(response))
|
"Expected Type: HTTPResponse, Actual Type: %s",
|
||||||
self.write_error(ServerError('Invalid response type'))
|
self.url,
|
||||||
|
type(response),
|
||||||
|
)
|
||||||
|
self.write_error(ServerError("Invalid response type"))
|
||||||
except RuntimeError:
|
except RuntimeError:
|
||||||
if self._debug:
|
if self._debug:
|
||||||
logger.error('Connection lost before response written @ %s',
|
logger.error(
|
||||||
self.request.ip)
|
"Connection lost before response written @ %s",
|
||||||
|
self.request.ip,
|
||||||
|
)
|
||||||
keep_alive = False
|
keep_alive = False
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
self.bail_out(
|
self.bail_out(
|
||||||
"Writing response failed, connection closed {}".format(
|
"Writing response failed, connection closed {}".format(repr(e))
|
||||||
repr(e)))
|
)
|
||||||
finally:
|
finally:
|
||||||
if not keep_alive:
|
if not keep_alive:
|
||||||
self.transport.close()
|
self.transport.close()
|
||||||
self.transport = None
|
self.transport = None
|
||||||
else:
|
else:
|
||||||
self._keep_alive_timeout_handler = self.loop.call_later(
|
self._keep_alive_timeout_handler = self.loop.call_later(
|
||||||
self.keep_alive_timeout,
|
self.keep_alive_timeout, self.keep_alive_timeout_callback
|
||||||
self.keep_alive_timeout_callback)
|
)
|
||||||
self._last_response_time = current_time
|
self._last_response_time = current_time
|
||||||
self.cleanup()
|
self.cleanup()
|
||||||
|
|
||||||
|
@ -411,32 +457,37 @@ class HttpProtocol(asyncio.Protocol):
|
||||||
response = None
|
response = None
|
||||||
try:
|
try:
|
||||||
response = self.error_handler.response(self.request, exception)
|
response = self.error_handler.response(self.request, exception)
|
||||||
version = self.request.version if self.request else '1.1'
|
version = self.request.version if self.request else "1.1"
|
||||||
self.transport.write(response.output(version))
|
self.transport.write(response.output(version))
|
||||||
except RuntimeError:
|
except RuntimeError:
|
||||||
if self._debug:
|
if self._debug:
|
||||||
logger.error('Connection lost before error written @ %s',
|
logger.error(
|
||||||
self.request.ip if self.request else 'Unknown')
|
"Connection lost before error written @ %s",
|
||||||
|
self.request.ip if self.request else "Unknown",
|
||||||
|
)
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
self.bail_out(
|
self.bail_out(
|
||||||
"Writing error failed, connection closed {}".format(
|
"Writing error failed, connection closed {}".format(repr(e)),
|
||||||
repr(e)), from_error=True
|
from_error=True,
|
||||||
)
|
)
|
||||||
finally:
|
finally:
|
||||||
if self.parser and (self.keep_alive
|
if self.parser and (
|
||||||
or getattr(response, 'status', 0) == 408):
|
self.keep_alive or getattr(response, "status", 0) == 408
|
||||||
|
):
|
||||||
self.log_response(response)
|
self.log_response(response)
|
||||||
try:
|
try:
|
||||||
self.transport.close()
|
self.transport.close()
|
||||||
except AttributeError:
|
except AttributeError:
|
||||||
logger.debug('Connection lost before server could close it.')
|
logger.debug("Connection lost before server could close it.")
|
||||||
|
|
||||||
def bail_out(self, message, from_error=False):
|
def bail_out(self, message, from_error=False):
|
||||||
if from_error or self.transport.is_closing():
|
if from_error or self.transport.is_closing():
|
||||||
logger.error("Transport closed @ %s and exception "
|
logger.error(
|
||||||
|
"Transport closed @ %s and exception "
|
||||||
"experienced during error handling",
|
"experienced during error handling",
|
||||||
self.transport.get_extra_info('peername'))
|
self.transport.get_extra_info("peername"),
|
||||||
logger.debug('Exception:', exc_info=True)
|
)
|
||||||
|
logger.debug("Exception:", exc_info=True)
|
||||||
else:
|
else:
|
||||||
self.write_error(ServerError(message))
|
self.write_error(ServerError(message))
|
||||||
logger.error(message)
|
logger.error(message)
|
||||||
|
@ -497,17 +548,43 @@ def trigger_events(events, loop):
|
||||||
loop.run_until_complete(result)
|
loop.run_until_complete(result)
|
||||||
|
|
||||||
|
|
||||||
def serve(host, port, request_handler, error_handler, before_start=None,
|
def serve(
|
||||||
after_start=None, before_stop=None, after_stop=None, debug=False,
|
host,
|
||||||
request_timeout=60, response_timeout=60, keep_alive_timeout=5,
|
port,
|
||||||
ssl=None, sock=None, request_max_size=None, reuse_port=False,
|
request_handler,
|
||||||
loop=None, protocol=HttpProtocol, backlog=100,
|
error_handler,
|
||||||
register_sys_signals=True, run_multiple=False, run_async=False,
|
before_start=None,
|
||||||
connections=None, signal=Signal(), request_class=None,
|
after_start=None,
|
||||||
access_log=True, keep_alive=True, is_request_stream=False,
|
before_stop=None,
|
||||||
router=None, websocket_max_size=None, websocket_max_queue=None,
|
after_stop=None,
|
||||||
websocket_read_limit=2 ** 16, websocket_write_limit=2 ** 16,
|
debug=False,
|
||||||
state=None, graceful_shutdown_timeout=15.0):
|
request_timeout=60,
|
||||||
|
response_timeout=60,
|
||||||
|
keep_alive_timeout=5,
|
||||||
|
ssl=None,
|
||||||
|
sock=None,
|
||||||
|
request_max_size=None,
|
||||||
|
reuse_port=False,
|
||||||
|
loop=None,
|
||||||
|
protocol=HttpProtocol,
|
||||||
|
backlog=100,
|
||||||
|
register_sys_signals=True,
|
||||||
|
run_multiple=False,
|
||||||
|
run_async=False,
|
||||||
|
connections=None,
|
||||||
|
signal=Signal(),
|
||||||
|
request_class=None,
|
||||||
|
access_log=True,
|
||||||
|
keep_alive=True,
|
||||||
|
is_request_stream=False,
|
||||||
|
router=None,
|
||||||
|
websocket_max_size=None,
|
||||||
|
websocket_max_queue=None,
|
||||||
|
websocket_read_limit=2 ** 16,
|
||||||
|
websocket_write_limit=2 ** 16,
|
||||||
|
state=None,
|
||||||
|
graceful_shutdown_timeout=15.0,
|
||||||
|
):
|
||||||
"""Start asynchronous HTTP Server on an individual process.
|
"""Start asynchronous HTTP Server on an individual process.
|
||||||
|
|
||||||
:param host: Address to host on
|
:param host: Address to host on
|
||||||
|
@ -592,7 +669,7 @@ def serve(host, port, request_handler, error_handler, before_start=None,
|
||||||
ssl=ssl,
|
ssl=ssl,
|
||||||
reuse_port=reuse_port,
|
reuse_port=reuse_port,
|
||||||
sock=sock,
|
sock=sock,
|
||||||
backlog=backlog
|
backlog=backlog,
|
||||||
)
|
)
|
||||||
|
|
||||||
# Instead of pulling time at the end of every request,
|
# Instead of pulling time at the end of every request,
|
||||||
|
@ -623,11 +700,13 @@ def serve(host, port, request_handler, error_handler, before_start=None,
|
||||||
try:
|
try:
|
||||||
loop.add_signal_handler(_signal, loop.stop)
|
loop.add_signal_handler(_signal, loop.stop)
|
||||||
except NotImplementedError:
|
except NotImplementedError:
|
||||||
logger.warning('Sanic tried to use loop.add_signal_handler '
|
logger.warning(
|
||||||
'but it is not implemented on this platform.')
|
"Sanic tried to use loop.add_signal_handler "
|
||||||
|
"but it is not implemented on this platform."
|
||||||
|
)
|
||||||
pid = os.getpid()
|
pid = os.getpid()
|
||||||
try:
|
try:
|
||||||
logger.info('Starting worker [%s]', pid)
|
logger.info("Starting worker [%s]", pid)
|
||||||
loop.run_forever()
|
loop.run_forever()
|
||||||
finally:
|
finally:
|
||||||
logger.info("Stopping worker [%s]", pid)
|
logger.info("Stopping worker [%s]", pid)
|
||||||
|
@ -658,9 +737,7 @@ def serve(host, port, request_handler, error_handler, before_start=None,
|
||||||
coros = []
|
coros = []
|
||||||
for conn in connections:
|
for conn in connections:
|
||||||
if hasattr(conn, "websocket") and conn.websocket:
|
if hasattr(conn, "websocket") and conn.websocket:
|
||||||
coros.append(
|
coros.append(conn.websocket.close_connection())
|
||||||
conn.websocket.close_connection()
|
|
||||||
)
|
|
||||||
else:
|
else:
|
||||||
conn.close()
|
conn.close()
|
||||||
|
|
||||||
|
@ -681,18 +758,18 @@ def serve_multiple(server_settings, workers):
|
||||||
:param stop_event: if provided, is used as a stop signal
|
:param stop_event: if provided, is used as a stop signal
|
||||||
:return:
|
:return:
|
||||||
"""
|
"""
|
||||||
server_settings['reuse_port'] = True
|
server_settings["reuse_port"] = True
|
||||||
server_settings['run_multiple'] = True
|
server_settings["run_multiple"] = True
|
||||||
|
|
||||||
# Handling when custom socket is not provided.
|
# Handling when custom socket is not provided.
|
||||||
if server_settings.get('sock') is None:
|
if server_settings.get("sock") is None:
|
||||||
sock = socket()
|
sock = socket()
|
||||||
sock.setsockopt(SOL_SOCKET, SO_REUSEADDR, 1)
|
sock.setsockopt(SOL_SOCKET, SO_REUSEADDR, 1)
|
||||||
sock.bind((server_settings['host'], server_settings['port']))
|
sock.bind((server_settings["host"], server_settings["port"]))
|
||||||
sock.set_inheritable(True)
|
sock.set_inheritable(True)
|
||||||
server_settings['sock'] = sock
|
server_settings["sock"] = sock
|
||||||
server_settings['host'] = None
|
server_settings["host"] = None
|
||||||
server_settings['port'] = None
|
server_settings["port"] = None
|
||||||
|
|
||||||
def sig_handler(signal, frame):
|
def sig_handler(signal, frame):
|
||||||
logger.info("Received signal %s. Shutting down.", Signals(signal).name)
|
logger.info("Received signal %s. Shutting down.", Signals(signal).name)
|
||||||
|
@ -716,4 +793,4 @@ def serve_multiple(server_settings, workers):
|
||||||
# the above processes will block this until they're stopped
|
# the above processes will block this until they're stopped
|
||||||
for process in processes:
|
for process in processes:
|
||||||
process.terminate()
|
process.terminate()
|
||||||
server_settings.get('sock').close()
|
server_settings.get("sock").close()
|
||||||
|
|
|
@ -16,10 +16,19 @@ from sanic.handlers import ContentRangeHandler
|
||||||
from sanic.response import file, file_stream, HTTPResponse
|
from sanic.response import file, file_stream, HTTPResponse
|
||||||
|
|
||||||
|
|
||||||
def register(app, uri, file_or_directory, pattern,
|
def register(
|
||||||
use_modified_since, use_content_range,
|
app,
|
||||||
stream_large_files, name='static', host=None,
|
uri,
|
||||||
strict_slashes=None, content_type=None):
|
file_or_directory,
|
||||||
|
pattern,
|
||||||
|
use_modified_since,
|
||||||
|
use_content_range,
|
||||||
|
stream_large_files,
|
||||||
|
name="static",
|
||||||
|
host=None,
|
||||||
|
strict_slashes=None,
|
||||||
|
content_type=None,
|
||||||
|
):
|
||||||
# TODO: Though sanic is not a file server, I feel like we should at least
|
# TODO: Though sanic is not a file server, I feel like we should at least
|
||||||
# make a good effort here. Modified-since is nice, but we could
|
# make a good effort here. Modified-since is nice, but we could
|
||||||
# also look into etags, expires, and caching
|
# also look into etags, expires, and caching
|
||||||
|
@ -46,12 +55,12 @@ def register(app, uri, file_or_directory, pattern,
|
||||||
# If we're not trying to match a file directly,
|
# If we're not trying to match a file directly,
|
||||||
# serve from the folder
|
# serve from the folder
|
||||||
if not path.isfile(file_or_directory):
|
if not path.isfile(file_or_directory):
|
||||||
uri += '<file_uri:' + pattern + '>'
|
uri += "<file_uri:" + pattern + ">"
|
||||||
|
|
||||||
async def _handler(request, file_uri=None):
|
async def _handler(request, file_uri=None):
|
||||||
# Using this to determine if the URL is trying to break out of the path
|
# Using this to determine if the URL is trying to break out of the path
|
||||||
# served. os.path.realpath seems to be very slow
|
# served. os.path.realpath seems to be very slow
|
||||||
if file_uri and '../' in file_uri:
|
if file_uri and "../" in file_uri:
|
||||||
raise InvalidUsage("Invalid URL")
|
raise InvalidUsage("Invalid URL")
|
||||||
# Merge served directory and requested file if provided
|
# Merge served directory and requested file if provided
|
||||||
# Strip all / that in the beginning of the URL to help prevent python
|
# Strip all / that in the beginning of the URL to help prevent python
|
||||||
|
@ -59,15 +68,16 @@ def register(app, uri, file_or_directory, pattern,
|
||||||
root_path = file_path = file_or_directory
|
root_path = file_path = file_or_directory
|
||||||
if file_uri:
|
if file_uri:
|
||||||
file_path = path.join(
|
file_path = path.join(
|
||||||
file_or_directory, sub('^[/]*', '', file_uri))
|
file_or_directory, sub("^[/]*", "", file_uri)
|
||||||
|
)
|
||||||
|
|
||||||
# URL decode the path sent by the browser otherwise we won't be able to
|
# URL decode the path sent by the browser otherwise we won't be able to
|
||||||
# match filenames which got encoded (filenames with spaces etc)
|
# match filenames which got encoded (filenames with spaces etc)
|
||||||
file_path = path.abspath(unquote(file_path))
|
file_path = path.abspath(unquote(file_path))
|
||||||
if not file_path.startswith(path.abspath(unquote(root_path))):
|
if not file_path.startswith(path.abspath(unquote(root_path))):
|
||||||
raise FileNotFound('File not found',
|
raise FileNotFound(
|
||||||
path=file_or_directory,
|
"File not found", path=file_or_directory, relative_url=file_uri
|
||||||
relative_url=file_uri)
|
)
|
||||||
try:
|
try:
|
||||||
headers = {}
|
headers = {}
|
||||||
# Check if the client has been sent this file before
|
# Check if the client has been sent this file before
|
||||||
|
@ -76,29 +86,31 @@ def register(app, uri, file_or_directory, pattern,
|
||||||
if use_modified_since:
|
if use_modified_since:
|
||||||
stats = await stat(file_path)
|
stats = await stat(file_path)
|
||||||
modified_since = strftime(
|
modified_since = strftime(
|
||||||
'%a, %d %b %Y %H:%M:%S GMT', gmtime(stats.st_mtime))
|
"%a, %d %b %Y %H:%M:%S GMT", gmtime(stats.st_mtime)
|
||||||
if request.headers.get('If-Modified-Since') == modified_since:
|
)
|
||||||
|
if request.headers.get("If-Modified-Since") == modified_since:
|
||||||
return HTTPResponse(status=304)
|
return HTTPResponse(status=304)
|
||||||
headers['Last-Modified'] = modified_since
|
headers["Last-Modified"] = modified_since
|
||||||
_range = None
|
_range = None
|
||||||
if use_content_range:
|
if use_content_range:
|
||||||
_range = None
|
_range = None
|
||||||
if not stats:
|
if not stats:
|
||||||
stats = await stat(file_path)
|
stats = await stat(file_path)
|
||||||
headers['Accept-Ranges'] = 'bytes'
|
headers["Accept-Ranges"] = "bytes"
|
||||||
headers['Content-Length'] = str(stats.st_size)
|
headers["Content-Length"] = str(stats.st_size)
|
||||||
if request.method != 'HEAD':
|
if request.method != "HEAD":
|
||||||
try:
|
try:
|
||||||
_range = ContentRangeHandler(request, stats)
|
_range = ContentRangeHandler(request, stats)
|
||||||
except HeaderNotFound:
|
except HeaderNotFound:
|
||||||
pass
|
pass
|
||||||
else:
|
else:
|
||||||
del headers['Content-Length']
|
del headers["Content-Length"]
|
||||||
for key, value in _range.headers.items():
|
for key, value in _range.headers.items():
|
||||||
headers[key] = value
|
headers[key] = value
|
||||||
headers['Content-Type'] = content_type \
|
headers["Content-Type"] = (
|
||||||
or guess_type(file_path)[0] or 'text/plain'
|
content_type or guess_type(file_path)[0] or "text/plain"
|
||||||
if request.method == 'HEAD':
|
)
|
||||||
|
if request.method == "HEAD":
|
||||||
return HTTPResponse(headers=headers)
|
return HTTPResponse(headers=headers)
|
||||||
else:
|
else:
|
||||||
if stream_large_files:
|
if stream_large_files:
|
||||||
|
@ -110,19 +122,25 @@ def register(app, uri, file_or_directory, pattern,
|
||||||
if not stats:
|
if not stats:
|
||||||
stats = await stat(file_path)
|
stats = await stat(file_path)
|
||||||
if stats.st_size >= threshold:
|
if stats.st_size >= threshold:
|
||||||
return await file_stream(file_path, headers=headers,
|
return await file_stream(
|
||||||
_range=_range)
|
file_path, headers=headers, _range=_range
|
||||||
|
)
|
||||||
return await file(file_path, headers=headers, _range=_range)
|
return await file(file_path, headers=headers, _range=_range)
|
||||||
except ContentRangeError:
|
except ContentRangeError:
|
||||||
raise
|
raise
|
||||||
except Exception:
|
except Exception:
|
||||||
raise FileNotFound('File not found',
|
raise FileNotFound(
|
||||||
path=file_or_directory,
|
"File not found", path=file_or_directory, relative_url=file_uri
|
||||||
relative_url=file_uri)
|
)
|
||||||
|
|
||||||
# special prefix for static files
|
# special prefix for static files
|
||||||
if not name.startswith('_static_'):
|
if not name.startswith("_static_"):
|
||||||
name = '_static_{}'.format(name)
|
name = "_static_{}".format(name)
|
||||||
|
|
||||||
app.route(uri, methods=['GET', 'HEAD'], name=name, host=host,
|
app.route(
|
||||||
strict_slashes=strict_slashes)(_handler)
|
uri,
|
||||||
|
methods=["GET", "HEAD"],
|
||||||
|
name=name,
|
||||||
|
host=host,
|
||||||
|
strict_slashes=strict_slashes,
|
||||||
|
)(_handler)
|
||||||
|
|
|
@ -4,7 +4,7 @@ from sanic.exceptions import MethodNotSupported
|
||||||
from sanic.response import text
|
from sanic.response import text
|
||||||
|
|
||||||
|
|
||||||
HOST = '127.0.0.1'
|
HOST = "127.0.0.1"
|
||||||
PORT = 42101
|
PORT = 42101
|
||||||
|
|
||||||
|
|
||||||
|
@ -15,18 +15,22 @@ class SanicTestClient:
|
||||||
|
|
||||||
async def _local_request(self, method, uri, cookies=None, *args, **kwargs):
|
async def _local_request(self, method, uri, cookies=None, *args, **kwargs):
|
||||||
import aiohttp
|
import aiohttp
|
||||||
if uri.startswith(('http:', 'https:', 'ftp:', 'ftps://' '//')):
|
|
||||||
|
if uri.startswith(("http:", "https:", "ftp:", "ftps://" "//")):
|
||||||
url = uri
|
url = uri
|
||||||
else:
|
else:
|
||||||
url = 'http://{host}:{port}{uri}'.format(
|
url = "http://{host}:{port}{uri}".format(
|
||||||
host=HOST, port=self.port, uri=uri)
|
host=HOST, port=self.port, uri=uri
|
||||||
|
)
|
||||||
|
|
||||||
logger.info(url)
|
logger.info(url)
|
||||||
conn = aiohttp.TCPConnector(verify_ssl=False)
|
conn = aiohttp.TCPConnector(verify_ssl=False)
|
||||||
async with aiohttp.ClientSession(
|
async with aiohttp.ClientSession(
|
||||||
cookies=cookies, connector=conn) as session:
|
cookies=cookies, connector=conn
|
||||||
async with getattr(
|
) as session:
|
||||||
session, method.lower())(url, *args, **kwargs) as response:
|
async with getattr(session, method.lower())(
|
||||||
|
url, *args, **kwargs
|
||||||
|
) as response:
|
||||||
try:
|
try:
|
||||||
response.text = await response.text()
|
response.text = await response.text()
|
||||||
except UnicodeDecodeError as e:
|
except UnicodeDecodeError as e:
|
||||||
|
@ -34,50 +38,60 @@ class SanicTestClient:
|
||||||
|
|
||||||
try:
|
try:
|
||||||
response.json = await response.json()
|
response.json = await response.json()
|
||||||
except (JSONDecodeError,
|
except (
|
||||||
|
JSONDecodeError,
|
||||||
UnicodeDecodeError,
|
UnicodeDecodeError,
|
||||||
aiohttp.ClientResponseError):
|
aiohttp.ClientResponseError,
|
||||||
|
):
|
||||||
response.json = None
|
response.json = None
|
||||||
|
|
||||||
response.body = await response.read()
|
response.body = await response.read()
|
||||||
return response
|
return response
|
||||||
|
|
||||||
def _sanic_endpoint_test(
|
def _sanic_endpoint_test(
|
||||||
self, method='get', uri='/', gather_request=True,
|
self,
|
||||||
debug=False, server_kwargs={"auto_reload": False},
|
method="get",
|
||||||
*request_args, **request_kwargs):
|
uri="/",
|
||||||
|
gather_request=True,
|
||||||
|
debug=False,
|
||||||
|
server_kwargs={"auto_reload": False},
|
||||||
|
*request_args,
|
||||||
|
**request_kwargs
|
||||||
|
):
|
||||||
results = [None, None]
|
results = [None, None]
|
||||||
exceptions = []
|
exceptions = []
|
||||||
|
|
||||||
if gather_request:
|
if gather_request:
|
||||||
|
|
||||||
def _collect_request(request):
|
def _collect_request(request):
|
||||||
if results[0] is None:
|
if results[0] is None:
|
||||||
results[0] = request
|
results[0] = request
|
||||||
|
|
||||||
self.app.request_middleware.appendleft(_collect_request)
|
self.app.request_middleware.appendleft(_collect_request)
|
||||||
|
|
||||||
@self.app.exception(MethodNotSupported)
|
@self.app.exception(MethodNotSupported)
|
||||||
async def error_handler(request, exception):
|
async def error_handler(request, exception):
|
||||||
if request.method in ['HEAD', 'PATCH', 'PUT', 'DELETE']:
|
if request.method in ["HEAD", "PATCH", "PUT", "DELETE"]:
|
||||||
return text(
|
return text(
|
||||||
'', exception.status_code, headers=exception.headers
|
"", exception.status_code, headers=exception.headers
|
||||||
)
|
)
|
||||||
else:
|
else:
|
||||||
return self.app.error_handler.default(request, exception)
|
return self.app.error_handler.default(request, exception)
|
||||||
|
|
||||||
@self.app.listener('after_server_start')
|
@self.app.listener("after_server_start")
|
||||||
async def _collect_response(sanic, loop):
|
async def _collect_response(sanic, loop):
|
||||||
try:
|
try:
|
||||||
response = await self._local_request(
|
response = await self._local_request(
|
||||||
method, uri, *request_args,
|
method, uri, *request_args, **request_kwargs
|
||||||
**request_kwargs)
|
)
|
||||||
results[-1] = response
|
results[-1] = response
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
logger.exception('Exception')
|
logger.exception("Exception")
|
||||||
exceptions.append(e)
|
exceptions.append(e)
|
||||||
self.app.stop()
|
self.app.stop()
|
||||||
|
|
||||||
self.app.run(host=HOST, debug=debug, port=self.port, **server_kwargs)
|
self.app.run(host=HOST, debug=debug, port=self.port, **server_kwargs)
|
||||||
self.app.listeners['after_server_start'].pop()
|
self.app.listeners["after_server_start"].pop()
|
||||||
|
|
||||||
if exceptions:
|
if exceptions:
|
||||||
raise ValueError("Exception during request: {}".format(exceptions))
|
raise ValueError("Exception during request: {}".format(exceptions))
|
||||||
|
@ -89,31 +103,34 @@ class SanicTestClient:
|
||||||
except BaseException:
|
except BaseException:
|
||||||
raise ValueError(
|
raise ValueError(
|
||||||
"Request and response object expected, got ({})".format(
|
"Request and response object expected, got ({})".format(
|
||||||
results))
|
results
|
||||||
|
)
|
||||||
|
)
|
||||||
else:
|
else:
|
||||||
try:
|
try:
|
||||||
return results[-1]
|
return results[-1]
|
||||||
except BaseException:
|
except BaseException:
|
||||||
raise ValueError(
|
raise ValueError(
|
||||||
"Request object expected, got ({})".format(results))
|
"Request object expected, got ({})".format(results)
|
||||||
|
)
|
||||||
|
|
||||||
def get(self, *args, **kwargs):
|
def get(self, *args, **kwargs):
|
||||||
return self._sanic_endpoint_test('get', *args, **kwargs)
|
return self._sanic_endpoint_test("get", *args, **kwargs)
|
||||||
|
|
||||||
def post(self, *args, **kwargs):
|
def post(self, *args, **kwargs):
|
||||||
return self._sanic_endpoint_test('post', *args, **kwargs)
|
return self._sanic_endpoint_test("post", *args, **kwargs)
|
||||||
|
|
||||||
def put(self, *args, **kwargs):
|
def put(self, *args, **kwargs):
|
||||||
return self._sanic_endpoint_test('put', *args, **kwargs)
|
return self._sanic_endpoint_test("put", *args, **kwargs)
|
||||||
|
|
||||||
def delete(self, *args, **kwargs):
|
def delete(self, *args, **kwargs):
|
||||||
return self._sanic_endpoint_test('delete', *args, **kwargs)
|
return self._sanic_endpoint_test("delete", *args, **kwargs)
|
||||||
|
|
||||||
def patch(self, *args, **kwargs):
|
def patch(self, *args, **kwargs):
|
||||||
return self._sanic_endpoint_test('patch', *args, **kwargs)
|
return self._sanic_endpoint_test("patch", *args, **kwargs)
|
||||||
|
|
||||||
def options(self, *args, **kwargs):
|
def options(self, *args, **kwargs):
|
||||||
return self._sanic_endpoint_test('options', *args, **kwargs)
|
return self._sanic_endpoint_test("options", *args, **kwargs)
|
||||||
|
|
||||||
def head(self, *args, **kwargs):
|
def head(self, *args, **kwargs):
|
||||||
return self._sanic_endpoint_test('head', *args, **kwargs)
|
return self._sanic_endpoint_test("head", *args, **kwargs)
|
||||||
|
|
|
@ -48,6 +48,7 @@ class HTTPMethodView:
|
||||||
"""Return view function for use with the routing system, that
|
"""Return view function for use with the routing system, that
|
||||||
dispatches request to appropriate handler method.
|
dispatches request to appropriate handler method.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def view(*args, **kwargs):
|
def view(*args, **kwargs):
|
||||||
self = view.view_class(*class_args, **class_kwargs)
|
self = view.view_class(*class_args, **class_kwargs)
|
||||||
return self.dispatch_request(*args, **kwargs)
|
return self.dispatch_request(*args, **kwargs)
|
||||||
|
@ -94,11 +95,13 @@ class CompositionView:
|
||||||
for method in methods:
|
for method in methods:
|
||||||
if method not in HTTP_METHODS:
|
if method not in HTTP_METHODS:
|
||||||
raise InvalidUsage(
|
raise InvalidUsage(
|
||||||
'{} is not a valid HTTP method.'.format(method))
|
"{} is not a valid HTTP method.".format(method)
|
||||||
|
)
|
||||||
|
|
||||||
if method in self.handlers:
|
if method in self.handlers:
|
||||||
raise InvalidUsage(
|
raise InvalidUsage(
|
||||||
'Method {} is already registered.'.format(method))
|
"Method {} is already registered.".format(method)
|
||||||
|
)
|
||||||
self.handlers[method] = handler
|
self.handlers[method] = handler
|
||||||
|
|
||||||
def __call__(self, request, *args, **kwargs):
|
def __call__(self, request, *args, **kwargs):
|
||||||
|
|
|
@ -6,11 +6,16 @@ from websockets import ConnectionClosed # noqa
|
||||||
|
|
||||||
|
|
||||||
class WebSocketProtocol(HttpProtocol):
|
class WebSocketProtocol(HttpProtocol):
|
||||||
def __init__(self, *args, websocket_timeout=10,
|
def __init__(
|
||||||
|
self,
|
||||||
|
*args,
|
||||||
|
websocket_timeout=10,
|
||||||
websocket_max_size=None,
|
websocket_max_size=None,
|
||||||
websocket_max_queue=None,
|
websocket_max_queue=None,
|
||||||
websocket_read_limit=2 ** 16,
|
websocket_read_limit=2 ** 16,
|
||||||
websocket_write_limit=2 ** 16, **kwargs):
|
websocket_write_limit=2 ** 16,
|
||||||
|
**kwargs
|
||||||
|
):
|
||||||
super().__init__(*args, **kwargs)
|
super().__init__(*args, **kwargs)
|
||||||
self.websocket = None
|
self.websocket = None
|
||||||
self.websocket_timeout = websocket_timeout
|
self.websocket_timeout = websocket_timeout
|
||||||
|
@ -63,24 +68,26 @@ class WebSocketProtocol(HttpProtocol):
|
||||||
key = handshake.check_request(request.headers)
|
key = handshake.check_request(request.headers)
|
||||||
handshake.build_response(headers, key)
|
handshake.build_response(headers, key)
|
||||||
except InvalidHandshake:
|
except InvalidHandshake:
|
||||||
raise InvalidUsage('Invalid websocket request')
|
raise InvalidUsage("Invalid websocket request")
|
||||||
|
|
||||||
subprotocol = None
|
subprotocol = None
|
||||||
if subprotocols and 'Sec-Websocket-Protocol' in request.headers:
|
if subprotocols and "Sec-Websocket-Protocol" in request.headers:
|
||||||
# select a subprotocol
|
# select a subprotocol
|
||||||
client_subprotocols = [p.strip() for p in request.headers[
|
client_subprotocols = [
|
||||||
'Sec-Websocket-Protocol'].split(',')]
|
p.strip()
|
||||||
|
for p in request.headers["Sec-Websocket-Protocol"].split(",")
|
||||||
|
]
|
||||||
for p in client_subprotocols:
|
for p in client_subprotocols:
|
||||||
if p in subprotocols:
|
if p in subprotocols:
|
||||||
subprotocol = p
|
subprotocol = p
|
||||||
headers['Sec-Websocket-Protocol'] = subprotocol
|
headers["Sec-Websocket-Protocol"] = subprotocol
|
||||||
break
|
break
|
||||||
|
|
||||||
# write the 101 response back to the client
|
# write the 101 response back to the client
|
||||||
rv = b'HTTP/1.1 101 Switching Protocols\r\n'
|
rv = b"HTTP/1.1 101 Switching Protocols\r\n"
|
||||||
for k, v in headers.items():
|
for k, v in headers.items():
|
||||||
rv += k.encode('utf-8') + b': ' + v.encode('utf-8') + b'\r\n'
|
rv += k.encode("utf-8") + b": " + v.encode("utf-8") + b"\r\n"
|
||||||
rv += b'\r\n'
|
rv += b"\r\n"
|
||||||
request.transport.write(rv)
|
request.transport.write(rv)
|
||||||
|
|
||||||
# hook up the websocket protocol
|
# hook up the websocket protocol
|
||||||
|
@ -89,7 +96,7 @@ class WebSocketProtocol(HttpProtocol):
|
||||||
max_size=self.websocket_max_size,
|
max_size=self.websocket_max_size,
|
||||||
max_queue=self.websocket_max_queue,
|
max_queue=self.websocket_max_queue,
|
||||||
read_limit=self.websocket_read_limit,
|
read_limit=self.websocket_read_limit,
|
||||||
write_limit=self.websocket_write_limit
|
write_limit=self.websocket_write_limit,
|
||||||
)
|
)
|
||||||
self.websocket.subprotocol = subprotocol
|
self.websocket.subprotocol = subprotocol
|
||||||
self.websocket.connection_made(request.transport)
|
self.websocket.connection_made(request.transport)
|
||||||
|
|
|
@ -12,6 +12,7 @@ except ImportError:
|
||||||
|
|
||||||
try:
|
try:
|
||||||
import uvloop
|
import uvloop
|
||||||
|
|
||||||
asyncio.set_event_loop_policy(uvloop.EventLoopPolicy())
|
asyncio.set_event_loop_policy(uvloop.EventLoopPolicy())
|
||||||
except ImportError:
|
except ImportError:
|
||||||
pass
|
pass
|
||||||
|
@ -50,36 +51,43 @@ class GunicornWorker(base.Worker):
|
||||||
def run(self):
|
def run(self):
|
||||||
is_debug = self.log.loglevel == logging.DEBUG
|
is_debug = self.log.loglevel == logging.DEBUG
|
||||||
protocol = (
|
protocol = (
|
||||||
self.websocket_protocol if self.app.callable.websocket_enabled
|
self.websocket_protocol
|
||||||
else self.http_protocol)
|
if self.app.callable.websocket_enabled
|
||||||
|
else self.http_protocol
|
||||||
|
)
|
||||||
self._server_settings = self.app.callable._helper(
|
self._server_settings = self.app.callable._helper(
|
||||||
loop=self.loop,
|
loop=self.loop,
|
||||||
debug=is_debug,
|
debug=is_debug,
|
||||||
protocol=protocol,
|
protocol=protocol,
|
||||||
ssl=self.ssl_context,
|
ssl=self.ssl_context,
|
||||||
run_async=True)
|
run_async=True,
|
||||||
self._server_settings['signal'] = self.signal
|
)
|
||||||
self._server_settings.pop('sock')
|
self._server_settings["signal"] = self.signal
|
||||||
trigger_events(self._server_settings.get('before_start', []),
|
self._server_settings.pop("sock")
|
||||||
self.loop)
|
trigger_events(
|
||||||
self._server_settings['before_start'] = ()
|
self._server_settings.get("before_start", []), self.loop
|
||||||
|
)
|
||||||
|
self._server_settings["before_start"] = ()
|
||||||
|
|
||||||
self._runner = asyncio.ensure_future(self._run(), loop=self.loop)
|
self._runner = asyncio.ensure_future(self._run(), loop=self.loop)
|
||||||
try:
|
try:
|
||||||
self.loop.run_until_complete(self._runner)
|
self.loop.run_until_complete(self._runner)
|
||||||
self.app.callable.is_running = True
|
self.app.callable.is_running = True
|
||||||
trigger_events(self._server_settings.get('after_start', []),
|
trigger_events(
|
||||||
self.loop)
|
self._server_settings.get("after_start", []), self.loop
|
||||||
|
)
|
||||||
self.loop.run_until_complete(self._check_alive())
|
self.loop.run_until_complete(self._check_alive())
|
||||||
trigger_events(self._server_settings.get('before_stop', []),
|
trigger_events(
|
||||||
self.loop)
|
self._server_settings.get("before_stop", []), self.loop
|
||||||
|
)
|
||||||
self.loop.run_until_complete(self.close())
|
self.loop.run_until_complete(self.close())
|
||||||
except BaseException:
|
except BaseException:
|
||||||
traceback.print_exc()
|
traceback.print_exc()
|
||||||
finally:
|
finally:
|
||||||
try:
|
try:
|
||||||
trigger_events(self._server_settings.get('after_stop', []),
|
trigger_events(
|
||||||
self.loop)
|
self._server_settings.get("after_stop", []), self.loop
|
||||||
|
)
|
||||||
except BaseException:
|
except BaseException:
|
||||||
traceback.print_exc()
|
traceback.print_exc()
|
||||||
finally:
|
finally:
|
||||||
|
@ -90,8 +98,11 @@ class GunicornWorker(base.Worker):
|
||||||
async def close(self):
|
async def close(self):
|
||||||
if self.servers:
|
if self.servers:
|
||||||
# stop accepting connections
|
# stop accepting connections
|
||||||
self.log.info("Stopping server: %s, connections: %s",
|
self.log.info(
|
||||||
self.pid, len(self.connections))
|
"Stopping server: %s, connections: %s",
|
||||||
|
self.pid,
|
||||||
|
len(self.connections),
|
||||||
|
)
|
||||||
for server in self.servers:
|
for server in self.servers:
|
||||||
server.close()
|
server.close()
|
||||||
await server.wait_closed()
|
await server.wait_closed()
|
||||||
|
@ -105,8 +116,9 @@ class GunicornWorker(base.Worker):
|
||||||
# gracefully shutdown timeout
|
# gracefully shutdown timeout
|
||||||
start_shutdown = 0
|
start_shutdown = 0
|
||||||
graceful_shutdown_timeout = self.cfg.graceful_timeout
|
graceful_shutdown_timeout = self.cfg.graceful_timeout
|
||||||
while self.connections and \
|
while self.connections and (
|
||||||
(start_shutdown < graceful_shutdown_timeout):
|
start_shutdown < graceful_shutdown_timeout
|
||||||
|
):
|
||||||
await asyncio.sleep(0.1)
|
await asyncio.sleep(0.1)
|
||||||
start_shutdown = start_shutdown + 0.1
|
start_shutdown = start_shutdown + 0.1
|
||||||
|
|
||||||
|
@ -115,9 +127,7 @@ class GunicornWorker(base.Worker):
|
||||||
coros = []
|
coros = []
|
||||||
for conn in self.connections:
|
for conn in self.connections:
|
||||||
if hasattr(conn, "websocket") and conn.websocket:
|
if hasattr(conn, "websocket") and conn.websocket:
|
||||||
coros.append(
|
coros.append(conn.websocket.close_connection())
|
||||||
conn.websocket.close_connection()
|
|
||||||
)
|
|
||||||
else:
|
else:
|
||||||
conn.close()
|
conn.close()
|
||||||
_shutdown = asyncio.gather(*coros, loop=self.loop)
|
_shutdown = asyncio.gather(*coros, loop=self.loop)
|
||||||
|
@ -148,8 +158,9 @@ class GunicornWorker(base.Worker):
|
||||||
)
|
)
|
||||||
if self.max_requests and req_count > self.max_requests:
|
if self.max_requests and req_count > self.max_requests:
|
||||||
self.alive = False
|
self.alive = False
|
||||||
self.log.info("Max requests exceeded, shutting down: %s",
|
self.log.info(
|
||||||
self)
|
"Max requests exceeded, shutting down: %s", self
|
||||||
|
)
|
||||||
elif pid == os.getpid() and self.ppid != os.getppid():
|
elif pid == os.getpid() and self.ppid != os.getppid():
|
||||||
self.alive = False
|
self.alive = False
|
||||||
self.log.info("Parent changed, shutting down: %s", self)
|
self.log.info("Parent changed, shutting down: %s", self)
|
||||||
|
@ -175,23 +186,29 @@ class GunicornWorker(base.Worker):
|
||||||
def init_signals(self):
|
def init_signals(self):
|
||||||
# Set up signals through the event loop API.
|
# Set up signals through the event loop API.
|
||||||
|
|
||||||
self.loop.add_signal_handler(signal.SIGQUIT, self.handle_quit,
|
self.loop.add_signal_handler(
|
||||||
signal.SIGQUIT, None)
|
signal.SIGQUIT, self.handle_quit, signal.SIGQUIT, None
|
||||||
|
)
|
||||||
|
|
||||||
self.loop.add_signal_handler(signal.SIGTERM, self.handle_exit,
|
self.loop.add_signal_handler(
|
||||||
signal.SIGTERM, None)
|
signal.SIGTERM, self.handle_exit, signal.SIGTERM, None
|
||||||
|
)
|
||||||
|
|
||||||
self.loop.add_signal_handler(signal.SIGINT, self.handle_quit,
|
self.loop.add_signal_handler(
|
||||||
signal.SIGINT, None)
|
signal.SIGINT, self.handle_quit, signal.SIGINT, None
|
||||||
|
)
|
||||||
|
|
||||||
self.loop.add_signal_handler(signal.SIGWINCH, self.handle_winch,
|
self.loop.add_signal_handler(
|
||||||
signal.SIGWINCH, None)
|
signal.SIGWINCH, self.handle_winch, signal.SIGWINCH, None
|
||||||
|
)
|
||||||
|
|
||||||
self.loop.add_signal_handler(signal.SIGUSR1, self.handle_usr1,
|
self.loop.add_signal_handler(
|
||||||
signal.SIGUSR1, None)
|
signal.SIGUSR1, self.handle_usr1, signal.SIGUSR1, None
|
||||||
|
)
|
||||||
|
|
||||||
self.loop.add_signal_handler(signal.SIGABRT, self.handle_abort,
|
self.loop.add_signal_handler(
|
||||||
signal.SIGABRT, None)
|
signal.SIGABRT, self.handle_abort, signal.SIGABRT, None
|
||||||
|
)
|
||||||
|
|
||||||
# Don't let SIGTERM and SIGUSR1 disturb active requests
|
# Don't let SIGTERM and SIGUSR1 disturb active requests
|
||||||
# by interrupting system calls
|
# by interrupting system calls
|
||||||
|
|
4
setup.cfg
Normal file
4
setup.cfg
Normal file
|
@ -0,0 +1,4 @@
|
||||||
|
[flake8]
|
||||||
|
# https://github.com/ambv/black#slices
|
||||||
|
# https://github.com/ambv/black#line-breaks--binary-operators
|
||||||
|
ignore = E203, W503
|
2
setup.py
2
setup.py
|
@ -21,7 +21,7 @@ def open_local(paths, mode='r', encoding='utf8'):
|
||||||
|
|
||||||
with open_local(['sanic', '__init__.py'], encoding='latin1') as fp:
|
with open_local(['sanic', '__init__.py'], encoding='latin1') as fp:
|
||||||
try:
|
try:
|
||||||
version = re.findall(r"^__version__ = '([^']+)'\r?$",
|
version = re.findall(r"^__version__ = \"([^']+)\"\r?$",
|
||||||
fp.read(), re.M)[0]
|
fp.read(), re.M)[0]
|
||||||
except IndexError:
|
except IndexError:
|
||||||
raise RuntimeError('Unable to determine version.')
|
raise RuntimeError('Unable to determine version.')
|
||||||
|
|
|
@ -16,11 +16,13 @@ from sanic.constants import HTTP_METHODS
|
||||||
def get_file_path(static_file_directory, file_name):
|
def get_file_path(static_file_directory, file_name):
|
||||||
return os.path.join(static_file_directory, file_name)
|
return os.path.join(static_file_directory, file_name)
|
||||||
|
|
||||||
|
|
||||||
def get_file_content(static_file_directory, file_name):
|
def get_file_content(static_file_directory, file_name):
|
||||||
"""The content of the static file to check"""
|
"""The content of the static file to check"""
|
||||||
with open(get_file_path(static_file_directory, file_name), 'rb') as file:
|
with open(get_file_path(static_file_directory, file_name), 'rb') as file:
|
||||||
return file.read()
|
return file.read()
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.parametrize('method', HTTP_METHODS)
|
@pytest.mark.parametrize('method', HTTP_METHODS)
|
||||||
def test_versioned_routes_get(app, method):
|
def test_versioned_routes_get(app, method):
|
||||||
bp = Blueprint('test_text')
|
bp = Blueprint('test_text')
|
||||||
|
@ -57,22 +59,23 @@ def test_bp(app):
|
||||||
|
|
||||||
assert response.text == 'Hello'
|
assert response.text == 'Hello'
|
||||||
|
|
||||||
|
|
||||||
def test_bp_strict_slash(app):
|
def test_bp_strict_slash(app):
|
||||||
bp = Blueprint('test_text')
|
bp = Blueprint('test_text')
|
||||||
|
|
||||||
@bp.get('/get', strict_slashes=True)
|
@bp.get('/get', strict_slashes=True)
|
||||||
def handler(request):
|
def get_handler(request):
|
||||||
return text('OK')
|
return text('OK')
|
||||||
|
|
||||||
@bp.post('/post/', strict_slashes=True)
|
@bp.post('/post/', strict_slashes=True)
|
||||||
def handler(request):
|
def post_handler(request):
|
||||||
return text('OK')
|
return text('OK')
|
||||||
|
|
||||||
app.blueprint(bp)
|
app.blueprint(bp)
|
||||||
|
|
||||||
request, response = app.test_client.get('/get')
|
request, response = app.test_client.get('/get')
|
||||||
assert response.text == 'OK'
|
assert response.text == 'OK'
|
||||||
assert response.json == None
|
assert response.json is None
|
||||||
|
|
||||||
request, response = app.test_client.get('/get/')
|
request, response = app.test_client.get('/get/')
|
||||||
assert response.status == 404
|
assert response.status == 404
|
||||||
|
@ -83,15 +86,16 @@ def test_bp_strict_slash(app):
|
||||||
request, response = app.test_client.post('/post')
|
request, response = app.test_client.post('/post')
|
||||||
assert response.status == 404
|
assert response.status == 404
|
||||||
|
|
||||||
|
|
||||||
def test_bp_strict_slash_default_value(app):
|
def test_bp_strict_slash_default_value(app):
|
||||||
bp = Blueprint('test_text', strict_slashes=True)
|
bp = Blueprint('test_text', strict_slashes=True)
|
||||||
|
|
||||||
@bp.get('/get')
|
@bp.get('/get')
|
||||||
def handler(request):
|
def get_handler(request):
|
||||||
return text('OK')
|
return text('OK')
|
||||||
|
|
||||||
@bp.post('/post/')
|
@bp.post('/post/')
|
||||||
def handler(request):
|
def post_handler(request):
|
||||||
return text('OK')
|
return text('OK')
|
||||||
|
|
||||||
app.blueprint(bp)
|
app.blueprint(bp)
|
||||||
|
@ -102,15 +106,16 @@ def test_bp_strict_slash_default_value(app):
|
||||||
request, response = app.test_client.post('/post')
|
request, response = app.test_client.post('/post')
|
||||||
assert response.status == 404
|
assert response.status == 404
|
||||||
|
|
||||||
|
|
||||||
def test_bp_strict_slash_without_passing_default_value(app):
|
def test_bp_strict_slash_without_passing_default_value(app):
|
||||||
bp = Blueprint('test_text')
|
bp = Blueprint('test_text')
|
||||||
|
|
||||||
@bp.get('/get')
|
@bp.get('/get')
|
||||||
def handler(request):
|
def get_handler(request):
|
||||||
return text('OK')
|
return text('OK')
|
||||||
|
|
||||||
@bp.post('/post/')
|
@bp.post('/post/')
|
||||||
def handler(request):
|
def post_handler(request):
|
||||||
return text('OK')
|
return text('OK')
|
||||||
|
|
||||||
app.blueprint(bp)
|
app.blueprint(bp)
|
||||||
|
@ -121,15 +126,16 @@ def test_bp_strict_slash_without_passing_default_value(app):
|
||||||
request, response = app.test_client.post('/post')
|
request, response = app.test_client.post('/post')
|
||||||
assert response.text == 'OK'
|
assert response.text == 'OK'
|
||||||
|
|
||||||
|
|
||||||
def test_bp_strict_slash_default_value_can_be_overwritten(app):
|
def test_bp_strict_slash_default_value_can_be_overwritten(app):
|
||||||
bp = Blueprint('test_text', strict_slashes=True)
|
bp = Blueprint('test_text', strict_slashes=True)
|
||||||
|
|
||||||
@bp.get('/get', strict_slashes=False)
|
@bp.get('/get', strict_slashes=False)
|
||||||
def handler(request):
|
def get_handler(request):
|
||||||
return text('OK')
|
return text('OK')
|
||||||
|
|
||||||
@bp.post('/post/', strict_slashes=False)
|
@bp.post('/post/', strict_slashes=False)
|
||||||
def handler(request):
|
def post_handler(request):
|
||||||
return text('OK')
|
return text('OK')
|
||||||
|
|
||||||
app.blueprint(bp)
|
app.blueprint(bp)
|
||||||
|
@ -140,6 +146,7 @@ def test_bp_strict_slash_default_value_can_be_overwritten(app):
|
||||||
request, response = app.test_client.post('/post')
|
request, response = app.test_client.post('/post')
|
||||||
assert response.text == 'OK'
|
assert response.text == 'OK'
|
||||||
|
|
||||||
|
|
||||||
def test_bp_with_url_prefix(app):
|
def test_bp_with_url_prefix(app):
|
||||||
bp = Blueprint('test_text', url_prefix='/test1')
|
bp = Blueprint('test_text', url_prefix='/test1')
|
||||||
|
|
||||||
|
@ -173,15 +180,16 @@ def test_several_bp_with_url_prefix(app):
|
||||||
request, response = app.test_client.get('/test2/')
|
request, response = app.test_client.get('/test2/')
|
||||||
assert response.text == 'Hello2'
|
assert response.text == 'Hello2'
|
||||||
|
|
||||||
|
|
||||||
def test_bp_with_host(app):
|
def test_bp_with_host(app):
|
||||||
bp = Blueprint('test_bp_host', url_prefix='/test1', host="example.com")
|
bp = Blueprint('test_bp_host', url_prefix='/test1', host="example.com")
|
||||||
|
|
||||||
@bp.route('/')
|
@bp.route('/')
|
||||||
def handler(request):
|
def handler1(request):
|
||||||
return text('Hello')
|
return text('Hello')
|
||||||
|
|
||||||
@bp.route('/', host="sub.example.com")
|
@bp.route('/', host="sub.example.com")
|
||||||
def handler(request):
|
def handler2(request):
|
||||||
return text('Hello subdomain!')
|
return text('Hello subdomain!')
|
||||||
|
|
||||||
app.blueprint(bp)
|
app.blueprint(bp)
|
||||||
|
@ -212,14 +220,13 @@ def test_several_bp_with_host(app):
|
||||||
return text('Hello')
|
return text('Hello')
|
||||||
|
|
||||||
@bp2.route('/')
|
@bp2.route('/')
|
||||||
def handler2(request):
|
def handler1(request):
|
||||||
return text('Hello2')
|
return text('Hello2')
|
||||||
|
|
||||||
@bp2.route('/other/')
|
@bp2.route('/other/')
|
||||||
def handler2(request):
|
def handler2(request):
|
||||||
return text('Hello3')
|
return text('Hello3')
|
||||||
|
|
||||||
|
|
||||||
app.blueprint(bp)
|
app.blueprint(bp)
|
||||||
app.blueprint(bp2)
|
app.blueprint(bp2)
|
||||||
|
|
||||||
|
@ -242,6 +249,7 @@ def test_several_bp_with_host(app):
|
||||||
headers=headers)
|
headers=headers)
|
||||||
assert response.text == 'Hello3'
|
assert response.text == 'Hello3'
|
||||||
|
|
||||||
|
|
||||||
def test_bp_middleware(app):
|
def test_bp_middleware(app):
|
||||||
blueprint = Blueprint('test_middleware')
|
blueprint = Blueprint('test_middleware')
|
||||||
|
|
||||||
|
@ -260,6 +268,7 @@ def test_bp_middleware(app):
|
||||||
assert response.status == 200
|
assert response.status == 200
|
||||||
assert response.text == 'OK'
|
assert response.text == 'OK'
|
||||||
|
|
||||||
|
|
||||||
def test_bp_exception_handler(app):
|
def test_bp_exception_handler(app):
|
||||||
blueprint = Blueprint('test_middleware')
|
blueprint = Blueprint('test_middleware')
|
||||||
|
|
||||||
|
@ -284,7 +293,6 @@ def test_bp_exception_handler(app):
|
||||||
request, response = app.test_client.get('/1')
|
request, response = app.test_client.get('/1')
|
||||||
assert response.status == 400
|
assert response.status == 400
|
||||||
|
|
||||||
|
|
||||||
request, response = app.test_client.get('/2')
|
request, response = app.test_client.get('/2')
|
||||||
assert response.status == 200
|
assert response.status == 200
|
||||||
assert response.text == 'OK'
|
assert response.text == 'OK'
|
||||||
|
@ -292,6 +300,7 @@ def test_bp_exception_handler(app):
|
||||||
request, response = app.test_client.get('/3')
|
request, response = app.test_client.get('/3')
|
||||||
assert response.status == 200
|
assert response.status == 200
|
||||||
|
|
||||||
|
|
||||||
def test_bp_listeners(app):
|
def test_bp_listeners(app):
|
||||||
blueprint = Blueprint('test_middleware')
|
blueprint = Blueprint('test_middleware')
|
||||||
|
|
||||||
|
@ -327,6 +336,7 @@ def test_bp_listeners(app):
|
||||||
|
|
||||||
assert order == [1, 2, 3, 4, 5, 6]
|
assert order == [1, 2, 3, 4, 5, 6]
|
||||||
|
|
||||||
|
|
||||||
def test_bp_static(app):
|
def test_bp_static(app):
|
||||||
current_file = inspect.getfile(inspect.currentframe())
|
current_file = inspect.getfile(inspect.currentframe())
|
||||||
with open(current_file, 'rb') as file:
|
with open(current_file, 'rb') as file:
|
||||||
|
@ -342,6 +352,7 @@ def test_bp_static(app):
|
||||||
assert response.status == 200
|
assert response.status == 200
|
||||||
assert response.body == current_file_contents
|
assert response.body == current_file_contents
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.parametrize('file_name', ['test.html'])
|
@pytest.mark.parametrize('file_name', ['test.html'])
|
||||||
def test_bp_static_content_type(app, file_name):
|
def test_bp_static_content_type(app, file_name):
|
||||||
# This is done here, since no other test loads a file here
|
# This is done here, since no other test loads a file here
|
||||||
|
@ -363,6 +374,7 @@ def test_bp_static_content_type(app, file_name):
|
||||||
assert response.body == get_file_content(static_directory, file_name)
|
assert response.body == get_file_content(static_directory, file_name)
|
||||||
assert response.headers['Content-Type'] == 'text/html; charset=utf-8'
|
assert response.headers['Content-Type'] == 'text/html; charset=utf-8'
|
||||||
|
|
||||||
|
|
||||||
def test_bp_shorthand(app):
|
def test_bp_shorthand(app):
|
||||||
blueprint = Blueprint('test_shorhand_routes')
|
blueprint = Blueprint('test_shorhand_routes')
|
||||||
ev = asyncio.Event()
|
ev = asyncio.Event()
|
||||||
|
@ -373,37 +385,37 @@ def test_bp_shorthand(app):
|
||||||
return text('OK')
|
return text('OK')
|
||||||
|
|
||||||
@blueprint.put('/put')
|
@blueprint.put('/put')
|
||||||
def handler(request):
|
def put_handler(request):
|
||||||
assert request.stream is None
|
assert request.stream is None
|
||||||
return text('OK')
|
return text('OK')
|
||||||
|
|
||||||
@blueprint.post('/post')
|
@blueprint.post('/post')
|
||||||
def handler(request):
|
def post_handler(request):
|
||||||
assert request.stream is None
|
assert request.stream is None
|
||||||
return text('OK')
|
return text('OK')
|
||||||
|
|
||||||
@blueprint.head('/head')
|
@blueprint.head('/head')
|
||||||
def handler(request):
|
def head_handler(request):
|
||||||
assert request.stream is None
|
assert request.stream is None
|
||||||
return text('OK')
|
return text('OK')
|
||||||
|
|
||||||
@blueprint.options('/options')
|
@blueprint.options('/options')
|
||||||
def handler(request):
|
def options_handler(request):
|
||||||
assert request.stream is None
|
assert request.stream is None
|
||||||
return text('OK')
|
return text('OK')
|
||||||
|
|
||||||
@blueprint.patch('/patch')
|
@blueprint.patch('/patch')
|
||||||
def handler(request):
|
def patch_handler(request):
|
||||||
assert request.stream is None
|
assert request.stream is None
|
||||||
return text('OK')
|
return text('OK')
|
||||||
|
|
||||||
@blueprint.delete('/delete')
|
@blueprint.delete('/delete')
|
||||||
def handler(request):
|
def delete_handler(request):
|
||||||
assert request.stream is None
|
assert request.stream is None
|
||||||
return text('OK')
|
return text('OK')
|
||||||
|
|
||||||
@blueprint.websocket('/ws')
|
@blueprint.websocket('/ws')
|
||||||
async def handler(request, ws):
|
async def websocket_handler(request, ws):
|
||||||
assert request.stream is None
|
assert request.stream is None
|
||||||
ev.set()
|
ev.set()
|
||||||
|
|
||||||
|
@ -461,6 +473,7 @@ def test_bp_shorthand(app):
|
||||||
assert response.status == 101
|
assert response.status == 101
|
||||||
assert ev.is_set()
|
assert ev.is_set()
|
||||||
|
|
||||||
|
|
||||||
def test_bp_group(app):
|
def test_bp_group(app):
|
||||||
deep_0 = Blueprint('deep_0', url_prefix='/deep')
|
deep_0 = Blueprint('deep_0', url_prefix='/deep')
|
||||||
deep_1 = Blueprint('deep_1', url_prefix='/deep1')
|
deep_1 = Blueprint('deep_1', url_prefix='/deep1')
|
||||||
|
@ -470,14 +483,14 @@ def test_bp_group(app):
|
||||||
return text('D0_OK')
|
return text('D0_OK')
|
||||||
|
|
||||||
@deep_1.route('/bottom')
|
@deep_1.route('/bottom')
|
||||||
def handler(request):
|
def bottom_handler(request):
|
||||||
return text('D1B_OK')
|
return text('D1B_OK')
|
||||||
|
|
||||||
mid_0 = Blueprint.group(deep_0, deep_1, url_prefix='/mid')
|
mid_0 = Blueprint.group(deep_0, deep_1, url_prefix='/mid')
|
||||||
mid_1 = Blueprint('mid_tier', url_prefix='/mid1')
|
mid_1 = Blueprint('mid_tier', url_prefix='/mid1')
|
||||||
|
|
||||||
@mid_1.route('/')
|
@mid_1.route('/')
|
||||||
def handler(request):
|
def handler1(request):
|
||||||
return text('M1_OK')
|
return text('M1_OK')
|
||||||
|
|
||||||
top = Blueprint.group(mid_0, mid_1)
|
top = Blueprint.group(mid_0, mid_1)
|
||||||
|
@ -485,7 +498,7 @@ def test_bp_group(app):
|
||||||
app.blueprint(top)
|
app.blueprint(top)
|
||||||
|
|
||||||
@app.route('/')
|
@app.route('/')
|
||||||
def handler(request):
|
def handler2(request):
|
||||||
return text('TOP_OK')
|
return text('TOP_OK')
|
||||||
|
|
||||||
request, response = app.test_client.get('/')
|
request, response = app.test_client.get('/')
|
||||||
|
@ -505,24 +518,29 @@ def test_bp_group_with_default_url_prefix(app):
|
||||||
|
|
||||||
from sanic.response import json
|
from sanic.response import json
|
||||||
bp_resources = Blueprint('bp_resources')
|
bp_resources = Blueprint('bp_resources')
|
||||||
|
|
||||||
@bp_resources.get('/')
|
@bp_resources.get('/')
|
||||||
def list_resources_handler(request):
|
def list_resources_handler(request):
|
||||||
resource = {}
|
resource = {}
|
||||||
return json([resource])
|
return json([resource])
|
||||||
|
|
||||||
bp_resource = Blueprint('bp_resource', url_prefix='/<resource_id>')
|
bp_resource = Blueprint('bp_resource', url_prefix='/<resource_id>')
|
||||||
|
|
||||||
@bp_resource.get('/')
|
@bp_resource.get('/')
|
||||||
def get_resource_hander(request, resource_id):
|
def get_resource_hander(request, resource_id):
|
||||||
resource = {'resource_id': resource_id}
|
resource = {'resource_id': resource_id}
|
||||||
return json(resource)
|
return json(resource)
|
||||||
|
|
||||||
bp_resources_group = Blueprint.group(bp_resources, bp_resource, url_prefix='/resources')
|
bp_resources_group = Blueprint.group(bp_resources, bp_resource,
|
||||||
|
url_prefix='/resources')
|
||||||
bp_api_v1 = Blueprint('bp_api_v1')
|
bp_api_v1 = Blueprint('bp_api_v1')
|
||||||
|
|
||||||
@bp_api_v1.get('/info')
|
@bp_api_v1.get('/info')
|
||||||
def api_v1_info(request):
|
def api_v1_info(request):
|
||||||
return text('api_version: v1')
|
return text('api_version: v1')
|
||||||
|
|
||||||
bp_api_v1_group = Blueprint.group(bp_api_v1, bp_resources_group, url_prefix='/v1')
|
bp_api_v1_group = Blueprint.group(bp_api_v1, bp_resources_group,
|
||||||
|
url_prefix='/v1')
|
||||||
bp_api_group = Blueprint.group(bp_api_v1_group, url_prefix='/api')
|
bp_api_group = Blueprint.group(bp_api_v1_group, url_prefix='/api')
|
||||||
app.blueprint(bp_api_group)
|
app.blueprint(bp_api_group)
|
||||||
|
|
||||||
|
@ -534,5 +552,6 @@ def test_bp_group_with_default_url_prefix(app):
|
||||||
|
|
||||||
from uuid import uuid4
|
from uuid import uuid4
|
||||||
resource_id = str(uuid4())
|
resource_id = str(uuid4())
|
||||||
request, response = app.test_client.get('/api/v1/resources/{0}'.format(resource_id))
|
request, response = app.test_client.get(
|
||||||
|
'/api/v1/resources/{0}'.format(resource_id))
|
||||||
assert response.json == {'resource_id': resource_id}
|
assert response.json == {'resource_id': resource_id}
|
||||||
|
|
6
tox.ini
6
tox.ini
|
@ -1,5 +1,5 @@
|
||||||
[tox]
|
[tox]
|
||||||
envlist = py35, py36, py37, {py35,py36,py37}-no-ext, flake8, check
|
envlist = py35, py36, py37, {py35,py36,py37}-no-ext, lint, check
|
||||||
|
|
||||||
[testenv]
|
[testenv]
|
||||||
usedevelop = True
|
usedevelop = True
|
||||||
|
@ -21,12 +21,14 @@ commands =
|
||||||
- coverage combine --append
|
- coverage combine --append
|
||||||
coverage report -m
|
coverage report -m
|
||||||
|
|
||||||
[testenv:flake8]
|
[testenv:lint]
|
||||||
deps =
|
deps =
|
||||||
flake8
|
flake8
|
||||||
|
black
|
||||||
|
|
||||||
commands =
|
commands =
|
||||||
flake8 sanic
|
flake8 sanic
|
||||||
|
black --check --verbose sanic
|
||||||
|
|
||||||
[testenv:check]
|
[testenv:check]
|
||||||
deps =
|
deps =
|
||||||
|
|
Loading…
Reference in New Issue
Block a user