import random import string from pathlib import Path import pytest import yaml from starlette.middleware.base import BaseHTTPMiddleware from starlette.testclient import TestClient as StarletteTestClient import responder from responder.routes import Route, WebSocketRoute from responder.templates import Templates def test_api_basic_route(api): @api.route("/") def home(req, resp): resp.text = "hello world!" def test_route_repr(): def home(req, resp): """Home page""" resp.text = "Hello !" route = Route("/", home) assert route.__repr__() == f"" assert route.endpoint_name == home.__name__ assert route.description == home.__doc__ def test_websocket_route_repr(): def chat_endpoint(ws): """Chat""" pass route = WebSocketRoute("/", chat_endpoint) assert route.__repr__() == f"" assert route.endpoint_name == chat_endpoint.__name__ assert route.description == chat_endpoint.__doc__ def test_route_eq(): def home(req, resp): resp.text = "Hello !" assert Route("/", home) == Route("/", home) def chat(ws): pass assert WebSocketRoute("/", home) == WebSocketRoute("/", home) def test_route_int_convertor(api): @api.route("/items/{id:int}") def item(req, resp, *, id): # noqa: A002 resp.media = {"id": id, "type": type(id).__name__} r = api.requests.get(api.url_for(item, id=42)) assert r.json() == {"id": 42, "type": "int"} def test_route_float_convertor(api): @api.route("/price/{amount:float}") def price(req, resp, *, amount): resp.media = {"amount": amount} r = api.requests.get(api.url_for(price, amount=9.99)) assert r.json() == {"amount": 9.99} def test_route_path_convertor(api): @api.route("/files/{filepath:path}") def serve_file(req, resp, *, filepath): resp.text = filepath r = api.requests.get("http://;/files/docs/api/index.html") assert r.text == "docs/api/index.html" def test_route_uuid_convertor(api): @api.route("/users/{user_id:uuid}") def user(req, resp, *, user_id): resp.media = {"user_id": user_id} test_uuid = "12345678-1234-5678-1234-567812345678" r = api.requests.get(f"http://;/users/{test_uuid}") assert r.json() == {"user_id": test_uuid} def test_class_based_view_registration(api): @api.route("/") class ThingsResource: def on_request(req, resp): resp.text = "42" def test_class_based_view_parameters(api): @api.route("/{greeting}") class Greeting: pass resp = api.session().get("http://;/Hello") assert resp.status_code == api.status_codes.HTTP_405 def test_requests_session(api): assert api.session() assert api.requests def test_requests_session_works(api): TEXT = "spiral out" @api.route("/") def hello(req, resp): resp.text = TEXT assert api.requests.get("/").text == TEXT def test_status_code(api): @api.route("/") def hello(req, resp): resp.text = "keep going" resp.status_code = responder.status_codes.HTTP_416 assert api.requests.get("http://;/").status_code == responder.status_codes.HTTP_416 def test_json_media(api): dump = {"life": 42} @api.route("/") def media(req, resp): resp.media = dump r = api.requests.get("http://;/") assert "json" in r.headers["Content-Type"] assert r.json() == dump def test_yaml_media(api): dump = {"life": 42} @api.route("/") def media(req, resp): resp.media = dump r = api.requests.get("http://;/", headers={"Accept": "yaml"}) assert "yaml" in r.headers["Content-Type"] assert yaml.load(r.content, Loader=yaml.FullLoader) == dump # noqa: S506 def test_argumented_routing(api): @api.route("/{name}") def hello(req, resp, *, name): resp.text = f"Hello, {name}." r = api.requests.get(api.url_for(hello, name="sean")) assert r.text == "Hello, sean." def test_mote_argumented_routing(api): @api.route("/{greeting}/{name}") def hello(req, resp, *, greeting, name): resp.text = f"{greeting}, {name}." r = api.requests.get(api.url_for(hello, greeting="hello", name="lyndsy")) assert r.text == "hello, lyndsy." def test_request_and_get(api): @api.route("/") class ThingsResource: def on_request(self, req, resp): resp.headers.update({"DEATH": "666"}) def on_get(self, req, resp): resp.headers.update({"LIFE": "42"}) r = api.requests.get(api.url_for(ThingsResource)) assert "DEATH" in r.headers assert "LIFE" in r.headers def test_req_is_json(api): @api.route("/") async def view(req, resp): resp.media = {"is_json": req.is_json} r = api.requests.post( api.url_for(view), json={"hello": "world"}, ) assert r.json()["is_json"] is True r = api.requests.get(api.url_for(view)) assert r.json()["is_json"] is False def test_req_path_params(api): @api.route("/users/{user_id:int}") def view(req, resp, *, user_id): # noqa: A002 resp.media = {"from_kwargs": user_id, "from_req": req.path_params} r = api.requests.get("http://;/users/42") data = r.json() assert data["from_kwargs"] == 42 assert data["from_req"] == {"user_id": 42} def test_class_based_view_status_code(api): @api.route("/") class ThingsResource: def on_request(self, req, resp): resp.status_code = responder.status_codes.HTTP_416 assert api.requests.get("http://;/").status_code == responder.status_codes.HTTP_416 def test_query_params(api, url): @api.route("/") def route(req, resp): resp.media = {"params": req.params} r = api.requests.get(api.url_for(route), params={"q": "q"}) assert r.json()["params"] == {"q": "q"} r = api.requests.get(url("/?q=1&q=2&q=3")) assert r.json()["params"] == {"q": "3"} def test_async_function(api): content = "The Emerald Tablet of Hermes" @api.route("/") async def route(req, resp): resp.text = content r = api.requests.get(api.url_for(route)) assert r.text == content def test_media_parsing(api): dump = {"hello": "sam"} @api.route("/") def route(req, resp): resp.media = dump r = api.requests.get(api.url_for(route)) assert r.json() == dump r = api.requests.get(api.url_for(route), headers={"Accept": "application/x-yaml"}) assert r.text == "hello: sam\n" def test_background(api): @api.route("/") def route(req, resp): @api.background.task def task(): import time time.sleep(3) task() api.text = "ok" r = api.requests.get(api.url_for(route)) assert r.status_code < 300 def test_async_background(api): result = {"value": None} @api.route("/") async def route(req, resp): async def set_value(): result["value"] = 42 await api.background(set_value) resp.media = {"dispatched": True} r = api.requests.get(api.url_for(route)) assert r.json() == {"dispatched": True} assert result["value"] == 42 def test_multiple_routes(api): @api.route("/1") def route1(req, resp): resp.text = "1" @api.route("/2") def route2(req, resp): resp.text = "2" r = api.requests.get(api.url_for(route1)) assert r.text == "1" r = api.requests.get(api.url_for(route2)) assert r.text == "2" def test_json_uploads(api): @api.route("/") async def route(req, resp): resp.media = await req.media() dump = {"complicated": "times"} r = api.requests.post(api.url_for(route), json=dump) assert r.json() == dump def test_yaml_uploads(api): @api.route("/") async def route(req, resp): resp.media = await req.media() dump = {"complicated": "times"} r = api.requests.post( api.url_for(route), content=yaml.dump(dump), headers={"Content-Type": "application/x-yaml"}, ) assert r.json() == dump def test_form_uploads(api): @api.route("/") async def route(req, resp): resp.media = await req.media() dump = {"complicated": "times"} r = api.requests.post(api.url_for(route), data=dump) assert r.json() == dump # requests with boundary files = {"complicated": (None, "times")} r = api.requests.post(api.url_for(route), files=files) assert r.json() == {"complicated": "times"} def test_json_downloads(api): dump = {"testing": "123"} @api.route("/") def route(req, resp): resp.media = dump r = api.requests.get(api.url_for(route), headers={"Content-Type": "application/json"}) assert r.json() == dump def test_yaml_downloads(api): dump = {"testing": "123"} @api.route("/") def route(req, resp): resp.media = dump r = api.requests.get( api.url_for(route), headers={"Content-Type": "application/x-yaml"} ) assert yaml.safe_load(r.content) == dump def test_schema_generation_explicit(needs_openapi): import marshmallow import responder from responder.ext.openapi import OpenAPISchema api = responder.API() schema = OpenAPISchema(app=api, title="Web Service", version="1.0", openapi="3.0.2") @schema.schema("Pet") class PetSchema(marshmallow.Schema): name = marshmallow.fields.Str() @api.route("/") def route(req, resp): """A cute furry animal endpoint. --- get: description: Get a random pet responses: 200: description: A pet to be returned schema: $ref: "#/components/schemas/Pet" """ resp.media = PetSchema().dump({"name": "little orange"}) r = api.requests.get("http://;/schema.yml") dump = yaml.safe_load(r.content) assert dump assert dump["openapi"] == "3.0.2" def test_schema_generation(needs_openapi): from marshmallow import Schema, fields import responder api = responder.API(title="Web Service", openapi="3.0.2") @api.schema("Pet") class PetSchema(Schema): name = fields.Str() @api.route("/") def route(req, resp): """A cute furry animal endpoint. --- get: description: Get a random pet responses: 200: description: A pet to be returned schema: $ref: "#/components/schemas/Pet" """ resp.media = PetSchema().dump({"name": "little orange"}) r = api.requests.get("http://;/schema.yml") dump = yaml.safe_load(r.content) assert dump assert dump["openapi"] == "3.0.2" def test_documentation_explicit(needs_openapi): import marshmallow import responder from responder.ext.openapi import OpenAPISchema description = "This is a sample server for a pet store." terms_of_service = "http://example.com/terms/" contact = { "name": "API Support", "url": "http://www.example.com/support", "email": "support@example.com", } license_ = { "name": "Apache 2.0", "url": "https://www.apache.org/licenses/LICENSE-2.0.html", } api = responder.API(allowed_hosts=["testserver", ";"]) schema = OpenAPISchema( app=api, title="Web Service", version="1.0", openapi="3.0.2", docs_route="/docs", description=description, terms_of_service=terms_of_service, contact=contact, license=license_, ) @schema.schema("Pet") class PetSchema(marshmallow.Schema): name = marshmallow.fields.Str() @api.route("/") def route(req, resp): """A cute furry animal endpoint. --- get: description: Get a random pet responses: 200: description: A pet to be returned schema: $ref: "#/components/schemas/Pet" """ resp.media = PetSchema().dump({"name": "little orange"}) r = api.requests.get("/docs") assert "html" in r.text def test_documentation(needs_openapi): from marshmallow import Schema, fields import responder description = "This is a sample server for a pet store." terms_of_service = "http://example.com/terms/" contact = { "name": "API Support", "url": "http://www.example.com/support", "email": "support@example.com", } license_ = { "name": "Apache 2.0", "url": "https://www.apache.org/licenses/LICENSE-2.0.html", } api = responder.API( title="Web Service", version="1.0", openapi="3.0.2", docs_route="/docs", description=description, terms_of_service=terms_of_service, contact=contact, license=license_, allowed_hosts=["testserver", ";"], ) @api.schema("Pet") class PetSchema(Schema): name = fields.Str() @api.route("/") def route(req, resp): """A cute furry animal endpoint. --- get: description: Get a random pet responses: 200: description: A pet to be returned schema: $ref: "#/components/schemas/Pet" """ resp.media = PetSchema().dump({"name": "little orange"}) r = api.requests.get("/docs") assert "html" in r.text def test_mount_wsgi_app(flask): # Use localhost so Werkzeug's trusted-host check accepts the request. api = responder.API(allowed_hosts=["localhost"]) @api.route("/") def hello(req, resp): resp.text = "hello" api.mount("/flask", flask) r = api.requests.get("http://localhost/flask") assert r.status_code < 300 def test_async_class_based_views(api): @api.route("/") class Resource: async def on_post(self, req, resp): resp.text = await req.text data = "frame" r = api.requests.post(api.url_for(Resource), content=data) assert r.text == data def test_cookies(api): @api.route("/") def cookies(req, resp): resp.media = {"cookies": req.cookies} resp.cookies["sent"] = "true" resp.set_cookie( "hello", "world", expires=123, path="/", max_age=123, secure=False, httponly=True, ) api.requests.cookies.set("hello", "universe") r = api.requests.get(api.url_for(cookies)) assert r.json() == {"cookies": {"hello": "universe"}} assert "sent" in r.cookies assert "hello" in r.cookies r = api.requests.get(api.url_for(cookies)) assert r.json() == {"cookies": {"hello": "world", "sent": "true"}} def test_sessions(api): @api.route("/") def view(req, resp): resp.session["hello"] = "world" resp.media = resp.session r = api.requests.get(api.url_for(view)) assert "session" in r.cookies r = api.requests.get(api.url_for(view)) assert r.json() == {"hello": "world"} def test_template_string_rendering(api): @api.route("/") def view(req, resp): resp.content = api.template_string("{{ var }}", var="hello") r = api.requests.get(api.url_for(view)) assert r.text == "hello" def test_template_rendering(template_path): api = responder.API(templates_dir=template_path.parent) @api.route("/") def view(req, resp): resp.content = api.template(template_path.name, var="hello") r = api.requests.get(api.url_for(view)) assert r.text == "hello" def test_template(api, template_path): templates = Templates(directory=template_path.parent) @api.route("/{var}/") def view(req, resp, var): resp.html = templates.render(template_path.name, var=var) r = api.requests.get("/test/") assert r.text == "test" def test_template_async(api, template_path): templates = Templates(directory=template_path.parent, enable_async=True) @api.route("/{var}/async") async def view_async(req, resp, var): resp.html = await templates.render_async(template_path.name, var=var) r = api.requests.get("/test/async") assert r.text == "test" def test_file_uploads(api): @api.route("/") async def upload(req, resp): files = await req.media("files") result = {} result["hello"] = files["hello"]["content"].decode("utf-8") resp.media = {"files": result} files = {"hello": ("hello.txt", b"world", "text/plain")} r = api.requests.post(api.url_for(upload), files=files) assert r.json() == {"files": {"hello": "world"}} def test_500(api): @api.route("/") def view(req, resp): raise ValueError dumb_client = StarletteTestClient( api, base_url="http://;", raise_server_exceptions=False ) r = dumb_client.get(api.url_for(view)) assert r.status_code >= 300 assert r.status_code == responder.status_codes.HTTP_500 def test_exception_handler(): api = responder.API(allowed_hosts=[";"]) @api.exception_handler(ValueError) async def handle_value_error(req, resp, exc): resp.status_code = 400 resp.media = {"error": str(exc)} @api.route("/") def view(req, resp): raise ValueError("bad input") client = StarletteTestClient(api, base_url="http://;", raise_server_exceptions=False) r = client.get(api.url_for(view)) assert r.status_code == 400 assert r.json() == {"error": "bad input"} def test_404(api): r = api.requests.get("/foo") assert r.status_code == responder.status_codes.HTTP_404 def test_websockets_text(api): payload = "Hello via websocket!" @api.route("/ws", websocket=True) async def websocket(ws): await ws.accept() await ws.send_text(payload) await ws.close() client = StarletteTestClient(api) with client.websocket_connect("ws://;/ws") as ws: data = ws.receive_text() assert data == payload def test_websockets_bytes(api): payload = b"Hello via websocket!" @api.route("/ws", websocket=True) async def websocket(ws): await ws.accept() await ws.send_bytes(payload) await ws.close() client = StarletteTestClient(api) with client.websocket_connect("ws://;/ws") as ws: data = ws.receive_bytes() assert data == payload def test_websockets_json(api): payload = {"Hello": "via websocket!"} @api.route("/ws", websocket=True) async def websocket(ws): await ws.accept() await ws.send_json(payload) await ws.close() client = StarletteTestClient(api) with client.websocket_connect("ws://;/ws") as ws: data = ws.receive_json() assert data == payload def test_before_websockets(api): payload = {"Hello": "via websocket!"} @api.route("/ws", websocket=True) async def websocket(ws): await ws.send_json(payload) await ws.close() @api.before_request(websocket=True) async def before_request(ws): await ws.accept() await ws.send_json({"before": "request"}) client = StarletteTestClient(api) with client.websocket_connect("ws://;/ws") as ws: data = ws.receive_json() assert data == {"before": "request"} data = ws.receive_json() assert data == payload def test_startup(api): who = [None] @api.route("/{greeting}") async def greet_world(req, resp, *, greeting): resp.text = f"{greeting}, {who[0]}!" @api.on_event("startup") async def run_startup(): who[0] = "world" with api.requests as session: r = session.get("http://;/hello") assert r.text == "hello, world!" def test_lifespan_context_manager(): from contextlib import asynccontextmanager state = {"started": False, "stopped": False} @asynccontextmanager async def lifespan(app): state["started"] = True yield state["stopped"] = True api = responder.API(lifespan=lifespan, allowed_hosts=[";"]) @api.route("/") def index(req, resp): resp.media = {"started": state["started"]} with api.requests as session: r = session.get("http://;/") assert r.json() == {"started": True} assert state["stopped"] is True def test_resp_file(api, tmp_path): test_file = tmp_path / "hello.txt" test_file.write_text("hello from file") @api.route("/download") def download(req, resp): resp.file(test_file) r = api.requests.get(api.url_for(download)) assert r.text == "hello from file" assert "text/plain" in r.headers["content-type"] def test_resp_file_binary(api, tmp_path): test_file = tmp_path / "image.png" test_file.write_bytes(b"\x89PNG\r\n\x1a\n") @api.route("/image") def image(req, resp): resp.file(test_file, content_type="image/png") r = api.requests.get(api.url_for(image)) assert r.content == b"\x89PNG\r\n\x1a\n" assert r.headers["content-type"] == "image/png" def test_redirects(api, session): @api.route("/2") def two(req, resp): api.redirect(resp, location="/1") @api.route("/1") def one(req, resp): resp.text = "redirected" assert session.get("/2").url == "http://;/1" def test_session_thoroughly(api, session): @api.route("/set") def setter(req, resp): resp.session["hello"] = "world" api.redirect(resp, location="/get") @api.route("/get") def getter(req, resp): resp.media = {"session": req.session} r = session.get(api.url_for(setter)) r = session.get(api.url_for(getter)) assert r.json() == {"session": {"hello": "world"}} def test_before_response(api, session): @api.route("/get") def get(req, resp): resp.media = req.session @api.route(before_request=True) async def async_before_request(req, resp): resp.headers["x-pizza"] = "1" @api.route(before_request=True) def before_request(req, resp): resp.headers["x-pizza"] = "1" r = session.get(api.url_for(get)) assert "x-pizza" in r.headers def test_route_methods_filter(api): @api.route("/data", methods=["GET"]) def get_data(req, resp): resp.media = {"method": "get"} @api.route("/data", methods=["POST"], check_existing=False) def post_data(req, resp): resp.media = {"method": "post"} r = api.requests.get(api.url_for(get_data)) assert r.json() == {"method": "get"} r = api.requests.post(api.url_for(post_data)) assert r.json() == {"method": "post"} def test_before_request_short_circuit(api): """If a before_request hook sets a status code, the route handler is skipped.""" called = {"handler": False} @api.route(before_request=True) def auth_check(req, resp): resp.status_code = 401 resp.media = {"error": "unauthorized"} @api.route("/protected") def protected(req, resp): called["handler"] = True resp.text = "secret" r = api.requests.get(api.url_for(protected)) assert r.status_code == 401 assert r.json() == {"error": "unauthorized"} assert called["handler"] is False @pytest.mark.parametrize("enable_hsts", [True, False]) @pytest.mark.parametrize("cors", [True, False]) def test_allowed_hosts(enable_hsts, cors): api = responder.API( allowed_hosts=[";", "tenant.;"], enable_hsts=enable_hsts, cors=cors ) @api.route("/") def get(req, resp): pass # Exact match r = api.requests.get(api.url_for(get)) assert r.status_code == 200 # Reset the session api._session = None r = api.session(base_url="http://tenant.;").get(api.url_for(get)) assert r.status_code == 200 # Reset the session api._session = None r = api.session(base_url="http://unkownhost").get(api.url_for(get)) assert r.status_code == 400 # Reset the session api._session = None r = api.session(base_url="http://unkown_tenant.;").get(api.url_for(get)) assert r.status_code == 400 api = responder.API(allowed_hosts=["*.;"]) @api.route("/") def get(req, resp): pass # Wildcard domains # Using http://; r = api.requests.get(api.url_for(get)) assert r.status_code == 400 # Reset the session api._session = None r = api.session(base_url="http://tenant1.;").get(api.url_for(get)) assert r.status_code == 200 # Reset the session api._session = None r = api.session(base_url="http://tenant2.;").get(api.url_for(get)) assert r.status_code == 200 def create_asset(static_dir: Path, name=None, parent_dir: str = None) -> Path: if name is None: name = "".join(random.choices(string.ascii_letters, k=6)) # noqa: S311 # :3 ext = "".join(random.choices(string.ascii_letters, k=2)) # noqa: S311 name = f"{name}.{ext}" if parent_dir is None: parent_dir = static_dir else: parent_dir = static_dir / parent_dir parent_dir.mkdir() asset = parent_dir / name asset.write_text("body { color: blue; }", encoding="utf-8") return Path(asset) @pytest.mark.parametrize("static_route", [None, "/static", "/custom/static/route"]) def test_staticfiles(tmp_path, static_route): static_dir = tmp_path / "static" static_dir.mkdir() asset1 = create_asset(static_dir) parent_dir = "css" asset2 = create_asset(static_dir, name="asset2", parent_dir=parent_dir) api = responder.API(static_dir=str(static_dir), static_route=static_route) session = api.session() static_route = api.static_route # ok r = session.get(f"{static_route}/{asset1.name}") assert r.status_code == api.status_codes.HTTP_200 r = session.get(f"{static_route}/{parent_dir}/{asset2.name}") assert r.status_code == api.status_codes.HTTP_200 # Asset not found r = session.get(f"{static_route}/not_found.css") assert r.status_code == api.status_codes.HTTP_404 # Not found on dir listing r = session.get(f"{static_route}") assert r.status_code == api.status_codes.HTTP_404 r = session.get(f"{static_route}/{parent_dir}") assert r.status_code == api.status_codes.HTTP_404 def test_staticfiles_add_directory(tmp_path): static_dir = tmp_path / "static" static_dir.mkdir() extra_dir = tmp_path / "extra" extra_dir.mkdir() (static_dir / "main.css").write_text("body {}") (extra_dir / "extra.css").write_text(".extra {}") api = responder.API(static_dir=str(static_dir)) api.static_app.add_directory(str(extra_dir)) session = api.session() r = session.get(f"{api.static_route}/main.css") assert r.status_code == 200 r = session.get(f"{api.static_route}/extra.css") assert r.status_code == 200 def test_staticfiles_none_dir(tmp_path): api = responder.API(static_dir=None) session = api.session() static_dir = tmp_path / "static" static_dir.mkdir() asset = create_asset(static_dir) static_route = api.static_route # ok r = session.get(f"{static_route}/{asset.name}") assert r.status_code == api.status_codes.HTTP_404 # dir listing r = session.get(f"{static_route}") assert r.status_code == api.status_codes.HTTP_404 # SPA with pytest.raises(Exception): # noqa: B017 api.add_route("/spa", static=True) def test_static_index_html(tmp_path): static_dir = tmp_path / "static" static_dir.mkdir() (static_dir / "index.html").write_text("

