Merge branch 'master' into mkdocs

This commit is contained in:
Jeff Buttars 2018-08-28 08:09:42 -06:00
commit 9cfb1af592
8 changed files with 292 additions and 57 deletions

View File

@ -20,7 +20,7 @@
Starlette is a small library for working with [ASGI](https://asgi.readthedocs.io/en/latest/). Starlette is a small library for working with [ASGI](https://asgi.readthedocs.io/en/latest/).
It gives you `Request` and `Response` classes, request routing, websocket support, It gives you `Request` and `Response` classes, websocket support, routing,
static files support, and a test client. static files support, and a test client.
**Requirements:** **Requirements:**
@ -36,7 +36,7 @@ pip3 install starlette
**Example:** **Example:**
```python ```python
from starlette import Response from starlette.response import Response
class App: class App:
@ -72,6 +72,7 @@ You can run the application with any ASGI server, including [uvicorn](http://www
* [Static Files](#static-files) * [Static Files](#static-files)
* [Test Client](#test-client) * [Test Client](#test-client)
* [Debugging](#debugging) * [Debugging](#debugging)
* [Applications](#applications)
--- ---
@ -111,7 +112,7 @@ class App:
Takes some text or bytes and returns an HTML response. Takes some text or bytes and returns an HTML response.
```python ```python
from starlette import HTMLResponse from starlette.response import HTMLResponse
class App: class App:
@ -128,7 +129,7 @@ class App:
Takes some text or bytes and returns an plain text response. Takes some text or bytes and returns an plain text response.
```python ```python
from starlette import PlainTextResponse from starlette.response import PlainTextResponse
class App: class App:
@ -145,7 +146,7 @@ class App:
Takes some data and returns an `application/json` encoded response. Takes some data and returns an `application/json` encoded response.
```python ```python
from starlette import JSONResponse from starlette.response import JSONResponse
class App: class App:
@ -162,7 +163,7 @@ class App:
Returns an HTTP redirect. Uses a 302 status code by default. Returns an HTTP redirect. Uses a 302 status code by default.
```python ```python
from starlette import PlainTextResponse, RedirectResponse from starlette.response import PlainTextResponse, RedirectResponse
class App: class App:
@ -182,7 +183,8 @@ class App:
Takes an async generator and streams the response body. Takes an async generator and streams the response body.
```python ```python
from starlette import Request, StreamingResponse from starlette.request import Request
from starlette.response import StreamingResponse
import asyncio import asyncio
@ -218,7 +220,7 @@ Takes a different set of arguments to instantiate than the other response types:
File responses will include appropriate `Content-Length`, `Last-Modified` and `ETag` headers. File responses will include appropriate `Content-Length`, `Last-Modified` and `ETag` headers.
```python ```python
from starlette import FileResponse from starlette.response import FileResponse
class App: class App:
@ -514,7 +516,8 @@ The test client allows you to make requests against your ASGI application,
using the `requests` library. using the `requests` library.
```python ```python
from starlette import HTMLResponse, TestClient from starlette.response import HTMLResponse
from starlette.testclient import TestClient
class App: class App:
@ -562,7 +565,7 @@ class App:
def test_app(): def test_app():
client = TestClient(App) client = TestClient(App)
with client.wsconnect('/') as session: with client.websocket_connect('/') as session:
data = session.receive_text() data = session.receive_text()
assert data == 'Hello, world!' assert data == 'Hello, world!'
``` ```
@ -576,10 +579,16 @@ always raised by the test client.
#### Establishing a test session #### Establishing a test session
* `.wsconnect(url, subprotocols=None, **options)` - Takes the same set of arguments as `requests.get()`. * `.websocket_connect(url, subprotocols=None, **options)` - Takes the same set of arguments as `requests.get()`.
May raise `starlette.websockets.Disconnect` if the application does not accept the websocket connection. May raise `starlette.websockets.Disconnect` if the application does not accept the websocket connection.
#### Sending data
* `.send_text(data)` - Send the given text to the application.
* `.send_bytes(data)` - Send the given bytes to the application.
* `.send_json(data)` - Send the given data to the application.
#### Receiving data #### Receiving data
* `.receive_text()` - Wait for incoming text sent by the application and return it. * `.receive_text()` - Wait for incoming text sent by the application and return it.
@ -615,4 +624,48 @@ app = DebugMiddleware(App)
--- ---
## Applications
Starlette also includes an `App` class that nicely ties together all of
its other functionality.
```python
from starlette.app import App
from starlette.response import PlainTextResponse
from starlette.staticfiles import StaticFiles
app = App()
app.mount("/static", StaticFiles(directory="static"))
@app.route('/')
def homepage(request):
return PlainTextResponse('Hello, world!')
@app.route('/user/{username}')
def user(request, username):
return PlainTextResponse('Hello, %s!' % username)
@app.websocket_route('/ws')
async def websocket_endpoint(session):
await session.accept()
await session.send_text('Hello, websocket!')
await session.close()
```
### Adding routes to the application
You can use any of the following to add handled routes to the application:
* `.add_route(path, func, methods=["GET"])` - Add an HTTP route. The function may be either a coroutine or a regular function, with a signature like `func(request **kwargs) -> response`.
* `.add_websocket_route(path, func)` - Add a websocket session route. The function must be a coroutine, with a signature like `func(session, **kwargs)`.
* `.mount(prefix, app)` - Include an ASGI app, mounted under the given path prefix
* `.route(path)` - Add an HTTP route, decorator style.
* `.websocket_route(path)` - Add a WebSocket route, decorator style.
---
<p align="center"><i>Starlette is <a href="https://github.com/tomchristie/starlette/blob/master/LICENSE.md">BSD licensed</a> code.<br/>Designed & built in Brighton, England.</i><br/>&mdash; ⭐️ &mdash;</p> <p align="center"><i>Starlette is <a href="https://github.com/tomchristie/starlette/blob/master/LICENSE.md">BSD licensed</a> code.<br/>Designed & built in Brighton, England.</i><br/>&mdash; ⭐️ &mdash;</p>

View File

@ -1,3 +1,4 @@
from starlette.app import App
from starlette.response import ( from starlette.response import (
FileResponse, FileResponse,
HTMLResponse, HTMLResponse,
@ -12,6 +13,7 @@ from starlette.testclient import TestClient
__all__ = ( __all__ = (
"App",
"FileResponse", "FileResponse",
"HTMLResponse", "HTMLResponse",
"JSONResponse", "JSONResponse",
@ -22,4 +24,4 @@ __all__ = (
"Request", "Request",
"TestClient", "TestClient",
) )
__version__ = "0.1.17" __version__ = "0.2.0"

77
starlette/app.py Normal file
View File

@ -0,0 +1,77 @@
from starlette.request import Request
from starlette.routing import Path, PathPrefix, Router
from starlette.types import ASGIApp, ASGIInstance, Receive, Scope, Send
from starlette.websockets import WebSocketSession
import asyncio
def request_response(func):
"""
Taks a function or coroutine `func(request, **kwargs) -> response`,
and returns an ASGI application.
"""
is_coroutine = asyncio.iscoroutinefunction(func)
def app(scope: Scope) -> ASGIInstance:
async def awaitable(receive: Receive, send: Send) -> None:
request = Request(scope, receive=receive)
kwargs = scope.get("kwargs", {})
if is_coroutine:
response = await func(request, **kwargs)
else:
response = func(request, **kwargs)
await response(receive, send)
return awaitable
return app
def websocket_session(func):
"""
Takes a coroutine `func(session, **kwargs)`, and returns an ASGI application.
"""
def app(scope: Scope) -> ASGIInstance:
async def awaitable(receive: Receive, send: Send) -> None:
session = WebSocketSession(scope, receive=receive, send=send)
kwargs = scope.get("kwargs", {})
await func(session, **kwargs)
return awaitable
return app
class App:
def __init__(self) -> None:
self.router = Router(routes=[])
def mount(self, path: str, app: ASGIApp):
prefix = PathPrefix(path, app=app)
self.router.routes.append(prefix)
def add_route(self, path: str, route, methods=None) -> None:
if methods is None:
methods = ["GET"]
instance = Path(path, request_response(route), protocol="http", methods=methods)
self.router.routes.append(instance)
def add_websocket_route(self, path: str, route) -> None:
instance = Path(path, websocket_session(route), protocol="websocket")
self.router.routes.append(instance)
def route(self, path: str):
def decorator(func):
self.add_route(path, func)
return decorator
def websocket_route(self, path: str):
def decorator(func):
self.add_websocket_route(path, func)
return decorator
def __call__(self, scope: Scope) -> ASGIInstance:
return self.router(scope)

View File

@ -1,4 +1,4 @@
from starlette import Response from starlette.response import Response
from starlette.types import Scope, ASGIApp, ASGIInstance from starlette.types import Scope, ASGIApp, ASGIInstance
import re import re
import typing import typing
@ -14,16 +14,22 @@ class Route:
class Path(Route): class Path(Route):
def __init__( def __init__(
self, path: str, app: ASGIApp, methods: typing.Sequence[str] = () self,
path: str,
app: ASGIApp,
methods: typing.Sequence[str] = (),
protocol: str = None,
) -> None: ) -> None:
self.path = path self.path = path
self.app = app self.app = app
self.protocol = protocol
self.methods = methods self.methods = methods
regex = "^" + path + "$" regex = "^" + path + "$"
regex = re.sub("{([a-zA-Z_][a-zA-Z0-9_]*)}", r"(?P<\1>[^/]+)", regex) regex = re.sub("{([a-zA-Z_][a-zA-Z0-9_]*)}", r"(?P<\1>[^/]+)", regex)
self.path_regex = re.compile(regex) self.path_regex = re.compile(regex)
def matches(self, scope: Scope) -> typing.Tuple[bool, Scope]: def matches(self, scope: Scope) -> typing.Tuple[bool, Scope]:
if self.protocol is None or scope["type"] == self.protocol:
match = self.path_regex.match(scope["path"]) match = self.path_regex.match(scope["path"])
if match: if match:
kwargs = dict(scope.get("kwargs", {})) kwargs = dict(scope.get("kwargs", {}))
@ -81,6 +87,12 @@ class Router:
return self.not_found(scope) return self.not_found(scope)
def not_found(self, scope: Scope) -> ASGIInstance: def not_found(self, scope: Scope) -> ASGIInstance:
if scope["type"] == "websocket":
async def close(receive, send):
await send({"type": "websocket.close"})
return close
return Response("Not found", 404, media_type="text/plain") return Response("Not found", 404, media_type="text/plain")

View File

@ -140,11 +140,11 @@ class WebSocketTestSession:
self._receive_queue = queue.Queue() self._receive_queue = queue.Queue()
self._send_queue = queue.Queue() self._send_queue = queue.Queue()
self._thread = threading.Thread(target=self._run) self._thread = threading.Thread(target=self._run)
self._receive_queue.put({"type": "websocket.connect"}) self.send({"type": "websocket.connect"})
self._thread.start() self._thread.start()
message = self._send_queue.get() message = self.receive()
self._raise_on_close_or_exception(message) self._raise_on_close(message)
self.accepted_subprotocol = message["subprotocol"] self.accepted_subprotocol = message.get("subprotocol", None)
def __enter__(self): def __enter__(self):
return self return self
@ -174,38 +174,45 @@ class WebSocketTestSession:
async def _asgi_send(self, message): async def _asgi_send(self, message):
self._send_queue.put(message) self._send_queue.put(message)
def _raise_on_close_or_exception(self, message): def _raise_on_close(self, message):
if isinstance(message, BaseException):
raise message
if message["type"] == "websocket.close": if message["type"] == "websocket.close":
raise WebSocketDisconnect(message["code"]) raise WebSocketDisconnect(message.get("code", 1000))
def send(self, message):
self._receive_queue.put(message)
def send_text(self, data): def send_text(self, data):
self._receive_queue.put({"type": "websocket.receive", "text": data}) self.send({"type": "websocket.receive", "text": data})
def send_bytes(self, data): def send_bytes(self, data):
self._receive_queue.put({"type": "websocket.receive", "bytes": data}) self.send({"type": "websocket.receive", "bytes": data})
def send_json(self, data): def send_json(self, data):
encoded = json.dumps(data).encode("utf-8") encoded = json.dumps(data).encode("utf-8")
self._receive_queue.put({"type": "websocket.receive", "bytes": encoded}) self.send({"type": "websocket.receive", "bytes": encoded})
def close(self, code=1000): def close(self, code=1000):
self._receive_queue.put({"type": "websocket.disconnect", "code": code}) self.send({"type": "websocket.disconnect", "code": code})
def receive(self):
message = self._send_queue.get()
if isinstance(message, BaseException):
raise message
return message
def receive_text(self): def receive_text(self):
message = self._send_queue.get() message = self.receive()
self._raise_on_close_or_exception(message) self._raise_on_close(message)
return message["text"] return message["text"]
def receive_bytes(self): def receive_bytes(self):
message = self._send_queue.get() message = self.receive()
self._raise_on_close_or_exception(message) self._raise_on_close(message)
return message["bytes"] return message["bytes"]
def receive_json(self): def receive_json(self):
message = self._send_queue.get() message = self.receive()
self._raise_on_close_or_exception(message) self._raise_on_close(message)
encoded = message["bytes"] encoded = message["bytes"]
return json.loads(encoded.decode("utf-8")) return json.loads(encoded.decode("utf-8"))
@ -225,7 +232,9 @@ class _TestClient(requests.Session):
url = urljoin(self.base_url, url) url = urljoin(self.base_url, url)
return super().request(method, url, **kwargs) return super().request(method, url, **kwargs)
def wsconnect(self, url: str, subprotocols=None, **kwargs) -> WebSocketTestSession: def websocket_connect(
self, url: str, subprotocols=None, **kwargs
) -> WebSocketTestSession:
url = urljoin("ws://testserver", url) url = urljoin("ws://testserver", url)
headers = kwargs.get("headers", {}) headers = kwargs.get("headers", {})
headers.setdefault("connection", "upgrade") headers.setdefault("connection", "upgrade")

75
tests/test_app.py Normal file
View File

@ -0,0 +1,75 @@
from starlette import App
from starlette.response import PlainTextResponse
from starlette.staticfiles import StaticFiles
from starlette.testclient import TestClient
import os
app = App()
@app.route("/func")
def func_homepage(request):
return PlainTextResponse("Hello, world!")
@app.route("/async")
async def async_homepage(request):
return PlainTextResponse("Hello, world!")
@app.route("/user/{username}")
def user_page(request, username):
return PlainTextResponse("Hello, %s!" % username)
@app.websocket_route("/ws")
async def websocket_endpoint(session):
await session.accept()
await session.send_text("Hello, world!")
await session.close()
client = TestClient(app)
def test_func_route():
response = client.get("/func")
assert response.status_code == 200
assert response.text == "Hello, world!"
def test_async_route():
response = client.get("/async")
assert response.status_code == 200
assert response.text == "Hello, world!"
def test_route_kwargs():
response = client.get("/user/tomchristie")
assert response.status_code == 200
assert response.text == "Hello, tomchristie!"
def test_websocket_route():
with client.websocket_connect("/ws") as session:
text = session.receive_text()
assert text == "Hello, world!"
def test_400():
response = client.get("/404")
assert response.status_code == 404
def test_app_mount(tmpdir):
path = os.path.join(tmpdir, "example.txt")
with open(path, "w") as file:
file.write("<file content>")
app = App()
app.mount("/static", StaticFiles(directory=tmpdir))
client = TestClient(app)
response = client.get("/static/example.txt")
assert response.status_code == 200
assert response.text == "<file content>"

View File

@ -1,6 +1,7 @@
from starlette import Response, TestClient from starlette import Response, TestClient
from starlette.routing import Path, PathPrefix, Router, ProtocolRouter from starlette.routing import Path, PathPrefix, Router, ProtocolRouter
from starlette.websockets import WebSocketSession from starlette.websockets import WebSocketSession, WebSocketDisconnect
import pytest
def homepage(scope): def homepage(scope):
@ -78,7 +79,10 @@ def websocket_endpoint(scope):
mixed_protocol_app = ProtocolRouter( mixed_protocol_app = ProtocolRouter(
{"http": http_endpoint, "websocket": websocket_endpoint} {
"http": Router([Path("/", app=http_endpoint)]),
"websocket": Router([Path("/", app=websocket_endpoint)]),
}
) )
@ -89,5 +93,8 @@ def test_protocol_switch():
assert response.status_code == 200 assert response.status_code == 200
assert response.text == "Hello, world" assert response.text == "Hello, world"
with client.wsconnect("/") as session: with client.websocket_connect("/") as session:
assert session.receive_json() == {"hello": "world"} assert session.receive_json() == {"hello": "world"}
with pytest.raises(WebSocketDisconnect):
client.websocket_connect("/404")

View File

@ -14,7 +14,7 @@ def test_session_url():
return asgi return asgi
client = TestClient(app) client = TestClient(app)
with client.wsconnect("/123?a=abc") as session: with client.websocket_connect("/123?a=abc") as session:
data = session.receive_json() data = session.receive_json()
assert data == {"url": "ws://testserver/123?a=abc"} assert data == {"url": "ws://testserver/123?a=abc"}
@ -31,7 +31,7 @@ def test_session_query_params():
return asgi return asgi
client = TestClient(app) client = TestClient(app)
with client.wsconnect("/?a=abc&b=456") as session: with client.websocket_connect("/?a=abc&b=456") as session:
data = session.receive_json() data = session.receive_json()
assert data == {"params": {"a": "abc", "b": "456"}} assert data == {"params": {"a": "abc", "b": "456"}}
@ -48,7 +48,7 @@ def test_session_headers():
return asgi return asgi
client = TestClient(app) client = TestClient(app)
with client.wsconnect("/") as session: with client.websocket_connect("/") as session:
expected_headers = { expected_headers = {
"accept": "*/*", "accept": "*/*",
"accept-encoding": "gzip, deflate", "accept-encoding": "gzip, deflate",
@ -73,7 +73,7 @@ def test_session_port():
return asgi return asgi
client = TestClient(app) client = TestClient(app)
with client.wsconnect("ws://example.com:123/123?a=abc") as session: with client.websocket_connect("ws://example.com:123/123?a=abc") as session:
data = session.receive_json() data = session.receive_json()
assert data == {"port": 123} assert data == {"port": 123}
@ -90,7 +90,7 @@ def test_session_send_and_receive_text():
return asgi return asgi
client = TestClient(app) client = TestClient(app)
with client.wsconnect("/") as session: with client.websocket_connect("/") as session:
session.send_text("Hello, world!") session.send_text("Hello, world!")
data = session.receive_text() data = session.receive_text()
assert data == "Message was: Hello, world!" assert data == "Message was: Hello, world!"
@ -108,7 +108,7 @@ def test_session_send_and_receive_bytes():
return asgi return asgi
client = TestClient(app) client = TestClient(app)
with client.wsconnect("/") as session: with client.websocket_connect("/") as session:
session.send_bytes(b"Hello, world!") session.send_bytes(b"Hello, world!")
data = session.receive_bytes() data = session.receive_bytes()
assert data == b"Message was: Hello, world!" assert data == b"Message was: Hello, world!"
@ -126,7 +126,7 @@ def test_session_send_and_receive_json():
return asgi return asgi
client = TestClient(app) client = TestClient(app)
with client.wsconnect("/") as session: with client.websocket_connect("/") as session:
session.send_json({"hello": "world"}) session.send_json({"hello": "world"})
data = session.receive_json() data = session.receive_json()
assert data == {"message": {"hello": "world"}} assert data == {"message": {"hello": "world"}}
@ -148,7 +148,7 @@ def test_client_close():
return asgi return asgi
client = TestClient(app) client = TestClient(app)
with client.wsconnect("/") as session: with client.websocket_connect("/") as session:
session.close(code=1001) session.close(code=1001)
assert close_code == 1001 assert close_code == 1001
@ -163,7 +163,7 @@ def test_application_close():
return asgi return asgi
client = TestClient(app) client = TestClient(app)
with client.wsconnect("/") as session: with client.websocket_connect("/") as session:
with pytest.raises(WebSocketDisconnect) as exc: with pytest.raises(WebSocketDisconnect) as exc:
session.receive_text() session.receive_text()
assert exc.value.code == 1001 assert exc.value.code == 1001
@ -179,7 +179,7 @@ def test_rejected_connection():
client = TestClient(app) client = TestClient(app)
with pytest.raises(WebSocketDisconnect) as exc: with pytest.raises(WebSocketDisconnect) as exc:
client.wsconnect("/") client.websocket_connect("/")
assert exc.value.code == 1001 assert exc.value.code == 1001
@ -194,7 +194,7 @@ def test_subprotocol():
return asgi return asgi
client = TestClient(app) client = TestClient(app)
with client.wsconnect("/", subprotocols=["soap", "wamp"]) as session: with client.websocket_connect("/", subprotocols=["soap", "wamp"]) as session:
assert session.accepted_subprotocol == "wamp" assert session.accepted_subprotocol == "wamp"
@ -207,7 +207,7 @@ def test_session_exception():
client = TestClient(app) client = TestClient(app)
with pytest.raises(AssertionError): with pytest.raises(AssertionError):
client.wsconnect("/123?a=abc") client.websocket_connect("/123?a=abc")
def test_duplicate_close(): def test_duplicate_close():
@ -222,7 +222,7 @@ def test_duplicate_close():
client = TestClient(app) client = TestClient(app)
with pytest.raises(RuntimeError): with pytest.raises(RuntimeError):
with client.wsconnect("/") as session: with client.websocket_connect("/") as session:
pass pass
@ -239,7 +239,7 @@ def test_duplicate_disconnect():
client = TestClient(app) client = TestClient(app)
with pytest.raises(RuntimeError): with pytest.raises(RuntimeError):
with client.wsconnect("/") as session: with client.websocket_connect("/") as session:
session.close() session.close()