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 23 commits
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
180 changes: 105 additions & 75 deletions graphene_sqlalchemy/batching.py
Original file line number Diff line number Diff line change
@@ -1,3 +1,7 @@
"""The dataloader uses "select in loading" strategy to load related entities."""
from asyncio import get_event_loop
from typing import Dict

import aiodataloader
import sqlalchemy
from sqlalchemy.orm import Session, strategies
Expand All @@ -6,82 +10,108 @@
from .utils import is_sqlalchemy_version_less_than


def get_batch_resolver(relationship_prop):
class RelationshipLoader(aiodataloader.DataLoader):
cache = False

def __init__(self, relationship_prop, selectin_loader):
super().__init__()
self.relationship_prop = relationship_prop
self.selectin_loader = selectin_loader

async def batch_load_fn(self, parents):
"""
Batch loads the relationships of all the parents as one SQL statement.

There is no way to do this out-of-the-box with SQLAlchemy but
we can piggyback on some internal APIs of the `selectin`
eager loading strategy. It's a bit hacky but it's preferable
than re-implementing and maintainnig a big chunk of the `selectin`
loader logic ourselves.

The approach here is to build a regular query that
selects the parent and `selectin` load the relationship.
But instead of having the query emits 2 `SELECT` statements
when callling `all()`, we skip the first `SELECT` statement
and jump right before the `selectin` loader is called.
To accomplish this, we have to construct objects that are
normally built in the first part of the query in order
to call directly `SelectInLoader._load_for_path`.

TODO Move this logic to a util in the SQLAlchemy repo as per
SQLAlchemy's main maitainer suggestion.
See https://git.io/JewQ7
"""
child_mapper = self.relationship_prop.mapper
parent_mapper = self.relationship_prop.parent
session = Session.object_session(parents[0])

# These issues are very unlikely to happen in practice...
for parent in parents:
# assert parent.__mapper__ is parent_mapper
# All instances must share the same session
assert session is Session.object_session(parent)
# The behavior of `selectin` is undefined if the parent is dirty
assert parent not in session.dirty

# Should the boolean be set to False? Does it matter for our purposes?
states = [(sqlalchemy.inspect(parent), True) for parent in parents]

# For our purposes, the query_context will only used to get the session
query_context = None
if is_sqlalchemy_version_less_than('1.4'):
query_context = QueryContext(session.query(parent_mapper.entity))
else:
parent_mapper_query = session.query(parent_mapper.entity)
query_context = parent_mapper_query._compile_context()

if is_sqlalchemy_version_less_than('1.4'):
self.selectin_loader._load_for_path(
query_context,
parent_mapper._path_registry,
states,
None,
child_mapper,
)
else:
self.selectin_loader._load_for_path(
query_context,
parent_mapper._path_registry,
states,
None,
child_mapper,
None,
)
return [
getattr(parent, self.relationship_prop.key) for parent in parents
]


# Cache this across `batch_load_fn` calls
# This is so SQL string generation is cached under-the-hood via `bakery`
# Caching the relationship loader for each relationship prop.
RELATIONSHIP_LOADERS_CACHE: Dict[
sqlalchemy.orm.relationships.RelationshipProperty, RelationshipLoader
] = {}

# Cache this across `batch_load_fn` calls
# This is so SQL string generation is cached under-the-hood via `bakery`
selectin_loader = strategies.SelectInLoader(relationship_prop, (('lazy', 'selectin'),))

class RelationshipLoader(aiodataloader.DataLoader):
cache = False

async def batch_load_fn(self, parents):
"""
Batch loads the relationships of all the parents as one SQL statement.

There is no way to do this out-of-the-box with SQLAlchemy but
we can piggyback on some internal APIs of the `selectin`
eager loading strategy. It's a bit hacky but it's preferable
than re-implementing and maintainnig a big chunk of the `selectin`
loader logic ourselves.

The approach here is to build a regular query that
selects the parent and `selectin` load the relationship.
But instead of having the query emits 2 `SELECT` statements
when callling `all()`, we skip the first `SELECT` statement
and jump right before the `selectin` loader is called.
To accomplish this, we have to construct objects that are
normally built in the first part of the query in order
to call directly `SelectInLoader._load_for_path`.

TODO Move this logic to a util in the SQLAlchemy repo as per
SQLAlchemy's main maitainer suggestion.
See https://git.io/JewQ7
"""
child_mapper = relationship_prop.mapper
parent_mapper = relationship_prop.parent
session = Session.object_session(parents[0])

