Skip to content

Commit

Permalink
Merge pull request pytroll#6 from pkhalaj/enhancement/more-tests-and-…
Browse files Browse the repository at this point in the history
…cleanup

Enhance code quality
  • Loading branch information
mraspaud authored May 31, 2024
2 parents 603ee30 + 781891d commit 1f0b8d0
Show file tree
Hide file tree
Showing 22 changed files with 556 additions and 246 deletions.
52 changes: 51 additions & 1 deletion docs/source/conf.py
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@
# documentation root, use os.path.abspath to make it absolute, like shown here.

import os
import re

from sphinx.ext import apidoc

Expand Down Expand Up @@ -62,14 +63,63 @@
# so a file named "default.css" will overwrite the builtin "default.css".
# html_static_path = ["_static"]


# Specify which special members have to be kept
special_members_dict = {
"Document": {"init"},
"ResponseError": {"init", "or"},
"PipelineBooleanDict": {"init", "or", "and"},
"PipelineAttribute": {"init", "or", "and", "eq", "gt", "ge", "lt", "le"},
"Pipelines": {"init", "add", "iadd"}
}

# Add trailing and leading "__" to all the aforementioned members
for cls, methods in special_members_dict.items():
special_members_dict[cls] = {f"__{method}__" for method in methods}

# Make a set of all allowed special members
all_special_members = set()
for methods in special_members_dict.values():
all_special_members |= methods

autodoc_default_options = {
"members": True,
"member-order": "bysource",
"private-members": True,
"special-members": True,
"undoc-members": True,
"undoc-members": False,
}


def is_special_member(member_name: str) -> bool:
"""Checks if the given member is special, i.e. its name has the following format ``__<some-str>__``."""
return bool(re.compile(r"^__\w+__$").match(member_name))


def skip(app, typ, member_name, obj, flag, options):
"""The filter function to determine whether to keep the member in the documentation.
``True`` means skip the member.
"""
if is_special_member(member_name):

if member_name not in all_special_members:
return True

obj_name = obj.__qualname__.split(".")[0]
if methods_set := special_members_dict.get(obj_name, None):
if member_name in methods_set:
return False # Keep the member
return True

return None


def setup(app):
"""Sets up the sphinx app."""
app.connect("autodoc-skip-member", skip)


root_doc = "index"

output_dir = os.path.join(".")
Expand Down
2 changes: 1 addition & 1 deletion docs/source/index.rst
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
Welcome to Pytroll documentation!
Pytroll-db Documentation
===========================================

.. toctree::
Expand Down
2 changes: 1 addition & 1 deletion trolldb/__init__.py
Original file line number Diff line number Diff line change
@@ -1 +1 @@
"""trolldb package."""
"""The database interface of the Pytroll package."""
2 changes: 1 addition & 1 deletion trolldb/api/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -5,5 +5,5 @@
For more information and documentation, please refer to the following sub-packages and modules:
- :obj:`trolldb.api.routes`: The package which defines the API routes.
- :obj:`trollddb.api.api`: The module which defines the API server and how it is run via the given configuration.
- :obj:`trolldb.api.api`: The module which defines the API server and how it is run via the given configuration.
"""
50 changes: 24 additions & 26 deletions trolldb/api/api.py
Original file line number Diff line number Diff line change
Expand Up @@ -14,27 +14,28 @@
"""

import asyncio
import sys
import time
from contextlib import contextmanager
from multiprocessing import Process
from typing import Union
from typing import Any, Generator, NoReturn

import uvicorn
from fastapi import FastAPI, status
from fastapi.responses import PlainTextResponse
from loguru import logger
from pydantic import FilePath, validate_call
from pydantic import ValidationError

from trolldb.api.routes import api_router
from trolldb.config.config import AppConfig, Timeout, parse_config_yaml_file
from trolldb.config.config import AppConfig, Timeout
from trolldb.database.mongodb import mongodb_context
from trolldb.errors.errors import ResponseError

API_INFO = dict(
title="pytroll-db",
summary="The database API of Pytroll",
description=
"The API allows you to perform CRUD operations as well as querying the database"
"The API allows you to perform CRUD operations as well as querying the database"
"At the moment only MongoDB is supported. It is based on the following Python packages"
"\n * **PyMongo** (https://github.com/mongodb/mongo-python-driver)"
"\n * **motor** (https://github.com/mongodb/motor)",
Expand All @@ -43,11 +44,11 @@
url="https://www.gnu.org/licenses/gpl-3.0.en.html"
)
)
"""These will appear int the auto-generated documentation and are passed to the ``FastAPI`` class as keyword args."""
"""These will appear in the auto-generated documentation and are passed to the ``FastAPI`` class as keyword args."""