Home

") api = responder.API(static_dir=str(static_dir), allowed_hosts=[";"]) api.add_route("/", static=True) r = api.requests.get("http://;/") assert r.text == "

Home

" def test_response_html_property(api): @api.route("/") def view(req, resp): resp.html = "

Hello !

" assert resp.content == "

Hello !

" assert resp.mimetype == "text/html" r = api.requests.get(api.url_for(view)) assert r.content == b"

Hello !

" assert r.headers["Content-Type"] == "text/html" def test_response_text_property(api): @api.route("/") def view(req, resp): resp.text = "

Hello !

" assert resp.content == "

Hello !

" assert resp.mimetype == "text/plain" r = api.requests.get(api.url_for(view)) assert r.content == b"

Hello !

" assert r.headers["Content-Type"] == "text/plain" def test_stream(api, session): async def shout_stream(who): for c in who.upper(): yield c @api.route("/{who}") async def greeting(req, resp, *, who): resp.stream(shout_stream, who) r = session.get("/morocco") assert r.text == "MOROCCO" @api.route("/") async def home(req, resp): # Raise when it's not an async generator with pytest.raises(AssertionError): def foo(): pass resp.stream(foo) with pytest.raises(AssertionError): async def foo(): pass resp.stream(foo) with pytest.raises(AssertionError): def foo(): yield "oopsie" resp.stream(foo) def test_empty_req_text(api): content = "It's working" @api.route("/") async def home(req, resp): await req.text resp.text = content r = api.requests.post("/") assert r.text == content def test_api_request_state(api, url): class StateMiddleware(BaseHTTPMiddleware): async def dispatch(self, request, call_next): request.state.test1 = 42 request.state.test2 = "Foo" return await call_next(request) api.add_middleware(StateMiddleware) @api.route("/") def home(req, resp): resp.text = "{}_{}".format(req.state.test2, req.state.test1) assert api.requests.get(url("/")).text == "Foo_42" def test_path_matches_route(api): @api.route("/hello") def home(req, resp): resp.text = "hello world!" route = api.path_matches_route({"type": "http", "path": "/hello"}) assert route.endpoint_name == "home" assert not api.path_matches_route({"type": "http", "path": "/foo"}) def test_route_without_endpoint(api): # test that a route without endpoint gets a default static response api.add_route("/") route = api.router.routes[0] assert route.endpoint_name == "_static_response"