Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add Filters #357

Merged
merged 90 commits into from
Dec 4, 2023
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
90 commits
Select commit Hold shift + click to select a range
ac57fd4
Enable sorting when batching is enabled
PaulSchweizer Jul 23, 2022
6417061
Deprecate UnsortedSQLAlchemyConnectionField and resetting Relationshi…
PaulSchweizer Jul 31, 2022
535afbe
Use field_name instead of column.key to build sort enum names to ensu…
PaulSchweizer Jul 31, 2022
b91fede
Adjust batching test to honor different selet in query structure in s…
PaulSchweizer Aug 6, 2022
5b8d068
add filter tests for discussion
sabard Jul 25, 2022
861613e
add typing for custom filter
sabard Aug 2, 2022
1f47f92
update 1:n and n:m filter tests to use RelationshipFilter syntax
sabard Aug 2, 2022
e0cd465
add models and mark tests to fail
sabard Aug 2, 2022
32254b6
make filter tests run
sabard Aug 2, 2022
172985c
Added draft methods & classes for filter registry
erikwrede Aug 11, 2022
6319278
Drafted abstract filters, generation of filter fields from methods.
erikwrede Aug 11, 2022
69680a9
Prototype: Filter Schema generation
erikwrede Aug 12, 2022
c31bbac
use re sub instead of removesuffix for python<3.9 support
sabard Aug 12, 2022
f532857
fix syntax for python<3.9 support
sabard Aug 12, 2022
7e2bf6e
field-level filtering working for equals and not equals
sabard Aug 12, 2022
b21e449
Made filter query generation modular & fix flake8
erikwrede Aug 13, 2022
4fbbe81
Fixed variable name in execute_filters
erikwrede Aug 13, 2022
675a264
Drafted :1 and :many relationship filter construction
erikwrede Aug 13, 2022
60bfd3b
Prototype: :1 relationship filtering is working
erikwrede Aug 13, 2022
a30d77d
error on simple filter test
sabard Aug 16, 2022
fcce1f7
get simple and 1:1 filter tests passing
sabard Oct 5, 2022
4862737
fix a couple errors to get basic, incomplete 1:n test working
sabard Oct 24, 2022
15cfbd7
revert scalar_subquery to as_scalar for sqlalchemy<1.4 tests
sabard Oct 24, 2022
122d182
initial implementation of and/or logic
sabard Nov 21, 2022
11f7d91
revert filter logic so 1:n test passes
sabard Nov 21, 2022
c187c11
replicated tox error in CI
sabard Nov 21, 2022
6263540
partial: support custom filter fields, custom types of filter inputs
erikwrede Dec 1, 2022
4c7efb8
chore: add missing workflow setting
erikwrede Dec 1, 2022
c16ccf6
chore: change workflows back to the original setting
erikwrede Dec 1, 2022
9b86332
fix: 3.7 type hints
erikwrede Dec 1, 2022
ae28fe6
fix: relationship contains filters for n:m working
erikwrede Dec 19, 2022
d334593
test: add nested filter test and reverse relationship tests
sabard Dec 19, 2022
a4621e2
fix: chain or/and in contains
sabard Dec 19, 2022
0f34c05
test: try filtering ids on each containsExactly subquery
sabard Jan 1, 2023
7f7e98a
test: working group_by/having clause for specific case
sabard Jan 1, 2023
98db652
fix: group_by/having working for all tests but containsExactly 2
sabard Jan 1, 2023
ca1b498
chore: remove some debugging statements
sabard Jan 2, 2023
8f012f4
test: cover additional filter types
sabard Jan 2, 2023
14c657c
fix: use notin_
sabard Jan 2, 2023
2e321e9
chore: prepare for sqlalchemy2.0 adjustments
erikwrede Dec 4, 2022
e4b1a7f
update envlist for tox,reduce number of python versions
erikwrede Dec 4, 2022
812b28b
fix: set sqlalchemy max version to 2.1
erikwrede Dec 4, 2022
7288117
fix: all unit tests running
erikwrede Jan 2, 2023
e562cc2
fix: corrected sql version check for batching
erikwrede Jan 2, 2023
e525a8d
fix: added pragma no cover to version checks
erikwrede Jan 2, 2023
b84aa9f
chore: test with all python versions
erikwrede Jan 2, 2023
b832fff
wip: test: add hybrid_prop tests
sabard Jan 2, 2023
5fc1be3
test: fix hybrid_prop converter test
sabard Jan 3, 2023
156fb68
test: fix hybrid_prop test by using string typevars
sabard Jan 3, 2023
5d81991
test: revert test models
sabard Jan 5, 2023
b5c9054
Merge branch 'master' into add-filters
sabard Jan 9, 2023
ec62082
fix: filters support interfaces with BaseType
sabard Jan 16, 2023
b88280d
test: fix basic tests for async
sabard Jan 16, 2023
8e45a3a
fix: remove print statements
sabard Jan 16, 2023
ca23083
Merge branch 'master' into add-filters
sabard Jan 16, 2023
2495448
chore: cleanup
sabard Jan 16, 2023
f37dd8b
fix: make converter tests work
sabard Jan 16, 2023
5942291
fix: revert ci
sabard Jan 16, 2023
acfd97d
Merge branch 'master' into add-filters
erikwrede Jan 27, 2023
ace0aba
add initial filter docs
sabard Feb 13, 2023
fa0eecc
fix: typo
sabard Feb 13, 2023
3db7411
fix: test nits
sabard Feb 15, 2023
5662015
add basic filter example app
sabard Feb 27, 2023
9e8f1a5
add like methods to StringFilter
sabard Feb 27, 2023
d2360a1
raise errors in filters tests
sabard Feb 28, 2023
589c7d7
remove contains_exactly logic
sabard Mar 27, 2023
68194cc
Merge branch 'master' into add-filters
sabard Apr 10, 2023
48b1c6c
fix lint
sabard Apr 10, 2023
cab376a
fix: sqla 1.4 async filter tests passing with distinct
sabard Apr 24, 2023
db9f794
cleanup: automatically register field filters
sabard Apr 24, 2023
21eba2e
fix: convert type vars in converter.py
erikwrede Apr 24, 2023
39834ed
chore: fix review comments
erikwrede May 14, 2023
a9915d6
chore: update dependencies and fix test
erikwrede May 14, 2023
4712e10
chore: update sqa-utils fix
erikwrede May 14, 2023
cf0ba76
Merge branch 'master' into sqa-2.0
erikwrede May 14, 2023
a0abf8a
fix: adjust test after sqlalchemy 2.0 update
erikwrede May 14, 2023
474f6fa
Merge branch 'sqa-2.0' into add-filters
erikwrede May 31, 2023
1052b52
fix: keep aliases during and + or filtering
erikwrede Jun 4, 2023
0fb1db3
chore: make flake8 happy
erikwrede Jun 4, 2023
87bbd6f
test: breaking tests on enums
sabard Jun 5, 2023
06c90cb
fix: create special enum filters. Code pending refactor.
erikwrede Jul 28, 2023
4e34a79
fix: use typing extensions
erikwrede Jul 28, 2023
c38ebb3
refactor: cleanup filter type generation
erikwrede Oct 6, 2023
064adc7
feat(filters): support filter aliasing (PR #378)
erikwrede Oct 6, 2023
1aef748
chore: remove print statements
erikwrede Oct 6, 2023
18a7c54
chore: move base filter creation
erikwrede Oct 6, 2023
4063ea6
Merge branch 'master' into add-filters
erikwrede Oct 6, 2023
a2b8a9b
chore: run pre commit on PR (hook update due to incompatible hooks wi…
erikwrede Oct 6, 2023
ad9c1aa
Merge branch 'master' into add-filters
erikwrede Nov 20, 2023
e698d7c
chore: fix newly added merge conflict due to association proxies
erikwrede Dec 4, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
initial implementation of and/or logic
  • Loading branch information
sabard committed Nov 21, 2022
commit 122d1828ca08828a0367bb0916435af8ca759d13
177 changes: 134 additions & 43 deletions graphene_sqlalchemy/filters.py
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@
import re
from typing import Any, Dict, List, Tuple, Type, TypeVar, Union, get_type_hints

from sqlalchemy import not_
from sqlalchemy import and_, not_, or_
from sqlalchemy.orm import Query, aliased

import graphene
Expand All @@ -13,17 +13,24 @@

class AbstractType:
"""Dummy class for generic filters"""

pass


class ObjectTypeFilter(graphene.InputObjectType):
@classmethod
def __init_subclass_with_meta__(cls, filter_fields=None, model=None, _meta=None, **options):
def __init_subclass_with_meta__(
cls, filter_fields=None, model=None, _meta=None, **options
):

# Init meta options class if it doesn't exist already
if not _meta:
_meta = InputObjectTypeOptions(cls)

# TODO do this dynamically based off the field name, but also value type
filter_fields["and"] = graphene.InputField(graphene.List(cls))
filter_fields["or"] = graphene.InputField(graphene.List(cls))

# Add all fields to the meta options. graphene.InputObjectType will take care of the rest
if _meta.fields:
_meta.fields.update(filter_fields)
Expand All @@ -35,12 +42,57 @@ def __init_subclass_with_meta__(cls, filter_fields=None, model=None, _meta=None,
super(ObjectTypeFilter, cls).__init_subclass_with_meta__(_meta=_meta, **options)

@classmethod
def and_logic(cls, query, field, val: list["ObjectTypeFilter"]):
# TODO
pass
def and_logic(
cls,
query,
filter_type: ObjectTypeFilter,
vals: graphene.List["ObjectTypeFilter"],
):
# # Get the model to join on the Filter Query
# joined_model = filter_type._meta.model
# # Always alias the model
# joined_model_alias = aliased(joined_model)

clauses = []
for val in vals:
# # Join the aliased model onto the query
# query = query.join(model_field.of_type(joined_model_alias))

query, _clauses = filter_type.execute_filters(
query, val
) # , model_alias=joined_model_alias)
clauses += _clauses

return query, [and_(*clauses)]

@classmethod
def or_logic(
cls,
query,
filter_type: ObjectTypeFilter,
vals: graphene.List["ObjectTypeFilter"],
):
# # Get the model to join on the Filter Query
# joined_model = filter_type._meta.model
# # Always alias the model
# joined_model_alias = aliased(joined_model)

clauses = []
for val in vals:
# # Join the aliased model onto the query
# query = query.join(model_field.of_type(joined_model_alias))

query, _clauses = filter_type.execute_filters(
query, val
) # , model_alias=joined_model_alias)
clauses += _clauses

return query, [or_(*clauses)]

@classmethod
def execute_filters(cls: Type[FieldFilter], query, filter_dict: Dict, model_alias=None) -> Tuple[Query, List[Any]]:
def execute_filters(
cls: Type[FieldFilter], query, filter_dict: Dict, model_alias=None
) -> Tuple[Query, List[Any]]:
model = cls._meta.model
if model_alias:
model = model_alias
Expand All @@ -58,34 +110,53 @@ def execute_filters(cls: Type[FieldFilter], query, filter_dict: Dict, model_alia
# raise Exception
# TODO we need to save the relationship props in the meta fields array
erikwrede marked this conversation as resolved.
Show resolved Hide resolved
# to conduct joins and alias the joins (in case there are duplicate joins: A->B A->C B->C)
model_field = getattr(model, field)
if issubclass(field_filter_type, ObjectTypeFilter):
# Get the model to join on the Filter Query
joined_model = field_filter_type._meta.model
# Always alias the model
joined_model_alias = aliased(joined_model)

# Join the aliased model onto the query
query = query.join(model_field.of_type(joined_model_alias))

# Pass the joined query down to the next object type filter for processing
query, _clauses = field_filter_type.execute_filters(query, filt_dict, model_alias=joined_model_alias)
clauses.extend(_clauses)
if issubclass(field_filter_type, RelationshipFilter):
# TODO see above; not yet working
relationship_prop = field_filter_type._meta.model
query, _clauses = field_filter_type.execute_filters(query, model_field, filt_dict, relationship_prop)
if field == "and":
query, _clauses = cls.and_logic(
query, field_filter_type.of_type, filt_dict
)
clauses.extend(_clauses)
elif issubclass(field_filter_type, FieldFilter):
query, _clauses = field_filter_type.execute_filters(query, model_field, filt_dict)
elif field == "or":
query, _clauses = cls.or_logic(
query, field_filter_type.of_type, filt_dict
)
clauses.extend(_clauses)
else:
model_field = getattr(model, field)
if issubclass(field_filter_type, ObjectTypeFilter):
# Get the model to join on the Filter Query
joined_model = field_filter_type._meta.model
# Always alias the model
joined_model_alias = aliased(joined_model)

# Join the aliased model onto the query
query = query.join(model_field.of_type(joined_model_alias))

# Pass the joined query down to the next object type filter for processing
query, _clauses = field_filter_type.execute_filters(
query, filt_dict, model_alias=joined_model_alias
)
clauses.extend(_clauses)
if issubclass(field_filter_type, RelationshipFilter):
# TODO see above; not yet working
relationship_prop = field_filter_type._meta.model
query, _clauses = field_filter_type.execute_filters(
query, model_field, filt_dict, relationship_prop
)
clauses.extend(_clauses)
elif issubclass(field_filter_type, FieldFilter):
query, _clauses = field_filter_type.execute_filters(
query, model_field, filt_dict
)
clauses.extend(_clauses)

return query, clauses


class RelationshipFilter(graphene.InputObjectType):
@classmethod
def __init_subclass_with_meta__(cls, object_type_filter=None, model=None, _meta=None, **options):
def __init_subclass_with_meta__(
cls, object_type_filter=None, model=None, _meta=None, **options
):
if not object_type_filter:
raise Exception("Relationship Filters must be specific to an object type")
# Init meta options class if it doesn't exist already
Expand All @@ -103,18 +174,26 @@ def __init_subclass_with_meta__(cls, object_type_filter=None, model=None, _meta=
# Check if attribute is a function
if callable(func_attr) and filter_function_regex.match(func):
# add function and attribute name to the list
filter_functions.append((re.sub("_filter$", "", func), get_type_hints(func_attr)))
filter_functions.append(
(re.sub("_filter$", "", func), get_type_hints(func_attr))
)

relationship_filters = {}

# Generate Graphene Fields from the filter functions based on type hints
for field_name, _annotations in filter_functions:
assert "val" in _annotations, "Each filter method must have a value field with valid type annotations"
assert (
"val" in _annotations
), "Each filter method must have a value field with valid type annotations"
# If type is generic, replace with actual type of filter class
if is_list(_annotations["val"]):
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

reminder: cross-check with python list type hints of converter.py

relationship_filters.update({field_name: graphene.InputField(graphene.List(object_type_filter))})
relationship_filters.update(
{field_name: graphene.InputField(graphene.List(object_type_filter))}
)
else:
relationship_filters.update({field_name: graphene.InputField(object_type_filter)})
relationship_filters.update(
{field_name: graphene.InputField(object_type_filter)}
)

# Add all fields to the meta options. graphene.InputObjectType will take care of the rest
if _meta.fields:
Expand All @@ -124,22 +203,24 @@ def __init_subclass_with_meta__(cls, object_type_filter=None, model=None, _meta=

_meta.model = model

super(RelationshipFilter, cls).__init_subclass_with_meta__(_meta=_meta, **options)
super(RelationshipFilter, cls).__init_subclass_with_meta__(
_meta=_meta, **options
)

@classmethod
def contains_filter(cls, query, field, val: List[AbstractType]):
clauses = []
for v in val:
query, clauses = v.execute_filters(query, dict(v))
clauses += clauses
query, _clauses = v.execute_filters(query, dict(v))
clauses += _clauses
return clauses

@classmethod
def contains_exactly_filter(cls, query, field, val: List[AbstractType]):
clauses = []
for v in val:
query, clauses = v.execute_filters(query, dict(v))
clauses += clauses
query, _clauses = v.execute_filters(query, dict(v))
clauses += _clauses
return clauses

@classmethod
Expand All @@ -154,7 +235,7 @@ def execute_filters(
return query.join(field), clauses


any_field_filter = TypeVar('any_field_filter', bound="FieldFilter")
any_field_filter = TypeVar("any_field_filter", bound="FieldFilter")


class FieldFilter(graphene.InputObjectType):
Expand All @@ -177,8 +258,8 @@ def __init_subclass_with_meta__(cls, type=None, _meta=None, **options):
# Check if attribute is a function
if callable(func_attr) and filter_function_regex.match(func):
# add function and attribute name to the list
filter_functions.append((
re.sub("_filter$", "", func), func_attr.__annotations__)
filter_functions.append(
(re.sub("_filter$", "", func), func_attr.__annotations__)
)

# Init meta options class if it doesn't exist already
Expand All @@ -189,7 +270,9 @@ def __init_subclass_with_meta__(cls, type=None, _meta=None, **options):
print(f"Generating Fields for {cls.__name__} with type {type} ")
# Generate Graphene Fields from the filter functions based on type hints
for field_name, _annotations in filter_functions:
assert "val" in _annotations, "Each filter method must have a value field with valid type annotations"
assert (
"val" in _annotations
), "Each filter method must have a value field with valid type annotations"
# If type is generic, replace with actual type of filter class
print(f"Field: {field_name} with annotation {_annotations['val']}")
if _annotations["val"] == "AbstractType":
Expand All @@ -199,7 +282,9 @@ def __init_subclass_with_meta__(cls, type=None, _meta=None, **options):
else:
# TODO this is a place holder, we need to convert the type of val to a valid graphene
# type that we can pass to the InputField. We could re-use converter.convert_hybrid_property_return_type
new_filter_fields.update({field_name: graphene.InputField(graphene.String)})
new_filter_fields.update(
{field_name: graphene.InputField(graphene.String)}
)

# Add all fields to the meta options. graphene.InputbjectType will take care of the rest
if _meta.fields:
Expand All @@ -212,15 +297,21 @@ def __init_subclass_with_meta__(cls, type=None, _meta=None, **options):

# Abstract methods can be marked using AbstractType. See comment on the init method
@classmethod
def eq_filter(cls, query, field, val: AbstractType) -> Union[Tuple[Query, Any], Any]:
def eq_filter(
cls, query, field, val: AbstractType
) -> Union[Tuple[Query, Any], Any]:
return field == val

@classmethod
def n_eq_filter(cls, query, field, val: AbstractType) -> Union[Tuple[Query, Any], Any]:
def n_eq_filter(
cls, query, field, val: AbstractType
) -> Union[Tuple[Query, Any], Any]:
return not_(field == val)

@classmethod
def execute_filters(cls: Type[FieldFilter], query, field, filter_dict: any_field_filter) -> Tuple[Query, List[Any]]:
def execute_filters(
cls: Type[FieldFilter], query, field, filter_dict: any_field_filter
) -> Tuple[Query, List[Any]]:
clauses = []
for filt, val in filter_dict.items():
clause = getattr(cls, filt + "_filter")(query, field, val)
Expand Down
Loading