2018-10-29 14:46:42 +00:00
|
|
|
import os
|
|
|
|
|
2021-06-28 20:36:13 +00:00
|
|
|
import pytest
|
|
|
|
|
2018-09-05 10:39:38 +00:00
|
|
|
from starlette.applications import Starlette
|
2018-10-29 14:46:42 +00:00
|
|
|
from starlette.endpoints import HTTPEndpoint
|
2018-09-04 10:52:29 +00:00
|
|
|
from starlette.exceptions import HTTPException
|
2019-01-30 15:15:59 +00:00
|
|
|
from starlette.middleware.trustedhost import TrustedHostMiddleware
|
2018-09-05 09:29:04 +00:00
|
|
|
from starlette.responses import JSONResponse, PlainTextResponse
|
2019-01-30 15:15:59 +00:00
|
|
|
from starlette.routing import Host, Mount, Route, Router, WebSocketRoute
|
2018-08-28 13:34:18 +00:00
|
|
|
from starlette.staticfiles import StaticFiles
|
|
|
|
|
2018-09-05 09:29:04 +00:00
|
|
|
app = Starlette()
|
2018-08-28 13:34:18 +00:00
|
|
|
|
|
|
|
|
2019-01-30 15:15:59 +00:00
|
|
|
app.add_middleware(TrustedHostMiddleware, allowed_hosts=["testserver", "*.example.org"])
|
2018-10-05 11:04:11 +00:00
|
|
|
|
|
|
|
|
2018-11-08 11:59:15 +00:00
|
|
|
@app.exception_handler(500)
|
2018-09-04 10:52:29 +00:00
|
|
|
async def error_500(request, exc):
|
|
|
|
return JSONResponse({"detail": "Server Error"}, status_code=500)
|
|
|
|
|
|
|
|
|
2018-11-08 11:59:15 +00:00
|
|
|
@app.exception_handler(405)
|
|
|
|
async def method_not_allowed(request, exc):
|
|
|
|
return JSONResponse({"detail": "Custom message"}, status_code=405)
|
|
|
|
|
|
|
|
|
2018-09-04 10:52:29 +00:00
|
|
|
@app.exception_handler(HTTPException)
|
2018-11-08 11:59:15 +00:00
|
|
|
async def http_exception(request, exc):
|
2018-09-04 10:52:29 +00:00
|
|
|
return JSONResponse({"detail": exc.detail}, status_code=exc.status_code)
|
|
|
|
|
|
|
|
|
2018-08-28 13:34:18 +00:00
|
|
|
@app.route("/func")
|
|
|
|
def func_homepage(request):
|
|
|
|
return PlainTextResponse("Hello, world!")
|
|
|
|
|
|
|
|
|
|
|
|
@app.route("/async")
|
|
|
|
async def async_homepage(request):
|
|
|
|
return PlainTextResponse("Hello, world!")
|
|
|
|
|
|
|
|
|
2018-09-04 10:52:29 +00:00
|
|
|
@app.route("/class")
|
2018-09-05 09:29:04 +00:00
|
|
|
class Homepage(HTTPEndpoint):
|
2018-09-04 10:52:29 +00:00
|
|
|
def get(self, request):
|
|
|
|
return PlainTextResponse("Hello, world!")
|
|
|
|
|
|
|
|
|
2018-10-29 09:22:45 +00:00
|
|
|
users = Router()
|
|
|
|
|
|
|
|
|
|
|
|
@users.route("/")
|
|
|
|
def all_users_page(request):
|
|
|
|
return PlainTextResponse("Hello, everyone!")
|
|
|
|
|
|
|
|
|
|
|
|
@users.route("/{username}")
|
|
|
|
def user_page(request):
|
|
|
|
username = request.path_params["username"]
|
2019-01-22 16:24:46 +00:00
|
|
|
return PlainTextResponse(f"Hello, {username}!")
|
2018-08-28 13:34:18 +00:00
|
|
|
|
|
|
|
|
2018-10-29 09:22:45 +00:00
|
|
|
app.mount("/users", users)
|
|
|
|
|
|
|
|
|
2019-01-30 15:15:59 +00:00
|
|
|
subdomain = Router()
|
|
|
|
|
|
|
|
|
|
|
|
@subdomain.route("/")
|
|
|
|
def custom_subdomain(request):
|
|
|
|
return PlainTextResponse("Subdomain: " + request.path_params["subdomain"])
|
|
|
|
|
|
|
|
|
|
|
|
app.host("{subdomain}.example.org", subdomain)
|
|
|
|
|
|
|
|
|
2018-09-04 10:52:29 +00:00
|
|
|
@app.route("/500")
|
2018-10-16 14:56:28 +00:00
|
|
|
def runtime_error(request):
|
2018-09-04 10:52:29 +00:00
|
|
|
raise RuntimeError()
|
|
|
|
|
|
|
|
|
2018-08-28 13:34:18 +00:00
|
|
|
@app.websocket_route("/ws")
|
|
|
|
async def websocket_endpoint(session):
|
|
|
|
await session.accept()
|
|
|
|
await session.send_text("Hello, world!")
|
|
|
|
await session.close()
|
|
|
|
|
|
|
|
|
2021-06-28 20:36:13 +00:00
|
|
|
@pytest.fixture
|
|
|
|
def client(test_client_factory):
|
|
|
|
with test_client_factory(app) as client:
|
|
|
|
yield client
|
2018-08-28 13:34:18 +00:00
|
|
|
|
|
|
|
|
2018-10-29 11:14:42 +00:00
|
|
|
def test_url_path_for():
|
|
|
|
assert app.url_path_for("func_homepage") == "/func"
|
2018-10-29 09:22:45 +00:00
|
|
|
|
|
|
|
|
2021-06-28 20:36:13 +00:00
|
|
|
def test_func_route(client):
|
2018-08-28 13:34:18 +00:00
|
|
|
response = client.get("/func")
|
|
|
|
assert response.status_code == 200
|
|
|
|
assert response.text == "Hello, world!"
|
|
|
|
|
2019-01-14 11:07:23 +00:00
|
|
|
response = client.head("/func")
|
|
|
|
assert response.status_code == 200
|
|
|
|
assert response.text == ""
|
|
|
|
|
2018-08-28 13:34:18 +00:00
|
|
|
|
2021-06-28 20:36:13 +00:00
|
|
|
def test_async_route(client):
|
2018-08-28 13:34:18 +00:00
|
|
|
response = client.get("/async")
|
|
|
|
assert response.status_code == 200
|
|
|
|
assert response.text == "Hello, world!"
|
|
|
|
|
|
|
|
|
2021-06-28 20:36:13 +00:00
|
|
|
def test_class_route(client):
|
2018-09-04 10:52:29 +00:00
|
|
|
response = client.get("/class")
|
|
|
|
assert response.status_code == 200
|
|
|
|
assert response.text == "Hello, world!"
|
|
|
|
|
|
|
|
|
2021-06-28 20:36:13 +00:00
|
|
|
def test_mounted_route(client):
|
2018-10-29 09:22:45 +00:00
|
|
|
response = client.get("/users/")
|
|
|
|
assert response.status_code == 200
|
|
|
|
assert response.text == "Hello, everyone!"
|
|
|
|
|
|
|
|
|
2021-06-28 20:36:13 +00:00
|
|
|
def test_mounted_route_path_params(client):
|
2018-10-29 09:22:45 +00:00
|
|
|
response = client.get("/users/tomchristie")
|
2018-08-28 13:34:18 +00:00
|
|
|
assert response.status_code == 200
|
|
|
|
assert response.text == "Hello, tomchristie!"
|
|
|
|
|
|
|
|
|
2021-06-28 20:36:13 +00:00
|
|
|
def test_subdomain_route(test_client_factory):
|
|
|
|
client = test_client_factory(app, base_url="https://foo.example.org/")
|
2019-01-30 15:15:59 +00:00
|
|
|
|
|
|
|
response = client.get("/")
|
|
|
|
assert response.status_code == 200
|
|
|
|
assert response.text == "Subdomain: foo"
|
|
|
|
|
|
|
|
|
2021-06-28 20:36:13 +00:00
|
|
|
def test_websocket_route(client):
|
2018-08-28 13:45:06 +00:00
|
|
|
with client.websocket_connect("/ws") as session:
|
2018-08-28 13:34:18 +00:00
|
|
|
text = session.receive_text()
|
|
|
|
assert text == "Hello, world!"
|
|
|
|
|
|
|
|
|
2021-06-28 20:36:13 +00:00
|
|
|
def test_400(client):
|
2018-08-28 13:34:18 +00:00
|
|
|
response = client.get("/404")
|
|
|
|
assert response.status_code == 404
|
2018-09-04 10:52:29 +00:00
|
|
|
assert response.json() == {"detail": "Not Found"}
|
|
|
|
|
|
|
|
|
2021-06-28 20:36:13 +00:00
|
|
|
def test_405(client):
|
2018-09-04 10:52:29 +00:00
|
|
|
response = client.post("/func")
|
|
|
|
assert response.status_code == 405
|
2018-11-08 11:59:15 +00:00
|
|
|
assert response.json() == {"detail": "Custom message"}
|
2018-09-04 10:52:29 +00:00
|
|
|
|
|
|
|
response = client.post("/class")
|
|
|
|
assert response.status_code == 405
|
2018-11-08 11:59:15 +00:00
|
|
|
assert response.json() == {"detail": "Custom message"}
|
2018-09-04 10:52:29 +00:00
|
|
|
|
|
|
|
|
2021-06-28 20:36:13 +00:00
|
|
|
def test_500(test_client_factory):
|
|
|
|
client = test_client_factory(app, raise_server_exceptions=False)
|
2018-09-04 10:52:29 +00:00
|
|
|
response = client.get("/500")
|
|
|
|
assert response.status_code == 500
|
|
|
|
assert response.json() == {"detail": "Server Error"}
|
2018-08-28 13:34:18 +00:00
|
|
|
|
|
|
|
|
2021-06-28 20:36:13 +00:00
|
|
|
def test_middleware(test_client_factory):
|
|
|
|
client = test_client_factory(app, base_url="http://incorrecthost")
|
2018-10-05 11:04:11 +00:00
|
|
|
response = client.get("/func")
|
|
|
|
assert response.status_code == 400
|
|
|
|
assert response.text == "Invalid host header"
|
|
|
|
|
|
|
|
|
2018-10-29 09:22:45 +00:00
|
|
|
def test_routes():
|
|
|
|
assert app.routes == [
|
|
|
|
Route("/func", endpoint=func_homepage, methods=["GET"]),
|
|
|
|
Route("/async", endpoint=async_homepage, methods=["GET"]),
|
|
|
|
Route("/class", endpoint=Homepage),
|
|
|
|
Mount(
|
|
|
|
"/users",
|
|
|
|
app=Router(
|
|
|
|
routes=[
|
|
|
|
Route("/", endpoint=all_users_page),
|
|
|
|
Route("/{username}", endpoint=user_page),
|
|
|
|
]
|
|
|
|
),
|
|
|
|
),
|
2019-01-30 15:15:59 +00:00
|
|
|
Host(
|
|
|
|
"{subdomain}.example.org",
|
|
|
|
app=Router(routes=[Route("/", endpoint=custom_subdomain)]),
|
|
|
|
),
|
2018-10-29 09:22:45 +00:00
|
|
|
Route("/500", endpoint=runtime_error, methods=["GET"]),
|
|
|
|
WebSocketRoute("/ws", endpoint=websocket_endpoint),
|
|
|
|
]
|
|
|
|
|
|
|
|
|
2021-06-28 20:36:13 +00:00
|
|
|
def test_app_mount(tmpdir, test_client_factory):
|
2018-08-28 13:34:18 +00:00
|
|
|
path = os.path.join(tmpdir, "example.txt")
|
|
|
|
with open(path, "w") as file:
|
|
|
|
file.write("<file content>")
|
|
|
|
|
2018-09-05 09:29:04 +00:00
|
|
|
app = Starlette()
|
2018-10-29 09:22:45 +00:00
|
|
|
app.mount("/static", StaticFiles(directory=tmpdir))
|
2018-09-05 09:29:04 +00:00
|
|
|
|
2021-06-28 20:36:13 +00:00
|
|
|
client = test_client_factory(app)
|
2018-09-04 10:52:29 +00:00
|
|
|
|
2018-08-28 13:34:18 +00:00
|
|
|
response = client.get("/static/example.txt")
|
|
|
|
assert response.status_code == 200
|
|
|
|
assert response.text == "<file content>"
|
2018-09-04 10:52:29 +00:00
|
|
|
|
|
|
|
response = client.post("/static/example.txt")
|
|
|
|
assert response.status_code == 405
|
|
|
|
assert response.text == "Method Not Allowed"
|
|
|
|
|
|
|
|
|
2021-06-28 20:36:13 +00:00
|
|
|
def test_app_debug(test_client_factory):
|
2018-09-05 09:29:04 +00:00
|
|
|
app = Starlette()
|
2018-09-04 10:52:29 +00:00
|
|
|
app.debug = True
|
|
|
|
|
|
|
|
@app.route("/")
|
|
|
|
async def homepage(request):
|
|
|
|
raise RuntimeError()
|
|
|
|
|
2021-06-28 20:36:13 +00:00
|
|
|
client = test_client_factory(app, raise_server_exceptions=False)
|
2018-09-04 10:52:29 +00:00
|
|
|
response = client.get("/")
|
|
|
|
assert response.status_code == 500
|
|
|
|
assert "RuntimeError" in response.text
|
|
|
|
assert app.debug
|
2018-10-18 07:43:21 +00:00
|
|
|
|
|
|
|
|
2021-06-28 20:36:13 +00:00
|
|
|
def test_app_add_route(test_client_factory):
|
2018-10-26 09:30:54 +00:00
|
|
|
app = Starlette()
|
|
|
|
|
|
|
|
async def homepage(request):
|
|
|
|
return PlainTextResponse("Hello, World!")
|
|
|
|
|
|
|
|
app.add_route("/", homepage)
|
2021-06-28 20:36:13 +00:00
|
|
|
client = test_client_factory(app)
|
2018-10-26 09:30:54 +00:00
|
|
|
response = client.get("/")
|
|
|
|
assert response.status_code == 200
|
|
|
|
assert response.text == "Hello, World!"
|
|
|
|
|
|
|
|
|
2021-06-28 20:36:13 +00:00
|
|
|
def test_app_add_websocket_route(test_client_factory):
|
2018-10-26 09:30:54 +00:00
|
|
|
app = Starlette()
|
|
|
|
|
|
|
|
async def websocket_endpoint(session):
|
|
|
|
await session.accept()
|
|
|
|
await session.send_text("Hello, world!")
|
|
|
|
await session.close()
|
|
|
|
|
|
|
|
app.add_websocket_route("/ws", websocket_endpoint)
|
2021-06-28 20:36:13 +00:00
|
|
|
client = test_client_factory(app)
|
2018-10-26 09:30:54 +00:00
|
|
|
|
|
|
|
with client.websocket_connect("/ws") as session:
|
|
|
|
text = session.receive_text()
|
|
|
|
assert text == "Hello, world!"
|
|
|
|
|
|
|
|
|
2021-06-28 20:36:13 +00:00
|
|
|
def test_app_add_event_handler(test_client_factory):
|
2018-10-18 07:43:21 +00:00
|
|
|
startup_complete = False
|
|
|
|
cleanup_complete = False
|
|
|
|
app = Starlette()
|
|
|
|
|
|
|
|
def run_startup():
|
|
|
|
nonlocal startup_complete
|
|
|
|
startup_complete = True
|
|
|
|
|
|
|
|
def run_cleanup():
|
|
|
|
nonlocal cleanup_complete
|
|
|
|
cleanup_complete = True
|
|
|
|
|
|
|
|
app.add_event_handler("startup", run_startup)
|
2018-10-22 14:08:04 +00:00
|
|
|
app.add_event_handler("shutdown", run_cleanup)
|
2018-10-18 07:43:21 +00:00
|
|
|
|
|
|
|
assert not startup_complete
|
|
|
|
assert not cleanup_complete
|
2021-06-28 20:36:13 +00:00
|
|
|
with test_client_factory(app):
|
2018-10-18 07:43:21 +00:00
|
|
|
assert startup_complete
|
|
|
|
assert not cleanup_complete
|
|
|
|
assert startup_complete
|
|
|
|
assert cleanup_complete
|
2020-05-04 10:31:54 +00:00
|
|
|
|
|
|
|
|
2021-06-28 20:36:13 +00:00
|
|
|
def test_app_async_lifespan(test_client_factory):
|
2020-05-04 10:31:54 +00:00
|
|
|
startup_complete = False
|
|
|
|
cleanup_complete = False
|
|
|
|
|
|
|
|
async def lifespan(app):
|
|
|
|
nonlocal startup_complete, cleanup_complete
|
|
|
|
startup_complete = True
|
|
|
|
yield
|
|
|
|
cleanup_complete = True
|
|
|
|
|
|
|
|
app = Starlette(lifespan=lifespan)
|
|
|
|
|
|
|
|
assert not startup_complete
|
|
|
|
assert not cleanup_complete
|
2021-06-28 20:36:13 +00:00
|
|
|
with test_client_factory(app):
|
2020-05-04 10:31:54 +00:00
|
|
|
assert startup_complete
|
|
|
|
assert not cleanup_complete
|
|
|
|
assert startup_complete
|
|
|
|
assert cleanup_complete
|
|
|
|
|
|
|
|
|
2021-06-28 20:36:13 +00:00
|
|
|
def test_app_sync_lifespan(test_client_factory):
|
2020-05-04 10:31:54 +00:00
|
|
|
startup_complete = False
|
|
|
|
cleanup_complete = False
|
|
|
|
|
|
|
|
def lifespan(app):
|
|
|
|
nonlocal startup_complete, cleanup_complete
|
|
|
|
startup_complete = True
|
|
|
|
yield
|
|
|
|
cleanup_complete = True
|
|
|
|
|
|
|
|
app = Starlette(lifespan=lifespan)
|
|
|
|
|
|
|
|
assert not startup_complete
|
|
|
|
assert not cleanup_complete
|
2021-06-28 20:36:13 +00:00
|
|
|
with test_client_factory(app):
|
2020-05-04 10:31:54 +00:00
|
|
|
assert startup_complete
|
|
|
|
assert not cleanup_complete
|
|
|
|
assert startup_complete
|
|
|
|
assert cleanup_complete
|