1 # SPDX-License-Identifier: GPL-2.0-only
 
   3 # This file is part of Nominatim. (https://nominatim.org)
 
   5 # Copyright (C) 2023 by the Nominatim developer community.
 
   6 # For a full list of authors see the git log.
 
   8 Server implementation using the starlette webserver framework.
 
  10 from typing import Any, Optional, Mapping, Callable, cast, Coroutine
 
  11 from pathlib import Path
 
  13 from starlette.applications import Starlette
 
  14 from starlette.routing import Route
 
  15 from starlette.exceptions import HTTPException
 
  16 from starlette.responses import Response
 
  17 from starlette.requests import Request
 
  18 from starlette.middleware import Middleware
 
  19 from starlette.middleware.cors import CORSMiddleware
 
  21 from nominatim.api import NominatimAPIAsync
 
  22 import nominatim.api.v1 as api_impl
 
  23 from nominatim.config import Configuration
 
  25 class ParamWrapper(api_impl.ASGIAdaptor):
 
  26     """ Adaptor class for server glue to Starlette framework.
 
  29     def __init__(self, request: Request) -> None:
 
  30         self.request = request
 
  33     def get(self, name: str, default: Optional[str] = None) -> Optional[str]:
 
  34         return self.request.query_params.get(name, default=default)
 
  37     def get_header(self, name: str, default: Optional[str] = None) -> Optional[str]:
 
  38         return self.request.headers.get(name, default)
 
  41     def error(self, msg: str, status: int = 400) -> HTTPException:
 
  42         return HTTPException(status, detail=msg,
 
  43                              headers={'content-type': self.content_type})
 
  46     def create_response(self, status: int, output: str) -> Response:
 
  47         return Response(output, status_code=status, media_type=self.content_type)
 
  50     def config(self) -> Configuration:
 
  51         return cast(Configuration, self.request.app.state.API.config)
 
  54 def _wrap_endpoint(func: api_impl.EndpointFunc)\
 
  55         -> Callable[[Request], Coroutine[Any, Any, Response]]:
 
  56     async def _callback(request: Request) -> Response:
 
  57         return cast(Response, await func(request.app.state.API, ParamWrapper(request)))
 
  62 def get_application(project_dir: Path,
 
  63                     environ: Optional[Mapping[str, str]] = None,
 
  64                     debug: bool = True) -> Starlette:
 
  65     """ Create a Nominatim falcon ASGI application.
 
  67     config = Configuration(project_dir, environ)
 
  70     legacy_urls = config.get_bool('SERVE_LEGACY_URLS')
 
  71     for name, func in api_impl.ROUTES:
 
  72         endpoint = _wrap_endpoint(func)
 
  73         routes.append(Route(f"/{name}", endpoint=endpoint))
 
  75             routes.append(Route(f"/{name}.php", endpoint=endpoint))
 
  78     if config.get_bool('CORS_NOACCESSCONTROL'):
 
  79         middleware.append(Middleware(CORSMiddleware, allow_origins=['*']))
 
  81     async def _shutdown() -> None:
 
  82         await app.state.API.close()
 
  84     app = Starlette(debug=debug, routes=routes, middleware=middleware,
 
  85                     on_shutdown=[_shutdown])
 
  87     app.state.API = NominatimAPIAsync(project_dir, environ)
 
  92 def run_wsgi() -> Starlette:
 
  93     """ Entry point for uvicorn.
 
  95     return get_application(Path('.'), debug=False)