@validate_call
def run_server(config: Union[AppConfig, FilePath], **kwargs) -> None:
@logger.catch(onerror=lambda _: sys.exit(1))
def run_server(config: AppConfig, **kwargs) -> None:
"""Runs the API server with all the routes and connection to the database.
It first creates a FastAPI application and runs it using `uvicorn <https://www.uvicorn.org/>`_ which is
Expand All @@ -56,40 +57,34 @@ def run_server(config: Union[AppConfig, FilePath], **kwargs) -> None:
Args:
config:
The configuration of the application which includes both the server and database configurations. Its type
should be a :class:`FilePath`, which is a valid path to an existing config file which will parsed as a
``.YAML`` file.
The configuration of the application which includes both the server and database configurations.
**kwargs:
The keyword arguments are the same as those accepted by the
`FastAPI class <https://fastapi.tiangolo.com/reference/fastapi/#fastapi.FastAPI>`_ and are directly passed
to it. These keyword arguments will be first concatenated with the configurations of the API server which
are read from the ``config`` argument. The keyword arguments which are passed explicitly to the function
take precedence over ``config``. Finally, ``API_INFO``, which are hard-coded information for the API server,
will be concatenated and takes precedence over all.
Raises:
ValidationError:
If the function is not called with arguments of valid type.
take precedence over ``config``. Finally, :obj:`API_INFO`, which are hard-coded information for the API
server, will be concatenated and takes precedence over all.
Example:
.. code-block:: python
from api.api import run_server
from trolldb.api.api import run_server
from trolldb.config.config import parse_config
if __name__ == "__main__":
run_server("config.yaml")
run_server(parse_config("config.yaml"))
"""
logger.info("Attempt to run the API server ...")
if not isinstance(config, AppConfig):
config = parse_config_yaml_file(config)

# Concatenate the keyword arguments for the API server in the order of precedence (lower to higher).
app = FastAPI(**(config.api_server._asdict() | kwargs | API_INFO))

app.include_router(api_router)

