2020-09-30 13:11:27 +01:00
|
|
|
import pytest
|
|
|
|
|
|
|
|
from sanic import Sanic
|
2021-11-17 17:36:36 +00:00
|
|
|
from sanic.config import Config
|
2021-09-29 21:53:49 +01:00
|
|
|
from sanic.errorpages import HTMLRenderer, exception_response
|
|
|
|
from sanic.exceptions import NotFound, SanicException
|
2021-11-16 11:07:33 +00:00
|
|
|
from sanic.handlers import ErrorHandler
|
2020-09-30 13:11:27 +01:00
|
|
|
from sanic.request import Request
|
2021-09-29 21:53:49 +01:00
|
|
|
from sanic.response import HTTPResponse, html, json, text
|
2020-09-30 13:11:27 +01:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
|
|
|
def app():
|
|
|
|
app = Sanic("error_page_testing")
|
|
|
|
|
|
|
|
@app.route("/error", methods=["GET", "POST"])
|
|
|
|
def err(request):
|
|
|
|
raise Exception("something went wrong")
|
|
|
|
|
|
|
|
return app
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
|
|
|
def fake_request(app):
|
2021-09-29 21:53:49 +01:00
|
|
|
return Request(b"/foobar", {"accept": "*/*"}, "1.1", "GET", None, app)
|
2020-09-30 13:11:27 +01:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"fallback,content_type, exception, status",
|
|
|
|
(
|
|
|
|
(None, "text/html; charset=utf-8", Exception, 500),
|
|
|
|
("html", "text/html; charset=utf-8", Exception, 500),
|
|
|
|
("auto", "text/html; charset=utf-8", Exception, 500),
|
|
|
|
("text", "text/plain; charset=utf-8", Exception, 500),
|
|
|
|
("json", "application/json", Exception, 500),
|
|
|
|
(None, "text/html; charset=utf-8", NotFound, 404),
|
|
|
|
("html", "text/html; charset=utf-8", NotFound, 404),
|
|
|
|
("auto", "text/html; charset=utf-8", NotFound, 404),
|
|
|
|
("text", "text/plain; charset=utf-8", NotFound, 404),
|
|
|
|
("json", "application/json", NotFound, 404),
|
|
|
|
),
|
|
|
|
)
|
|
|
|
def test_should_return_html_valid_setting(
|
|
|
|
fake_request, fallback, content_type, exception, status
|
|
|
|
):
|
|
|
|
if fallback:
|
|
|
|
fake_request.app.config.FALLBACK_ERROR_FORMAT = fallback
|
|
|
|
|
|
|
|
try:
|
|
|
|
raise exception("bad stuff")
|
|
|
|
except Exception as e:
|
2021-09-29 21:53:49 +01:00
|
|
|
response = exception_response(
|
|
|
|
fake_request,
|
|
|
|
e,
|
|
|
|
True,
|
|
|
|
base=HTMLRenderer,
|
|
|
|
fallback=fake_request.app.config.FALLBACK_ERROR_FORMAT,
|
|
|
|
)
|
2020-09-30 13:11:27 +01:00
|
|
|
|
|
|
|
assert isinstance(response, HTTPResponse)
|
|
|
|
assert response.status == status
|
|
|
|
assert response.content_type == content_type
|
|
|
|
|
|
|
|
|
|
|
|
def test_auto_fallback_with_data(app):
|
|
|
|
app.config.FALLBACK_ERROR_FORMAT = "auto"
|
|
|
|
|
|
|
|
_, response = app.test_client.get("/error")
|
|
|
|
assert response.status == 500
|
2022-01-12 14:28:43 +00:00
|
|
|
assert response.content_type == "text/plain; charset=utf-8"
|
2020-09-30 13:11:27 +01:00
|
|
|
|
|
|
|
_, response = app.test_client.post("/error", json={"foo": "bar"})
|
|
|
|
assert response.status == 500
|
|
|
|
assert response.content_type == "application/json"
|
|
|
|
|
|
|
|
_, response = app.test_client.post("/error", data={"foo": "bar"})
|
|
|
|
assert response.status == 500
|
2022-01-12 14:28:43 +00:00
|
|
|
assert response.content_type == "text/plain; charset=utf-8"
|
2020-09-30 13:11:27 +01:00
|
|
|
|
|
|
|
|
|
|
|
def test_auto_fallback_with_content_type(app):
|
|
|
|
app.config.FALLBACK_ERROR_FORMAT = "auto"
|
|
|
|
|
|
|
|
_, response = app.test_client.get(
|
2021-09-29 21:53:49 +01:00
|
|
|
"/error", headers={"content-type": "application/json", "accept": "*/*"}
|
2020-09-30 13:11:27 +01:00
|
|
|
)
|
|
|
|
assert response.status == 500
|
|
|
|
assert response.content_type == "application/json"
|
|
|
|
|
|
|
|
_, response = app.test_client.get(
|
2021-09-29 21:53:49 +01:00
|
|
|
"/error", headers={"content-type": "foo/bar", "accept": "*/*"}
|
|
|
|
)
|
|
|
|
assert response.status == 500
|
2022-01-12 14:28:43 +00:00
|
|
|
assert response.content_type == "text/plain; charset=utf-8"
|
2021-09-29 21:53:49 +01:00
|
|
|
|
|
|
|
|
|
|
|
def test_route_error_format_set_on_auto(app):
|
|
|
|
@app.get("/text")
|
|
|
|
def text_response(request):
|
|
|
|
return text(request.route.ctx.error_format)
|
|
|
|
|
|
|
|
@app.get("/json")
|
|
|
|
def json_response(request):
|
|
|
|
return json({"format": request.route.ctx.error_format})
|
|
|
|
|
|
|
|
@app.get("/html")
|
|
|
|
def html_response(request):
|
|
|
|
return html(request.route.ctx.error_format)
|
|
|
|
|
|
|
|
_, response = app.test_client.get("/text")
|
|
|
|
assert response.text == "text"
|
|
|
|
|
|
|
|
_, response = app.test_client.get("/json")
|
|
|
|
assert response.json["format"] == "json"
|
|
|
|
|
|
|
|
_, response = app.test_client.get("/html")
|
|
|
|
assert response.text == "html"
|
|
|
|
|
|
|
|
|
|
|
|
def test_route_error_response_from_auto_route(app):
|
|
|
|
@app.get("/text")
|
|
|
|
def text_response(request):
|
|
|
|
raise Exception("oops")
|
|
|
|
return text("Never gonna see this")
|
|
|
|
|
|
|
|
@app.get("/json")
|
|
|
|
def json_response(request):
|
|
|
|
raise Exception("oops")
|
|
|
|
return json({"message": "Never gonna see this"})
|
|
|
|
|
|
|
|
@app.get("/html")
|
|
|
|
def html_response(request):
|
|
|
|
raise Exception("oops")
|
|
|
|
return html("<h1>Never gonna see this</h1>")
|
|
|
|
|
|
|
|
_, response = app.test_client.get("/text")
|
|
|
|
assert response.content_type == "text/plain; charset=utf-8"
|
|
|
|
|
|
|
|
_, response = app.test_client.get("/json")
|
|
|
|
assert response.content_type == "application/json"
|
|
|
|
|
|
|
|
_, response = app.test_client.get("/html")
|
|
|
|
assert response.content_type == "text/html; charset=utf-8"
|
|
|
|
|
|
|
|
|
|
|
|
def test_route_error_response_from_explicit_format(app):
|
|
|
|
@app.get("/text", error_format="json")
|
|
|
|
def text_response(request):
|
|
|
|
raise Exception("oops")
|
|
|
|
return text("Never gonna see this")
|
|
|
|
|
|
|
|
@app.get("/json", error_format="text")
|
|
|
|
def json_response(request):
|
|
|
|
raise Exception("oops")
|
|
|
|
return json({"message": "Never gonna see this"})
|
|
|
|
|
|
|
|
_, response = app.test_client.get("/text")
|
|
|
|
assert response.content_type == "application/json"
|
|
|
|
|
|
|
|
_, response = app.test_client.get("/json")
|
|
|
|
assert response.content_type == "text/plain; charset=utf-8"
|
|
|
|
|
|
|
|
|
|
|
|
def test_unknown_fallback_format(app):
|
|
|
|
with pytest.raises(SanicException, match="Unknown format: bad"):
|
|
|
|
app.config.FALLBACK_ERROR_FORMAT = "bad"
|
|
|
|
|
|
|
|
|
|
|
|
def test_route_error_format_unknown(app):
|
|
|
|
with pytest.raises(SanicException, match="Unknown format: bad"):
|
|
|
|
|
|
|
|
@app.get("/text", error_format="bad")
|
|
|
|
def handler(request):
|
|
|
|
...
|
|
|
|
|
|
|
|
|
2022-01-12 14:28:43 +00:00
|
|
|
def test_fallback_with_content_type_html(app):
|
|
|
|
app.config.FALLBACK_ERROR_FORMAT = "auto"
|
|
|
|
|
|
|
|
_, response = app.test_client.get(
|
|
|
|
"/error",
|
|
|
|
headers={"content-type": "application/json", "accept": "text/html"},
|
|
|
|
)
|
|
|
|
assert response.status == 500
|
|
|
|
assert response.content_type == "text/html; charset=utf-8"
|
|
|
|
|
|
|
|
|
2021-09-29 21:53:49 +01:00
|
|
|
def test_fallback_with_content_type_mismatch_accept(app):
|
|
|
|
app.config.FALLBACK_ERROR_FORMAT = "auto"
|
|
|
|
|
|
|
|
_, response = app.test_client.get(
|
|
|
|
"/error",
|
|
|
|
headers={"content-type": "application/json", "accept": "text/plain"},
|
2020-09-30 13:11:27 +01:00
|
|
|
)
|
|
|
|
assert response.status == 500
|
|
|
|
assert response.content_type == "text/plain; charset=utf-8"
|
2021-09-29 21:53:49 +01:00
|
|
|
|
|
|
|
_, response = app.test_client.get(
|
|
|
|
"/error",
|
2022-01-12 14:28:43 +00:00
|
|
|
headers={"content-type": "text/html", "accept": "foo/bar"},
|
2021-09-29 21:53:49 +01:00
|
|
|
)
|
|
|
|
assert response.status == 500
|
2022-01-12 14:28:43 +00:00
|
|
|
assert response.content_type == "text/plain; charset=utf-8"
|
2021-09-29 21:53:49 +01:00
|
|
|
|
|
|
|
app.router.reset()
|
|
|
|
|
|
|
|
@app.route("/alt1")
|
|
|
|
@app.route("/alt2", error_format="text")
|
|
|
|
@app.route("/alt3", error_format="html")
|
|
|
|
def handler(_):
|
|
|
|
raise Exception("problem here")
|
|
|
|
# Yes, we know this return value is unreachable. This is on purpose.
|
|
|
|
return json({})
|
|
|
|
|
|
|
|
app.router.finalize()
|
|
|
|
|
|
|
|
_, response = app.test_client.get(
|
|
|
|
"/alt1",
|
|
|
|
headers={"accept": "foo/bar"},
|
|
|
|
)
|
|
|
|
assert response.status == 500
|
2022-01-12 14:28:43 +00:00
|
|
|
assert response.content_type == "text/plain; charset=utf-8"
|
2021-09-29 21:53:49 +01:00
|
|
|
_, response = app.test_client.get(
|
|
|
|
"/alt1",
|
|
|
|
headers={"accept": "foo/bar,*/*"},
|
|
|
|
)
|
|
|
|
assert response.status == 500
|
|
|
|
assert response.content_type == "application/json"
|
|
|
|
|
|
|
|
_, response = app.test_client.get(
|
|
|
|
"/alt2",
|
|
|
|
headers={"accept": "foo/bar"},
|
|
|
|
)
|
|
|
|
assert response.status == 500
|
2022-01-12 14:28:43 +00:00
|
|
|
assert response.content_type == "text/plain; charset=utf-8"
|
2021-09-29 21:53:49 +01:00
|
|
|
_, response = app.test_client.get(
|
|
|
|
"/alt2",
|
|
|
|
headers={"accept": "foo/bar,*/*"},
|
|
|
|
)
|
|
|
|
assert response.status == 500
|
|
|
|
assert response.content_type == "text/plain; charset=utf-8"
|
|
|
|
|
|
|
|
_, response = app.test_client.get(
|
|
|
|
"/alt3",
|
|
|
|
headers={"accept": "foo/bar"},
|
|
|
|
)
|
|
|
|
assert response.status == 500
|
2022-01-12 14:28:43 +00:00
|
|
|
assert response.content_type == "text/plain; charset=utf-8"
|
|
|
|
|
|
|
|
_, response = app.test_client.get(
|
|
|
|
"/alt3",
|
|
|
|
headers={"accept": "foo/bar,text/html"},
|
|
|
|
)
|
|
|
|
assert response.status == 500
|
2021-09-29 21:53:49 +01:00
|
|
|
assert response.content_type == "text/html; charset=utf-8"
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"accept,content_type,expected",
|
|
|
|
(
|
|
|
|
(None, None, "text/plain; charset=utf-8"),
|
|
|
|
("foo/bar", None, "text/html; charset=utf-8"),
|
|
|
|
("application/json", None, "application/json"),
|
|
|
|
("application/json,text/plain", None, "application/json"),
|
|
|
|
("text/plain,application/json", None, "application/json"),
|
|
|
|
("text/plain,foo/bar", None, "text/plain; charset=utf-8"),
|
|
|
|
# Following test is valid after v22.3
|
|
|
|
# ("text/plain,text/html", None, "text/plain; charset=utf-8"),
|
|
|
|
("*/*", "foo/bar", "text/html; charset=utf-8"),
|
|
|
|
("*/*", "application/json", "application/json"),
|
|
|
|
),
|
|
|
|
)
|
|
|
|
def test_combinations_for_auto(fake_request, accept, content_type, expected):
|
|
|
|
if accept:
|
|
|
|
fake_request.headers["accept"] = accept
|
|
|
|
else:
|
|
|
|
del fake_request.headers["accept"]
|
|
|
|
|
|
|
|
if content_type:
|
|
|
|
fake_request.headers["content-type"] = content_type
|
|
|
|
|
|
|
|
try:
|
|
|
|
raise Exception("bad stuff")
|
|
|
|
except Exception as e:
|
|
|
|
response = exception_response(
|
|
|
|
fake_request,
|
|
|
|
e,
|
|
|
|
True,
|
|
|
|
base=HTMLRenderer,
|
|
|
|
fallback="auto",
|
|
|
|
)
|
|
|
|
|
|
|
|
assert response.content_type == expected
|
2021-11-16 11:07:33 +00:00
|
|
|
|
|
|
|
|
|
|
|
def test_allow_fallback_error_format_set_main_process_start(app):
|
|
|
|
@app.main_process_start
|
|
|
|
async def start(app, _):
|
|
|
|
app.config.FALLBACK_ERROR_FORMAT = "text"
|
|
|
|
|
2021-12-18 16:58:14 +00:00
|
|
|
_, response = app.test_client.get("/error")
|
2021-11-16 11:07:33 +00:00
|
|
|
assert response.status == 500
|
|
|
|
assert response.content_type == "text/plain; charset=utf-8"
|
|
|
|
|
|
|
|
|
2021-12-18 16:58:14 +00:00
|
|
|
def test_setting_fallback_on_config_changes_as_expected(app):
|
|
|
|
app.error_handler = ErrorHandler()
|
2021-11-16 11:07:33 +00:00
|
|
|
|
2021-12-18 16:58:14 +00:00
|
|
|
_, response = app.test_client.get("/error")
|
2022-01-12 14:28:43 +00:00
|
|
|
assert response.content_type == "text/plain; charset=utf-8"
|
|
|
|
|
|
|
|
app.config.FALLBACK_ERROR_FORMAT = "html"
|
|
|
|
_, response = app.test_client.get("/error")
|
2021-12-18 16:58:14 +00:00
|
|
|
assert response.content_type == "text/html; charset=utf-8"
|
2021-11-16 11:07:33 +00:00
|
|
|
|
|
|
|
app.config.FALLBACK_ERROR_FORMAT = "text"
|
2021-12-18 16:58:14 +00:00
|
|
|
_, response = app.test_client.get("/error")
|
|
|
|
assert response.content_type == "text/plain; charset=utf-8"
|
2021-11-17 17:36:36 +00:00
|
|
|
|
|
|
|
|
|
|
|
def test_allow_fallback_error_format_in_config_injection():
|
|
|
|
class MyConfig(Config):
|
|
|
|
FALLBACK_ERROR_FORMAT = "text"
|
|
|
|
|
|
|
|
app = Sanic("test", config=MyConfig())
|
|
|
|
|
|
|
|
@app.route("/error", methods=["GET", "POST"])
|
|
|
|
def err(request):
|
|
|
|
raise Exception("something went wrong")
|
|
|
|
|
|
|
|
request, response = app.test_client.get("/error")
|
|
|
|
assert response.status == 500
|
|
|
|
assert response.content_type == "text/plain; charset=utf-8"
|
|
|
|
|
|
|
|
|
|
|
|
def test_allow_fallback_error_format_in_config_replacement(app):
|
|
|
|
class MyConfig(Config):
|
|
|
|
FALLBACK_ERROR_FORMAT = "text"
|
|
|
|
|
|
|
|
app.config = MyConfig()
|
|
|
|
|
|
|
|
request, response = app.test_client.get("/error")
|
|
|
|
assert response.status == 500
|
|
|
|
assert response.content_type == "text/plain; charset=utf-8"
|
2021-12-18 16:58:14 +00:00
|
|
|
|
|
|
|
|
|
|
|
def test_config_fallback_before_and_after_startup(app):
|
|
|
|
app.config.FALLBACK_ERROR_FORMAT = "json"
|
|
|
|
|
|
|
|
@app.main_process_start
|
|
|
|
async def start(app, _):
|
|
|
|
app.config.FALLBACK_ERROR_FORMAT = "text"
|
|
|
|
|
|
|
|
_, response = app.test_client.get("/error")
|
|
|
|
assert response.status == 500
|
|
|
|
assert response.content_type == "text/plain; charset=utf-8"
|
|
|
|
|
|
|
|
|
2022-01-06 10:40:52 +00:00
|
|
|
def test_config_fallback_using_update_dict(app):
|
|
|
|
app.config.update({"FALLBACK_ERROR_FORMAT": "text"})
|
|
|
|
|
|
|
|
_, response = app.test_client.get("/error")
|
|
|
|
assert response.status == 500
|
|
|
|
assert response.content_type == "text/plain; charset=utf-8"
|
|
|
|
|
|
|
|
|
|
|
|
def test_config_fallback_using_update_kwarg(app):
|
|
|
|
app.config.update(FALLBACK_ERROR_FORMAT="text")
|
|
|
|
|
|
|
|
_, response = app.test_client.get("/error")
|
|
|
|
assert response.status == 500
|
|
|
|
assert response.content_type == "text/plain; charset=utf-8"
|
|
|
|
|
|
|
|
|
2021-12-18 16:58:14 +00:00
|
|
|
def test_config_fallback_bad_value(app):
|
|
|
|
message = "Unknown format: fake"
|
|
|
|
with pytest.raises(SanicException, match=message):
|
|
|
|
app.config.FALLBACK_ERROR_FORMAT = "fake"
|