Source code for galaxy.webapps.galaxy.api.common

"""This module contains utility functions shared across the api package."""

from typing import (
    Any,
    Dict,
    List,
    Optional,
    Set,
)

from fastapi import (
    Path,
    Query,
    Request,
)
from typing_extensions import Annotated

from galaxy.schema import (
    FilterQueryParams,
    SerializationParams,
    ValueFilterQueryParams,
)
from galaxy.schema.fields import DecodedDatabaseIdField
from galaxy.schema.schema import UpdateDatasetPermissionsPayload
from galaxy.util import listify

HistoryIDPathParam = Annotated[
    DecodedDatabaseIdField,
    Path(..., title="History ID", description="The encoded database identifier of the History."),
]

HistoryDatasetIDPathParam = Annotated[
    DecodedDatabaseIdField, Path(..., title="History Dataset ID", description="The ID of the History Dataset.")
]


HistoryItemIDPathParam = Annotated[
    DecodedDatabaseIdField, Path(..., title="History Item ID", description="The ID of the item (`HDA`/`HDCA`)")
]

HistoryHDCAIDPathParam = Annotated[
    DecodedDatabaseIdField, Path(..., title="History Dataset Collection ID", description="The ID of the `HDCA`.")
]


DatasetCollectionElementIdPathParam = Annotated[
    DecodedDatabaseIdField,
    Path(..., title="Dataset Collection Element ID", description="The encoded ID of the dataset collection element."),
]


UserIdPathParam = Annotated[
    DecodedDatabaseIdField,
    Path(..., title="User ID", description="The ID of the user."),
]


GroupIDPathParam = Annotated[
    DecodedDatabaseIdField,
    Path(..., title="Group ID", description="The ID of the group."),
]


RoleIDPathParam = Annotated[
    DecodedDatabaseIdField,
    Path(..., title="Role ID", description="The ID of the role."),
]


LibraryIdPathParam = Annotated[
    DecodedDatabaseIdField,
    Path(..., title="Library ID", description="The ID of the Library."),
]

NotificationIdPathParam = Annotated[
    DecodedDatabaseIdField,
    Path(..., title="Notification ID", description="The ID of the Notification."),
]


PageIdPathParam = Annotated[
    DecodedDatabaseIdField,
    Path(..., title="Page ID", description="The ID of the Page."),
]

QuotaIdPathParam = Annotated[
    DecodedDatabaseIdField,
    Path(..., title="Quota ID", description="The ID of the Quota."),
]

SerializationViewQueryParam = Annotated[
    Optional[str],
    Query(
        title="View",
        description="View to be passed to the serializer",
    ),
]

SerializationKeysQueryParam: Optional[str] = Query(
    None,
    title="Keys",
    description="Comma-separated list of keys to be passed to the serializer",
)

FilterQueryQueryParam: Optional[List[str]] = Query(
    default=None,
    title="Filter Query",
    description="Generally a property name to filter by followed by an (often optional) hyphen and operator string.",
    examples=["create_time-gt"],
)

FilterValueQueryParam: Optional[List[str]] = Query(
    default=None,
    title="Filter Value",
    description="The value to filter by.",
    examples=["2015-01-29"],
)

OffsetQueryParam: Optional[int] = Query(
    default=0,
    ge=0,
    title="Offset",
    description="Starts at the beginning skip the first ( offset - 1 ) items and begin returning at the Nth item",
)

LimitQueryParam: Optional[int] = Query(
    default=None,
    ge=1,
    title="Limit",
    description="The maximum number of items to return.",
)

OrderQueryParam: Optional[str] = Query(
    default=None,
    title="Order",
    description=(
        "String containing one of the valid ordering attributes followed (optionally) "
        "by '-asc' or '-dsc' for ascending and descending order respectively. "
        "Orders can be stacked as a comma-separated list of values."
    ),
)


[docs]def parse_serialization_params( view: Optional[str] = None, keys: Optional[str] = None, default_view: Optional[str] = None, **_, # Additional params are ignored ) -> SerializationParams: key_list = None if keys: key_list = keys.split(",") return SerializationParams(view=view, keys=key_list, default_view=default_view)
[docs]def query_serialization_params( view: SerializationViewQueryParam = None, keys: Optional[str] = SerializationKeysQueryParam, ) -> SerializationParams: return parse_serialization_params(view=view, keys=keys)
[docs]def get_value_filter_query_params( q: Optional[List[str]] = FilterQueryQueryParam, qv: Optional[List[str]] = FilterValueQueryParam, ) -> ValueFilterQueryParams: """ This function is meant to be used as a Dependency. See https://fastapi.tiangolo.com/tutorial/dependencies/#first-steps """ return ValueFilterQueryParams( q=q, qv=qv, )
[docs]def get_filter_query_params( q: Optional[List[str]] = FilterQueryQueryParam, qv: Optional[List[str]] = FilterValueQueryParam, offset: Optional[int] = OffsetQueryParam, limit: Optional[int] = LimitQueryParam, order: Optional[str] = OrderQueryParam, ) -> FilterQueryParams: """ This function is meant to be used as a Dependency. See https://fastapi.tiangolo.com/tutorial/dependencies/#first-steps """ return FilterQueryParams( q=q, qv=qv, offset=offset, limit=limit, order=order, )
[docs]def get_update_permission_payload(payload: Dict[str, Any]) -> UpdateDatasetPermissionsPayload: """Converts the generic payload dictionary into a UpdateDatasetPermissionsPayload model with custom parsing. This is an attempt on supporting multiple aliases for the permissions params.""" # There are several allowed names for the same role list parameter, i.e.: `access`, `access_ids`, `access_ids[]` # The `access_ids[]` name is not pydantic friendly, so this will be modelled as an alias but we can only set one alias # TODO: Maybe we should choose only one way/naming and deprecate the others? payload["access_ids"] = payload.get("access_ids[]") or payload.get("access") payload["manage_ids"] = payload.get("manage_ids[]") or payload.get("manage") payload["modify_ids"] = payload.get("modify_ids[]") or payload.get("modify") update_payload = UpdateDatasetPermissionsPayload(**payload) return update_payload
[docs]def get_query_parameters_from_request_excluding(request: Request, exclude: Set[str]) -> dict: """Gets all the request query parameters excluding the given parameters names in `exclude` set. This is useful when an endpoint uses arbitrary or dynamic query parameters that cannot be anticipated or documented beforehand. The `exclude` set can be used to avoid including those parameters that are already handled by the endpoint. """ extra_params = request.query_params._dict for param_name in exclude: extra_params.pop(param_name, None) return extra_params
[docs]def query_parameter_as_list(query): """Used as FastAPI dependable for query parameters that need to behave as a list of values separated by comma or as multiple instances of the same parameter. .. important:: the ``query`` annotation provided must define the ``alias`` exactly as the name of the actual parameter name. Usage example:: ValueQueryParam = Query( default=None, alias="value", # Important! this is the parameter name that will be displayed in the API docs title="My Value", description="A single value, a comma-separated list of values or a list of values.", ) @router.get("/api/my_route") def index( self, values: Optional[List[str]] = Depends(query_parameter_as_list(ValueQueryParam)), ): ... This will render in the API docs as a single string query parameter but will make the following requests equivalent: - ``api/my_route?value=val1,val2,val3`` - ``api/my_route?value=val1&value=val2&value=val3`` """ def parse_elements( elements: Optional[List[str]] = query, ) -> Optional[List[Any]]: if query.default != Ellipsis and not elements: return query.default if elements and len(elements) == 1: return listify(elements[0]) return elements return parse_elements