From c5e5966a57f9b5903efbd49d2d884e1574f6d13b Mon Sep 17 00:00:00 2001 From: Sabar Dasgupta Date: Mon, 4 Mar 2024 13:44:17 -0500 Subject: [PATCH 1/6] remove extraneous comments and enum code --- graphene_sqlalchemy/filters.py | 10 ++-------- graphene_sqlalchemy/registry.py | 13 ------------- 2 files changed, 2 insertions(+), 21 deletions(-) diff --git a/graphene_sqlalchemy/filters.py b/graphene_sqlalchemy/filters.py index bb42272..552c070 100644 --- a/graphene_sqlalchemy/filters.py +++ b/graphene_sqlalchemy/filters.py @@ -309,10 +309,7 @@ def execute_filters( class SQLEnumFilter(FieldFilter): - """Basic Filter for Scalars in Graphene. - We want this filter to use Dynamic fields so it provides the base - filtering methods ("eq, nEq") for different types of scalars. - The Dynamic fields will resolve to Meta.filtered_type""" + """Basic Filter for SQL Enums in Graphene.""" class Meta: graphene_type = graphene.Enum @@ -332,10 +329,7 @@ def n_eq_filter( class PyEnumFilter(FieldFilter): - """Basic Filter for Scalars in Graphene. - We want this filter to use Dynamic fields so it provides the base - filtering methods ("eq, nEq") for different types of scalars. - The Dynamic fields will resolve to Meta.filtered_type""" + """Basic Filter for Python Enums in Graphene.""" class Meta: graphene_type = graphene.Enum diff --git a/graphene_sqlalchemy/registry.py b/graphene_sqlalchemy/registry.py index b959d22..09e23a4 100644 --- a/graphene_sqlalchemy/registry.py +++ b/graphene_sqlalchemy/registry.py @@ -181,19 +181,6 @@ def get_filter_for_scalar_type( return filter_type - # TODO register enums automatically - def register_filter_for_enum_type( - self, enum_type: Type[graphene.Enum], filter_obj: Type["FieldFilter"] - ): - from .filters import FieldFilter - - if not issubclass(enum_type, graphene.Enum): - raise TypeError("Expected Enum, but got: {!r}".format(enum_type)) - - if not issubclass(filter_obj, FieldFilter): - raise TypeError("Expected FieldFilter, but got: {!r}".format(filter_obj)) - self._registry_scalar_filters[enum_type] = filter_obj - # Filter Base Types def register_filter_for_base_type( self, From 9b61945b3e31d022e54c10b51b8e270629246cfd Mon Sep 17 00:00:00 2001 From: Sabar Dasgupta Date: Mon, 4 Mar 2024 13:44:40 -0500 Subject: [PATCH 2/6] add enum to filter example --- examples/filters/database.py | 4 ++-- examples/filters/models.py | 5 ++++- 2 files changed, 6 insertions(+), 3 deletions(-) diff --git a/examples/filters/database.py b/examples/filters/database.py index 8f6522f..c842c41 100644 --- a/examples/filters/database.py +++ b/examples/filters/database.py @@ -26,8 +26,8 @@ def init_db(): person1 = Person(name="A") person2 = Person(name="B") - pet1 = Pet(name="Spot") - pet2 = Pet(name="Milo") + pet1 = Pet(name="Spot", kind="dog") + pet2 = Pet(name="Milo", kind="cat") toy1 = Toy(name="disc") toy2 = Toy(name="ball") diff --git a/examples/filters/models.py b/examples/filters/models.py index 1b22956..f5d4bf9 100644 --- a/examples/filters/models.py +++ b/examples/filters/models.py @@ -1,14 +1,17 @@ import sqlalchemy from database import Base -from sqlalchemy import Column, ForeignKey, Integer, String +from sqlalchemy import Column, Enum, ForeignKey, Integer, String from sqlalchemy.orm import relationship +PetKind = Enum("cat", "dog", name="pet_kind") + class Pet(Base): __tablename__ = "pets" id = Column(Integer(), primary_key=True) name = Column(String(30)) age = Column(Integer()) + kind = Column(PetKind) person_id = Column(Integer(), ForeignKey("people.id")) From 4080d2050b1f3b2331b608cc6d5e7869db9a13c1 Mon Sep 17 00:00:00 2001 From: Sabar Dasgupta Date: Mon, 4 Mar 2024 14:43:54 -0500 Subject: [PATCH 3/6] fix typo in docs --- docs/filters.rst | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/filters.rst b/docs/filters.rst index ac36803..47cb872 100644 --- a/docs/filters.rst +++ b/docs/filters.rst @@ -152,7 +152,7 @@ this query will return all pets which have a person named "Ben" in their ``peopl } -and this one will return all pets which hvae a person list that contains exactly the people "Ada" and "Ben" and no fewer or people with other names. +and this one will return all pets which have a person list that contains exactly the people "Ada" and "Ben" and no fewer or people with other names. .. code:: From 7d872351b3f2998b3f32ba4234e6f70a1a32aa8c Mon Sep 17 00:00:00 2001 From: Sabar Dasgupta Date: Mon, 4 Mar 2024 14:45:05 -0500 Subject: [PATCH 4/6] test RelationshipFilter exceptions --- graphene_sqlalchemy/filters.py | 13 ++++++---- graphene_sqlalchemy/tests/test_filters.py | 30 ++++++++++++++++++++++- graphene_sqlalchemy/types.py | 22 +++++++---------- 3 files changed, 46 insertions(+), 19 deletions(-) diff --git a/graphene_sqlalchemy/filters.py b/graphene_sqlalchemy/filters.py index 552c070..d7ed3d1 100644 --- a/graphene_sqlalchemy/filters.py +++ b/graphene_sqlalchemy/filters.py @@ -170,7 +170,7 @@ def execute_filters( field_filter_type = input_field.type else: field_filter_type = cls._meta.fields[field].type - # raise Exception + # TODO we need to save the relationship props in the meta fields array # to conduct joins and alias the joins (in case there are duplicate joins: A->B A->C B->C) if field == "and": @@ -428,7 +428,8 @@ def __init_subclass_with_meta__( cls, base_type_filter=None, model=None, _meta=None, **options ): if not base_type_filter: - raise Exception("Relationship Filters must be specific to an object type") + raise TypeError("Relationship Filters must be specific to an object type.") + # Init meta options class if it doesn't exist already if not _meta: _meta = InputObjectTypeOptions(cls) @@ -440,9 +441,11 @@ def __init_subclass_with_meta__( # 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" + if "val" not in _annotations: + raise TypeError( + "Each filter method must have a 'val' field with valid type annotations." + ) + # If type is generic, replace with actual type of filter class if is_list(_annotations["val"]): relationship_filters.update( diff --git a/graphene_sqlalchemy/tests/test_filters.py b/graphene_sqlalchemy/tests/test_filters.py index 4acf89a..bec6204 100644 --- a/graphene_sqlalchemy/tests/test_filters.py +++ b/graphene_sqlalchemy/tests/test_filters.py @@ -5,7 +5,7 @@ from graphene import Connection, relay from ..fields import SQLAlchemyConnectionField -from ..filters import FloatFilter +from ..filters import FloatFilter, RelationshipFilter from ..types import ORMField, SQLAlchemyObjectType from .models import ( Article, @@ -1199,3 +1199,31 @@ async def test_additional_filters(session): schema = graphene.Schema(query=Query) result = await schema.execute_async(query, context_value={"session": session}) assert_and_raise_result(result, expected) + + +# Test that exceptions are called correctly +@pytest.mark.asyncio +async def test_filter_relationship_no_base_type(session): + with pytest.raises( + TypeError, + match=r"(.*)Relationship Filters must be specific to an object type.(.*)", + ): + RelationshipFilter.create_type( + "InvalidRelationshipFilter", base_type_filter=None, model=Article + ) + + +@pytest.mark.asyncio +async def test_filter_(session): + with pytest.raises( + TypeError, + match=r"(.*)Each filter method must have a 'val' field with valid type annotations.(.*)", + ): + + class InvalidFilter(FloatFilter): + class Meta: + graphene_type = graphene.Float + + @classmethod + def invalid_filter(cls, query, field) -> bool: + return False diff --git a/graphene_sqlalchemy/types.py b/graphene_sqlalchemy/types.py index 7053988..533e348 100644 --- a/graphene_sqlalchemy/types.py +++ b/graphene_sqlalchemy/types.py @@ -135,19 +135,15 @@ def get_or_create_relationship_filter( relationship_filter = registry.get_relationship_filter_for_base_type(base_type) if not relationship_filter: - try: - base_type_filter = registry.get_filter_for_base_type(base_type) - relationship_filter = RelationshipFilter.create_type( - f"{base_type.__name__}RelationshipFilter", - base_type_filter=base_type_filter, - model=base_type._meta.model, - ) - registry.register_relationship_filter_for_base_type( - base_type, relationship_filter - ) - except Exception as e: - print("e") - raise e + base_type_filter = registry.get_filter_for_base_type(base_type) + relationship_filter = RelationshipFilter.create_type( + f"{base_type.__name__}RelationshipFilter", + base_type_filter=base_type_filter, + model=base_type._meta.model, + ) + registry.register_relationship_filter_for_base_type( + base_type, relationship_filter + ) return relationship_filter From bd2b3d0a0e344e32ee8b7451701c6c7c0a9fe9c0 Mon Sep 17 00:00:00 2001 From: Sabar Dasgupta Date: Mon, 4 Mar 2024 14:58:23 -0500 Subject: [PATCH 5/6] use TypeError over asserts and fix filter exception tests --- graphene_sqlalchemy/filters.py | 16 ++++++++++------ graphene_sqlalchemy/tests/test_filters.py | 2 +- 2 files changed, 11 insertions(+), 7 deletions(-) diff --git a/graphene_sqlalchemy/filters.py b/graphene_sqlalchemy/filters.py index d7ed3d1..fbca5dd 100644 --- a/graphene_sqlalchemy/filters.py +++ b/graphene_sqlalchemy/filters.py @@ -79,9 +79,11 @@ def __init_subclass_with_meta__( new_filter_fields = {} # Generate Graphene Fields from the filter functions based on type hints for field_name, _annotations in logic_functions: - assert ( - "val" in _annotations - ), "Each filter method must have a value field with valid type annotations" + if "val" not in _annotations: + raise TypeError( + "Each filter method must have a 'val' field with valid type annotations." + ) + # If type is generic, replace with actual type of filter class replace_type_vars = {BaseTypeFilterSelf: cls} @@ -252,9 +254,11 @@ def __init_subclass_with_meta__(cls, graphene_type=None, _meta=None, **options): new_filter_fields = {} # 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" + if "val" not in _annotations: + raise TypeError( + "Each filter method must have a 'val' field with valid type annotations." + ) + # If type is generic, replace with actual type of filter class replace_type_vars = {ScalarFilterInputType: _meta.graphene_type} field_type = convert_sqlalchemy_type( diff --git a/graphene_sqlalchemy/tests/test_filters.py b/graphene_sqlalchemy/tests/test_filters.py index bec6204..a1018f3 100644 --- a/graphene_sqlalchemy/tests/test_filters.py +++ b/graphene_sqlalchemy/tests/test_filters.py @@ -1214,7 +1214,7 @@ async def test_filter_relationship_no_base_type(session): @pytest.mark.asyncio -async def test_filter_(session): +async def test_filter_invalid_filter_method(session): with pytest.raises( TypeError, match=r"(.*)Each filter method must have a 'val' field with valid type annotations.(.*)", From 3414623bde9867046545fc98515ab6450a90c0c0 Mon Sep 17 00:00:00 2001 From: Sabar Dasgupta Date: Mon, 4 Mar 2024 15:48:22 -0500 Subject: [PATCH 6/6] remove extraneous exceptions --- graphene_sqlalchemy/filters.py | 11 ----------- graphene_sqlalchemy/tests/test_filters.py | 4 +++- 2 files changed, 3 insertions(+), 12 deletions(-) diff --git a/graphene_sqlalchemy/filters.py b/graphene_sqlalchemy/filters.py index fbca5dd..27060e9 100644 --- a/graphene_sqlalchemy/filters.py +++ b/graphene_sqlalchemy/filters.py @@ -79,13 +79,7 @@ def __init_subclass_with_meta__( new_filter_fields = {} # Generate Graphene Fields from the filter functions based on type hints for field_name, _annotations in logic_functions: - if "val" not in _annotations: - raise TypeError( - "Each filter method must have a 'val' field with valid type annotations." - ) - # If type is generic, replace with actual type of filter class - replace_type_vars = {BaseTypeFilterSelf: cls} field_type = convert_sqlalchemy_type( _annotations.get("val", str), replace_type_vars=replace_type_vars @@ -445,11 +439,6 @@ def __init_subclass_with_meta__( # Generate Graphene Fields from the filter functions based on type hints for field_name, _annotations in filter_functions: - if "val" not in _annotations: - raise TypeError( - "Each filter method must have a 'val' field with valid type annotations." - ) - # If type is generic, replace with actual type of filter class if is_list(_annotations["val"]): relationship_filters.update( diff --git a/graphene_sqlalchemy/tests/test_filters.py b/graphene_sqlalchemy/tests/test_filters.py index a1018f3..dd5ad8a 100644 --- a/graphene_sqlalchemy/tests/test_filters.py +++ b/graphene_sqlalchemy/tests/test_filters.py @@ -1215,12 +1215,14 @@ async def test_filter_relationship_no_base_type(session): @pytest.mark.asyncio async def test_filter_invalid_filter_method(session): + + # Field filter with pytest.raises( TypeError, match=r"(.*)Each filter method must have a 'val' field with valid type annotations.(.*)", ): - class InvalidFilter(FloatFilter): + class InvalidFieldFilter(FloatFilter): class Meta: graphene_type = graphene.Float