import pytest import ruamel.yaml as yaml from validate import check_query_f, check_query @pytest.mark.usefixtures('per_class_tests_db_state') class TestGraphqlIntrospection: def test_introspection(self, hge_ctx): with open(self.dir() + "/introspection.yaml") as c: conf = yaml.safe_load(c) resp, _ = check_query(hge_ctx, conf) hasArticle = False hasArticleAuthorFKRel = False hasArticleAuthorManualRel = False for t in resp['data']['__schema']['types']: if t['name'] == 'article': hasArticle = True for fld in t['fields']: if fld['name'] == 'author_obj_rel_manual': hasArticleAuthorManualRel = True assert fld['type']['kind'] == 'OBJECT' elif fld['name'] == 'author_obj_rel_fk': hasArticleAuthorFKRel = True assert fld['type']['kind'] == 'NON_NULL' assert hasArticle assert hasArticleAuthorFKRel assert hasArticleAuthorManualRel def test_introspection_user(self, hge_ctx): check_query_f(hge_ctx, self.dir() + "/introspection_user_role.yaml") @classmethod def dir(cls): return "queries/graphql_introspection" @pytest.mark.usefixtures('per_class_tests_db_state') class TestNullableObjectRelationshipInSchema: @classmethod def dir(cls): return "queries/graphql_introspection/nullable_object_relationship" def test_introspection(self, hge_ctx): with open(self.dir() + "/../introspection.yaml") as c: conf = yaml.safe_load(c) resp, _ = check_query(hge_ctx, conf) for t in resp['data']['__schema']['types']: if t['name'] == 'table2': for fld in t['fields']: if fld['name'] == 'via_table1': # graphql schema introspection doesn't explictly mark # the fields to be nullable (it does in the non-null # case). So checking this should be sufficient to detect # if its nullable. If this is not nullable, the # top-level kind would be `NON_NULL` and its `ofType` # would have the actual type assert fld['type']['kind'] == 'OBJECT' def getTypeNameFromType(typeObject): if typeObject['name'] != None: return typeObject['name'] elif isinstance(typeObject['ofType'],dict): return getTypeNameFromType(typeObject['ofType']) else: raise Exception("typeObject doesn't have name and ofType is not an object") @pytest.mark.usefixtures('per_class_tests_db_state') class TestGraphqlIntrospectionWithCustomTableName: # test to check some of the type names that are generated # while tracking a table with a custom name def test_introspection(self, hge_ctx): with open(self.dir() + "/introspection.yaml") as c: conf = yaml.safe_load(c) resp, _ = check_query(hge_ctx, conf) hasMultiSelect = False hasAggregate = False hasSelectByPk = False hasQueryRoot = False for t in resp['data']['__schema']['types']: if t['name'] == 'query_root': hasQueryRoot = True for field in t['fields']: if field['name'] == 'user_address': hasMultiSelect = True assert 'args' in field for args in field['args']: if args['name'] == 'distinct_on': assert "user_address_select_column" == getTypeNameFromType(args['type']) elif args['name'] == 'order_by': assert "user_address_order_by" == getTypeNameFromType(args['type']) elif args['name'] == 'where': assert 'user_address_bool_exp' == getTypeNameFromType(args['type']) elif field['name'] == 'user_address_aggregate': hasAggregate = True assert "user_address_aggregate" == getTypeNameFromType(field['type']) elif field['name'] == 'user_address_by_pk': assert "user_address" == getTypeNameFromType(field['type']) hasSelectByPk = True elif t['name'] == 'mutation_root': for field in t['fields']: if field['name'] == 'insert_user_address': hasMultiInsert = True assert "user_address_mutation_response" == getTypeNameFromType(field['type']) for args in field['args']: if args['name'] == 'object': assert "user_address_insert_input" == getTypeNameFromType(args['type']) elif field['name'] == 'update_user_address_by_pk': hasUpdateByPk = True assert "user_address" == getTypeNameFromType(field['type']) for args in field['args']: if args['name'] == 'object': assert "user_address" == getTypeNameFromType(args['type']) assert hasQueryRoot assert hasMultiSelect assert hasAggregate assert hasSelectByPk assert hasMultiInsert assert hasUpdateByPk @classmethod def dir(cls): return "queries/graphql_introspection/custom_table_name"