Skip to content

Commit

Permalink
Fix Nested Filter-Sort and Model Querying Issues (#29)
Browse files Browse the repository at this point in the history
* refactor: make filter operators constant

* fix: resolve filtering issues

- ensure each filter is applied only once per query
- fix failures on nested filtering

* fix: allow nested sorting
  • Loading branch information
dhruv-ahuja committed Aug 26, 2024
1 parent a6c095e commit fcc07b5
Show file tree
Hide file tree
Showing 3 changed files with 61 additions and 37 deletions.
18 changes: 18 additions & 0 deletions src/config/constants/app.py
Original file line number Diff line number Diff line change
@@ -1,8 +1,10 @@
from typing import Literal
from uuid import uuid4
import datetime as dt
import operator

from beanie.odm.interfaces.find import FindType, DocumentProjectionType
from beanie.odm.operators.find.evaluation import RegEx as RegExOperator
from beanie.odm.queries.find import FindMany


Expand Down Expand Up @@ -31,3 +33,19 @@
FIND_MANY_QUERY = FindMany[FindType] | FindMany[DocumentProjectionType]

FILTER_OPERATION = Literal["=", "!=", ">", ">=", "<", "<=", "like"]
FILTER_OPERATION_MAP = {
"=": operator.eq,
"!=": operator.ne,
">": operator.gt,
"<": operator.lt,
">=": operator.ge,
"<=": operator.le,
"like": RegExOperator,
}
NESTED_FILTER_OPERATION_MAP = {
"=": "$eq",
">": "$gt",
">=": "$gte",
"<": "$lt",
"<=": "$lte",
}
26 changes: 10 additions & 16 deletions src/services/poe.py
Original file line number Diff line number Diff line change
Expand Up @@ -42,30 +42,24 @@ async def get_items(
) -> tuple[list[ItemBase], int]:
"""Gets items by given category group, and the total items' count in the database."""

query = Item.find()
chainer = QueryChainer(query, Item)

if filter_sort_input is None:
items_count = await query.find(fetch_links=True).count()
items = await chainer.paginate(pagination).query.find(fetch_links=True).project(ItemBase).to_list()
items_count = await Item.find().count()
items = await QueryChainer(Item.find(), Item).paginate(pagination).query.find().project(ItemBase).to_list()

return items, items_count

base_query_chain = chainer.filter(filter_sort_input.filter_).sort(filter_sort_input.sort)

# * clone the query for use with total record counts and pagination calculations
count_query = (
base_query_chain.filter(filter_sort_input.filter_)
items_query = (
QueryChainer(Item.find(), Item)
.filter(filter_sort_input.filter_)
.sort(filter_sort_input.sort)
.clone()
.query.find(fetch_links=True)
.count()
.paginate(pagination)
.query.project(ItemBase)
.to_list()
)

paginated_query = base_query_chain.paginate(pagination).query.find(fetch_links=True).project(ItemBase).to_list()
count_query = QueryChainer(Item.find(), Item).filter(filter_sort_input.filter_).query.count()

try:
items = await paginated_query
items = await items_query
items_count = await count_query
except Exception as exc:
logger.error(f"error getting items from database; filter_sort: {filter_sort_input}: {exc}")
Expand Down
54 changes: 33 additions & 21 deletions src/utils/services.py
Original file line number Diff line number Diff line change
@@ -1,15 +1,15 @@
import copy
import operator
from typing import Self, Type, cast

from beanie import Document
from beanie.odm.operators.find.evaluation import RegEx as RegExOperator
from bson import Decimal128
from loguru import logger
import orjson
import pymongo
from redis.asyncio import Redis, RedisError

from src.config.constants.app import FIND_MANY_QUERY
from src.config.constants.app import FILTER_OPERATION_MAP, FIND_MANY_QUERY, NESTED_FILTER_OPERATION_MAP
from src.schemas.requests import FilterInputType, FilterSchema, PaginationInput, SortInputType, SortSchema
from src.schemas.responses import E, T, BaseResponse

Expand Down Expand Up @@ -71,7 +71,8 @@ def sort_on_query(query: FIND_MANY_QUERY, model: Type[Document], sort: SortInput
field = entry.field
operation = pymongo.ASCENDING if entry.operation == "asc" else pymongo.DESCENDING

model_field = getattr(model, field)
is_nested = "." in field
model_field = field if is_nested else getattr(model, field)
expression = (model_field, operation)

sort_expressions.append(expression)
Expand All @@ -80,6 +81,23 @@ def sort_on_query(query: FIND_MANY_QUERY, model: Type[Document], sort: SortInput
return query


def _build_nested_query(entry: FilterSchema, query: FIND_MANY_QUERY) -> FIND_MANY_QUERY:
"""Builds queries for nested fields, using raw BSON query syntax to ensure nested fields are parsed properly."""

field = entry.field
operation = entry.operation
value = entry.value

if operation != "like":
operation_function = NESTED_FILTER_OPERATION_MAP[operation]
filter_query = {field: {operation_function: Decimal128(value)}}
else:
filter_query = {field: {"$regex": value, "$options": "i"}}

query = query.find(filter_query)
return query


def filter_on_query(query: FIND_MANY_QUERY, model: Type[Document], filter_: FilterInputType) -> FIND_MANY_QUERY:
"""Parses, gathers and chains filter operations on the input query. Skips the process if filter input is empty.\n
Maps the operation list to operator arguments that allow using the operator dynamically, to create expressions
Expand All @@ -89,31 +107,25 @@ def filter_on_query(query: FIND_MANY_QUERY, model: Type[Document], filter_: Filt
if not isinstance(filter_, list):
return query

operation_map = {
"=": operator.eq,
"!=": operator.ne,
">": operator.gt,
"<": operator.lt,
">=": operator.ge,
"<=": operator.le,
"like": RegExOperator,
}

for entry in filter_:
field = entry.field
operation = entry.operation
operation_function = operation_map[operation]
operation_function = FILTER_OPERATION_MAP[operation]
value = entry.value

model_field = getattr(model, field)

if operation != "like":
query = query.find(operation_function(model_field, value))
is_nested = "." in field
if is_nested:
query = _build_nested_query(entry, query)
else:
operation_function = RegExOperator
options = "i" # case-insensitive search
model_field = getattr(model, field)

if operation != "like":
query = query.find(operation_function(model_field, value))
else:
operation_function = RegExOperator
options = "i" # case-insensitive search

query = query.find(operation_function(model_field, value, options=options))
query = query.find(operation_function(model_field, value, options=options))

return query

Expand Down

0 comments on commit fcc07b5

Please sign in to comment.