Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Convenience method to allow customizing route dependencies #295

Merged
merged 19 commits into from
Feb 15, 2022
Merged
21 changes: 19 additions & 2 deletions stac_fastapi/api/stac_fastapi/api/app.py
Original file line number Diff line number Diff line change
@@ -1,10 +1,11 @@
"""fastapi app creation."""
from typing import Any, Callable, Dict, List, Optional, Type, Union
from typing import Any, Callable, Dict, List, Optional, Tuple, Type, Union

import attr
from brotli_asgi import BrotliMiddleware
from fastapi import APIRouter, FastAPI
from fastapi.openapi.utils import get_openapi
from fastapi.params import Depends
from pydantic import BaseModel
from stac_pydantic import Collection, Item, ItemCollection
from stac_pydantic.api import ConformanceClasses, LandingPage, Search
Expand All @@ -22,7 +23,12 @@
SearchGetRequest,
_create_request_model,
)
from stac_fastapi.api.routes import create_async_endpoint, create_sync_endpoint
from stac_fastapi.api.routes import (
Scope,
add_route_dependencies,
create_async_endpoint,
create_sync_endpoint,
)

# TODO: make this module not depend on `stac_fastapi.extensions`
from stac_fastapi.extensions.core import FieldsExtension
Expand Down Expand Up @@ -77,6 +83,7 @@ class StacApi:
item_collection_uri: Type[ItemCollectionUri] = attr.ib(default=ItemCollectionUri)
response_class: Type[Response] = attr.ib(default=JSONResponse)
middlewares: List = attr.ib(default=attr.Factory(lambda: [BrotliMiddleware]))
route_dependencies: List[Tuple[List[Scope], List[Depends]]] = attr.ib(default=[])

def get_extension(self, extension: Type[ApiExtension]) -> Optional[ApiExtension]:
"""Get an extension.
Expand Down Expand Up @@ -312,6 +319,12 @@ async def ping():

self.app.include_router(mgmt_router, tags=["Liveliness/Readiness"])

def add_route_dependencies(
self, scopes: List[Scope], dependencies=List[Depends]
alukach marked this conversation as resolved.
Show resolved Hide resolved
) -> None:
"""Add custom dependencies to routes."""
return add_route_dependencies(self.app.router.routes, scopes, dependencies)

def __attrs_post_init__(self):
"""Post-init hook.

Expand Down Expand Up @@ -353,3 +366,7 @@ def __attrs_post_init__(self):
# add middlewares
for middleware in self.middlewares:
self.app.add_middleware(middleware)

# customize route dependencies
for scopes, dependencies in self.route_dependencies:
self.add_route_dependencies(scopes=scopes, dependencies=dependencies)
50 changes: 48 additions & 2 deletions stac_fastapi/api/stac_fastapi/api/routes.py
Original file line number Diff line number Diff line change
@@ -1,10 +1,12 @@
"""route factories."""
from typing import Any, Callable, Dict, Type, Union
from typing import Any, Callable, Dict, List, Optional, Type, TypedDict, Union

from fastapi import Depends
from fastapi import Depends, params
from fastapi.dependencies.utils import get_parameterless_sub_dependant
from pydantic import BaseModel
from starlette.requests import Request
from starlette.responses import JSONResponse, Response
from starlette.routing import BaseRoute, Match

from stac_fastapi.api.models import APIRequest

Expand Down Expand Up @@ -94,3 +96,47 @@ def _endpoint(
return _wrap_response(func(request_data, request=request), response_class)

return _endpoint


class Scope(TypedDict, total=False):
"""More strict version of Starlette's Scope."""

# https://github.com/encode/starlette/blob/6af5c515e0a896cbf3f86ee043b88f6c24200bcf/starlette/types.py#L3
path: str
method: str
type: Optional[str]


def add_route_dependencies(
routes: List[BaseRoute], scopes: List[Scope], dependencies=List[params.Depends]
) -> None:
"""Add dependencies to routes.

Allows a developer to add dependencies to a route after the route has been
defined.

Returns:
None
"""
for scope in scopes:
for route in routes:

match, _ = route.matches({"type": "http", **scope})
if match != Match.FULL:
continue

# Mimicking how APIRoute handles dependencies:
# https://github.com/tiangolo/fastapi/blob/1760da0efa55585c19835d81afa8ca386036c325/fastapi/routing.py#L408-L412
for depends in dependencies[::-1]:
route.dependant.dependencies.insert(
vincentsarago marked this conversation as resolved.
Show resolved Hide resolved
0,
get_parameterless_sub_dependant(
depends=depends, path=route.path_format
vincentsarago marked this conversation as resolved.
Show resolved Hide resolved
),
)

# Register dependencies directly on route so that they aren't ignored if
# the routes are later associated with an app (e.g. app.include_router(router))
# https://github.com/tiangolo/fastapi/blob/58ab733f19846b4875c5b79bfb1f4d1cb7f4823f/fastapi/applications.py#L337-L360
# https://github.com/tiangolo/fastapi/blob/58ab733f19846b4875c5b79bfb1f4d1cb7f4823f/fastapi/routing.py#L677-L678
route.dependencies.extend(dependencies)