2018-12-07 13:05:31 +00:00
|
|
|
Starlette offers a simple but powerful interface for handling authentication
|
|
|
|
and permissions. Once you've installed `AuthenticationMiddleware` with an
|
|
|
|
appropriate authentication backend the `request.user` and `request.auth`
|
|
|
|
interfaces will be available in your endpoints.
|
|
|
|
|
|
|
|
|
|
|
|
```python
|
2021-04-28 09:42:02 +00:00
|
|
|
from starlette.applications import Starlette
|
2018-12-07 13:05:31 +00:00
|
|
|
from starlette.authentication import (
|
2022-01-18 14:48:48 +00:00
|
|
|
AuthCredentials, AuthenticationBackend, AuthenticationError, SimpleUser
|
2018-12-07 13:05:31 +00:00
|
|
|
)
|
2019-11-13 12:25:18 +00:00
|
|
|
from starlette.middleware import Middleware
|
2018-12-10 16:51:43 +00:00
|
|
|
from starlette.middleware.authentication import AuthenticationMiddleware
|
2018-12-07 13:05:31 +00:00
|
|
|
from starlette.responses import PlainTextResponse
|
2019-11-13 12:25:18 +00:00
|
|
|
from starlette.routing import Route
|
2018-12-07 13:05:31 +00:00
|
|
|
import base64
|
|
|
|
import binascii
|
|
|
|
|
|
|
|
|
|
|
|
class BasicAuthBackend(AuthenticationBackend):
|
2022-01-18 14:48:48 +00:00
|
|
|
async def authenticate(self, conn):
|
|
|
|
if "Authorization" not in conn.headers:
|
2018-12-07 13:05:31 +00:00
|
|
|
return
|
|
|
|
|
2022-01-18 14:48:48 +00:00
|
|
|
auth = conn.headers["Authorization"]
|
2018-12-07 13:05:31 +00:00
|
|
|
try:
|
|
|
|
scheme, credentials = auth.split()
|
|
|
|
if scheme.lower() != 'basic':
|
|
|
|
return
|
|
|
|
decoded = base64.b64decode(credentials).decode("ascii")
|
2018-12-10 16:51:43 +00:00
|
|
|
except (ValueError, UnicodeDecodeError, binascii.Error) as exc:
|
2018-12-07 13:05:31 +00:00
|
|
|
raise AuthenticationError('Invalid basic auth credentials')
|
|
|
|
|
|
|
|
username, _, password = decoded.partition(":")
|
2019-11-13 12:25:18 +00:00
|
|
|
# TODO: You'd want to verify the username and password here.
|
2018-12-07 13:05:31 +00:00
|
|
|
return AuthCredentials(["authenticated"]), SimpleUser(username)
|
|
|
|
|
|
|
|
|
|
|
|
async def homepage(request):
|
|
|
|
if request.user.is_authenticated:
|
2019-11-13 12:25:18 +00:00
|
|
|
return PlainTextResponse('Hello, ' + request.user.display_name)
|
|
|
|
return PlainTextResponse('Hello, you')
|
|
|
|
|
|
|
|
routes = [
|
|
|
|
Route("/", endpoint=homepage)
|
|
|
|
]
|
|
|
|
|
|
|
|
middleware = [
|
|
|
|
Middleware(AuthenticationMiddleware, backend=BasicAuthBackend())
|
|
|
|
]
|
|
|
|
|
|
|
|
app = Starlette(routes=routes, middleware=middleware)
|
2018-12-07 13:05:31 +00:00
|
|
|
```
|
|
|
|
|
|
|
|
## Users
|
|
|
|
|
|
|
|
Once `AuthenticationMiddleware` is installed the `request.user` interface
|
|
|
|
will be available to endpoints or other middleware.
|
|
|
|
|
|
|
|
This interface should subclass `BaseUser`, which provides two properties,
|
|
|
|
as well as whatever other information your user model includes.
|
|
|
|
|
|
|
|
* `.is_authenticated`
|
|
|
|
* `.display_name`
|
|
|
|
|
|
|
|
Starlette provides two built-in user implementations: `UnauthenticatedUser()`,
|
|
|
|
and `SimpleUser(username)`.
|
|
|
|
|
|
|
|
## AuthCredentials
|
|
|
|
|
|
|
|
It is important that authentication credentials are treated as separate concept
|
|
|
|
from users. An authentication scheme should be able to restrict or grant
|
|
|
|
particular privileges independently of the user identity.
|
|
|
|
|
|
|
|
The `AuthCredentials` class provides the basic interface that `request.auth`
|
|
|
|
exposes:
|
|
|
|
|
|
|
|
* `.scopes`
|
|
|
|
|
|
|
|
## Permissions
|
|
|
|
|
|
|
|
Permissions are implemented as an endpoint decorator, that enforces that the
|
|
|
|
incoming request includes the required authentication scopes.
|
|
|
|
|
|
|
|
```python
|
|
|
|
from starlette.authentication import requires
|
|
|
|
|
|
|
|
|
|
|
|
@requires('authenticated')
|
|
|
|
async def dashboard(request):
|
|
|
|
...
|
|
|
|
```
|
|
|
|
|
|
|
|
You can include either one or multiple required scopes:
|
|
|
|
|
|
|
|
```python
|
|
|
|
from starlette.authentication import requires
|
|
|
|
|
|
|
|
|
|
|
|
@requires(['authenticated', 'admin'])
|
|
|
|
async def dashboard(request):
|
|
|
|
...
|
|
|
|
```
|
|
|
|
|
|
|
|
By default 403 responses will be returned when permissions are not granted.
|
|
|
|
In some cases you might want to customize this, for example to hide information
|
|
|
|
about the URL layout from unauthenticated users.
|
|
|
|
|
|
|
|
```python
|
|
|
|
from starlette.authentication import requires
|
|
|
|
|
|
|
|
|
|
|
|
@requires(['authenticated', 'admin'], status_code=404)
|
|
|
|
async def dashboard(request):
|
|
|
|
...
|
|
|
|
```
|
|
|
|
|
2022-05-17 09:36:34 +00:00
|
|
|
!!! note
|
|
|
|
The `status_code` parameter is not supported with WebSockets. The 403 (Forbidden)
|
|
|
|
status code will always be used for those.
|
|
|
|
|
2018-12-07 13:05:31 +00:00
|
|
|
Alternatively you might want to redirect unauthenticated users to a different
|
|
|
|
page.
|
|
|
|
|
|
|
|
```python
|
|
|
|
from starlette.authentication import requires
|
|
|
|
|
|
|
|
|
|
|
|
async def homepage(request):
|
|
|
|
...
|
|
|
|
|
|
|
|
|
|
|
|
@requires('authenticated', redirect='homepage')
|
|
|
|
async def dashboard(request):
|
|
|
|
...
|
|
|
|
```
|
2018-12-18 12:32:28 +00:00
|
|
|
|
2022-02-16 11:19:08 +00:00
|
|
|
When redirecting users, the page you redirect them to will include URL they originally requested at the `next` query param:
|
|
|
|
|
|
|
|
```python
|
|
|
|
from starlette.authentication import requires
|
|
|
|
from starlette.responses import RedirectResponse
|
|
|
|
|
|
|
|
|
|
|
|
@requires('authenticated', redirect='login')
|
|
|
|
async def admin(request):
|
|
|
|
...
|
|
|
|
|
|
|
|
|
|
|
|
async def login(request):
|
|
|
|
if request.method == "POST":
|
|
|
|
# Now that the user is authenticated,
|
|
|
|
# we can send them to their original request destination
|
|
|
|
if request.user.is_authenticated:
|
|
|
|
next_url = request.query_params.get("next")
|
|
|
|
if next_url:
|
|
|
|
return RedirectResponse(next_url)
|
|
|
|
return RedirectResponse("/")
|
|
|
|
```
|
|
|
|
|
2019-01-10 13:46:46 +00:00
|
|
|
For class-based endpoints, you should wrap the decorator
|
|
|
|
around a method on the class.
|
|
|
|
|
|
|
|
```python
|
2022-01-18 14:48:48 +00:00
|
|
|
from starlette.authentication import requires
|
|
|
|
from starlette.endpoints import HTTPEndpoint
|
|
|
|
|
|
|
|
|
2019-01-10 13:46:46 +00:00
|
|
|
class Dashboard(HTTPEndpoint):
|
|
|
|
@requires("authenticated")
|
|
|
|
async def get(self, request):
|
|
|
|
...
|
|
|
|
```
|
|
|
|
|
2018-12-18 12:32:28 +00:00
|
|
|
## Custom authentication error responses
|
|
|
|
|
2019-01-10 13:46:46 +00:00
|
|
|
You can customise the error response sent when a `AuthenticationError` is
|
2018-12-18 12:32:28 +00:00
|
|
|
raised by an auth backend:
|
|
|
|
|
|
|
|
```python
|
2022-07-13 04:29:57 +00:00
|
|
|
from starlette.applications import Starlette
|
|
|
|
from starlette.middleware import Middleware
|
2022-01-18 14:48:48 +00:00
|
|
|
from starlette.middleware.authentication import AuthenticationMiddleware
|
|
|
|
from starlette.requests import Request
|
|
|
|
from starlette.responses import JSONResponse
|
|
|
|
|
|
|
|
|
2018-12-18 12:32:28 +00:00
|
|
|
def on_auth_error(request: Request, exc: Exception):
|
|
|
|
return JSONResponse({"error": str(exc)}, status_code=401)
|
|
|
|
|
2022-07-13 04:29:57 +00:00
|
|
|
app = Starlette(
|
|
|
|
middleware=[
|
|
|
|
Middleware(AuthenticationMiddleware, backend=BasicAuthBackend(), on_error=on_auth_error),
|
|
|
|
],
|
|
|
|
)
|
2019-01-10 13:46:46 +00:00
|
|
|
```
|