1114 lines
29 KiB
Python
1114 lines
29 KiB
Python
import asyncio
|
|
import inspect
|
|
import os
|
|
|
|
import pytest
|
|
|
|
from sanic.app import Sanic
|
|
from sanic.blueprints import Blueprint
|
|
from sanic.constants import HTTP_METHODS
|
|
from sanic.exceptions import BadRequest, NotFound, SanicException, ServerError
|
|
from sanic.request import Request
|
|
from sanic.response import json, text
|
|
|
|
# ------------------------------------------------------------ #
|
|
# GET
|
|
# ------------------------------------------------------------ #
|
|
|
|
|
|
def test_bp(app: Sanic):
|
|
bp = Blueprint("test_text")
|
|
|
|
@bp.route("/")
|
|
def handler(request):
|
|
return text("Hello")
|
|
|
|
app.blueprint(bp)
|
|
request, response = app.test_client.get("/")
|
|
|
|
assert response.text == "Hello"
|
|
|
|
|
|
def test_bp_app_access(app: Sanic):
|
|
bp = Blueprint("test")
|
|
|
|
with pytest.raises(
|
|
SanicException,
|
|
match="<Blueprint test> has not yet been registered to an app",
|
|
):
|
|
bp.apps
|
|
|
|
app.blueprint(bp)
|
|
|
|
assert app in bp.apps
|
|
|
|
|
|
@pytest.fixture(scope="module")
|
|
def static_file_directory():
|
|
"""The static directory to serve"""
|
|
current_file = inspect.getfile(inspect.currentframe())
|
|
current_directory = os.path.dirname(os.path.abspath(current_file))
|
|
static_directory = os.path.join(current_directory, "static")
|
|
return static_directory
|
|
|
|
|
|
def get_file_path(static_file_directory, file_name):
|
|
return os.path.join(static_file_directory, file_name)
|
|
|
|
|
|
def get_file_content(static_file_directory, file_name):
|
|
"""The content of the static file to check"""
|
|
with open(get_file_path(static_file_directory, file_name), "rb") as file:
|
|
return file.read()
|
|
|
|
|
|
@pytest.mark.parametrize("method", HTTP_METHODS)
|
|
def test_versioned_routes_get(app, method):
|
|
bp = Blueprint("test_text")
|
|
|
|
method = method.lower()
|
|
|
|
func = getattr(bp, method)
|
|
if callable(func):
|
|
|
|
@func(f"/{method}", version=1)
|
|
def handler(request):
|
|
return text("OK")
|
|
|
|
else:
|
|
raise Exception(f"{func} is not callable")
|
|
|
|
app.blueprint(bp)
|
|
|
|
client_method = getattr(app.test_client, method)
|
|
|
|
request, response = client_method(f"/v1/{method}")
|
|
assert response.status == 200
|
|
|
|
|
|
def test_bp_strict_slash(app: Sanic):
|
|
bp = Blueprint("test_text")
|
|
|
|
@bp.get("/get", strict_slashes=True)
|
|
def get_handler(request):
|
|
return text("OK")
|
|
|
|
@bp.post("/post/", strict_slashes=True)
|
|
def post_handler(request):
|
|
return text("OK")
|
|
|
|
app.blueprint(bp)
|
|
|
|
request, response = app.test_client.get("/get")
|
|
assert response.text == "OK"
|
|
assert response.json is None
|
|
|
|
request, response = app.test_client.get("/get/")
|
|
assert response.status == 404
|
|
|
|
request, response = app.test_client.post("/post/")
|
|
assert response.text == "OK"
|
|
|
|
request, response = app.test_client.post("/post")
|
|
assert response.status == 404
|
|
|
|
|
|
def test_bp_strict_slash_default_value(app: Sanic):
|
|
bp = Blueprint("test_text", strict_slashes=True)
|
|
|
|
@bp.get("/get")
|
|
def get_handler(request):
|
|
return text("OK")
|
|
|
|
@bp.post("/post/")
|
|
def post_handler(request):
|
|
return text("OK")
|
|
|
|
app.blueprint(bp)
|
|
|
|
request, response = app.test_client.get("/get/")
|
|
assert response.status == 404
|
|
|
|
request, response = app.test_client.post("/post")
|
|
assert response.status == 404
|
|
|
|
|
|
def test_bp_strict_slash_without_passing_default_value(app: Sanic):
|
|
bp = Blueprint("test_text")
|
|
|
|
@bp.get("/get")
|
|
def get_handler(request):
|
|
return text("OK")
|
|
|
|
@bp.post("/post/")
|
|
def post_handler(request):
|
|
return text("OK")
|
|
|
|
app.blueprint(bp)
|
|
|
|
request, response = app.test_client.get("/get/")
|
|
assert response.text == "OK"
|
|
|
|
request, response = app.test_client.post("/post")
|
|
assert response.text == "OK"
|
|
|
|
|
|
def test_bp_strict_slash_default_value_can_be_overwritten(app: Sanic):
|
|
bp = Blueprint("test_text", strict_slashes=True)
|
|
|
|
@bp.get("/get", strict_slashes=False)
|
|
def get_handler(request):
|
|
return text("OK")
|
|
|
|
@bp.post("/post/", strict_slashes=False)
|
|
def post_handler(request):
|
|
return text("OK")
|
|
|
|
app.blueprint(bp)
|
|
|
|
request, response = app.test_client.get("/get/")
|
|
assert response.text == "OK"
|
|
|
|
request, response = app.test_client.post("/post")
|
|
assert response.text == "OK"
|
|
|
|
|
|
def test_bp_with_url_prefix(app: Sanic):
|
|
bp = Blueprint("test_text", url_prefix="/test1")
|
|
|
|
@bp.route("/")
|
|
def handler(request):
|
|
return text("Hello")
|
|
|
|
app.blueprint(bp)
|
|
request, response = app.test_client.get("/test1/")
|
|
|
|
assert response.text == "Hello"
|
|
|
|
|
|
def test_several_bp_with_url_prefix(app: Sanic):
|
|
bp = Blueprint("test_text", url_prefix="/test1")
|
|
bp2 = Blueprint("test_text2", url_prefix="/test2")
|
|
|
|
@bp.route("/")
|
|
def handler(request):
|
|
return text("Hello")
|
|
|
|
@bp2.route("/")
|
|
def handler2(request):
|
|
return text("Hello2")
|
|
|
|
app.blueprint(bp)
|
|
app.blueprint(bp2)
|
|
request, response = app.test_client.get("/test1/")
|
|
assert response.text == "Hello"
|
|
|
|
request, response = app.test_client.get("/test2/")
|
|
assert response.text == "Hello2"
|
|
|
|
|
|
def test_bp_with_host(app: Sanic):
|
|
bp = Blueprint("test_bp_host", url_prefix="/test1", host="example.com")
|
|
|
|
@bp.route("/")
|
|
def handler1(request):
|
|
return text("Hello")
|
|
|
|
@bp.route("/", host="sub.example.com")
|
|
def handler2(request):
|
|
return text("Hello subdomain!")
|
|
|
|
app.blueprint(bp)
|
|
headers = {"Host": "example.com"}
|
|
|
|
request, response = app.test_client.get("/test1/", headers=headers)
|
|
assert response.text == "Hello"
|
|
|
|
headers = {"Host": "sub.example.com"}
|
|
request, response = app.test_client.get("/test1/", headers=headers)
|
|
assert response.body == b"Hello subdomain!"
|
|
|
|
|
|
def test_several_bp_with_host(app: Sanic):
|
|
bp = Blueprint(
|
|
"test_text",
|
|
url_prefix="/test",
|
|
host="example.com",
|
|
strict_slashes=True,
|
|
)
|
|
bp2 = Blueprint(
|
|
"test_text2",
|
|
url_prefix="/test",
|
|
host="sub.example.com",
|
|
strict_slashes=True,
|
|
)
|
|
|
|
@bp.route("/")
|
|
def handler(request):
|
|
return text("Hello")
|
|
|
|
@bp2.route("/")
|
|
def handler1(request):
|
|
return text("Hello2")
|
|
|
|
@bp2.route("/other/")
|
|
def handler2(request):
|
|
return text("Hello3")
|
|
|
|
app.blueprint(bp)
|
|
app.blueprint(bp2)
|
|
|
|
assert bp.host == "example.com"
|
|
headers = {"Host": "example.com"}
|
|
request, response = app.test_client.get("/test/", headers=headers)
|
|
|
|
assert response.text == "Hello"
|
|
|
|
assert bp2.host == "sub.example.com"
|
|
headers = {"Host": "sub.example.com"}
|
|
request, response = app.test_client.get("/test/", headers=headers)
|
|
|
|
assert response.text == "Hello2"
|
|
request, response = app.test_client.get("/test/other/", headers=headers)
|
|
assert response.text == "Hello3"
|
|
|
|
|
|
def test_bp_with_host_list(app: Sanic):
|
|
bp = Blueprint(
|
|
"test_bp_host",
|
|
url_prefix="/test1",
|
|
host=["example.com", "sub.example.com"],
|
|
)
|
|
|
|
@bp.route("/")
|
|
def handler1(request):
|
|
return text("Hello")
|
|
|
|
@bp.route("/", host=["sub1.example.com"])
|
|
def handler2(request):
|
|
return text("Hello subdomain!")
|
|
|
|
app.blueprint(bp)
|
|
headers = {"Host": "example.com"}
|
|
request, response = app.test_client.get("/test1/", headers=headers)
|
|
assert response.text == "Hello"
|
|
|
|
headers = {"Host": "sub.example.com"}
|
|
request, response = app.test_client.get("/test1/", headers=headers)
|
|
assert response.text == "Hello"
|
|
|
|
headers = {"Host": "sub1.example.com"}
|
|
request, response = app.test_client.get("/test1/", headers=headers)
|
|
|
|
assert response.text == "Hello subdomain!"
|
|
|
|
route_names = [r.name for r in app.router.routes]
|
|
assert "test_bp_with_host_list.test_bp_host.handler1" in route_names
|
|
assert "test_bp_with_host_list.test_bp_host.handler2" in route_names
|
|
|
|
|
|
def test_several_bp_with_host_list(app: Sanic):
|
|
bp = Blueprint(
|
|
"test_text",
|
|
url_prefix="/test",
|
|
host=["example.com", "sub.example.com"],
|
|
)
|
|
bp2 = Blueprint(
|
|
"test_text2",
|
|
url_prefix="/test",
|
|
host=["sub1.example.com", "sub2.example.com"],
|
|
)
|
|
|
|
@bp.route("/")
|
|
def handler(request):
|
|
return text("Hello")
|
|
|
|
@bp2.route("/")
|
|
def handler1(request):
|
|
return text("Hello2")
|
|
|
|
@bp2.route("/other/")
|
|
def handler2(request):
|
|
return text("Hello3")
|
|
|
|
app.blueprint(bp)
|
|
app.blueprint(bp2)
|
|
|
|
assert bp.host == ["example.com", "sub.example.com"]
|
|
headers = {"Host": "example.com"}
|
|
request, response = app.test_client.get("/test/", headers=headers)
|
|
assert response.text == "Hello"
|
|
|
|
assert bp.host == ["example.com", "sub.example.com"]
|
|
headers = {"Host": "sub.example.com"}
|
|
request, response = app.test_client.get("/test/", headers=headers)
|
|
assert response.text == "Hello"
|
|
|
|
assert bp2.host == ["sub1.example.com", "sub2.example.com"]
|
|
headers = {"Host": "sub1.example.com"}
|
|
request, response = app.test_client.get("/test/", headers=headers)
|
|
assert response.text == "Hello2"
|
|
request, response = app.test_client.get("/test/other/", headers=headers)
|
|
assert response.text == "Hello3"
|
|
|
|
assert bp2.host == ["sub1.example.com", "sub2.example.com"]
|
|
headers = {"Host": "sub2.example.com"}
|
|
request, response = app.test_client.get("/test/", headers=headers)
|
|
assert response.text == "Hello2"
|
|
request, response = app.test_client.get("/test/other/", headers=headers)
|
|
assert response.text == "Hello3"
|
|
|
|
|
|
def test_bp_middleware(app: Sanic):
|
|
blueprint = Blueprint("test_bp_middleware")
|
|
|
|
@blueprint.middleware("response")
|
|
async def process_response(request, response):
|
|
return text("OK")
|
|
|
|
@app.route("/")
|
|
async def handler(request):
|
|
return text("FAIL")
|
|
|
|
app.blueprint(blueprint)
|
|
|
|
request, response = app.test_client.get("/")
|
|
|
|
assert response.status == 200
|
|
assert response.text == "FAIL"
|
|
|
|
|
|
def test_bp_middleware_with_route(app: Sanic):
|
|
blueprint = Blueprint("test_bp_middleware")
|
|
|
|
@blueprint.middleware("response")
|
|
async def process_response(request, response):
|
|
return text("OK")
|
|
|
|
@app.route("/")
|
|
async def handler(request):
|
|
return text("FAIL")
|
|
|
|
@blueprint.route("/bp")
|
|
async def bp_handler(request):
|
|
return text("FAIL")
|
|
|
|
app.blueprint(blueprint)
|
|
|
|
request, response = app.test_client.get("/bp")
|
|
|
|
assert response.status == 200
|
|
assert response.text == "OK"
|
|
|
|
|
|
def test_bp_middleware_order(app: Sanic):
|
|
blueprint = Blueprint("test_bp_middleware_order")
|
|
order = []
|
|
|
|
@blueprint.middleware("request")
|
|
def mw_1(request):
|
|
order.append(1)
|
|
|
|
@blueprint.middleware("request")
|
|
def mw_2(request):
|
|
order.append(2)
|
|
|
|
@blueprint.middleware("request")
|
|
def mw_3(request):
|
|
order.append(3)
|
|
|
|
@blueprint.middleware("response")
|
|
def mw_4(request, response):
|
|
order.append(6)
|
|
|
|
@blueprint.middleware("response")
|
|
def mw_5(request, response):
|
|
order.append(5)
|
|
|
|
@blueprint.middleware("response")
|
|
def mw_6(request, response):
|
|
order.append(4)
|
|
|
|
@blueprint.route("/")
|
|
def process_response(request):
|
|
return text("OK")
|
|
|
|
app.blueprint(blueprint)
|
|
order.clear()
|
|
request, response = app.test_client.get("/")
|
|
|
|
assert response.status == 200
|
|
assert order == [1, 2, 3, 4, 5, 6]
|
|
|
|
|
|
def test_bp_exception_handler(app: Sanic):
|
|
blueprint = Blueprint("test_middleware")
|
|
|
|
@blueprint.route("/1")
|
|
def handler_1(request):
|
|
raise BadRequest("OK")
|
|
|
|
@blueprint.route("/2")
|
|
def handler_2(request):
|
|
raise ServerError("OK")
|
|
|
|
@blueprint.route("/3")
|
|
def handler_3(request):
|
|
raise NotFound("OK")
|
|
|
|
@blueprint.exception(NotFound, ServerError)
|
|
def handler_exception(request, exception):
|
|
return text("OK")
|
|
|
|
app.blueprint(blueprint)
|
|
|
|
request, response = app.test_client.get("/1")
|
|
assert response.status == 400
|
|
|
|
request, response = app.test_client.get("/2")
|
|
assert response.status == 200
|
|
assert response.text == "OK"
|
|
|
|
request, response = app.test_client.get("/3")
|
|
assert response.status == 200
|
|
|
|
|
|
def test_bp_exception_handler_applied(app: Sanic):
|
|
class Error(Exception):
|
|
pass
|
|
|
|
handled = Blueprint("handled")
|
|
nothandled = Blueprint("nothandled")
|
|
|
|
@handled.exception(Error)
|
|
def handle_error(req, e):
|
|
return text(f"handled {e}")
|
|
|
|
@handled.route("/ok")
|
|
def ok(request):
|
|
raise Error("uh oh")
|
|
|
|
@nothandled.route("/notok")
|
|
def notok(request):
|
|
raise Error("uh oh")
|
|
|
|
app.blueprint(handled)
|
|
app.blueprint(nothandled)
|
|
|
|
_, response = app.test_client.get("/ok")
|
|
assert response.status == 200
|
|
assert response.text == "handled uh oh"
|
|
|
|
_, response = app.test_client.get("/notok")
|
|
assert response.status == 500
|
|
|
|
|
|
def test_bp_exception_handler_not_applied(app: Sanic):
|
|
class Error(Exception):
|
|
pass
|
|
|
|
handled = Blueprint("handled")
|
|
nothandled = Blueprint("nothandled")
|
|
|
|
@handled.exception(Error)
|
|
def handle_error(req, e):
|
|
return text(f"handled {e}")
|
|
|
|
@nothandled.route("/notok")
|
|
def notok(request):
|
|
raise Error("uh oh")
|
|
|
|
app.blueprint(handled)
|
|
app.blueprint(nothandled)
|
|
|
|
_, response = app.test_client.get("/notok")
|
|
assert response.status == 500
|
|
|
|
|
|
def test_bp_listeners(app: Sanic):
|
|
app.route("/")(lambda x: x)
|
|
blueprint = Blueprint("test_middleware")
|
|
|
|
order = []
|
|
|
|
@blueprint.listener("before_server_start")
|
|
def handler_1(sanic, loop):
|
|
order.append(1)
|
|
|
|
@blueprint.listener("after_server_start")
|
|
def handler_2(sanic, loop):
|
|
order.append(2)
|
|
|
|
@blueprint.listener("after_server_start")
|
|
def handler_3(sanic, loop):
|
|
order.append(3)
|
|
|
|
@blueprint.listener("before_server_stop")
|
|
def handler_4(sanic, loop):
|
|
order.append(5)
|
|
|
|
@blueprint.listener("before_server_stop")
|
|
def handler_5(sanic, loop):
|
|
order.append(4)
|
|
|
|
@blueprint.listener("after_server_stop")
|
|
def handler_6(sanic, loop):
|
|
order.append(6)
|
|
|
|
app.blueprint(blueprint)
|
|
|
|
request, response = app.test_client.get("/")
|
|
|
|
assert order == [1, 2, 3, 4, 5, 6]
|
|
|
|
|
|
def test_bp_static(app: Sanic):
|
|
current_file = inspect.getfile(inspect.currentframe())
|
|
with open(current_file, "rb") as file:
|
|
current_file_contents = file.read()
|
|
|
|
blueprint = Blueprint("test_static")
|
|
|
|
blueprint.static("/testing.file", current_file)
|
|
|
|
app.blueprint(blueprint)
|
|
|
|
request, response = app.test_client.get("/testing.file")
|
|
assert response.status == 200
|
|
assert response.body == current_file_contents
|
|
|
|
|
|
@pytest.mark.parametrize("file_name", ["test.html"])
|
|
def test_bp_static_content_type(app, file_name):
|
|
# This is done here, since no other test loads a file here
|
|
current_file = inspect.getfile(inspect.currentframe())
|
|
current_directory = os.path.dirname(os.path.abspath(current_file))
|
|
static_directory = os.path.join(current_directory, "static")
|
|
|
|
blueprint = Blueprint("test_static")
|
|
blueprint.static(
|
|
"/testing.file",
|
|
get_file_path(static_directory, file_name),
|
|
content_type="text/html; charset=utf-8",
|
|
)
|
|
|
|
app.blueprint(blueprint)
|
|
|
|
request, response = app.test_client.get("/testing.file")
|
|
assert response.status == 200
|
|
assert response.body == get_file_content(static_directory, file_name)
|
|
assert response.headers["Content-Type"] == "text/html; charset=utf-8"
|
|
|
|
|
|
def test_bp_shorthand(app: Sanic):
|
|
blueprint = Blueprint("test_shorhand_routes")
|
|
ev = asyncio.Event()
|
|
|
|
@blueprint.get("/get")
|
|
def handler(request):
|
|
return text("OK")
|
|
|
|
@blueprint.put("/put")
|
|
def put_handler(request):
|
|
return text("OK")
|
|
|
|
@blueprint.post("/post")
|
|
def post_handler(request):
|
|
return text("OK")
|
|
|
|
@blueprint.head("/head")
|
|
def head_handler(request):
|
|
return text("OK")
|
|
|
|
@blueprint.options("/options")
|
|
def options_handler(request):
|
|
return text("OK")
|
|
|
|
@blueprint.patch("/patch")
|
|
def patch_handler(request):
|
|
return text("OK")
|
|
|
|
@blueprint.delete("/delete")
|
|
def delete_handler(request):
|
|
return text("OK")
|
|
|
|
@blueprint.websocket("/ws/", strict_slashes=True)
|
|
async def websocket_handler(request, ws):
|
|
ev.set()
|
|
|
|
app.blueprint(blueprint)
|
|
|
|
request, response = app.test_client.get("/get")
|
|
assert response.body == b"OK"
|
|
|
|
request, response = app.test_client.post("/get")
|
|
assert response.status == 405
|
|
|
|
request, response = app.test_client.put("/put")
|
|
assert response.body == b"OK"
|
|
|
|
request, response = app.test_client.get("/post")
|
|
assert response.status == 405
|
|
|
|
request, response = app.test_client.post("/post")
|
|
assert response.body == b"OK"
|
|
|
|
request, response = app.test_client.get("/post")
|
|
assert response.status == 405
|
|
|
|
request, response = app.test_client.head("/head")
|
|
assert response.status == 200
|
|
|
|
request, response = app.test_client.get("/head")
|
|
assert response.status == 405
|
|
|
|
request, response = app.test_client.options("/options")
|
|
assert response.body == b"OK"
|
|
|
|
request, response = app.test_client.get("/options")
|
|
assert response.status == 405
|
|
|
|
request, response = app.test_client.patch("/patch")
|
|
assert response.body == b"OK"
|
|
|
|
request, response = app.test_client.get("/patch")
|
|
assert response.status == 405
|
|
|
|
request, response = app.test_client.delete("/delete")
|
|
assert response.body == b"OK"
|
|
|
|
request, response = app.test_client.get("/delete")
|
|
assert response.status == 405
|
|
|
|
request, response = app.test_client.websocket("/ws/")
|
|
assert response.opened is True
|
|
assert ev.is_set()
|
|
|
|
|
|
def test_bp_group(app: Sanic):
|
|
deep_0 = Blueprint("deep_0", url_prefix="/deep")
|
|
deep_1 = Blueprint("deep_1", url_prefix="/deep1")
|
|
|
|
@deep_0.route("/")
|
|
def handler(request):
|
|
return text("D0_OK")
|
|
|
|
@deep_1.route("/bottom")
|
|
def bottom_handler(request):
|
|
return text("D1B_OK")
|
|
|
|
mid_0 = Blueprint.group(deep_0, deep_1, url_prefix="/mid")
|
|
mid_1 = Blueprint("mid_tier", url_prefix="/mid1")
|
|
|
|
@mid_1.route("/")
|
|
def handler1(request):
|
|
return text("M1_OK")
|
|
|
|
top = Blueprint.group(mid_0, mid_1)
|
|
|
|
app.blueprint(top)
|
|
|
|
@app.route("/")
|
|
def handler2(request):
|
|
return text("TOP_OK")
|
|
|
|
request, response = app.test_client.get("/")
|
|
assert response.text == "TOP_OK"
|
|
|
|
request, response = app.test_client.get("/mid1")
|
|
assert response.text == "M1_OK"
|
|
|
|
request, response = app.test_client.get("/mid/deep")
|
|
assert response.text == "D0_OK"
|
|
|
|
request, response = app.test_client.get("/mid/deep1/bottom")
|
|
assert response.text == "D1B_OK"
|
|
|
|
|
|
def test_bp_group_with_default_url_prefix(app: Sanic):
|
|
from sanic.response import json
|
|
|
|
bp_resources = Blueprint("bp_resources")
|
|
|
|
@bp_resources.get("/")
|
|
def list_resources_handler(request):
|
|
resource = {}
|
|
return json([resource])
|
|
|
|
bp_resource = Blueprint("bp_resource", url_prefix="/<resource_id>")
|
|
|
|
@bp_resource.get("/")
|
|
def get_resource_hander(request, resource_id):
|
|
resource = {"resource_id": resource_id}
|
|
return json(resource)
|
|
|
|
bp_resources_group = Blueprint.group(
|
|
bp_resources, bp_resource, url_prefix="/resources"
|
|
)
|
|
bp_api_v1 = Blueprint("bp_api_v1")
|
|
|
|
@bp_api_v1.get("/info")
|
|
def api_v1_info(request):
|
|
return text("api_version: 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")
|
|
app.blueprint(bp_api_group)
|
|
|
|
request, response = app.test_client.get("/api/v1/info")
|
|
assert response.text == "api_version: v1"
|
|
|
|
request, response = app.test_client.get("/api/v1/resources")
|
|
assert response.json == [{}]
|
|
|
|
from uuid import uuid4
|
|
|
|
resource_id = str(uuid4())
|
|
request, response = app.test_client.get(f"/api/v1/resources/{resource_id}")
|
|
assert response.json == {"resource_id": resource_id}
|
|
|
|
|
|
def test_blueprint_middleware_with_args(app: Sanic):
|
|
bp = Blueprint(name="with_args_bp", url_prefix="/wa")
|
|
|
|
@bp.middleware
|
|
def middleware_with_no_tag(request: Request):
|
|
if request.headers.get("content-type") == "application/json":
|
|
request.headers["accepts"] = "plain/text"
|
|
else:
|
|
request.headers["accepts"] = "application/json"
|
|
|
|
@bp.route("/")
|
|
def default_route(request):
|
|
if request.headers.get("accepts") == "application/json":
|
|
return json({"test": "value"})
|
|
else:
|
|
return text("value")
|
|
|
|
app.blueprint(bp)
|
|
|
|
_, response = app.test_client.get(
|
|
"/wa", headers={"content-type": "application/json"}
|
|
)
|
|
assert response.text == "value"
|
|
|
|
_, response = app.test_client.get(
|
|
"/wa", headers={"content-type": "plain/text"}
|
|
)
|
|
assert response.json.get("test") == "value"
|
|
|
|
|
|
@pytest.mark.parametrize("file_name", ["test.file"])
|
|
def test_static_blueprint_name(static_file_directory, file_name):
|
|
app = Sanic("app")
|
|
current_file = inspect.getfile(inspect.currentframe())
|
|
with open(current_file, "rb") as file:
|
|
file.read()
|
|
|
|
bp = Blueprint(name="static", url_prefix="/static", strict_slashes=False)
|
|
|
|
bp.static(
|
|
"/test.file/",
|
|
get_file_path(static_file_directory, file_name),
|
|
name="static.testing",
|
|
strict_slashes=True,
|
|
)
|
|
|
|
app.blueprint(bp)
|
|
|
|
uri = app.url_for("static", name="static.testing")
|
|
assert uri == "/static/test.file/"
|
|
|
|
_, response = app.test_client.get("/static/test.file")
|
|
assert response.status == 404
|
|
|
|
_, response = app.test_client.get("/static/test.file/")
|
|
assert response.status == 200
|
|
|
|
|
|
@pytest.mark.parametrize("file_name", ["test.file"])
|
|
def test_static_blueprintp_mw(app: Sanic, static_file_directory, file_name):
|
|
current_file = inspect.getfile(inspect.currentframe()) # type: ignore
|
|
with open(current_file, "rb") as file:
|
|
file.read()
|
|
|
|
triggered = False
|
|
|
|
bp = Blueprint(name="test_mw", url_prefix="")
|
|
|
|
@bp.middleware("request")
|
|
def bp_mw1(request):
|
|
nonlocal triggered
|
|
triggered = True
|
|
|
|
bp.static(
|
|
"/test.file",
|
|
get_file_path(static_file_directory, file_name),
|
|
strict_slashes=True,
|
|
name="static",
|
|
)
|
|
|
|
app.blueprint(bp)
|
|
|
|
uri = app.url_for("test_mw.static")
|
|
assert uri == "/test.file"
|
|
|
|
_, response = app.test_client.get("/test.file")
|
|
assert triggered is True
|
|
|
|
|
|
def test_websocket_route(app: Sanic):
|
|
event = asyncio.Event()
|
|
|
|
async def websocket_handler(request, ws):
|
|
assert ws.subprotocol is None
|
|
event.set()
|
|
|
|
bp = Blueprint(name="handler", url_prefix="/ws")
|
|
bp.add_websocket_route(websocket_handler, "/test", name="test")
|
|
|
|
app.blueprint(bp)
|
|
|
|
_, response = app.test_client.websocket("/ws/test")
|
|
assert response.opened is True
|
|
assert event.is_set()
|
|
|
|
|
|
def test_duplicate_blueprint(app: Sanic):
|
|
bp_name = "bp"
|
|
bp = Blueprint(bp_name)
|
|
bp1 = Blueprint(bp_name)
|
|
|
|
app.blueprint(bp)
|
|
|
|
with pytest.raises(AssertionError) as excinfo:
|
|
app.blueprint(bp1)
|
|
|
|
assert str(excinfo.value) == (
|
|
f'A blueprint with the name "{bp_name}" is already registered. '
|
|
"Blueprint names must be unique."
|
|
)
|
|
|
|
|
|
def test_strict_slashes_behavior_adoption():
|
|
app = Sanic("app")
|
|
app.strict_slashes = True
|
|
bp = Blueprint("bp")
|
|
bp2 = Blueprint("bp2", strict_slashes=False)
|
|
|
|
@app.get("/test")
|
|
def handler_test(request):
|
|
return text("Test")
|
|
|
|
@app.get("/f1", strict_slashes=False)
|
|
def f1(request):
|
|
return text("f1")
|
|
|
|
@bp.get("/one", strict_slashes=False)
|
|
def one(request):
|
|
return text("one")
|
|
|
|
@bp.get("/second")
|
|
def second(request):
|
|
return text("second")
|
|
|
|
@bp2.get("/third")
|
|
def third(request):
|
|
return text("third")
|
|
|
|
app.blueprint(bp)
|
|
app.blueprint(bp2)
|
|
|
|
assert app.test_client.get("/test")[1].status == 200
|
|
assert app.test_client.get("/test/")[1].status == 404
|
|
|
|
assert app.test_client.get("/one")[1].status == 200
|
|
assert app.test_client.get("/one/")[1].status == 200
|
|
|
|
assert app.test_client.get("/second")[1].status == 200
|
|
assert app.test_client.get("/second/")[1].status == 404
|
|
|
|
assert app.test_client.get("/third")[1].status == 200
|
|
assert app.test_client.get("/third/")[1].status == 200
|
|
|
|
assert app.test_client.get("/f1")[1].status == 200
|
|
assert app.test_client.get("/f1/")[1].status == 200
|
|
|
|
|
|
def test_blueprint_group_versioning():
|
|
app = Sanic(name="blueprint-group-test")
|
|
|
|
bp1 = Blueprint(name="bp1", url_prefix="/bp1")
|
|
bp2 = Blueprint(name="bp2", url_prefix="/bp2", version=2)
|
|
|
|
bp3 = Blueprint(name="bp3", url_prefix="/bp3")
|
|
|
|
@bp3.get("/r1")
|
|
async def bp3_r1(request):
|
|
return json({"from": "bp3/r1"})
|
|
|
|
@bp1.get("/pre-group")
|
|
async def pre_group(request):
|
|
return json({"from": "bp1/pre-group"})
|
|
|
|
group = Blueprint.group([bp1, bp2], url_prefix="/group1", version=1)
|
|
|
|
group2 = Blueprint.group([bp3])
|
|
|
|
@bp1.get("/r1")
|
|
async def r1(request):
|
|
return json({"from": "bp1/r1"})
|
|
|
|
@bp2.get("/r2")
|
|
async def r2(request):
|
|
return json({"from": "bp2/r2"})
|
|
|
|
@bp2.get("/r3", version=3)
|
|
async def r3(request):
|
|
return json({"from": "bp2/r3"})
|
|
|
|
app.blueprint([group, group2])
|
|
|
|
assert app.test_client.get("/v1/group1/bp1/r1/")[1].status == 200
|
|
assert app.test_client.get("/v2/group1/bp2/r2")[1].status == 200
|
|
assert app.test_client.get("/v1/group1/bp1/pre-group")[1].status == 200
|
|
assert app.test_client.get("/v3/group1/bp2/r3")[1].status == 200
|
|
assert app.test_client.get("/bp3/r1")[1].status == 200
|
|
|
|
assert group.version == 1
|
|
assert group2.strict_slashes is None
|
|
|
|
|
|
def test_blueprint_group_strict_slashes():
|
|
app = Sanic(name="blueprint-group-test")
|
|
bp1 = Blueprint(name="bp1", url_prefix=None, strict_slashes=False)
|
|
|
|
bp2 = Blueprint(
|
|
name="bp2", version=3, url_prefix="/bp2", strict_slashes=None
|
|
)
|
|
|
|
bp3 = Blueprint(
|
|
name="bp3", version=None, url_prefix="/bp3/", strict_slashes=None
|
|
)
|
|
|
|
@bp1.get("/r1")
|
|
async def bp1_r1(request):
|
|
return json({"from": "bp1/r1"})
|
|
|
|
@bp2.get("/r1")
|
|
async def bp2_r1(request):
|
|
return json({"from": "bp2/r1"})
|
|
|
|
@bp2.get("/r2/")
|
|
async def bp2_r2(request):
|
|
return json({"from": "bp2/r2"})
|
|
|
|
@bp3.get("/r1")
|
|
async def bp3_r1(request):
|
|
return json({"from": "bp3/r1"})
|
|
|
|
group = Blueprint.group(
|
|
[bp1, bp2],
|
|
url_prefix="/slash-check/",
|
|
version=1.3,
|
|
strict_slashes=True,
|
|
)
|
|
|
|
group2 = Blueprint.group(
|
|
[bp3], url_prefix="/other-prefix/", version="v2", strict_slashes=False
|
|
)
|
|
|
|
app.blueprint(group)
|
|
app.blueprint(group2)
|
|
|
|
assert app.test_client.get("/v1.3/slash-check/r1")[1].status == 200
|
|
assert app.test_client.get("/v1.3/slash-check/r1/")[1].status == 200
|
|
assert app.test_client.get("/v3/slash-check/bp2/r1")[1].status == 200
|
|
assert app.test_client.get("/v3/slash-check/bp2/r1/")[1].status == 404
|
|
assert app.test_client.get("/v3/slash-check/bp2/r2")[1].status == 404
|
|
assert app.test_client.get("/v3/slash-check/bp2/r2/")[1].status == 200
|
|
assert app.test_client.get("/v2/other-prefix/bp3/r1")[1].status == 200
|
|
|
|
|
|
def test_blueprint_registered_multiple_apps():
|
|
app1 = Sanic("app1")
|
|
app2 = Sanic("app2")
|
|
bp = Blueprint("bp")
|
|
|
|
@bp.get("/")
|
|
async def handler(request):
|
|
return text(request.route.name)
|
|
|
|
app1.blueprint(bp)
|
|
app2.blueprint(bp)
|
|
|
|
for app in (app1, app2):
|
|
_, response = app.test_client.get("/")
|
|
assert response.text == f"{app.name}.bp.handler"
|
|
|
|
|
|
def test_bp_set_attribute_warning():
|
|
bp = Blueprint("bp")
|
|
message = (
|
|
"Setting variables on Blueprint instances is not allowed. You should "
|
|
"change your Blueprint instance to use instance.ctx.foo instead."
|
|
)
|
|
with pytest.raises(AttributeError, match=message):
|
|
bp.foo = 1
|
|
|
|
|
|
def test_early_registration(app: Sanic):
|
|
assert len(app.router.routes) == 0
|
|
|
|
bp = Blueprint("bp")
|
|
|
|
@bp.get("/one")
|
|
async def one(_):
|
|
return text("one")
|
|
|
|
app.blueprint(bp)
|
|
|
|
assert len(app.router.routes) == 1
|
|
|
|
@bp.get("/two")
|
|
async def two(_):
|
|
return text("two")
|
|
|
|
@bp.get("/three")
|
|
async def three(_):
|
|
return text("three")
|
|
|
|
assert len(app.router.routes) == 3
|
|
|
|
for path in ("one", "two", "three"):
|
|
_, response = app.test_client.get(f"/{path}")
|
|
assert response.text == path
|
|
|
|
|
|
def test_remove_double_slashes_defined_on_bp(app: Sanic):
|
|
bp = Blueprint("bp", url_prefix="/foo/", strict_slashes=True)
|
|
|
|
@bp.get("/")
|
|
async def handler(_):
|
|
...
|
|
|
|
app.blueprint(bp)
|
|
app.router.finalize()
|
|
|
|
assert app.router.routes[0].path == "foo/"
|
|
|
|
|
|
def test_remove_double_slashes_defined_on_register(app: Sanic):
|
|
bp = Blueprint("bp")
|
|
|
|
@bp.get("/")
|
|
async def index(_):
|
|
...
|
|
|
|
app.blueprint(bp, url_prefix="/foo/", strict_slashes=True)
|
|
app.router.finalize()
|
|
|
|
assert app.router.routes[0].path == "foo/"
|