# These issues are very unlikely to happen in practice...
for parent in parents:
# assert parent.__mapper__ is parent_mapper
# All instances must share the same session
assert session is Session.object_session(parent)
# The behavior of `selectin` is undefined if the parent is dirty
assert parent not in session.dirty

# Should the boolean be set to False? Does it matter for our purposes?
states = [(sqlalchemy.inspect(parent), True) for parent in parents]

# For our purposes, the query_context will only used to get the session
query_context = None
if is_sqlalchemy_version_less_than('1.4'):
query_context = QueryContext(session.query(parent_mapper.entity))
else:
parent_mapper_query = session.query(parent_mapper.entity)
query_context = parent_mapper_query._compile_context()

if is_sqlalchemy_version_less_than('1.4'):
selectin_loader._load_for_path(
query_context,
parent_mapper._path_registry,
states,
None,
child_mapper
)
else:
selectin_loader._load_for_path(
query_context,
parent_mapper._path_registry,
states,
None,
child_mapper,
None
)

return [getattr(parent, relationship_prop.key) for parent in parents]

loader = RelationshipLoader()

def get_batch_resolver(relationship_prop):
"""get the resolve function for the given relationship."""

def _get_loader(relationship_prop):
"""Retrieve the cached loader of the given relationship."""
loader = RELATIONSHIP_LOADERS_CACHE.get(relationship_prop, None)
if loader is None or loader.loop != get_event_loop():
selectin_loader = strategies.SelectInLoader(
relationship_prop, (('lazy', 'selectin'),)
)
loader = RelationshipLoader(
relationship_prop=relationship_prop,
selectin_loader=selectin_loader,
)
RELATIONSHIP_LOADERS_CACHE[relationship_prop] = loader
return loader

loader = _get_loader(relationship_prop)

async def resolve(root, info, **args):
return await loader.load(root)
Expand Down
4 changes: 2 additions & 2 deletions graphene_sqlalchemy/converter.py
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,7 @@
default_connection_field_factory)
from .registry import get_global_registry
from .resolvers import get_attr_resolver, get_custom_resolver
from .utils import (DummyImport, registry_sqlalchemy_model_from_str,
from .utils import (DummyImport, is_list, registry_sqlalchemy_model_from_str,
safe_isinstance, singledispatchbymatchfunction,
value_equals)

Expand Down Expand Up @@ -420,7 +420,7 @@ def convert_sqlalchemy_hybrid_property_union(arg):
get_global_registry())


@convert_sqlalchemy_hybrid_property_type.register(lambda x: getattr(x, '__origin__', None) in [list, typing.List])
@convert_sqlalchemy_hybrid_property_type.register(is_list)
def convert_sqlalchemy_hybrid_property_type_list_t(arg):
# type is either list[T] or List[T], generic argument at __args__[0]
internal_type = arg.__args__[0]
Expand Down
4 changes: 2 additions & 2 deletions graphene_sqlalchemy/enums.py
Original file line number Diff line number Diff line change
Expand Up @@ -144,9 +144,9 @@ def sort_enum_for_object_type(
column = orm_field.columns[0]
if only_indexed and not (column.primary_key or column.index):
continue
asc_name = get_name(column.key, True)
asc_name = get_name(field_name, True)
asc_value = EnumValue(asc_name, column.asc())
desc_name = get_name(column.key, False)
desc_name = get_name(field_name, False)
desc_value = EnumValue(desc_name, column.desc())
if column.primary_key:
default.append(asc_value)
Expand Down
Loading