Create requests-async based TestClient, remove aiohttp dependency, drop Python 3.5
Update all tests to be compatible with requests-async Cleanup testing client changes with black and isort Remove Python 3.5 and other meta doc cleanup rename pyproject and fix pep517 error Add black config to tox.ini Cleanup tests and remove aiohttp tox.ini change for easier development commands Remove aiohttp from changelog and requirements Cleanup imports and Makefile
This commit is contained in:
parent
6a4a3f617f
commit
ccd4c9615c
|
@ -2,11 +2,6 @@ version: "{branch}.{build}"
|
||||||
|
|
||||||
environment:
|
environment:
|
||||||
matrix:
|
matrix:
|
||||||
- TOXENV: py35-no-ext
|
|
||||||
PYTHON: "C:\\Python35-x64"
|
|
||||||
PYTHON_VERSION: "3.5.x"
|
|
||||||
PYTHON_ARCH: "64"
|
|
||||||
|
|
||||||
- TOXENV: py36-no-ext
|
- TOXENV: py36-no-ext
|
||||||
PYTHON: "C:\\Python36-x64"
|
PYTHON: "C:\\Python36-x64"
|
||||||
PYTHON_VERSION: "3.6.x"
|
PYTHON_VERSION: "3.6.x"
|
||||||
|
|
|
@ -5,10 +5,6 @@ cache:
|
||||||
- $HOME/.cache/pip
|
- $HOME/.cache/pip
|
||||||
matrix:
|
matrix:
|
||||||
include:
|
include:
|
||||||
- env: TOX_ENV=py35
|
|
||||||
python: 3.5
|
|
||||||
- env: TOX_ENV=py35-no-ext
|
|
||||||
python: 3.5
|
|
||||||
- env: TOX_ENV=py36
|
- env: TOX_ENV=py36
|
||||||
python: 3.6
|
python: 3.6
|
||||||
- env: TOX_ENV=py36-no-ext
|
- env: TOX_ENV=py36-no-ext
|
||||||
|
|
15
CHANGELOG.md
15
CHANGELOG.md
|
@ -1,3 +1,18 @@
|
||||||
|
Version 19.6
|
||||||
|
------------
|
||||||
|
19.6.0
|
||||||
|
- Changes:
|
||||||
|
- Remove `aiohttp` dependencey and create new `SanicTestClient` based upon
|
||||||
|
[`requests-async`](https://github.com/encode/requests-async).
|
||||||
|
|
||||||
|
- Deprecation:
|
||||||
|
- Support for Python 3.5
|
||||||
|
|
||||||
|
Note: Sanic will not support Python 3.5 from version 19.6 and forward. However,
|
||||||
|
version 18.12LTS will have its support period extended thru December 2020, and
|
||||||
|
therefore passing Python's official support version 3.5, which is set to expire
|
||||||
|
in September 2020.
|
||||||
|
|
||||||
Version 19.3
|
Version 19.3
|
||||||
-------------
|
-------------
|
||||||
19.3.1
|
19.3.1
|
||||||
|
|
2
LICENSE
2
LICENSE
|
@ -1,6 +1,6 @@
|
||||||
MIT License
|
MIT License
|
||||||
|
|
||||||
Copyright (c) 2016-present Channel Cat
|
Copyright (c) 2016-present Sanic Community
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
|
6
Makefile
6
Makefile
|
@ -47,12 +47,12 @@ ifdef include_tests
|
||||||
isort -rc sanic tests
|
isort -rc sanic tests
|
||||||
else
|
else
|
||||||
$(info Sorting Imports)
|
$(info Sorting Imports)
|
||||||
isort -rc sanic
|
isort -rc sanic tests
|
||||||
endif
|
endif
|
||||||
endif
|
endif
|
||||||
|
|
||||||
black:
|
black:
|
||||||
black --config ./pyproject.toml sanic tests
|
black --config ./.black.toml sanic tests
|
||||||
|
|
||||||
fix-import: black
|
fix-import: black
|
||||||
isort -rc sanic
|
isort -rc sanic tests
|
||||||
|
|
|
@ -17,6 +17,8 @@ Sanic | Build fast. Run fast.
|
||||||
- | |PyPI| |PyPI version| |Wheel| |Supported implementations| |Code style black|
|
- | |PyPI| |PyPI version| |Wheel| |Supported implementations| |Code style black|
|
||||||
* - Support
|
* - Support
|
||||||
- | |Forums| |Join the chat at https://gitter.im/sanic-python/Lobby|
|
- | |Forums| |Join the chat at https://gitter.im/sanic-python/Lobby|
|
||||||
|
* - Stats
|
||||||
|
- | |Downloads|
|
||||||
|
|
||||||
.. |Forums| image:: https://img.shields.io/badge/forums-community-ff0068.svg
|
.. |Forums| image:: https://img.shields.io/badge/forums-community-ff0068.svg
|
||||||
:target: https://community.sanicframework.org/
|
:target: https://community.sanicframework.org/
|
||||||
|
@ -42,10 +44,13 @@ Sanic | Build fast. Run fast.
|
||||||
.. |Supported implementations| image:: https://img.shields.io/pypi/implementation/sanic.svg
|
.. |Supported implementations| image:: https://img.shields.io/pypi/implementation/sanic.svg
|
||||||
:alt: Supported implementations
|
:alt: Supported implementations
|
||||||
:target: https://pypi.python.org/pypi/sanic
|
:target: https://pypi.python.org/pypi/sanic
|
||||||
|
.. |Downloads| image:: https://pepy.tech/badge/sanic/month
|
||||||
|
:alt: Downloads
|
||||||
|
:target: https://pepy.tech/project/sanic
|
||||||
|
|
||||||
.. end-badges
|
.. end-badges
|
||||||
|
|
||||||
Sanic is a Python web server and web framework that's written to go fast. It allows the usage of the ``async/await`` syntax added in Python 3.5, which makes your code non-blocking and speedy.
|
Sanic is a **Python 3.6+** web server and web framework that's written to go fast. It allows the usage of the ``async/await`` syntax added in Python 3.5, which makes your code non-blocking and speedy.
|
||||||
|
|
||||||
`Source code on GitHub <https://github.com/huge-success/sanic/>`_ | `Help and discussion board <https://community.sanicframework.org/>`_.
|
`Source code on GitHub <https://github.com/huge-success/sanic/>`_ | `Help and discussion board <https://community.sanicframework.org/>`_.
|
||||||
|
|
||||||
|
|
|
@ -1,3 +1,18 @@
|
||||||
|
Version 19.6
|
||||||
|
------------
|
||||||
|
19.6.0
|
||||||
|
- Changes:
|
||||||
|
- Remove `aiohttp` dependencey and create new `SanicTestClient` based upon
|
||||||
|
[`requests-async`](https://github.com/encode/requests-async).
|
||||||
|
|
||||||
|
- Deprecation:
|
||||||
|
- Support for Python 3.5
|
||||||
|
|
||||||
|
Note: Sanic will not support Python 3.5 from version 19.6 and forward. However,
|
||||||
|
version 18.12LTS will have its support period extended thru December 2020, and
|
||||||
|
therefore passing Python's official support version 3.5, which is set to expire
|
||||||
|
in September 2020.
|
||||||
|
|
||||||
Version 19.3
|
Version 19.3
|
||||||
-------------
|
-------------
|
||||||
19.3.1
|
19.3.1
|
||||||
|
|
|
@ -1,8 +1,8 @@
|
||||||
# Testing
|
# Testing
|
||||||
|
|
||||||
Sanic endpoints can be tested locally using the `test_client` object, which
|
Sanic endpoints can be tested locally using the `test_client` object, which
|
||||||
depends on the additional [aiohttp](https://aiohttp.readthedocs.io/en/stable/)
|
depends on the additional [`requests-async`](https://github.com/encode/requests-async)
|
||||||
library.
|
library, which implements an API that mirrors the `requests` library.
|
||||||
|
|
||||||
The `test_client` exposes `get`, `post`, `put`, `delete`, `patch`, `head` and `options` methods
|
The `test_client` exposes `get`, `post`, `put`, `delete`, `patch`, `head` and `options` methods
|
||||||
for you to run against your application. A simple example (using pytest) is like follows:
|
for you to run against your application. A simple example (using pytest) is like follows:
|
||||||
|
@ -21,7 +21,7 @@ def test_index_put_not_allowed():
|
||||||
```
|
```
|
||||||
|
|
||||||
Internally, each time you call one of the `test_client` methods, the Sanic app is run at `127.0.0.1:42101` and
|
Internally, each time you call one of the `test_client` methods, the Sanic app is run at `127.0.0.1:42101` and
|
||||||
your test request is executed against your application, using `aiohttp`.
|
your test request is executed against your application, using `requests-async`.
|
||||||
|
|
||||||
The `test_client` methods accept the following arguments and keyword arguments:
|
The `test_client` methods accept the following arguments and keyword arguments:
|
||||||
|
|
||||||
|
@ -33,7 +33,7 @@ The `test_client` methods accept the following arguments and keyword arguments:
|
||||||
- `server_kwargs` *(default `{}`) a dict of additional arguments to pass into `app.run` before the test request is run.
|
- `server_kwargs` *(default `{}`) a dict of additional arguments to pass into `app.run` before the test request is run.
|
||||||
- `debug` *(default `False`)* A boolean which determines whether to run the server in debug mode.
|
- `debug` *(default `False`)* A boolean which determines whether to run the server in debug mode.
|
||||||
|
|
||||||
The function further takes the `*request_args` and `**request_kwargs`, which are passed directly to the aiohttp ClientSession request.
|
The function further takes the `*request_args` and `**request_kwargs`, which are passed directly to the request.
|
||||||
|
|
||||||
For example, to supply data to a GET request, you would do the following:
|
For example, to supply data to a GET request, you would do the following:
|
||||||
|
|
||||||
|
@ -55,8 +55,8 @@ def test_post_json_request_includes_data():
|
||||||
|
|
||||||
|
|
||||||
More information about
|
More information about
|
||||||
the available arguments to aiohttp can be found
|
the available arguments to `requests-async` can be found
|
||||||
[in the documentation for ClientSession](https://aiohttp.readthedocs.io/en/stable/client_reference.html#client-session).
|
[in the documentation for `requests`](https://2.python-requests.org/en/master/).
|
||||||
|
|
||||||
|
|
||||||
## Using a random port
|
## Using a random port
|
||||||
|
|
|
@ -1,6 +1,9 @@
|
||||||
from json import JSONDecodeError
|
from json import JSONDecodeError
|
||||||
from socket import socket
|
from socket import socket
|
||||||
|
|
||||||
|
import requests_async as requests
|
||||||
|
import websockets
|
||||||
|
|
||||||
from sanic.exceptions import MethodNotSupported
|
from sanic.exceptions import MethodNotSupported
|
||||||
from sanic.log import logger
|
from sanic.log import logger
|
||||||
from sanic.response import text
|
from sanic.response import text
|
||||||
|
@ -16,32 +19,41 @@ class SanicTestClient:
|
||||||
self.app = app
|
self.app = app
|
||||||
self.port = port
|
self.port = port
|
||||||
|
|
||||||
async def _local_request(self, method, url, cookies=None, *args, **kwargs):
|
def get_new_session(self):
|
||||||
import aiohttp
|
return requests.Session()
|
||||||
|
|
||||||
|
async def _local_request(self, method, url, *args, **kwargs):
|
||||||
logger.info(url)
|
logger.info(url)
|
||||||
conn = aiohttp.TCPConnector(ssl=False)
|
raw_cookies = kwargs.pop("raw_cookies", None)
|
||||||
async with aiohttp.ClientSession(
|
|
||||||
cookies=cookies, connector=conn
|
if method == "websocket":
|
||||||
) as session:
|
async with websockets.connect(url, *args, **kwargs) as websocket:
|
||||||
async with getattr(session, method.lower())(
|
websocket.opened = websocket.open
|
||||||
url, *args, **kwargs
|
return websocket
|
||||||
) as response:
|
else:
|
||||||
try:
|
async with self.get_new_session() as session:
|
||||||
response.text = await response.text()
|
|
||||||
except UnicodeDecodeError:
|
|
||||||
response.text = None
|
|
||||||
|
|
||||||
try:
|
try:
|
||||||
response.json = await response.json()
|
response = await getattr(session, method.lower())(
|
||||||
except (
|
url, verify=False, *args, **kwargs
|
||||||
JSONDecodeError,
|
)
|
||||||
UnicodeDecodeError,
|
except NameError:
|
||||||
aiohttp.ClientResponseError,
|
raise Exception(response.status_code)
|
||||||
):
|
|
||||||
|
try:
|
||||||
|
response.json = response.json()
|
||||||
|
except (JSONDecodeError, UnicodeDecodeError):
|
||||||
response.json = None
|
response.json = None
|
||||||
|
|
||||||
response.body = await response.read()
|
response.body = await response.read()
|
||||||
|
response.status = response.status_code
|
||||||
|
response.content_type = response.headers.get("content-type")
|
||||||
|
|
||||||
|
if raw_cookies:
|
||||||
|
response.raw_cookies = {}
|
||||||
|
for cookie in response.cookies:
|
||||||
|
response.raw_cookies[cookie.name] = cookie
|
||||||
|
|
||||||
return response
|
return response
|
||||||
|
|
||||||
def _sanic_endpoint_test(
|
def _sanic_endpoint_test(
|
||||||
|
@ -83,11 +95,15 @@ class SanicTestClient:
|
||||||
server_kwargs = dict(sock=sock, **server_kwargs)
|
server_kwargs = dict(sock=sock, **server_kwargs)
|
||||||
host, port = sock.getsockname()
|
host, port = sock.getsockname()
|
||||||
|
|
||||||
if uri.startswith(("http:", "https:", "ftp:", "ftps://", "//")):
|
if uri.startswith(
|
||||||
|
("http:", "https:", "ftp:", "ftps://", "//", "ws:", "wss:")
|
||||||
|
):
|
||||||
url = uri
|
url = uri
|
||||||
else:
|
else:
|
||||||
url = "http://{host}:{port}{uri}".format(
|
uri = uri if uri.startswith("/") else "/{uri}".format(uri=uri)
|
||||||
host=host, port=port, uri=uri
|
scheme = "ws" if method == "websocket" else "http"
|
||||||
|
url = "{scheme}://{host}:{port}{uri}".format(
|
||||||
|
scheme=scheme, host=host, port=port, uri=uri
|
||||||
)
|
)
|
||||||
|
|
||||||
@self.app.listener("after_server_start")
|
@self.app.listener("after_server_start")
|
||||||
|
@ -146,3 +162,6 @@ class SanicTestClient:
|
||||||
|
|
||||||
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)
|
||||||
|
|
||||||
|
def websocket(self, *args, **kwargs):
|
||||||
|
return self._sanic_endpoint_test("websocket", *args, **kwargs)
|
||||||
|
|
|
@ -14,7 +14,7 @@ multi_line_output = 3
|
||||||
not_skip = __init__.py
|
not_skip = __init__.py
|
||||||
|
|
||||||
[version]
|
[version]
|
||||||
current_version = 0.8.3
|
current_version = 19.3.1
|
||||||
file = sanic/__init__.py
|
file = sanic/__init__.py
|
||||||
current_version_pattern = __version__ = "{current_version}"
|
current_version_pattern = __version__ = "{current_version}"
|
||||||
new_version_pattern = __version__ = "{new_version}"
|
new_version_pattern = __version__ = "{new_version}"
|
||||||
|
|
14
setup.py
14
setup.py
|
@ -50,12 +50,12 @@ with open_local(["README.rst"]) as rm:
|
||||||
setup_kwargs = {
|
setup_kwargs = {
|
||||||
"name": "sanic",
|
"name": "sanic",
|
||||||
"version": version,
|
"version": version,
|
||||||
"url": "http://github.com/channelcat/sanic/",
|
"url": "http://github.com/huge-success/sanic/",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"author": "Channel Cat",
|
"author": "Sanic Community",
|
||||||
"author_email": "channelcat@gmail.com",
|
"author_email": "admhpkns@gmail.com",
|
||||||
"description": (
|
"description": (
|
||||||
"A microframework based on uvloop, httptools, and learnings of flask"
|
"A web server and web framework that's written to go fast. Build fast. Run fast."
|
||||||
),
|
),
|
||||||
"long_description": long_description,
|
"long_description": long_description,
|
||||||
"packages": ["sanic"],
|
"packages": ["sanic"],
|
||||||
|
@ -64,7 +64,6 @@ setup_kwargs = {
|
||||||
"Development Status :: 4 - Beta",
|
"Development Status :: 4 - Beta",
|
||||||
"Environment :: Web Environment",
|
"Environment :: Web Environment",
|
||||||
"License :: OSI Approved :: MIT License",
|
"License :: OSI Approved :: MIT License",
|
||||||
"Programming Language :: Python :: 3.5",
|
|
||||||
"Programming Language :: Python :: 3.6",
|
"Programming Language :: Python :: 3.6",
|
||||||
"Programming Language :: Python :: 3.7",
|
"Programming Language :: Python :: 3.7",
|
||||||
],
|
],
|
||||||
|
@ -90,7 +89,8 @@ tests_require = [
|
||||||
"multidict>=4.0,<5.0",
|
"multidict>=4.0,<5.0",
|
||||||
"gunicorn",
|
"gunicorn",
|
||||||
"pytest-cov",
|
"pytest-cov",
|
||||||
"aiohttp>=2.3.0,<=3.2.1",
|
"httpcore==0.1.1",
|
||||||
|
"requests-async==0.4.0",
|
||||||
"beautifulsoup4",
|
"beautifulsoup4",
|
||||||
uvloop,
|
uvloop,
|
||||||
ujson,
|
ujson,
|
||||||
|
@ -119,7 +119,7 @@ extras_require = {
|
||||||
"recommonmark",
|
"recommonmark",
|
||||||
"sphinxcontrib-asyncio",
|
"sphinxcontrib-asyncio",
|
||||||
"docutils",
|
"docutils",
|
||||||
"pygments"
|
"pygments",
|
||||||
],
|
],
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,8 +1,10 @@
|
||||||
from random import choice, seed
|
from random import choice, seed
|
||||||
|
|
||||||
from pytest import mark
|
from pytest import mark
|
||||||
|
|
||||||
import sanic.router
|
import sanic.router
|
||||||
|
|
||||||
|
|
||||||
seed("Pack my box with five dozen liquor jugs.")
|
seed("Pack my box with five dozen liquor jugs.")
|
||||||
|
|
||||||
# Disable Caching for testing purpose
|
# Disable Caching for testing purpose
|
||||||
|
|
|
@ -9,6 +9,7 @@ import pytest
|
||||||
from sanic import Sanic
|
from sanic import Sanic
|
||||||
from sanic.router import RouteExists, Router
|
from sanic.router import RouteExists, Router
|
||||||
|
|
||||||
|
|
||||||
random.seed("Pack my box with five dozen liquor jugs.")
|
random.seed("Pack my box with five dozen liquor jugs.")
|
||||||
|
|
||||||
if sys.platform in ["win32", "cygwin"]:
|
if sys.platform in ["win32", "cygwin"]:
|
||||||
|
|
|
@ -1,10 +1,13 @@
|
||||||
# Run with python3 simple_server.py PORT
|
# Run with python3 simple_server.py PORT
|
||||||
|
|
||||||
from aiohttp import web
|
|
||||||
import asyncio
|
import asyncio
|
||||||
import sys
|
import sys
|
||||||
import uvloop
|
|
||||||
import ujson as json
|
import ujson as json
|
||||||
|
import uvloop
|
||||||
|
|
||||||
|
from aiohttp import web
|
||||||
|
|
||||||
|
|
||||||
loop = uvloop.new_event_loop()
|
loop = uvloop.new_event_loop()
|
||||||
asyncio.set_event_loop(loop)
|
asyncio.set_event_loop(loop)
|
||||||
|
|
|
@ -1,8 +1,9 @@
|
||||||
# Run with: gunicorn --workers=1 --worker-class=meinheld.gmeinheld.MeinheldWorker -b :8000 simple_server:app
|
# Run with: gunicorn --workers=1 --worker-class=meinheld.gmeinheld.MeinheldWorker -b :8000 simple_server:app
|
||||||
import bottle
|
import bottle
|
||||||
from bottle import route, run
|
|
||||||
import ujson
|
import ujson
|
||||||
|
|
||||||
|
from bottle import route, run
|
||||||
|
|
||||||
|
|
||||||
@route("/")
|
@route("/")
|
||||||
def index():
|
def index():
|
||||||
|
|
|
@ -1,10 +1,13 @@
|
||||||
# Run with: python3 -O simple_server.py
|
# Run with: python3 -O simple_server.py
|
||||||
import asyncio
|
import asyncio
|
||||||
from kyoukai import Kyoukai, HTTPRequestContext
|
|
||||||
import logging
|
import logging
|
||||||
|
|
||||||
import ujson
|
import ujson
|
||||||
import uvloop
|
import uvloop
|
||||||
|
|
||||||
|
from kyoukai import HTTPRequestContext, Kyoukai
|
||||||
|
|
||||||
|
|
||||||
loop = uvloop.new_event_loop()
|
loop = uvloop.new_event_loop()
|
||||||
asyncio.set_event_loop(loop)
|
asyncio.set_event_loop(loop)
|
||||||
|
|
||||||
|
|
|
@ -1,16 +1,18 @@
|
||||||
import asyncpg
|
|
||||||
import sys
|
|
||||||
import os
|
|
||||||
import inspect
|
import inspect
|
||||||
|
import os
|
||||||
|
import sys
|
||||||
|
import timeit
|
||||||
|
|
||||||
|
import asyncpg
|
||||||
|
|
||||||
|
from sanic.response import json
|
||||||
|
|
||||||
|
|
||||||
currentdir = os.path.dirname(
|
currentdir = os.path.dirname(
|
||||||
os.path.abspath(inspect.getfile(inspect.currentframe()))
|
os.path.abspath(inspect.getfile(inspect.currentframe()))
|
||||||
)
|
)
|
||||||
sys.path.insert(0, currentdir + "/../../../")
|
sys.path.insert(0, currentdir + "/../../../")
|
||||||
|
|
||||||
import timeit
|
|
||||||
|
|
||||||
from sanic.response import json
|
|
||||||
|
|
||||||
print(json({"test": True}).output())
|
print(json({"test": True}).output())
|
||||||
|
|
||||||
|
|
|
@ -1,14 +1,16 @@
|
||||||
import sys
|
|
||||||
import os
|
|
||||||
import inspect
|
import inspect
|
||||||
|
import os
|
||||||
|
import sys
|
||||||
|
|
||||||
|
from sanic import Sanic
|
||||||
|
from sanic.response import json
|
||||||
|
|
||||||
|
|
||||||
currentdir = os.path.dirname(
|
currentdir = os.path.dirname(
|
||||||
os.path.abspath(inspect.getfile(inspect.currentframe()))
|
os.path.abspath(inspect.getfile(inspect.currentframe()))
|
||||||
)
|
)
|
||||||
sys.path.insert(0, currentdir + "/../../../")
|
sys.path.insert(0, currentdir + "/../../../")
|
||||||
|
|
||||||
from sanic import Sanic
|
|
||||||
from sanic.response import json
|
|
||||||
|
|
||||||
app = Sanic("test")
|
app = Sanic("test")
|
||||||
|
|
||||||
|
|
|
@ -1,15 +1,17 @@
|
||||||
import sys
|
|
||||||
import os
|
|
||||||
import inspect
|
import inspect
|
||||||
|
import os
|
||||||
|
import sys
|
||||||
|
|
||||||
|
from sanic import Sanic
|
||||||
|
from sanic.exceptions import ServerError
|
||||||
|
from sanic.response import json, text
|
||||||
|
|
||||||
|
|
||||||
currentdir = os.path.dirname(
|
currentdir = os.path.dirname(
|
||||||
os.path.abspath(inspect.getfile(inspect.currentframe()))
|
os.path.abspath(inspect.getfile(inspect.currentframe()))
|
||||||
)
|
)
|
||||||
sys.path.insert(0, currentdir + "/../../../")
|
sys.path.insert(0, currentdir + "/../../../")
|
||||||
|
|
||||||
from sanic import Sanic
|
|
||||||
from sanic.response import json, text
|
|
||||||
from sanic.exceptions import ServerError
|
|
||||||
|
|
||||||
app = Sanic("test")
|
app = Sanic("test")
|
||||||
|
|
||||||
|
@ -56,8 +58,6 @@ def query_string(request):
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
import sys
|
|
||||||
|
|
||||||
app.run(host="0.0.0.0", port=sys.argv[1])
|
app.run(host="0.0.0.0", port=sys.argv[1])
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
# Run with: python simple_server.py
|
# Run with: python simple_server.py
|
||||||
import ujson
|
import ujson
|
||||||
|
|
||||||
from tornado import ioloop, web
|
from tornado import ioloop, web
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -2,15 +2,16 @@
|
||||||
""" Minimal helloworld application.
|
""" Minimal helloworld application.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
from wheezy.http import HTTPResponse
|
import ujson
|
||||||
from wheezy.http import WSGIApplication
|
|
||||||
|
from wheezy.http import HTTPResponse, WSGIApplication
|
||||||
from wheezy.http.response import json_response
|
from wheezy.http.response import json_response
|
||||||
from wheezy.routing import url
|
from wheezy.routing import url
|
||||||
from wheezy.web.handlers import BaseHandler
|
from wheezy.web.handlers import BaseHandler
|
||||||
from wheezy.web.middleware import bootstrap_defaults
|
from wheezy.web.middleware import (
|
||||||
from wheezy.web.middleware import path_routing_middleware_factory
|
bootstrap_defaults,
|
||||||
|
path_routing_middleware_factory,
|
||||||
import ujson
|
)
|
||||||
|
|
||||||
|
|
||||||
class WelcomeHandler(BaseHandler):
|
class WelcomeHandler(BaseHandler):
|
||||||
|
|
|
@ -3,6 +3,7 @@ import logging
|
||||||
import sys
|
import sys
|
||||||
|
|
||||||
from inspect import isawaitable
|
from inspect import isawaitable
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
|
|
||||||
from sanic.exceptions import SanicException
|
from sanic.exceptions import SanicException
|
||||||
|
@ -11,7 +12,7 @@ from sanic.response import text
|
||||||
|
|
||||||
def uvloop_installed():
|
def uvloop_installed():
|
||||||
try:
|
try:
|
||||||
import uvloop
|
import uvloop # noqa
|
||||||
|
|
||||||
return True
|
return True
|
||||||
except ImportError:
|
except ImportError:
|
||||||
|
|
|
@ -3,7 +3,8 @@ from pytest import raises
|
||||||
from sanic.app import Sanic
|
from sanic.app import Sanic
|
||||||
from sanic.blueprints import Blueprint
|
from sanic.blueprints import Blueprint
|
||||||
from sanic.request import Request
|
from sanic.request import Request
|
||||||
from sanic.response import text, HTTPResponse
|
from sanic.response import HTTPResponse, text
|
||||||
|
|
||||||
|
|
||||||
MIDDLEWARE_INVOKE_COUNTER = {"request": 0, "response": 0}
|
MIDDLEWARE_INVOKE_COUNTER = {"request": 0, "response": 0}
|
||||||
|
|
||||||
|
|
|
@ -7,9 +7,9 @@ import pytest
|
||||||
from sanic.app import Sanic
|
from sanic.app import Sanic
|
||||||
from sanic.blueprints import Blueprint
|
from sanic.blueprints import Blueprint
|
||||||
from sanic.constants import HTTP_METHODS
|
from sanic.constants import HTTP_METHODS
|
||||||
from sanic.exceptions import NotFound, ServerError, InvalidUsage
|
from sanic.exceptions import InvalidUsage, NotFound, ServerError
|
||||||
from sanic.request import Request
|
from sanic.request import Request
|
||||||
from sanic.response import text, json
|
from sanic.response import json, text
|
||||||
from sanic.views import CompositionView
|
from sanic.views import CompositionView
|
||||||
|
|
||||||
|
|
||||||
|
@ -467,16 +467,8 @@ def test_bp_shorthand(app):
|
||||||
request, response = app.test_client.get("/delete")
|
request, response = app.test_client.get("/delete")
|
||||||
assert response.status == 405
|
assert response.status == 405
|
||||||
|
|
||||||
request, response = app.test_client.get(
|
request, response = app.test_client.websocket("/ws/")
|
||||||
"/ws/",
|
assert response.opened is True
|
||||||
headers={
|
|
||||||
"Upgrade": "websocket",
|
|
||||||
"Connection": "upgrade",
|
|
||||||
"Sec-WebSocket-Key": "dGhlIHNhbXBsZSBub25jZQ==",
|
|
||||||
"Sec-WebSocket-Version": "13",
|
|
||||||
},
|
|
||||||
)
|
|
||||||
assert response.status == 101
|
|
||||||
assert ev.is_set()
|
assert ev.is_set()
|
||||||
|
|
||||||
|
|
||||||
|
@ -595,14 +587,13 @@ def test_blueprint_middleware_with_args(app: Sanic):
|
||||||
"/wa", headers={"content-type": "plain/text"}
|
"/wa", headers={"content-type": "plain/text"}
|
||||||
)
|
)
|
||||||
assert response.json.get("test") == "value"
|
assert response.json.get("test") == "value"
|
||||||
d = {}
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.parametrize("file_name", ["test.file"])
|
@pytest.mark.parametrize("file_name", ["test.file"])
|
||||||
def test_static_blueprint_name(app: Sanic, static_file_directory, file_name):
|
def test_static_blueprint_name(app: Sanic, static_file_directory, file_name):
|
||||||
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:
|
||||||
current_file_contents = file.read()
|
file.read()
|
||||||
|
|
||||||
bp = Blueprint(name="static", url_prefix="/static", strict_slashes=False)
|
bp = Blueprint(name="static", url_prefix="/static", strict_slashes=False)
|
||||||
|
|
||||||
|
@ -662,16 +653,8 @@ def test_websocket_route(app: Sanic):
|
||||||
|
|
||||||
app.blueprint(bp)
|
app.blueprint(bp)
|
||||||
|
|
||||||
_, response = app.test_client.get(
|
_, response = app.test_client.websocket("/ws/test")
|
||||||
"/ws/test",
|
assert response.opened is True
|
||||||
headers={
|
|
||||||
"Upgrade": "websocket",
|
|
||||||
"Connection": "upgrade",
|
|
||||||
"Sec-WebSocket-Key": "dGhlIHNhbXBsZSBub25jZQ==",
|
|
||||||
"Sec-WebSocket-Version": "13",
|
|
||||||
},
|
|
||||||
)
|
|
||||||
assert response.status == 101
|
|
||||||
assert event.is_set()
|
assert event.is_set()
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -1,12 +1,13 @@
|
||||||
|
from contextlib import contextmanager
|
||||||
from os import environ
|
from os import environ
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
from contextlib import contextmanager
|
|
||||||
from tempfile import TemporaryDirectory
|
from tempfile import TemporaryDirectory
|
||||||
from textwrap import dedent
|
from textwrap import dedent
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
|
|
||||||
from sanic import Sanic
|
from sanic import Sanic
|
||||||
from sanic.config import Config, DEFAULT_CONFIG
|
from sanic.config import DEFAULT_CONFIG, Config
|
||||||
from sanic.exceptions import PyFileError
|
from sanic.exceptions import PyFileError
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -1,8 +1,11 @@
|
||||||
from datetime import datetime, timedelta
|
from datetime import datetime, timedelta
|
||||||
from http.cookies import SimpleCookie
|
from http.cookies import SimpleCookie
|
||||||
from sanic.response import text
|
|
||||||
import pytest
|
import pytest
|
||||||
from sanic.cookies import Cookie, DEFAULT_MAX_AGE
|
|
||||||
|
from sanic.cookies import Cookie
|
||||||
|
from sanic.response import text
|
||||||
|
|
||||||
|
|
||||||
# ------------------------------------------------------------ #
|
# ------------------------------------------------------------ #
|
||||||
# GET
|
# GET
|
||||||
|
@ -100,7 +103,7 @@ def test_cookie_deletion(app):
|
||||||
|
|
||||||
assert int(response_cookies["i_want_to_die"]["max-age"]) == 0
|
assert int(response_cookies["i_want_to_die"]["max-age"]) == 0
|
||||||
with pytest.raises(KeyError):
|
with pytest.raises(KeyError):
|
||||||
_ = response.cookies["i_never_existed"]
|
response.cookies["i_never_existed"]
|
||||||
|
|
||||||
|
|
||||||
def test_cookie_reserved_cookie():
|
def test_cookie_reserved_cookie():
|
||||||
|
@ -135,7 +138,7 @@ def test_cookie_set_same_key(app):
|
||||||
|
|
||||||
request, response = app.test_client.get("/", cookies=cookies)
|
request, response = app.test_client.get("/", cookies=cookies)
|
||||||
assert response.status == 200
|
assert response.status == 200
|
||||||
assert response.cookies["test"].value == "pass"
|
assert response.cookies["test"] == "pass"
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.parametrize("max_age", ["0", 30, 30.0, 30.1, "30", "test"])
|
@pytest.mark.parametrize("max_age", ["0", 30, 30.0, 30.1, "30", "test"])
|
||||||
|
@ -149,19 +152,42 @@ def test_cookie_max_age(app, max_age):
|
||||||
response.cookies["test"]["max-age"] = max_age
|
response.cookies["test"]["max-age"] = max_age
|
||||||
return response
|
return response
|
||||||
|
|
||||||
request, response = app.test_client.get("/", cookies=cookies)
|
request, response = app.test_client.get(
|
||||||
|
"/", cookies=cookies, raw_cookies=True
|
||||||
|
)
|
||||||
assert response.status == 200
|
assert response.status == 200
|
||||||
|
|
||||||
assert response.cookies["test"].value == "pass"
|
cookie = response.cookies.get("test")
|
||||||
|
if (
|
||||||
|
str(max_age).isdigit()
|
||||||
|
and int(max_age) == float(max_age)
|
||||||
|
and int(max_age) != 0
|
||||||
|
):
|
||||||
|
cookie_expires = datetime.utcfromtimestamp(
|
||||||
|
response.raw_cookies["test"].expires
|
||||||
|
).replace(microsecond=0)
|
||||||
|
|
||||||
if str(max_age).isdigit() and int(max_age) == float(max_age):
|
# Grabbing utcnow after the response may lead to it being off slightly.
|
||||||
assert response.cookies["test"]["max-age"] == str(max_age)
|
# Therefore, we 0 out the microseconds, and accept the test if there
|
||||||
|
# is a 1 second difference.
|
||||||
|
expires = datetime.utcnow().replace(microsecond=0) + timedelta(
|
||||||
|
seconds=int(max_age)
|
||||||
|
)
|
||||||
|
|
||||||
|
assert cookie == "pass"
|
||||||
|
assert (
|
||||||
|
cookie_expires == expires
|
||||||
|
or cookie_expires == expires + timedelta(seconds=-1)
|
||||||
|
)
|
||||||
else:
|
else:
|
||||||
assert response.cookies["test"]["max-age"] == str(DEFAULT_MAX_AGE)
|
assert cookie is None
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.parametrize("expires", [datetime.now() + timedelta(seconds=60)])
|
@pytest.mark.parametrize(
|
||||||
|
"expires", [datetime.utcnow() + timedelta(seconds=60)]
|
||||||
|
)
|
||||||
def test_cookie_expires(app, expires):
|
def test_cookie_expires(app, expires):
|
||||||
|
expires = expires.replace(microsecond=0)
|
||||||
cookies = {"test": "wait"}
|
cookies = {"test": "wait"}
|
||||||
|
|
||||||
@app.get("/")
|
@app.get("/")
|
||||||
|
@ -171,15 +197,16 @@ def test_cookie_expires(app, expires):
|
||||||
response.cookies["test"]["expires"] = expires
|
response.cookies["test"]["expires"] = expires
|
||||||
return response
|
return response
|
||||||
|
|
||||||
request, response = app.test_client.get("/", cookies=cookies)
|
request, response = app.test_client.get(
|
||||||
|
"/", cookies=cookies, raw_cookies=True
|
||||||
|
)
|
||||||
|
cookie_expires = datetime.utcfromtimestamp(
|
||||||
|
response.raw_cookies["test"].expires
|
||||||
|
).replace(microsecond=0)
|
||||||
|
|
||||||
assert response.status == 200
|
assert response.status == 200
|
||||||
|
assert response.cookies["test"] == "pass"
|
||||||
assert response.cookies["test"].value == "pass"
|
assert cookie_expires == expires
|
||||||
|
|
||||||
if isinstance(expires, datetime):
|
|
||||||
expires = expires.strftime("%a, %d-%b-%Y %T GMT")
|
|
||||||
|
|
||||||
assert response.cookies["test"]["expires"] == expires
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.parametrize("expires", ["Fri, 21-Dec-2018 15:30:00 GMT"])
|
@pytest.mark.parametrize("expires", ["Fri, 21-Dec-2018 15:30:00 GMT"])
|
||||||
|
|
|
@ -1,7 +1,9 @@
|
||||||
from sanic.response import text
|
|
||||||
from threading import Event
|
|
||||||
import asyncio
|
import asyncio
|
||||||
|
|
||||||
from queue import Queue
|
from queue import Queue
|
||||||
|
from threading import Event
|
||||||
|
|
||||||
|
from sanic.response import text
|
||||||
|
|
||||||
|
|
||||||
def test_create_task(app):
|
def test_create_task(app):
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
from sanic.server import HttpProtocol
|
|
||||||
from sanic.response import text
|
from sanic.response import text
|
||||||
|
from sanic.server import HttpProtocol
|
||||||
|
|
||||||
|
|
||||||
class CustomHttpProtocol(HttpProtocol):
|
class CustomHttpProtocol(HttpProtocol):
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
|
import pytest
|
||||||
|
|
||||||
from sanic.response import text
|
from sanic.response import text
|
||||||
from sanic.router import RouteExists
|
from sanic.router import RouteExists
|
||||||
import pytest
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.parametrize(
|
@pytest.mark.parametrize(
|
||||||
|
|
|
@ -1,10 +1,17 @@
|
||||||
import pytest
|
import pytest
|
||||||
|
|
||||||
from bs4 import BeautifulSoup
|
from bs4 import BeautifulSoup
|
||||||
|
|
||||||
from sanic import Sanic
|
from sanic import Sanic
|
||||||
|
from sanic.exceptions import (
|
||||||
|
Forbidden,
|
||||||
|
InvalidUsage,
|
||||||
|
NotFound,
|
||||||
|
ServerError,
|
||||||
|
Unauthorized,
|
||||||
|
abort,
|
||||||
|
)
|
||||||
from sanic.response import text
|
from sanic.response import text
|
||||||
from sanic.exceptions import InvalidUsage, ServerError, NotFound, Unauthorized
|
|
||||||
from sanic.exceptions import Forbidden, abort
|
|
||||||
|
|
||||||
|
|
||||||
class SanicExceptionTestException(Exception):
|
class SanicExceptionTestException(Exception):
|
||||||
|
|
|
@ -1,9 +1,11 @@
|
||||||
from sanic import Sanic
|
|
||||||
from sanic.response import text
|
|
||||||
from sanic.exceptions import InvalidUsage, ServerError, NotFound
|
|
||||||
from sanic.handlers import ErrorHandler
|
|
||||||
from bs4 import BeautifulSoup
|
from bs4 import BeautifulSoup
|
||||||
|
|
||||||
|
from sanic import Sanic
|
||||||
|
from sanic.exceptions import InvalidUsage, NotFound, ServerError
|
||||||
|
from sanic.handlers import ErrorHandler
|
||||||
|
from sanic.response import text
|
||||||
|
|
||||||
|
|
||||||
exception_handler_app = Sanic("test_exception_handler")
|
exception_handler_app = Sanic("test_exception_handler")
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -1,39 +1,143 @@
|
||||||
from json import JSONDecodeError
|
|
||||||
from sanic import Sanic
|
|
||||||
import asyncio
|
import asyncio
|
||||||
|
import functools
|
||||||
|
import socket
|
||||||
|
|
||||||
from asyncio import sleep as aio_sleep
|
from asyncio import sleep as aio_sleep
|
||||||
|
from http.client import _encode
|
||||||
|
from json import JSONDecodeError
|
||||||
|
|
||||||
|
import httpcore
|
||||||
|
import requests_async as requests
|
||||||
|
|
||||||
|
from httpcore import PoolTimeout
|
||||||
|
|
||||||
|
from sanic import Sanic, server
|
||||||
from sanic.response import text
|
from sanic.response import text
|
||||||
from sanic import server
|
from sanic.testing import HOST, PORT, SanicTestClient
|
||||||
import aiohttp
|
|
||||||
from aiohttp import TCPConnector
|
|
||||||
from sanic.testing import SanicTestClient, HOST, PORT
|
# import traceback
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
CONFIG_FOR_TESTS = {"KEEP_ALIVE_TIMEOUT": 2, "KEEP_ALIVE": True}
|
CONFIG_FOR_TESTS = {"KEEP_ALIVE_TIMEOUT": 2, "KEEP_ALIVE": True}
|
||||||
|
|
||||||
|
old_conn = None
|
||||||
|
|
||||||
class ReuseableTCPConnector(TCPConnector):
|
|
||||||
def __init__(self, *args, **kwargs):
|
|
||||||
super(ReuseableTCPConnector, self).__init__(*args, **kwargs)
|
|
||||||
self.old_proto = None
|
|
||||||
|
|
||||||
async def connect(self, req, *args, **kwargs):
|
class ReusableSanicConnectionPool(httpcore.ConnectionPool):
|
||||||
new_conn = await super(ReuseableTCPConnector, self).connect(
|
async def acquire_connection(self, url, ssl, timeout):
|
||||||
req, *args, **kwargs
|
global old_conn
|
||||||
)
|
if timeout.connect_timeout and not timeout.pool_timeout:
|
||||||
if self.old_proto is not None:
|
timeout.pool_timeout = timeout.connect_timeout
|
||||||
if self.old_proto != new_conn._protocol:
|
key = (url.scheme, url.hostname, url.port, ssl, timeout)
|
||||||
|
try:
|
||||||
|
connection = self._keepalive_connections[key].pop()
|
||||||
|
if not self._keepalive_connections[key]:
|
||||||
|
del self._keepalive_connections[key]
|
||||||
|
self.num_keepalive_connections -= 1
|
||||||
|
self.num_active_connections += 1
|
||||||
|
|
||||||
|
except (KeyError, IndexError):
|
||||||
|
ssl_context = await self.get_ssl_context(url, ssl)
|
||||||
|
try:
|
||||||
|
await asyncio.wait_for(
|
||||||
|
self._max_connections.acquire(), timeout.pool_timeout
|
||||||
|
)
|
||||||
|
except asyncio.TimeoutError:
|
||||||
|
raise PoolTimeout()
|
||||||
|
release = functools.partial(self.release_connection, key=key)
|
||||||
|
connection = httpcore.connections.Connection(
|
||||||
|
timeout=timeout, on_release=release
|
||||||
|
)
|
||||||
|
self.num_active_connections += 1
|
||||||
|
await connection.open(url.hostname, url.port, ssl=ssl_context)
|
||||||
|
|
||||||
|
if old_conn is not None:
|
||||||
|
if old_conn != connection:
|
||||||
raise RuntimeError(
|
raise RuntimeError(
|
||||||
"We got a new connection, wanted the same one!"
|
"We got a new connection, wanted the same one!"
|
||||||
)
|
)
|
||||||
print(new_conn.__dict__)
|
old_conn = connection
|
||||||
self.old_proto = new_conn._protocol
|
|
||||||
return new_conn
|
return connection
|
||||||
|
|
||||||
|
|
||||||
|
class ReusableSanicAdapter(requests.adapters.HTTPAdapter):
|
||||||
|
def __init__(self):
|
||||||
|
self.pool = ReusableSanicConnectionPool()
|
||||||
|
|
||||||
|
async def send(
|
||||||
|
self,
|
||||||
|
request,
|
||||||
|
stream=False,
|
||||||
|
timeout=None,
|
||||||
|
verify=True,
|
||||||
|
cert=None,
|
||||||
|
proxies=None,
|
||||||
|
):
|
||||||
|
|
||||||
|
method = request.method
|
||||||
|
url = request.url
|
||||||
|
headers = [
|
||||||
|
(_encode(k), _encode(v)) for k, v in request.headers.items()
|
||||||
|
]
|
||||||
|
|
||||||
|
if not request.body:
|
||||||
|
body = b""
|
||||||
|
elif isinstance(request.body, str):
|
||||||
|
body = _encode(request.body)
|
||||||
|
else:
|
||||||
|
body = request.body
|
||||||
|
|
||||||
|
if isinstance(timeout, tuple):
|
||||||
|
timeout_kwargs = {
|
||||||
|
"connect_timeout": timeout[0],
|
||||||
|
"read_timeout": timeout[1],
|
||||||
|
}
|
||||||
|
else:
|
||||||
|
timeout_kwargs = {
|
||||||
|
"connect_timeout": timeout,
|
||||||
|
"read_timeout": timeout,
|
||||||
|
}
|
||||||
|
|
||||||
|
ssl = httpcore.SSLConfig(cert=cert, verify=verify)
|
||||||
|
timeout = httpcore.TimeoutConfig(**timeout_kwargs)
|
||||||
|
|
||||||
|
try:
|
||||||
|
response = await self.pool.request(
|
||||||
|
method,
|
||||||
|
url,
|
||||||
|
headers=headers,
|
||||||
|
body=body,
|
||||||
|
stream=stream,
|
||||||
|
ssl=ssl,
|
||||||
|
timeout=timeout,
|
||||||
|
)
|
||||||
|
except (httpcore.BadResponse, socket.error) as err:
|
||||||
|
raise ConnectionError(err, request=request)
|
||||||
|
except httpcore.ConnectTimeout as err:
|
||||||
|
raise requests.exceptions.ConnectTimeout(err, request=request)
|
||||||
|
except httpcore.ReadTimeout as err:
|
||||||
|
raise requests.exceptions.ReadTimeout(err, request=request)
|
||||||
|
|
||||||
|
return self.build_response(request, response)
|
||||||
|
|
||||||
|
|
||||||
|
class ResusableSanicSession(requests.Session):
|
||||||
|
def __init__(self, *args, **kwargs) -> None:
|
||||||
|
super().__init__(*args, **kwargs)
|
||||||
|
adapter = ReusableSanicAdapter()
|
||||||
|
self.mount("http://", adapter)
|
||||||
|
self.mount("https://", adapter)
|
||||||
|
|
||||||
|
|
||||||
class ReuseableSanicTestClient(SanicTestClient):
|
class ReuseableSanicTestClient(SanicTestClient):
|
||||||
def __init__(self, app, loop=None):
|
def __init__(self, app, loop=None):
|
||||||
super(ReuseableSanicTestClient, self).__init__(app)
|
super().__init__(app)
|
||||||
if loop is None:
|
if loop is None:
|
||||||
loop = asyncio.get_event_loop()
|
loop = asyncio.get_event_loop()
|
||||||
self._loop = loop
|
self._loop = loop
|
||||||
|
@ -51,12 +155,11 @@ class ReuseableSanicTestClient(SanicTestClient):
|
||||||
debug=False,
|
debug=False,
|
||||||
server_kwargs={"return_asyncio_server": True},
|
server_kwargs={"return_asyncio_server": True},
|
||||||
*request_args,
|
*request_args,
|
||||||
**request_kwargs
|
**request_kwargs,
|
||||||
):
|
):
|
||||||
loop = self._loop
|
loop = self._loop
|
||||||
results = [None, None]
|
results = [None, None]
|
||||||
exceptions = []
|
exceptions = []
|
||||||
do_kill_server = request_kwargs.pop("end_server", False)
|
|
||||||
if gather_request:
|
if gather_request:
|
||||||
|
|
||||||
def _collect_request(request):
|
def _collect_request(request):
|
||||||
|
@ -65,21 +168,27 @@ class ReuseableSanicTestClient(SanicTestClient):
|
||||||
|
|
||||||
self.app.request_middleware.appendleft(_collect_request)
|
self.app.request_middleware.appendleft(_collect_request)
|
||||||
|
|
||||||
|
if uri.startswith(
|
||||||
|
("http:", "https:", "ftp:", "ftps://", "//", "ws:", "wss:")
|
||||||
|
):
|
||||||
|
url = uri
|
||||||
|
else:
|
||||||
|
uri = uri if uri.startswith("/") else "/{uri}".format(uri=uri)
|
||||||
|
scheme = "http"
|
||||||
|
url = "{scheme}://{host}:{port}{uri}".format(
|
||||||
|
scheme=scheme, host=HOST, port=PORT, uri=uri
|
||||||
|
)
|
||||||
|
|
||||||
@self.app.listener("after_server_start")
|
@self.app.listener("after_server_start")
|
||||||
async def _collect_response(loop):
|
async def _collect_response(loop):
|
||||||
try:
|
try:
|
||||||
if do_kill_server:
|
|
||||||
request_kwargs["end_session"] = True
|
|
||||||
response = await self._local_request(
|
response = await self._local_request(
|
||||||
method, uri, *request_args, **request_kwargs
|
method, url, *request_args, **request_kwargs
|
||||||
)
|
)
|
||||||
results[-1] = response
|
results[-1] = response
|
||||||
except Exception as e2:
|
except Exception as e2:
|
||||||
import traceback
|
# traceback.print_tb(e2.__traceback__)
|
||||||
|
|
||||||
traceback.print_tb(e2.__traceback__)
|
|
||||||
exceptions.append(e2)
|
exceptions.append(e2)
|
||||||
# Don't stop here! self.app.stop()
|
|
||||||
|
|
||||||
if self._server is not None:
|
if self._server is not None:
|
||||||
_server = self._server
|
_server = self._server
|
||||||
|
@ -94,27 +203,14 @@ class ReuseableSanicTestClient(SanicTestClient):
|
||||||
|
|
||||||
try:
|
try:
|
||||||
loop._stopping = False
|
loop._stopping = False
|
||||||
http_server = loop.run_until_complete(_server_co)
|
_server = loop.run_until_complete(_server_co)
|
||||||
except Exception as e1:
|
except Exception as e1:
|
||||||
import traceback
|
# traceback.print_tb(e1.__traceback__)
|
||||||
|
|
||||||
traceback.print_tb(e1.__traceback__)
|
|
||||||
raise e1
|
raise e1
|
||||||
self._server = _server = http_server
|
self._server = _server
|
||||||
server.trigger_events(self.app.listeners["after_server_start"], loop)
|
server.trigger_events(self.app.listeners["after_server_start"], loop)
|
||||||
self.app.listeners["after_server_start"].pop()
|
self.app.listeners["after_server_start"].pop()
|
||||||
|
|
||||||
if do_kill_server:
|
|
||||||
try:
|
|
||||||
_server.close()
|
|
||||||
self._server = None
|
|
||||||
loop.run_until_complete(_server.wait_closed())
|
|
||||||
self.app.stop()
|
|
||||||
except Exception as e3:
|
|
||||||
import traceback
|
|
||||||
|
|
||||||
traceback.print_tb(e3.__traceback__)
|
|
||||||
exceptions.append(e3)
|
|
||||||
if exceptions:
|
if exceptions:
|
||||||
raise ValueError("Exception during request: {}".format(exceptions))
|
raise ValueError("Exception during request: {}".format(exceptions))
|
||||||
|
|
||||||
|
@ -137,59 +233,61 @@ class ReuseableSanicTestClient(SanicTestClient):
|
||||||
"Request object expected, got ({})".format(results)
|
"Request object expected, got ({})".format(results)
|
||||||
)
|
)
|
||||||
|
|
||||||
|
def kill_server(self):
|
||||||
|
try:
|
||||||
|
if self._server:
|
||||||
|
self._server.close()
|
||||||
|
self._loop.run_until_complete(self._server.wait_closed())
|
||||||
|
self._server = None
|
||||||
|
self.app.stop()
|
||||||
|
|
||||||
|
if self._session:
|
||||||
|
self._loop.run_until_complete(self._session.close())
|
||||||
|
self._session = None
|
||||||
|
|
||||||
|
except Exception as e3:
|
||||||
|
raise e3
|
||||||
|
|
||||||
# Copied from SanicTestClient, but with some changes to reuse the
|
# Copied from SanicTestClient, but with some changes to reuse the
|
||||||
# same TCPConnection and the sane ClientSession more than once.
|
# same TCPConnection and the sane ClientSession more than once.
|
||||||
# Note, you cannot use the same session if you are in a _different_
|
# Note, you cannot use the same session if you are in a _different_
|
||||||
# loop, so the changes above are required too.
|
# loop, so the changes above are required too.
|
||||||
async def _local_request(self, method, uri, cookies=None, *args, **kwargs):
|
async def _local_request(self, method, url, *args, **kwargs):
|
||||||
|
raw_cookies = kwargs.pop("raw_cookies", None)
|
||||||
request_keepalive = kwargs.pop(
|
request_keepalive = kwargs.pop(
|
||||||
"request_keepalive", CONFIG_FOR_TESTS["KEEP_ALIVE_TIMEOUT"]
|
"request_keepalive", CONFIG_FOR_TESTS["KEEP_ALIVE_TIMEOUT"]
|
||||||
)
|
)
|
||||||
if uri.startswith(("http:", "https:", "ftp:", "ftps://" "//")):
|
|
||||||
url = uri
|
|
||||||
else:
|
|
||||||
url = "http://{host}:{port}{uri}".format(
|
|
||||||
host=HOST, port=self.port, uri=uri
|
|
||||||
)
|
|
||||||
do_kill_session = kwargs.pop("end_session", False)
|
|
||||||
if self._session:
|
if self._session:
|
||||||
session = self._session
|
_session = self._session
|
||||||
else:
|
else:
|
||||||
if self._tcp_connector:
|
_session = ResusableSanicSession()
|
||||||
conn = self._tcp_connector
|
self._session = _session
|
||||||
else:
|
async with _session as session:
|
||||||
conn = ReuseableTCPConnector(
|
try:
|
||||||
ssl=False,
|
response = await getattr(session, method.lower())(
|
||||||
loop=self._loop,
|
url,
|
||||||
keepalive_timeout=request_keepalive,
|
verify=False,
|
||||||
|
timeout=request_keepalive,
|
||||||
|
*args,
|
||||||
|
**kwargs,
|
||||||
)
|
)
|
||||||
self._tcp_connector = conn
|
except NameError:
|
||||||
session = aiohttp.ClientSession(
|
raise Exception(response.status_code)
|
||||||
cookies=cookies, connector=conn, loop=self._loop
|
|
||||||
)
|
|
||||||
self._session = session
|
|
||||||
|
|
||||||
async with getattr(session, method.lower())(
|
|
||||||
url, *args, **kwargs
|
|
||||||
) as response:
|
|
||||||
try:
|
|
||||||
response.text = await response.text()
|
|
||||||
except UnicodeDecodeError:
|
|
||||||
response.text = None
|
|
||||||
|
|
||||||
try:
|
try:
|
||||||
response.json = await response.json()
|
response.json = response.json()
|
||||||
except (
|
except (JSONDecodeError, UnicodeDecodeError):
|
||||||
JSONDecodeError,
|
|
||||||
UnicodeDecodeError,
|
|
||||||
aiohttp.ClientResponseError,
|
|
||||||
):
|
|
||||||
response.json = None
|
response.json = None
|
||||||
|
|
||||||
response.body = await response.read()
|
response.body = await response.read()
|
||||||
if do_kill_session:
|
response.status = response.status_code
|
||||||
await session.close()
|
response.content_type = response.headers.get("content-type")
|
||||||
self._session = None
|
|
||||||
|
if raw_cookies:
|
||||||
|
response.raw_cookies = {}
|
||||||
|
for cookie in response.cookies:
|
||||||
|
response.raw_cookies[cookie.name] = cookie
|
||||||
|
|
||||||
return response
|
return response
|
||||||
|
|
||||||
|
|
||||||
|
@ -229,9 +327,10 @@ def test_keep_alive_timeout_reuse():
|
||||||
assert response.status == 200
|
assert response.status == 200
|
||||||
assert response.text == "OK"
|
assert response.text == "OK"
|
||||||
loop.run_until_complete(aio_sleep(1))
|
loop.run_until_complete(aio_sleep(1))
|
||||||
request, response = client.get("/1", end_server=True)
|
request, response = client.get("/1")
|
||||||
assert response.status == 200
|
assert response.status == 200
|
||||||
assert response.text == "OK"
|
assert response.text == "OK"
|
||||||
|
client.kill_server()
|
||||||
|
|
||||||
|
|
||||||
def test_keep_alive_client_timeout():
|
def test_keep_alive_client_timeout():
|
||||||
|
@ -241,20 +340,21 @@ def test_keep_alive_client_timeout():
|
||||||
asyncio.set_event_loop(loop)
|
asyncio.set_event_loop(loop)
|
||||||
client = ReuseableSanicTestClient(keep_alive_app_client_timeout, loop)
|
client = ReuseableSanicTestClient(keep_alive_app_client_timeout, loop)
|
||||||
headers = {"Connection": "keep-alive"}
|
headers = {"Connection": "keep-alive"}
|
||||||
request, response = client.get("/1", headers=headers, request_keepalive=1)
|
|
||||||
assert response.status == 200
|
|
||||||
assert response.text == "OK"
|
|
||||||
loop.run_until_complete(aio_sleep(2))
|
|
||||||
exception = None
|
|
||||||
try:
|
try:
|
||||||
request, response = client.get(
|
request, response = client.get(
|
||||||
"/1", end_server=True, request_keepalive=1
|
"/1", headers=headers, request_keepalive=1
|
||||||
)
|
)
|
||||||
|
assert response.status == 200
|
||||||
|
assert response.text == "OK"
|
||||||
|
loop.run_until_complete(aio_sleep(2))
|
||||||
|
exception = None
|
||||||
|
request, response = client.get("/1", request_keepalive=1)
|
||||||
except ValueError as e:
|
except ValueError as e:
|
||||||
exception = e
|
exception = e
|
||||||
assert exception is not None
|
assert exception is not None
|
||||||
assert isinstance(exception, ValueError)
|
assert isinstance(exception, ValueError)
|
||||||
assert "got a new connection" in exception.args[0]
|
assert "got a new connection" in exception.args[0]
|
||||||
|
client.kill_server()
|
||||||
|
|
||||||
|
|
||||||
def test_keep_alive_server_timeout():
|
def test_keep_alive_server_timeout():
|
||||||
|
@ -266,15 +366,15 @@ def test_keep_alive_server_timeout():
|
||||||
asyncio.set_event_loop(loop)
|
asyncio.set_event_loop(loop)
|
||||||
client = ReuseableSanicTestClient(keep_alive_app_server_timeout, loop)
|
client = ReuseableSanicTestClient(keep_alive_app_server_timeout, loop)
|
||||||
headers = {"Connection": "keep-alive"}
|
headers = {"Connection": "keep-alive"}
|
||||||
request, response = client.get("/1", headers=headers, request_keepalive=60)
|
|
||||||
assert response.status == 200
|
|
||||||
assert response.text == "OK"
|
|
||||||
loop.run_until_complete(aio_sleep(3))
|
|
||||||
exception = None
|
|
||||||
try:
|
try:
|
||||||
request, response = client.get(
|
request, response = client.get(
|
||||||
"/1", request_keepalive=60, end_server=True
|
"/1", headers=headers, request_keepalive=60
|
||||||
)
|
)
|
||||||
|
assert response.status == 200
|
||||||
|
assert response.text == "OK"
|
||||||
|
loop.run_until_complete(aio_sleep(3))
|
||||||
|
exception = None
|
||||||
|
request, response = client.get("/1", request_keepalive=60)
|
||||||
except ValueError as e:
|
except ValueError as e:
|
||||||
exception = e
|
exception = e
|
||||||
assert exception is not None
|
assert exception is not None
|
||||||
|
@ -283,3 +383,4 @@ def test_keep_alive_server_timeout():
|
||||||
"Connection reset" in exception.args[0]
|
"Connection reset" in exception.args[0]
|
||||||
or "got a new connection" in exception.args[0]
|
or "got a new connection" in exception.args[0]
|
||||||
)
|
)
|
||||||
|
client.kill_server()
|
||||||
|
|
|
@ -1,17 +1,17 @@
|
||||||
import uuid
|
|
||||||
import logging
|
import logging
|
||||||
|
import uuid
|
||||||
|
|
||||||
from io import StringIO
|
|
||||||
from importlib import reload
|
from importlib import reload
|
||||||
|
from io import StringIO
|
||||||
import pytest
|
|
||||||
from unittest.mock import Mock
|
from unittest.mock import Mock
|
||||||
|
|
||||||
|
import pytest
|
||||||
|
|
||||||
import sanic
|
import sanic
|
||||||
from sanic.response import text
|
|
||||||
from sanic.log import LOGGING_CONFIG_DEFAULTS
|
|
||||||
from sanic import Sanic
|
from sanic import Sanic
|
||||||
from sanic.log import logger
|
from sanic.log import LOGGING_CONFIG_DEFAULTS, logger
|
||||||
|
from sanic.response import text
|
||||||
|
|
||||||
|
|
||||||
logging_format = """module: %(module)s; \
|
logging_format = """module: %(module)s; \
|
||||||
|
|
|
@ -1,8 +1,9 @@
|
||||||
import logging
|
|
||||||
import asyncio
|
import asyncio
|
||||||
|
import logging
|
||||||
|
|
||||||
from sanic.config import BASE_LOGO
|
from sanic.config import BASE_LOGO
|
||||||
|
|
||||||
|
|
||||||
try:
|
try:
|
||||||
import uvloop # noqa
|
import uvloop # noqa
|
||||||
|
|
||||||
|
|
|
@ -1,10 +1,12 @@
|
||||||
import logging
|
import logging
|
||||||
|
|
||||||
from asyncio import CancelledError
|
from asyncio import CancelledError
|
||||||
|
|
||||||
from sanic.exceptions import NotFound
|
from sanic.exceptions import NotFound
|
||||||
from sanic.request import Request
|
from sanic.request import Request
|
||||||
from sanic.response import HTTPResponse, text
|
from sanic.response import HTTPResponse, text
|
||||||
|
|
||||||
|
|
||||||
# ------------------------------------------------------------ #
|
# ------------------------------------------------------------ #
|
||||||
# GET
|
# GET
|
||||||
# ------------------------------------------------------------ #
|
# ------------------------------------------------------------ #
|
||||||
|
|
|
@ -1,11 +1,12 @@
|
||||||
import multiprocessing
|
import multiprocessing
|
||||||
|
import pickle
|
||||||
import random
|
import random
|
||||||
import signal
|
import signal
|
||||||
import pickle
|
|
||||||
import pytest
|
import pytest
|
||||||
|
|
||||||
from sanic.testing import HOST, PORT
|
|
||||||
from sanic.response import text
|
from sanic.response import text
|
||||||
|
from sanic.testing import HOST, PORT
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.skipif(
|
@pytest.mark.skipif(
|
||||||
|
|
|
@ -2,12 +2,13 @@
|
||||||
# -*- coding: utf-8 -*-
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
import asyncio
|
import asyncio
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
|
|
||||||
from sanic.blueprints import Blueprint
|
from sanic.blueprints import Blueprint
|
||||||
from sanic.response import text
|
|
||||||
from sanic.exceptions import URLBuildError
|
|
||||||
from sanic.constants import HTTP_METHODS
|
from sanic.constants import HTTP_METHODS
|
||||||
|
from sanic.exceptions import URLBuildError
|
||||||
|
from sanic.response import text
|
||||||
|
|
||||||
|
|
||||||
# ------------------------------------------------------------ #
|
# ------------------------------------------------------------ #
|
||||||
|
|
|
@ -1,7 +1,8 @@
|
||||||
import pytest
|
|
||||||
from urllib.parse import quote
|
from urllib.parse import quote
|
||||||
|
|
||||||
from sanic.response import text, redirect
|
import pytest
|
||||||
|
|
||||||
|
from sanic.response import redirect, text
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture
|
@pytest.fixture
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
import asyncio
|
import asyncio
|
||||||
import contextlib
|
import contextlib
|
||||||
|
|
||||||
from sanic.response import text, stream
|
from sanic.response import stream, text
|
||||||
|
|
||||||
|
|
||||||
async def test_request_cancel_when_connection_lost(loop, app, test_client):
|
async def test_request_cancel_when_connection_lost(loop, app, test_client):
|
||||||
|
|
|
@ -2,6 +2,7 @@ import random
|
||||||
|
|
||||||
from sanic.response import json
|
from sanic.response import json
|
||||||
|
|
||||||
|
|
||||||
try:
|
try:
|
||||||
from ujson import loads
|
from ujson import loads
|
||||||
except ImportError:
|
except ImportError:
|
||||||
|
|
|
@ -1,10 +1,8 @@
|
||||||
import asyncio
|
|
||||||
from sanic.blueprints import Blueprint
|
from sanic.blueprints import Blueprint
|
||||||
from sanic.views import CompositionView
|
|
||||||
from sanic.views import HTTPMethodView
|
|
||||||
from sanic.views import stream as stream_decorator
|
|
||||||
from sanic.response import stream, text
|
|
||||||
from sanic.request import StreamBuffer
|
from sanic.request import StreamBuffer
|
||||||
|
from sanic.response import stream, text
|
||||||
|
from sanic.views import CompositionView, HTTPMethodView
|
||||||
|
from sanic.views import stream as stream_decorator
|
||||||
|
|
||||||
|
|
||||||
data = "abc" * 10000000
|
data = "abc" * 10000000
|
||||||
|
|
|
@ -1,183 +1,73 @@
|
||||||
from json import JSONDecodeError
|
import asyncio
|
||||||
|
|
||||||
|
import httpcore
|
||||||
|
import requests_async as requests
|
||||||
|
|
||||||
from sanic import Sanic
|
from sanic import Sanic
|
||||||
import asyncio
|
|
||||||
from sanic.response import text
|
from sanic.response import text
|
||||||
import aiohttp
|
from sanic.testing import SanicTestClient
|
||||||
from aiohttp import TCPConnector
|
|
||||||
from sanic.testing import SanicTestClient, HOST
|
|
||||||
|
|
||||||
try:
|
|
||||||
try:
|
|
||||||
# direct use
|
|
||||||
import packaging
|
|
||||||
|
|
||||||
version = packaging.version
|
|
||||||
except (ImportError, AttributeError):
|
|
||||||
# setuptools v39.0 and above.
|
|
||||||
try:
|
|
||||||
from setuptools.extern import packaging
|
|
||||||
except ImportError:
|
|
||||||
# Before setuptools v39.0
|
|
||||||
from pkg_resources.extern import packaging
|
|
||||||
version = packaging.version
|
|
||||||
except ImportError:
|
|
||||||
raise RuntimeError("The 'packaging' library is missing.")
|
|
||||||
|
|
||||||
|
|
||||||
aiohttp_version = version.parse(aiohttp.__version__)
|
class DelayableSanicConnectionPool(httpcore.ConnectionPool):
|
||||||
|
def __init__(self, request_delay=None, *args, **kwargs):
|
||||||
|
self._request_delay = request_delay
|
||||||
|
super().__init__(*args, **kwargs)
|
||||||
|
|
||||||
|
async def request(
|
||||||
|
self,
|
||||||
|
method,
|
||||||
|
url,
|
||||||
|
headers=(),
|
||||||
|
body=b"",
|
||||||
|
stream=False,
|
||||||
|
ssl=None,
|
||||||
|
timeout=None,
|
||||||
|
):
|
||||||
|
if ssl is None:
|
||||||
|
ssl = self.ssl_config
|
||||||
|
if timeout is None:
|
||||||
|
timeout = self.timeout
|
||||||
|
|
||||||
class DelayableTCPConnector(TCPConnector):
|
parsed_url = httpcore.URL(url)
|
||||||
class RequestContextManager(object):
|
request = httpcore.Request(
|
||||||
def __new__(cls, req, delay):
|
method, parsed_url, headers=headers, body=body
|
||||||
cls = super(
|
|
||||||
DelayableTCPConnector.RequestContextManager, cls
|
|
||||||
).__new__(cls)
|
|
||||||
cls.req = req
|
|
||||||
cls.send_task = None
|
|
||||||
cls.resp = None
|
|
||||||
cls.orig_send = getattr(req, "send")
|
|
||||||
cls.orig_start = None
|
|
||||||
cls.delay = delay
|
|
||||||
cls._acting_as = req
|
|
||||||
return cls
|
|
||||||
|
|
||||||
def __getattr__(self, item):
|
|
||||||
acting_as = self._acting_as
|
|
||||||
return getattr(acting_as, item)
|
|
||||||
|
|
||||||
async def start(self, connection, read_until_eof=False):
|
|
||||||
if self.send_task is None:
|
|
||||||
raise RuntimeError("do a send() before you do a start()")
|
|
||||||
resp = await self.send_task
|
|
||||||
self.send_task = None
|
|
||||||
self.resp = resp
|
|
||||||
self._acting_as = self.resp
|
|
||||||
self.orig_start = getattr(resp, "start")
|
|
||||||
|
|
||||||
try:
|
|
||||||
if aiohttp_version >= version.parse("3.3.0"):
|
|
||||||
ret = await self.orig_start(connection)
|
|
||||||
else:
|
|
||||||
ret = await self.orig_start(connection, read_until_eof)
|
|
||||||
except Exception as e:
|
|
||||||
raise e
|
|
||||||
return ret
|
|
||||||
|
|
||||||
def close(self):
|
|
||||||
if self.resp is not None:
|
|
||||||
self.resp.close()
|
|
||||||
if self.send_task is not None:
|
|
||||||
self.send_task.cancel()
|
|
||||||
|
|
||||||
async def delayed_send(self, *args, **kwargs):
|
|
||||||
req = self.req
|
|
||||||
if self.delay and self.delay > 0:
|
|
||||||
# sync_sleep(self.delay)
|
|
||||||
await asyncio.sleep(self.delay)
|
|
||||||
t = req.loop.time()
|
|
||||||
print("sending at {}".format(t), flush=True)
|
|
||||||
next(iter(args)) # first arg is connection
|
|
||||||
|
|
||||||
try:
|
|
||||||
return await self.orig_send(*args, **kwargs)
|
|
||||||
except Exception as e:
|
|
||||||
if aiohttp_version < version.parse("3.1.0"):
|
|
||||||
return aiohttp.ClientResponse(req.method, req.url)
|
|
||||||
kw = dict(
|
|
||||||
writer=None,
|
|
||||||
continue100=None,
|
|
||||||
timer=None,
|
|
||||||
request_info=None,
|
|
||||||
traces=[],
|
|
||||||
loop=req.loop,
|
|
||||||
session=None,
|
|
||||||
)
|
|
||||||
if aiohttp_version < version.parse("3.3.0"):
|
|
||||||
kw["auto_decompress"] = None
|
|
||||||
return aiohttp.ClientResponse(req.method, req.url, **kw)
|
|
||||||
|
|
||||||
def _send(self, *args, **kwargs):
|
|
||||||
gen = self.delayed_send(*args, **kwargs)
|
|
||||||
task = self.req.loop.create_task(gen)
|
|
||||||
self.send_task = task
|
|
||||||
self._acting_as = task
|
|
||||||
return self
|
|
||||||
|
|
||||||
if aiohttp_version >= version.parse("3.1.0"):
|
|
||||||
# aiohttp changed the request.send method to async
|
|
||||||
async def send(self, *args, **kwargs):
|
|
||||||
return self._send(*args, **kwargs)
|
|
||||||
|
|
||||||
else:
|
|
||||||
send = _send
|
|
||||||
|
|
||||||
def __init__(self, *args, **kwargs):
|
|
||||||
_post_connect_delay = kwargs.pop("post_connect_delay", 0)
|
|
||||||
_pre_request_delay = kwargs.pop("pre_request_delay", 0)
|
|
||||||
super(DelayableTCPConnector, self).__init__(*args, **kwargs)
|
|
||||||
self._post_connect_delay = _post_connect_delay
|
|
||||||
self._pre_request_delay = _pre_request_delay
|
|
||||||
|
|
||||||
async def connect(self, req, *args, **kwargs):
|
|
||||||
d_req = DelayableTCPConnector.RequestContextManager(
|
|
||||||
req, self._pre_request_delay
|
|
||||||
)
|
)
|
||||||
conn = await super(DelayableTCPConnector, self).connect(
|
connection = await self.acquire_connection(
|
||||||
req, *args, **kwargs
|
parsed_url, ssl=ssl, timeout=timeout
|
||||||
)
|
)
|
||||||
if self._post_connect_delay and self._post_connect_delay > 0:
|
if self._request_delay:
|
||||||
await asyncio.sleep(self._post_connect_delay, loop=self._loop)
|
print(f"\t>> Sleeping ({self._request_delay})")
|
||||||
req.send = d_req.send
|
await asyncio.sleep(self._request_delay)
|
||||||
t = req.loop.time()
|
response = await connection.send(request)
|
||||||
print("Connected at {}".format(t), flush=True)
|
if not stream:
|
||||||
return conn
|
try:
|
||||||
|
await response.read()
|
||||||
|
finally:
|
||||||
|
await response.close()
|
||||||
|
return response
|
||||||
|
|
||||||
|
|
||||||
|
class DelayableSanicAdapter(requests.adapters.HTTPAdapter):
|
||||||
|
def __init__(self, request_delay=None):
|
||||||
|
self.pool = DelayableSanicConnectionPool(request_delay=request_delay)
|
||||||
|
|
||||||
|
|
||||||
|
class DelayableSanicSession(requests.Session):
|
||||||
|
def __init__(self, request_delay=None, *args, **kwargs) -> None:
|
||||||
|
super().__init__(*args, **kwargs)
|
||||||
|
adapter = DelayableSanicAdapter(request_delay=request_delay)
|
||||||
|
self.mount("http://", adapter)
|
||||||
|
self.mount("https://", adapter)
|
||||||
|
|
||||||
|
|
||||||
class DelayableSanicTestClient(SanicTestClient):
|
class DelayableSanicTestClient(SanicTestClient):
|
||||||
def __init__(self, app, loop, request_delay=1):
|
def __init__(self, app, request_delay=None):
|
||||||
super(DelayableSanicTestClient, self).__init__(app)
|
super().__init__(app)
|
||||||
self._request_delay = request_delay
|
self._request_delay = request_delay
|
||||||
self._loop = None
|
self._loop = None
|
||||||
|
|
||||||
async def _local_request(self, method, uri, cookies=None, *args, **kwargs):
|
def get_new_session(self):
|
||||||
if self._loop is None:
|
return DelayableSanicSession(request_delay=self._request_delay)
|
||||||
self._loop = asyncio.get_event_loop()
|
|
||||||
if uri.startswith(("http:", "https:", "ftp:", "ftps://" "//")):
|
|
||||||
url = uri
|
|
||||||
else:
|
|
||||||
url = "http://{host}:{port}{uri}".format(
|
|
||||||
host=HOST, port=self.port, uri=uri
|
|
||||||
)
|
|
||||||
conn = DelayableTCPConnector(
|
|
||||||
pre_request_delay=self._request_delay, ssl=False, loop=self._loop
|
|
||||||
)
|
|
||||||
async with aiohttp.ClientSession(
|
|
||||||
cookies=cookies, connector=conn, loop=self._loop
|
|
||||||
) as session:
|
|
||||||
# Insert a delay after creating the connection
|
|
||||||
# But before sending the request.
|
|
||||||
|
|
||||||
async with getattr(session, method.lower())(
|
|
||||||
url, *args, **kwargs
|
|
||||||
) as response:
|
|
||||||
try:
|
|
||||||
response.text = await response.text()
|
|
||||||
except UnicodeDecodeError:
|
|
||||||
response.text = None
|
|
||||||
|
|
||||||
try:
|
|
||||||
response.json = await response.json()
|
|
||||||
except (
|
|
||||||
JSONDecodeError,
|
|
||||||
UnicodeDecodeError,
|
|
||||||
aiohttp.ClientResponseError,
|
|
||||||
):
|
|
||||||
response.json = None
|
|
||||||
|
|
||||||
response.body = await response.read()
|
|
||||||
return response
|
|
||||||
|
|
||||||
|
|
||||||
request_timeout_default_app = Sanic("test_request_timeout_default")
|
request_timeout_default_app = Sanic("test_request_timeout_default")
|
||||||
|
@ -202,14 +92,14 @@ async def ws_handler1(request, ws):
|
||||||
|
|
||||||
|
|
||||||
def test_default_server_error_request_timeout():
|
def test_default_server_error_request_timeout():
|
||||||
client = DelayableSanicTestClient(request_timeout_default_app, None, 2)
|
client = DelayableSanicTestClient(request_timeout_default_app, 2)
|
||||||
request, response = client.get("/1")
|
request, response = client.get("/1")
|
||||||
assert response.status == 408
|
assert response.status == 408
|
||||||
assert response.text == "Error: Request Timeout"
|
assert response.text == "Error: Request Timeout"
|
||||||
|
|
||||||
|
|
||||||
def test_default_server_error_request_dont_timeout():
|
def test_default_server_error_request_dont_timeout():
|
||||||
client = DelayableSanicTestClient(request_no_timeout_app, None, 0.2)
|
client = DelayableSanicTestClient(request_no_timeout_app, 0.2)
|
||||||
request, response = client.get("/1")
|
request, response = client.get("/1")
|
||||||
assert response.status == 200
|
assert response.status == 200
|
||||||
assert response.text == "OK"
|
assert response.text == "OK"
|
||||||
|
@ -224,7 +114,7 @@ def test_default_server_error_websocket_request_timeout():
|
||||||
"Sec-WebSocket-Version": "13",
|
"Sec-WebSocket-Version": "13",
|
||||||
}
|
}
|
||||||
|
|
||||||
client = DelayableSanicTestClient(request_timeout_default_app, None, 2)
|
client = DelayableSanicTestClient(request_timeout_default_app, 2)
|
||||||
request, response = client.get("/ws1", headers=headers)
|
request, response = client.get("/ws1", headers=headers)
|
||||||
|
|
||||||
assert response.status == 408
|
assert response.status == 408
|
||||||
|
|
|
@ -1,19 +1,20 @@
|
||||||
import logging
|
import logging
|
||||||
import os
|
import os
|
||||||
import ssl
|
import ssl
|
||||||
|
|
||||||
from json import dumps as json_dumps
|
from json import dumps as json_dumps
|
||||||
from json import loads as json_loads
|
from json import loads as json_loads
|
||||||
from urllib.parse import urlparse
|
from urllib.parse import urlparse
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
|
|
||||||
from sanic import Sanic
|
from sanic import Blueprint, Sanic
|
||||||
from sanic import Blueprint
|
|
||||||
from sanic.exceptions import ServerError
|
from sanic.exceptions import ServerError
|
||||||
from sanic.request import DEFAULT_HTTP_CONTENT_TYPE, RequestParameters
|
from sanic.request import DEFAULT_HTTP_CONTENT_TYPE, RequestParameters
|
||||||
from sanic.response import json, text
|
from sanic.response import json, text
|
||||||
from sanic.testing import HOST, PORT
|
from sanic.testing import HOST, PORT
|
||||||
|
|
||||||
|
|
||||||
# ------------------------------------------------------------ #
|
# ------------------------------------------------------------ #
|
||||||
# GET
|
# GET
|
||||||
# ------------------------------------------------------------ #
|
# ------------------------------------------------------------ #
|
||||||
|
@ -529,36 +530,54 @@ def test_request_string_representation(app):
|
||||||
@pytest.mark.parametrize(
|
@pytest.mark.parametrize(
|
||||||
"payload,filename",
|
"payload,filename",
|
||||||
[
|
[
|
||||||
("------sanic\r\n"
|
(
|
||||||
'Content-Disposition: form-data; filename="filename"; name="test"\r\n'
|
"------sanic\r\n"
|
||||||
"\r\n"
|
'Content-Disposition: form-data; filename="filename"; name="test"\r\n'
|
||||||
"OK\r\n"
|
"\r\n"
|
||||||
"------sanic--\r\n", "filename"),
|
"OK\r\n"
|
||||||
("------sanic\r\n"
|
"------sanic--\r\n",
|
||||||
'content-disposition: form-data; filename="filename"; name="test"\r\n'
|
"filename",
|
||||||
"\r\n"
|
),
|
||||||
'content-type: application/json; {"field": "value"}\r\n'
|
(
|
||||||
"------sanic--\r\n", "filename"),
|
"------sanic\r\n"
|
||||||
("------sanic\r\n"
|
'content-disposition: form-data; filename="filename"; name="test"\r\n'
|
||||||
'Content-Disposition: form-data; filename=""; name="test"\r\n'
|
"\r\n"
|
||||||
"\r\n"
|
'content-type: application/json; {"field": "value"}\r\n'
|
||||||
"OK\r\n"
|
"------sanic--\r\n",
|
||||||
"------sanic--\r\n", ""),
|
"filename",
|
||||||
("------sanic\r\n"
|
),
|
||||||
'content-disposition: form-data; filename=""; name="test"\r\n'
|
(
|
||||||
"\r\n"
|
"------sanic\r\n"
|
||||||
'content-type: application/json; {"field": "value"}\r\n'
|
'Content-Disposition: form-data; filename=""; name="test"\r\n'
|
||||||
"------sanic--\r\n", ""),
|
"\r\n"
|
||||||
("------sanic\r\n"
|
"OK\r\n"
|
||||||
'Content-Disposition: form-data; filename*="utf-8\'\'filename_%C2%A0_test"; name="test"\r\n'
|
"------sanic--\r\n",
|
||||||
"\r\n"
|
"",
|
||||||
"OK\r\n"
|
),
|
||||||
"------sanic--\r\n", "filename_\u00A0_test"),
|
(
|
||||||
("------sanic\r\n"
|
"------sanic\r\n"
|
||||||
'content-disposition: form-data; filename*="utf-8\'\'filename_%C2%A0_test"; name="test"\r\n'
|
'content-disposition: form-data; filename=""; name="test"\r\n'
|
||||||
"\r\n"
|
"\r\n"
|
||||||
'content-type: application/json; {"field": "value"}\r\n'
|
'content-type: application/json; {"field": "value"}\r\n'
|
||||||
"------sanic--\r\n", "filename_\u00A0_test"),
|
"------sanic--\r\n",
|
||||||
|
"",
|
||||||
|
),
|
||||||
|
(
|
||||||
|
"------sanic\r\n"
|
||||||
|
'Content-Disposition: form-data; filename*="utf-8\'\'filename_%C2%A0_test"; name="test"\r\n'
|
||||||
|
"\r\n"
|
||||||
|
"OK\r\n"
|
||||||
|
"------sanic--\r\n",
|
||||||
|
"filename_\u00A0_test",
|
||||||
|
),
|
||||||
|
(
|
||||||
|
"------sanic\r\n"
|
||||||
|
'content-disposition: form-data; filename*="utf-8\'\'filename_%C2%A0_test"; name="test"\r\n'
|
||||||
|
"\r\n"
|
||||||
|
'content-type: application/json; {"field": "value"}\r\n'
|
||||||
|
"------sanic--\r\n",
|
||||||
|
"filename_\u00A0_test",
|
||||||
|
),
|
||||||
],
|
],
|
||||||
)
|
)
|
||||||
def test_request_multipart_files(app, payload, filename):
|
def test_request_multipart_files(app, payload, filename):
|
||||||
|
@ -743,7 +762,7 @@ def test_request_raw_args(app):
|
||||||
|
|
||||||
def test_request_query_args(app):
|
def test_request_query_args(app):
|
||||||
# test multiple params with the same key
|
# test multiple params with the same key
|
||||||
params = [('test', 'value1'), ('test', 'value2')]
|
params = [("test", "value1"), ("test", "value2")]
|
||||||
|
|
||||||
@app.get("/")
|
@app.get("/")
|
||||||
def handler(request):
|
def handler(request):
|
||||||
|
@ -754,7 +773,10 @@ def test_request_query_args(app):
|
||||||
assert request.query_args == params
|
assert request.query_args == params
|
||||||
|
|
||||||
# test cached value
|
# test cached value
|
||||||
assert request.parsed_not_grouped_args[(False, False, "utf-8", "replace")] == request.query_args
|
assert (
|
||||||
|
request.parsed_not_grouped_args[(False, False, "utf-8", "replace")]
|
||||||
|
== request.query_args
|
||||||
|
)
|
||||||
|
|
||||||
# test params directly in the url
|
# test params directly in the url
|
||||||
request, response = app.test_client.get("/?test=value1&test=value2")
|
request, response = app.test_client.get("/?test=value1&test=value2")
|
||||||
|
@ -762,7 +784,7 @@ def test_request_query_args(app):
|
||||||
assert request.query_args == params
|
assert request.query_args == params
|
||||||
|
|
||||||
# test unique params
|
# test unique params
|
||||||
params = [('test1', 'value1'), ('test2', 'value2')]
|
params = [("test1", "value1"), ("test2", "value2")]
|
||||||
|
|
||||||
request, response = app.test_client.get("/", params=params)
|
request, response = app.test_client.get("/", params=params)
|
||||||
|
|
||||||
|
@ -779,25 +801,22 @@ def test_request_query_args_custom_parsing(app):
|
||||||
def handler(request):
|
def handler(request):
|
||||||
return text("pass")
|
return text("pass")
|
||||||
|
|
||||||
request, response = app.test_client.get("/?test1=value1&test2=&test3=value3")
|
request, response = app.test_client.get(
|
||||||
|
"/?test1=value1&test2=&test3=value3"
|
||||||
|
)
|
||||||
|
|
||||||
assert request.get_query_args(
|
assert request.get_query_args(keep_blank_values=True) == [
|
||||||
keep_blank_values=True
|
("test1", "value1"),
|
||||||
) == [
|
("test2", ""),
|
||||||
('test1', 'value1'), ('test2', ''), ('test3', 'value3')
|
("test3", "value3"),
|
||||||
]
|
]
|
||||||
assert request.query_args == [
|
assert request.query_args == [("test1", "value1"), ("test3", "value3")]
|
||||||
('test1', 'value1'), ('test3', 'value3')
|
assert request.get_query_args(keep_blank_values=False) == [
|
||||||
]
|
("test1", "value1"),
|
||||||
assert request.get_query_args(
|
("test3", "value3"),
|
||||||
keep_blank_values=False
|
|
||||||
) == [
|
|
||||||
('test1', 'value1'), ('test3', 'value3')
|
|
||||||
]
|
]
|
||||||
|
|
||||||
assert request.get_args(
|
assert request.get_args(keep_blank_values=True) == RequestParameters(
|
||||||
keep_blank_values=True
|
|
||||||
) == RequestParameters(
|
|
||||||
{"test1": ["value1"], "test2": [""], "test3": ["value3"]}
|
{"test1": ["value1"], "test2": [""], "test3": ["value3"]}
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -805,9 +824,7 @@ def test_request_query_args_custom_parsing(app):
|
||||||
{"test1": ["value1"], "test3": ["value3"]}
|
{"test1": ["value1"], "test3": ["value3"]}
|
||||||
)
|
)
|
||||||
|
|
||||||
assert request.get_args(
|
assert request.get_args(keep_blank_values=False) == RequestParameters(
|
||||||
keep_blank_values=False
|
|
||||||
) == RequestParameters(
|
|
||||||
{"test1": ["value1"], "test3": ["value3"]}
|
{"test1": ["value1"], "test3": ["value3"]}
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
import asyncio
|
import asyncio
|
||||||
import inspect
|
import inspect
|
||||||
import os
|
import os
|
||||||
|
|
||||||
from collections import namedtuple
|
from collections import namedtuple
|
||||||
from mimetypes import guess_type
|
from mimetypes import guess_type
|
||||||
from random import choice
|
from random import choice
|
||||||
|
@ -8,6 +9,7 @@ from unittest.mock import MagicMock
|
||||||
from urllib.parse import unquote
|
from urllib.parse import unquote
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
|
|
||||||
from aiofiles import os as async_os
|
from aiofiles import os as async_os
|
||||||
|
|
||||||
from sanic.response import (
|
from sanic.response import (
|
||||||
|
@ -18,11 +20,11 @@ from sanic.response import (
|
||||||
json,
|
json,
|
||||||
raw,
|
raw,
|
||||||
stream,
|
stream,
|
||||||
text,
|
|
||||||
)
|
)
|
||||||
from sanic.server import HttpProtocol
|
from sanic.server import HttpProtocol
|
||||||
from sanic.testing import HOST, PORT
|
from sanic.testing import HOST, PORT
|
||||||
|
|
||||||
|
|
||||||
JSON_DATA = {"ok": True}
|
JSON_DATA = {"ok": True}
|
||||||
|
|
||||||
|
|
||||||
|
@ -77,10 +79,10 @@ def test_response_header(app):
|
||||||
|
|
||||||
request, response = app.test_client.get("/")
|
request, response = app.test_client.get("/")
|
||||||
assert dict(response.headers) == {
|
assert dict(response.headers) == {
|
||||||
"Connection": "keep-alive",
|
"connection": "keep-alive",
|
||||||
"Keep-Alive": str(app.config.KEEP_ALIVE_TIMEOUT),
|
"keep-alive": str(app.config.KEEP_ALIVE_TIMEOUT),
|
||||||
"Content-Length": "11",
|
"content-length": "11",
|
||||||
"Content-Type": "application/json",
|
"content-type": "application/json",
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -276,7 +278,7 @@ def test_stream_response_with_cookies(app):
|
||||||
return response
|
return response
|
||||||
|
|
||||||
request, response = app.test_client.get("/")
|
request, response = app.test_client.get("/")
|
||||||
assert response.cookies["test"].value == "pass"
|
assert response.cookies["test"] == "pass"
|
||||||
|
|
||||||
|
|
||||||
def test_stream_response_without_cookies(app):
|
def test_stream_response_without_cookies(app):
|
||||||
|
|
|
@ -1,7 +1,9 @@
|
||||||
from sanic import Sanic
|
|
||||||
import asyncio
|
import asyncio
|
||||||
from sanic.response import text
|
|
||||||
|
from sanic import Sanic
|
||||||
from sanic.exceptions import ServiceUnavailable
|
from sanic.exceptions import ServiceUnavailable
|
||||||
|
from sanic.response import text
|
||||||
|
|
||||||
|
|
||||||
response_timeout_app = Sanic("test_response_timeout")
|
response_timeout_app = Sanic("test_response_timeout")
|
||||||
response_timeout_default_app = Sanic("test_response_timeout_default")
|
response_timeout_default_app = Sanic("test_response_timeout_default")
|
||||||
|
|
|
@ -7,6 +7,7 @@ from sanic.constants import HTTP_METHODS
|
||||||
from sanic.response import json, text
|
from sanic.response import json, text
|
||||||
from sanic.router import ParameterNameConflicts, RouteDoesNotExist, RouteExists
|
from sanic.router import ParameterNameConflicts, RouteDoesNotExist, RouteExists
|
||||||
|
|
||||||
|
|
||||||
# ------------------------------------------------------------ #
|
# ------------------------------------------------------------ #
|
||||||
# UTF-8
|
# UTF-8
|
||||||
# ------------------------------------------------------------ #
|
# ------------------------------------------------------------ #
|
||||||
|
@ -468,16 +469,8 @@ def test_websocket_route(app, url):
|
||||||
assert ws.subprotocol is None
|
assert ws.subprotocol is None
|
||||||
ev.set()
|
ev.set()
|
||||||
|
|
||||||
request, response = app.test_client.get(
|
request, response = app.test_client.websocket(url)
|
||||||
"/ws",
|
assert response.opened is True
|
||||||
headers={
|
|
||||||
"Upgrade": "websocket",
|
|
||||||
"Connection": "upgrade",
|
|
||||||
"Sec-WebSocket-Key": "dGhlIHNhbXBsZSBub25jZQ==",
|
|
||||||
"Sec-WebSocket-Version": "13",
|
|
||||||
},
|
|
||||||
)
|
|
||||||
assert response.status == 101
|
|
||||||
assert ev.is_set()
|
assert ev.is_set()
|
||||||
|
|
||||||
|
|
||||||
|
@ -487,54 +480,24 @@ def test_websocket_route_with_subprotocols(app):
|
||||||
@app.websocket("/ws", subprotocols=["foo", "bar"])
|
@app.websocket("/ws", subprotocols=["foo", "bar"])
|
||||||
async def handler(request, ws):
|
async def handler(request, ws):
|
||||||
results.append(ws.subprotocol)
|
results.append(ws.subprotocol)
|
||||||
|
assert ws.subprotocol is not None
|
||||||
|
|
||||||
request, response = app.test_client.get(
|
request, response = app.test_client.websocket("/ws", subprotocols=["bar"])
|
||||||
"/ws",
|
assert response.opened is True
|
||||||
headers={
|
assert results == ["bar"]
|
||||||
"Upgrade": "websocket",
|
|
||||||
"Connection": "upgrade",
|
request, response = app.test_client.websocket(
|
||||||
"Sec-WebSocket-Key": "dGhlIHNhbXBsZSBub25jZQ==",
|
"/ws", subprotocols=["bar", "foo"]
|
||||||
"Sec-WebSocket-Version": "13",
|
|
||||||
"Sec-WebSocket-Protocol": "bar",
|
|
||||||
},
|
|
||||||
)
|
)
|
||||||
assert response.status == 101
|
assert response.opened is True
|
||||||
|
assert results == ["bar", "bar"]
|
||||||
|
|
||||||
request, response = app.test_client.get(
|
request, response = app.test_client.websocket("/ws", subprotocols=["baz"])
|
||||||
"/ws",
|
assert response.opened is True
|
||||||
headers={
|
assert results == ["bar", "bar", None]
|
||||||
"Upgrade": "websocket",
|
|
||||||
"Connection": "upgrade",
|
|
||||||
"Sec-WebSocket-Key": "dGhlIHNhbXBsZSBub25jZQ==",
|
|
||||||
"Sec-WebSocket-Version": "13",
|
|
||||||
"Sec-WebSocket-Protocol": "bar, foo",
|
|
||||||
},
|
|
||||||
)
|
|
||||||
assert response.status == 101
|
|
||||||
|
|
||||||
request, response = app.test_client.get(
|
|
||||||
"/ws",
|
|
||||||
headers={
|
|
||||||
"Upgrade": "websocket",
|
|
||||||
"Connection": "upgrade",
|
|
||||||
"Sec-WebSocket-Key": "dGhlIHNhbXBsZSBub25jZQ==",
|
|
||||||
"Sec-WebSocket-Version": "13",
|
|
||||||
"Sec-WebSocket-Protocol": "baz",
|
|
||||||
},
|
|
||||||
)
|
|
||||||
assert response.status == 101
|
|
||||||
|
|
||||||
request, response = app.test_client.get(
|
|
||||||
"/ws",
|
|
||||||
headers={
|
|
||||||
"Upgrade": "websocket",
|
|
||||||
"Connection": "upgrade",
|
|
||||||
"Sec-WebSocket-Key": "dGhlIHNhbXBsZSBub25jZQ==",
|
|
||||||
"Sec-WebSocket-Version": "13",
|
|
||||||
},
|
|
||||||
)
|
|
||||||
assert response.status == 101
|
|
||||||
|
|
||||||
|
request, response = app.test_client.websocket("/ws")
|
||||||
|
assert response.opened is True
|
||||||
assert results == ["bar", "bar", None, None]
|
assert results == ["bar", "bar", None, None]
|
||||||
|
|
||||||
|
|
||||||
|
@ -547,16 +510,8 @@ def test_add_webscoket_route(app, strict_slashes):
|
||||||
ev.set()
|
ev.set()
|
||||||
|
|
||||||
app.add_websocket_route(handler, "/ws", strict_slashes=strict_slashes)
|
app.add_websocket_route(handler, "/ws", strict_slashes=strict_slashes)
|
||||||
request, response = app.test_client.get(
|
request, response = app.test_client.websocket("/ws")
|
||||||
"/ws",
|
assert response.opened is True
|
||||||
headers={
|
|
||||||
"Upgrade": "websocket",
|
|
||||||
"Connection": "upgrade",
|
|
||||||
"Sec-WebSocket-Key": "dGhlIHNhbXBsZSBub25jZQ==",
|
|
||||||
"Sec-WebSocket-Version": "13",
|
|
||||||
},
|
|
||||||
)
|
|
||||||
assert response.status == 101
|
|
||||||
assert ev.is_set()
|
assert ev.is_set()
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -4,6 +4,7 @@ import pytest
|
||||||
|
|
||||||
from sanic.testing import HOST, PORT
|
from sanic.testing import HOST, PORT
|
||||||
|
|
||||||
|
|
||||||
AVAILABLE_LISTENERS = [
|
AVAILABLE_LISTENERS = [
|
||||||
"before_server_start",
|
"before_server_start",
|
||||||
"after_server_start",
|
"after_server_start",
|
||||||
|
|
|
@ -1,8 +1,10 @@
|
||||||
|
import asyncio
|
||||||
|
|
||||||
|
from queue import Queue
|
||||||
|
from unittest.mock import MagicMock
|
||||||
|
|
||||||
from sanic.response import HTTPResponse
|
from sanic.response import HTTPResponse
|
||||||
from sanic.testing import HOST, PORT
|
from sanic.testing import HOST, PORT
|
||||||
from unittest.mock import MagicMock
|
|
||||||
import asyncio
|
|
||||||
from queue import Queue
|
|
||||||
|
|
||||||
|
|
||||||
async def stop(app, loop):
|
async def stop(app, loop):
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
import inspect
|
import inspect
|
||||||
import os
|
import os
|
||||||
|
|
||||||
from time import gmtime, strftime
|
from time import gmtime, strftime
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
|
|
|
@ -1,7 +1,8 @@
|
||||||
import socket
|
import socket
|
||||||
|
|
||||||
from sanic.testing import PORT, SanicTestClient
|
|
||||||
from sanic.response import json, text
|
from sanic.response import json, text
|
||||||
|
from sanic.testing import PORT, SanicTestClient
|
||||||
|
|
||||||
|
|
||||||
# ------------------------------------------------------------ #
|
# ------------------------------------------------------------ #
|
||||||
# UTF-8
|
# UTF-8
|
||||||
|
@ -9,26 +10,26 @@ from sanic.response import json, text
|
||||||
|
|
||||||
|
|
||||||
def test_test_client_port_none(app):
|
def test_test_client_port_none(app):
|
||||||
@app.get('/get')
|
@app.get("/get")
|
||||||
def handler(request):
|
def handler(request):
|
||||||
return text('OK')
|
return text("OK")
|
||||||
|
|
||||||
test_client = SanicTestClient(app, port=None)
|
test_client = SanicTestClient(app, port=None)
|
||||||
|
|
||||||
request, response = test_client.get('/get')
|
request, response = test_client.get("/get")
|
||||||
assert response.text == 'OK'
|
assert response.text == "OK"
|
||||||
|
|
||||||
request, response = test_client.post('/get')
|
request, response = test_client.post("/get")
|
||||||
assert response.status == 405
|
assert response.status == 405
|
||||||
|
|
||||||
|
|
||||||
def test_test_client_port_default(app):
|
def test_test_client_port_default(app):
|
||||||
@app.get('/get')
|
@app.get("/get")
|
||||||
def handler(request):
|
def handler(request):
|
||||||
return json(request.transport.get_extra_info('sockname')[1])
|
return json(request.transport.get_extra_info("sockname")[1])
|
||||||
|
|
||||||
test_client = SanicTestClient(app)
|
test_client = SanicTestClient(app)
|
||||||
assert test_client.port == PORT
|
assert test_client.port == PORT
|
||||||
|
|
||||||
request, response = test_client.get('/get')
|
request, response = test_client.get("/get")
|
||||||
assert response.json == PORT
|
assert response.json == PORT
|
||||||
|
|
|
@ -1,14 +1,17 @@
|
||||||
import pytest as pytest
|
|
||||||
from urllib.parse import urlsplit, parse_qsl
|
|
||||||
|
|
||||||
from sanic.response import text
|
|
||||||
from sanic.views import HTTPMethodView
|
|
||||||
from sanic.blueprints import Blueprint
|
|
||||||
from sanic.testing import PORT as test_port, HOST as test_host
|
|
||||||
from sanic.exceptions import URLBuildError
|
|
||||||
|
|
||||||
import string
|
import string
|
||||||
|
|
||||||
|
from urllib.parse import parse_qsl, urlsplit
|
||||||
|
|
||||||
|
import pytest as pytest
|
||||||
|
|
||||||
|
from sanic.blueprints import Blueprint
|
||||||
|
from sanic.exceptions import URLBuildError
|
||||||
|
from sanic.response import text
|
||||||
|
from sanic.testing import HOST as test_host
|
||||||
|
from sanic.testing import PORT as test_port
|
||||||
|
from sanic.views import HTTPMethodView
|
||||||
|
|
||||||
|
|
||||||
URL_FOR_ARGS1 = dict(arg1=["v1", "v2"])
|
URL_FOR_ARGS1 = dict(arg1=["v1", "v2"])
|
||||||
URL_FOR_VALUE1 = "/myurl?arg1=v1&arg1=v2"
|
URL_FOR_VALUE1 = "/myurl?arg1=v1&arg1=v2"
|
||||||
URL_FOR_ARGS2 = dict(arg1=["v1", "v2"], _anchor="anchor")
|
URL_FOR_ARGS2 = dict(arg1=["v1", "v2"], _anchor="anchor")
|
||||||
|
@ -170,7 +173,7 @@ def test_fails_with_int_message(app):
|
||||||
|
|
||||||
expected_error = (
|
expected_error = (
|
||||||
r'Value "not_int" for parameter `foo` '
|
r'Value "not_int" for parameter `foo` '
|
||||||
r'does not match pattern for type `int`: -?\d+'
|
r"does not match pattern for type `int`: -?\d+"
|
||||||
)
|
)
|
||||||
assert str(e.value) == expected_error
|
assert str(e.value) == expected_error
|
||||||
|
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
from json import dumps as json_dumps
|
from json import dumps as json_dumps
|
||||||
|
|
||||||
from sanic.response import text
|
from sanic.response import text
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -1,11 +1,11 @@
|
||||||
import pytest as pytest
|
import pytest as pytest
|
||||||
|
|
||||||
from sanic.exceptions import InvalidUsage
|
|
||||||
from sanic.response import text, HTTPResponse
|
|
||||||
from sanic.views import HTTPMethodView, CompositionView
|
|
||||||
from sanic.blueprints import Blueprint
|
from sanic.blueprints import Blueprint
|
||||||
from sanic.request import Request
|
|
||||||
from sanic.constants import HTTP_METHODS
|
from sanic.constants import HTTP_METHODS
|
||||||
|
from sanic.exceptions import InvalidUsage
|
||||||
|
from sanic.request import Request
|
||||||
|
from sanic.response import HTTPResponse, text
|
||||||
|
from sanic.views import CompositionView, HTTPMethodView
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.parametrize("method", HTTP_METHODS)
|
@pytest.mark.parametrize("method", HTTP_METHODS)
|
||||||
|
|
|
@ -1,14 +1,17 @@
|
||||||
import time
|
import asyncio
|
||||||
import json
|
import json
|
||||||
import shlex
|
import shlex
|
||||||
import subprocess
|
import subprocess
|
||||||
|
import time
|
||||||
import urllib.request
|
import urllib.request
|
||||||
|
|
||||||
from unittest import mock
|
from unittest import mock
|
||||||
from sanic.worker import GunicornWorker
|
|
||||||
from sanic.app import Sanic
|
|
||||||
import asyncio
|
|
||||||
import pytest
|
import pytest
|
||||||
|
|
||||||
|
from sanic.app import Sanic
|
||||||
|
from sanic.worker import GunicornWorker
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture(scope="module")
|
@pytest.fixture(scope="module")
|
||||||
def gunicorn_worker():
|
def gunicorn_worker():
|
||||||
|
|
13
tox.ini
13
tox.ini
|
@ -1,24 +1,25 @@
|
||||||
[tox]
|
[tox]
|
||||||
envlist = py35, py36, py37, {py35,py36,py37}-no-ext, lint, check
|
envlist = py36, py37, {py36,py37}-no-ext, lint, check
|
||||||
|
|
||||||
[testenv]
|
[testenv]
|
||||||
usedevelop = True
|
usedevelop = True
|
||||||
setenv =
|
setenv =
|
||||||
{py35,py36,py37}-no-ext: SANIC_NO_UJSON=1
|
{py36,py37}-no-ext: SANIC_NO_UJSON=1
|
||||||
{py35,py36,py37}-no-ext: SANIC_NO_UVLOOP=1
|
{py36,py37}-no-ext: SANIC_NO_UVLOOP=1
|
||||||
deps =
|
deps =
|
||||||
coverage
|
coverage
|
||||||
pytest==4.1.0
|
pytest==4.1.0
|
||||||
pytest-cov
|
pytest-cov
|
||||||
pytest-sanic
|
pytest-sanic
|
||||||
pytest-sugar
|
pytest-sugar
|
||||||
aiohttp>=2.3,<=3.2.1
|
httpcore==0.1.1
|
||||||
|
requests-async==0.4.0
|
||||||
chardet<=2.3.0
|
chardet<=2.3.0
|
||||||
beautifulsoup4
|
beautifulsoup4
|
||||||
gunicorn
|
gunicorn
|
||||||
pytest-benchmark
|
pytest-benchmark
|
||||||
commands =
|
commands =
|
||||||
pytest tests --cov sanic --cov-report= {posargs}
|
pytest {posargs:tests --cov sanic}
|
||||||
- coverage combine --append
|
- coverage combine --append
|
||||||
coverage report -m
|
coverage report -m
|
||||||
coverage html -i
|
coverage html -i
|
||||||
|
@ -31,7 +32,7 @@ deps =
|
||||||
|
|
||||||
commands =
|
commands =
|
||||||
flake8 sanic
|
flake8 sanic
|
||||||
black --check --verbose sanic
|
black --config ./.black.toml --check --verbose sanic
|
||||||
isort --check-only --recursive sanic
|
isort --check-only --recursive sanic
|
||||||
|
|
||||||
[testenv:check]
|
[testenv:check]
|
||||||
|
|
Loading…
Reference in New Issue
Block a user