from __future__ import annotations
import typing as t
from globus_sdk import utils
# workaround for absence of Self type
# for the workaround and some background, see:
# https://github.com/python/mypy/issues/11871
SearchQueryT = t.TypeVar("SearchQueryT", bound="SearchQueryBase")
# an internal class for declaring multiple related types with shared methods
[docs]class SearchQueryBase(utils.PayloadWrapper):
"""
The base class for all Search query helpers.
Search has multiple types of query documents. Not all of their supported attributes
are shared, and they therefore do not inherit from one another.
This class implements common methods to all of them.
Query objects have a chainable API, in which methods return the query object after
modification. This allows usage like
>>> query = ...
>>> query = query.set_limit(10).set_advanced(False)
"""
[docs] def set_query(self: SearchQueryT, query: str) -> SearchQueryT:
"""
Set the query string for the query document.
:param query: the new query string
:type query: str
"""
self["q"] = query
return self
[docs] def set_limit(self: SearchQueryT, limit: int) -> SearchQueryT:
"""
Set the limit for the query document.
:param limit: a limit on the number of results returned in a single page
:type limit: int
"""
self["limit"] = limit
return self
[docs] def set_advanced(self: SearchQueryT, advanced: bool) -> SearchQueryT:
"""
Enable or disable advanced query string processing.
:param advanced: whether to enable (``True``) or not (``False``)
:type advanced: bool
"""
self["advanced"] = advanced
return self
[docs] def add_filter(
self: SearchQueryT,
field_name: str,
values: list[str],
*,
# pylint: disable=redefined-builtin
type: str = "match_all",
additional_fields: dict[str, t.Any] | None = None,
) -> SearchQueryT:
"""
Add a filter subdocument to the query.
:param field_name: the field on which to filter
:type field_name: str
:param values: the values to use in the filter
:type values: list of str
:param type: the type of filter to apply, defaults to "match_all"
:type type: str
:param additional_fields: additional data to include in the filter document
:type additional_fields: dict, optional
"""
self["filters"] = self.get("filters", [])
new_filter = {
"field_name": field_name,
"values": values,
"type": type,
**(additional_fields or {}),
}
self["filters"].append(new_filter)
return self
[docs]class SearchQuery(SearchQueryBase):
"""
A specialized dict which has helpers for creating and modifying a Search
Query document.
:param q: The query string. Required unless filters are used.
:type q: str, optional
:param limit: A limit on the number of results returned in a single page
:type limit: int
:param offset: An offset into the set of all results for the query
:type offset: int
:param advanced: Whether to enable (``True``) or not to enable (``False``) advanced
parsing of query strings. The default of ``False`` is robust and guarantees that
the query will not error with "bad query string" errors
:type advanced: bool, optional
:param additional_fields: additional data to include in the query document
:type additional_fields: dict, optional
Example usage:
>>> from globus_sdk import SearchClient, SearchQuery
>>> sc = SearchClient(...)
>>> index_id = ...
>>> query = (SearchQuery(q='example query')
>>> .set_limit(100).set_offset(10)
>>> .add_filter('path.to.field1', ['foo', 'bar']))
>>> result = sc.post_search(index_id, query)
"""
def __init__(
self,
q: str | None = None,
*,
limit: int | None = None,
offset: int | None = None,
advanced: bool | None = None,
additional_fields: dict[str, t.Any] | None = None,
):
super().__init__()
if q is not None:
self["q"] = q
if limit is not None:
self["limit"] = limit
if offset is not None:
self["offset"] = offset
if advanced is not None:
self["advanced"] = advanced
if additional_fields is not None:
self.update(additional_fields)
[docs] def set_offset(self, offset: int) -> SearchQuery:
"""
Set the offset for the query document.
:param offset: an offset into the set of all results for the query
:type offset: int
"""
self["offset"] = offset
return self
[docs] def add_facet(
self,
name: str,
field_name: str,
*,
# pylint: disable=redefined-builtin
type: str = "terms",
size: int | None = None,
date_interval: str | None = None,
histogram_range: tuple[t.Any, t.Any] | None = None,
additional_fields: dict[str, t.Any] | None = None,
) -> SearchQuery:
"""
Add a facet subdocument to the query.
:param name: the name for the facet in the result
:type name: str
:param field_name: the field on which to build the facet
:type field_name: str
:param type: the type of facet to apply, defaults to "terms"
:type type: str
:param size: the size parameter for the facet
:type size: int, optional
:param date_interval: the date interval for a date histogram facet
:type date_interval: str, optional
:param histogram_range: a low and high bound for a numeric histogram facet
:type histogram_range: tuple, optional
:param additional_fields: additional data to include in the facet document
:type additional_fields: dict, optional
"""
self["facets"] = self.get("facets", [])
facet: dict[str, t.Any] = {
"name": name,
"field_name": field_name,
"type": type,
**(additional_fields or {}),
}
if size is not None:
facet["size"] = size
if date_interval is not None:
facet["date_interval"] = date_interval
if histogram_range is not None:
low, high = histogram_range
facet["histogram_range"] = {"low": low, "high": high}
self["facets"].append(facet)
return self
[docs] def add_boost(
self,
field_name: str,
factor: str | int | float,
*,
additional_fields: dict[str, t.Any] | None = None,
) -> SearchQuery:
"""
Add a boost subdocument to the query.
:param field_name: the field to boost in result weighting
:type field_name: str
:param factor: the factor by which to adjust the field weight (where ``1.0`` is
the default weight)
:type factor: str, int, or float
:param additional_fields: additional data to include in the boost document
:type additional_fields: dict, optional
"""
self["boosts"] = self.get("boosts", [])
boost = {
"field_name": field_name,
"factor": factor,
**(additional_fields or {}),
}
self["boosts"].append(boost)
return self
[docs] def add_sort(
self,
field_name: str,
*,
order: str | None = None,
additional_fields: dict[str, t.Any] | None = None,
) -> SearchQuery:
"""
Add a sort subdocument to the query.
:param field_name: the field on which to sort
:type field_name: str
:param order: ascending or descending order, given as ``"asc"`` or ``"desc"``
:type order: str, optional
:param additional_fields: additional data to include in the sort document
:type additional_fields: dict, optional
"""
self["sort"] = self.get("sort", [])
sort = {"field_name": field_name, **(additional_fields or {})}
if order is not None:
sort["order"] = order
self["sort"].append(sort)
return self