@app.exception_handler(ResponseError)
async def auto_exception_handler(_, exc: ResponseError):
async def auto_handler_response_errors(_, exc: ResponseError) -> PlainTextResponse:
"""Catches all the exceptions raised as a ResponseError, e.g. accessing non-existing databases/collections."""
status_code, message = exc.get_error_details()
info = dict(
Expand All @@ -99,7 +94,13 @@ async def auto_exception_handler(_, exc: ResponseError):
logger.error(f"Response error caught by the API auto exception handler: {info}")
return PlainTextResponse(**info)

async def _serve():
@app.exception_handler(ValidationError)
async def auto_handler_pydantic_validation_errors(_, exc: ValidationError) -> PlainTextResponse:
"""Catches all the exceptions raised as a Pydantic ValidationError."""
logger.error(f"Response error caught by the API auto exception handler: {exc}")
return PlainTextResponse(str(exc), status_code=status.HTTP_500_INTERNAL_SERVER_ERROR)

async def _serve() -> NoReturn:
"""An auxiliary coroutine to be used in the asynchronous execution of the FastAPI application."""
async with mongodb_context(config.database):
logger.info("Attempt to start the uvicorn server ...")
Expand All @@ -116,7 +117,7 @@ async def _serve():


@contextmanager
def api_server_process_context(config: Union[AppConfig, FilePath], startup_time: Timeout = 2):
def api_server_process_context(config: AppConfig, startup_time: Timeout = 2) -> Generator[Process, Any, None]:
"""A synchronous context manager to run the API server in a separate process (non-blocking).
It uses the `multiprocessing <https://docs.python.org/3/library/multiprocessing.html>`_ package. The main use case
Expand All @@ -132,9 +133,6 @@ def api_server_process_context(config: Union[AppConfig, FilePath], startup_time:
large so that the tests will not time out.
"""
logger.info("Attempt to run the API server process in a context manager ...")
if not isinstance(config, AppConfig):
config = parse_config_yaml_file(config)

process = Process(target=run_server, args=(config,))
try:
process.start()
Expand Down
2 changes: 1 addition & 1 deletion trolldb/api/routes/__init__.py
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
"""routes package."""
"""The routes package of the API."""

from .router import api_router

Expand Down
10 changes: 1 addition & 9 deletions trolldb/api/routes/common.py
Original file line number Diff line number Diff line change
Expand Up @@ -2,19 +2,11 @@

from typing import Annotated, Union

from fastapi import Depends, Query, Response
from fastapi import Depends, Response
from motor.motor_asyncio import AsyncIOMotorCollection, AsyncIOMotorDatabase

from trolldb.database.mongodb import MongoDB

exclude_defaults_query = Query(
True,
title="Query string",
description=
"A boolean to exclude default databases from a MongoDB instance. Refer to "
"`trolldb.database.mongodb.MongoDB.default_database_names` for more information."
)


async def check_database(database_name: str | None = None) -> AsyncIOMotorDatabase:
"""A dependency for route handlers to check for the existence of a database given its name.
Expand Down
13 changes: 10 additions & 3 deletions trolldb/api/routes/databases.py
Original file line number Diff line number Diff line change
Expand Up @@ -4,10 +4,12 @@
For more information on the API server, see the automatically generated documentation by FastAPI.
"""

from fastapi import APIRouter
from typing import Annotated

from fastapi import APIRouter, Query
from pymongo.collection import _DocumentType

from trolldb.api.routes.common import CheckCollectionDependency, CheckDataBaseDependency, exclude_defaults_query
from trolldb.api.routes.common import CheckCollectionDependency, CheckDataBaseDependency
from trolldb.config.config import MongoObjectId
from trolldb.database.errors import (
Databases,
Expand All @@ -23,7 +25,12 @@
@router.get("/",
response_model=list[str],
summary="Gets the list of all database names")
async def database_names(exclude_defaults: bool = exclude_defaults_query) -> list[str]:
async def database_names(
exclude_defaults: Annotated[bool, Query(
title="Query parameter",
description="A boolean to exclude default databases from a MongoDB instance. Refer to "
"`trolldb.database.mongodb.MongoDB.default_database_names` for more information."
)] = True) -> list[str]:
"""Please consult the auto-generated documentation by FastAPI."""
db_names = await MongoDB.list_database_names()

Expand Down
21 changes: 8 additions & 13 deletions trolldb/api/routes/queries.py
Original file line number Diff line number Diff line change
Expand Up @@ -5,13 +5,14 @@
"""

import datetime
from typing import Annotated

from fastapi import APIRouter, Query

from trolldb.api.routes.common import CheckCollectionDependency
from trolldb.database.errors import database_collection_error_descriptor
from trolldb.database.mongodb import get_ids
from trolldb.database.piplines import PipelineAttribute, Pipelines
from trolldb.database.pipelines import PipelineAttribute, Pipelines

router = APIRouter()

Expand All @@ -22,14 +23,11 @@
summary="Gets the database UUIDs of the documents that match specifications determined by the query string")
async def queries(
collection: CheckCollectionDependency,
# We suppress ruff for the following four lines with `Query(default=None)`.
# Reason: This is the FastAPI way of defining optional queries and ruff is not happy about it!
platform: list[str] = Query(default=None), # noqa: B008
sensor: list[str] = Query(default=None), # noqa: B008
time_min: datetime.datetime = Query(default=None), # noqa: B008
time_max: datetime.datetime = Query(default=None)) -> list[str]: # noqa: B008
platform: Annotated[list[str] | None, Query()] = None,
sensor: Annotated[list[str] | None, Query()] = None,
time_min: Annotated[datetime.datetime, Query()] = None,
time_max: Annotated[datetime.datetime, Query()] = None) -> list[str]:
"""Please consult the auto-generated documentation by FastAPI."""
# We
pipelines = Pipelines()

if platform:
Expand All @@ -42,10 +40,7 @@ async def queries(
start_time = PipelineAttribute("start_time")
end_time = PipelineAttribute("end_time")
pipelines += (
(start_time >= time_min) |
(start_time <= time_max) |
(end_time >= time_min) |
(end_time <= time_max)
((start_time >= time_min) & (start_time <= time_max)) |
((end_time >= time_min) & (end_time <= time_max))
)

return await get_ids(collection.aggregate(pipelines))
Loading

0 comments on commit 1f0b8d0

Please sign in to comment.