From 74b6b5d66ab04d17d266b16beb72470b6b012414 Mon Sep 17 00:00:00 2001 From: Miles Sabin Date: Thu, 31 Aug 2023 14:46:39 +0100 Subject: [PATCH 1/2] Directives support, elaborator rework and query algebra simplification + Full support of custom query and schema directives. Directives are a major GraphQL feature (schema directives are an important part of typical GraphQL access control schemes) and it would be premature to declare a 1.0 release without support. + Simplify the query algebra. The query algebra has accumulated a number of odd cases. For the most part these are unproblematic, but Rename in particular has proven to be awkward. It manifests in user code as the need to use the PossiblyRenamedSelect extractor in effect handlers, and this is confusing and error prone. + Support threading of state through the query elaborator. This enables, amongst other things, a simpler implementation of CSS-style cascading field arguments (eg. to implement filter criteria which should be inherited by children). Directive support ----------------- Directives can be applied to both the GraphQL schema and to queries. Typically directives will be processed by one or more additional phases during query compilation. Included in the tests are two examples, one demonstrating query directives (that is, directives which are added to the query by the client), and one demonstrating schema directives (that is, directives which are added to the query by the application developer). + Added directives to all applicable operation, query and type elements. + Added full directive location, argument and repetition validation logic. + Schema parser/renderer now has full support for directives. + Renamed Directive to DirectiveDef. The type Directive is now an application of a directive. + Added DirectiveDefs for skip, include and deprecated rather than special casing in introspection and directive processing. + Properly handle directive locations in introspection. + Query minimizer now has full support for directives. + Added query directive test as a demo of custom query directives. + Added a schema directive based access control/authorization test as a demo of custom schema directives. Elaborator rework ----------------- To support the threading of state through the query elaboration process an elaboration monad Elab has been introduced. This, + Provides access to the schema, context, variables and fragments of a query. + Supports the transformation of the children of Select to provide semantics for field arguments. + Supports the addition of contextual data to the resulting query both to drive run time behaviour and to support propagation of context to the elaboration of children. + Supports the addition of queries for additional attributes to the resulting query. + Supports context queries against the query being elaborated. + Supports reporting of errors or warnings during elaboration. Query algebra changes --------------------- + Removed Skipped, UntypedNarrows and Wrap query algebra cases. + Added UntypedFragmentSpread and UntypedInlineFragment query algebra cases primarily to provide a position in the query algebra for directives with the FRAGMENT_SPREAD and INLINE_FRAGMENT location to live. + Removed Rename query algebra case and added alias field to Select, The args and directives fields of Select have been moved to a new UntypedSelect query algebra case. + Count nodes now correspond to values rather than fields and so no longer have a field name and must be nested within a Select. + Added operation name and directives to untyped operation types. + Renamed Query.mapFields to mapFieldsR and introduced a non-Result using mapFields. Type/schema changes ------------------- + The Field.isDeprecated and deprecationReason fields have been removed and redefined in terms of directives. + Schema.{queryType, mutationType, subscriptionType} now correctly handle the case where the declared type is nullable. + Added fieldInfo method to Type. + Removed redundant Type.withField and withUnderlyingField. + Renamed EnumValue to EnumValueDefinition. + Renamed EnumType.value to valueDefinition. EnumType.value new returns an EnumValue. + Added Value.VariableRef. + Collected all value elaboration and validation into object Value. Query parser/compiler changes ----------------------------- + QueryParser.parseText incompatible signature change. Now returns full set of operations and fragments. + QueryParser.parseDocument incompatible signature change. Now returns full set of operations and fragments. + QueryParser.parseXxx removed redundant fragments argument. + QueryParser directive processing generalized from skip/include to general directives. + QueryParser.parseValue logic moved to SchemaParser. + QueryCompiler.compileUntyped renamed to compileOperation. + Fragments are now comiled in the context of each operation where there are more than one. This means that fragments are now correctly evaluated in the scope of any operation specific variable bindings. + The parser now recognises arguments as optional in directive definitions. + The parser now supports directives on operations and variable definitions. Mapping/interpreter changes --------------------------- + Removed redundant QueryExecutor trait. + Renamed Mapping.compileAndRunOne and Mapping.compileAndRunAll to compileAndRun and compileAndRunSubscription respectively. + Removed Mapping.run methods because they don't properly accumulate GraphQL compilation warnings. + Added RootEffect.computeUnit for effectful queries/mutations which don't need to modify the query or any special handling for root cursor creation. + RootEffect.computeCursor and RootStream.computeCursor no longer take the query as a argument. + RootEffect.computeQuery and RootStream.computeQuery have been replaced by computeChild methods which transform only the child of the root query. Uses of computeQuery were the main places where PossiblyRenamedSelect had to be used by end users, because the name and alias of the root query had to be preserved. By only giving the end user the opportunity to compute the child query and having the library deal with the top level, this complication has been removed. + RootStream.computeCursor no longer takes the query as a argument. + QueryInterpreter.run no longer generates the full GraphQL response. This is now done in Mapping.compileAndRun/compileAndRunSubscription, allowing compilation warnings to be incorporated in the GraphQL response. + Various errors now correctly identifed as internal rather than GraphQL errors. Miscellaneous changes --------------------- + Added Result.pure and unit. + Fixes to the order of accumulation of GraphQL problems in Result. + Fixed ap of Parallel[Result]'s Applicative to accumulate problems correctly. + Unnested Context and Env from object Cursor because they are now used in non-cursor scenarios. + Added NullFieldCursor which transforms a cursor such that its children are null. + Added NullCursor which transforms a cursor such that it is null. + Moved QueryMinimizer to its own source file. + Updated tutorial and demo. --- .../scala/demo/starwars/StarWarsMapping.scala | 32 +- .../main/scala/demo/world/WorldMapping.scala | 131 +- .../main/paradox/tutorial/db-backed-model.md | 39 +- .../main/paradox/tutorial/in-memory-model.md | 75 +- .../circe/src/main/scala/circemapping.scala | 22 +- modules/circe/src/test/scala/CirceData.scala | 10 +- .../src/test/scala/CirceEffectData.scala | 8 +- modules/circe/src/test/scala/CirceSuite.scala | 2 +- modules/core/src/main/scala/ast.scala | 14 +- modules/core/src/main/scala/compiler.scala | 1234 ++++++++++------- modules/core/src/main/scala/cursor.scala | 280 ++-- .../core/src/main/scala/introspection.scala | 35 +- modules/core/src/main/scala/mapping.scala | 139 +- .../src/main/scala/mappingvalidator.scala | 2 +- modules/core/src/main/scala/minimizer.scala | 128 ++ modules/core/src/main/scala/operation.scala | 36 +- modules/core/src/main/scala/parser.scala | 30 +- modules/core/src/main/scala/problem.scala | 1 - modules/core/src/main/scala/query.scala | 251 ++-- .../src/main/scala/queryinterpreter.scala | 137 +- modules/core/src/main/scala/result.scala | 31 +- modules/core/src/main/scala/schema.scala | 770 +++++++--- .../core/src/main/scala/valuemapping.scala | 2 +- modules/core/src/test/scala/arb/AstArb.scala | 3 +- .../test/scala/compiler/AttributesSuite.scala | 4 +- .../test/scala/compiler/CascadeSuite.scala | 751 ++++++++++ .../test/scala/compiler/CompilerSuite.scala | 139 +- .../test/scala/compiler/DirectivesSuite.scala | 319 +++++ .../scala/compiler/EnvironmentSuite.scala | 26 +- .../test/scala/compiler/FragmentSuite.scala | 163 ++- .../scala/compiler/InputValuesSuite.scala | 31 +- .../test/scala/compiler/PredicatesSuite.scala | 26 +- .../compiler/PreserveArgsElaborator.scala | 42 + .../test/scala/compiler/QuerySizeSuite.scala | 18 +- .../test/scala/compiler/ScalarsSuite.scala | 89 +- .../scala/compiler/SkipIncludeSuite.scala | 72 +- .../test/scala/compiler/VariablesSuite.scala | 77 +- .../test/scala/composed/ComposedData.scala | 43 +- .../scala/composed/ComposedListSuite.scala | 39 +- .../directives/DirectiveValidationSuite.scala | 369 +++++ .../directives/QueryDirectivesSuite.scala | 198 +++ .../directives/SchemaDirectivesSuite.scala | 425 ++++++ .../test/scala/effects/ValueEffectData.scala | 2 +- .../introspection/IntrospectionSuite.scala | 16 +- .../src/test/scala/parser/ParserSuite.scala | 107 +- .../src/test/scala/schema/SchemaSuite.scala | 32 +- .../core/src/test/scala/sdl/SDLSuite.scala | 42 +- .../test/scala/starwars/StarWarsData.scala | 32 +- .../subscription/SubscriptionSuite.scala | 39 +- .../src/test/scala/DoobieSuites.scala | 8 +- .../src/main/scala-2/genericmapping2.scala | 2 +- .../src/main/scala-3/genericmapping3.scala | 2 +- .../src/main/scala/CursorBuilder.scala | 2 +- .../src/main/scala/genericmapping.scala | 2 +- .../src/test/scala/DerivationSuite.scala | 31 +- .../generic/src/test/scala/EffectsSuite.scala | 2 +- .../src/test/scala/RecursionSuite.scala | 10 +- .../generic/src/test/scala/ScalarsSuite.scala | 86 +- .../src/test/scala/SkunkDatabaseSuite.scala | 2 +- .../js-jvm/src/test/scala/SkunkSuites.scala | 8 +- .../subscription/SubscriptionMapping.scala | 19 +- .../subscription/SubscriptionSuite.scala | 4 +- .../shared/src/main/scala/SqlMapping.scala | 118 +- .../src/main/scala/SqlMappingValidator.scala | 2 +- .../src/test/scala/SqlArrayJoinSuite.scala | 3 +- .../src/test/scala/SqlCoalesceSuite.scala | 2 +- .../test/scala/SqlComposedWorldMapping.scala | 56 +- .../test/scala/SqlComposedWorldSuite.scala | 3 +- .../src/test/scala/SqlCompositeKeySuite.scala | 3 +- .../src/test/scala/SqlCursorJsonMapping.scala | 10 +- .../src/test/scala/SqlCursorJsonSuite.scala | 5 +- .../src/test/scala/SqlEmbedding2Mapping.scala | 33 +- .../src/test/scala/SqlEmbedding2Suite.scala | 3 +- .../src/test/scala/SqlEmbedding3Suite.scala | 3 +- .../src/test/scala/SqlEmbeddingSuite.scala | 3 +- .../scala/SqlFilterJoinAliasMapping.scala | 28 +- .../SqlFilterOrderOffsetLimit2Mapping.scala | 56 +- .../SqlFilterOrderOffsetLimit2Suite.scala | 3 +- .../SqlFilterOrderOffsetLimitMapping.scala | 40 +- .../SqlFilterOrderOffsetLimitSuite.scala | 3 +- .../src/test/scala/SqlGraphMapping.scala | 12 +- .../shared/src/test/scala/SqlGraphSuite.scala | 3 +- .../src/test/scala/SqlInterfacesMapping.scala | 10 +- .../src/test/scala/SqlInterfacesSuite.scala | 3 +- .../src/test/scala/SqlInterfacesSuite2.scala | 3 +- .../src/test/scala/SqlJsonbMapping.scala | 10 +- .../shared/src/test/scala/SqlJsonbSuite.scala | 3 +- .../src/test/scala/SqlLikeMapping.scala | 29 +- .../shared/src/test/scala/SqlLikeSuite.scala | 3 +- .../src/test/scala/SqlMixedMapping.scala | 10 +- .../shared/src/test/scala/SqlMixedSuite.scala | 3 +- .../src/test/scala/SqlMovieMapping.scala | 94 +- .../shared/src/test/scala/SqlMovieSuite.scala | 5 +- .../src/test/scala/SqlMutationMapping.scala | 61 +- .../src/test/scala/SqlMutationSuite.scala | 3 +- .../test/scala/SqlNestedEffectsMapping.scala | 32 +- .../test/scala/SqlNestedEffectsSuite.scala | 3 +- .../src/test/scala/SqlPaging1Mapping.scala | 93 +- .../src/test/scala/SqlPaging1Suite.scala | 3 +- .../src/test/scala/SqlPaging2Mapping.scala | 110 +- .../src/test/scala/SqlPaging2Suite.scala | 3 +- .../src/test/scala/SqlPaging3Mapping.scala | 168 +-- .../src/test/scala/SqlPaging3Suite.scala | 3 +- .../src/test/scala/SqlProjectionMapping.scala | 66 +- .../src/test/scala/SqlProjectionSuite.scala | 5 +- .../scala/SqlRecursiveInterfacesSuite.scala | 3 +- .../test/scala/SqlSiblingListsMapping.scala | 25 +- .../src/test/scala/SqlSiblingListsSuite.scala | 3 +- .../src/test/scala/SqlTreeMapping.scala | 12 +- .../shared/src/test/scala/SqlTreeSuite.scala | 3 +- .../shared/src/test/scala/SqlUnionSuite.scala | 3 +- .../test/scala/SqlWorldCompilerSuite.scala | 6 +- .../src/test/scala/SqlWorldMapping.scala | 127 +- .../shared/src/test/scala/SqlWorldSuite.scala | 5 +- profile/src/main/scala/Bench.scala | 115 +- 115 files changed, 5437 insertions(+), 2668 deletions(-) create mode 100644 modules/core/src/main/scala/minimizer.scala create mode 100644 modules/core/src/test/scala/compiler/CascadeSuite.scala create mode 100644 modules/core/src/test/scala/compiler/DirectivesSuite.scala create mode 100644 modules/core/src/test/scala/compiler/PreserveArgsElaborator.scala create mode 100644 modules/core/src/test/scala/directives/DirectiveValidationSuite.scala create mode 100644 modules/core/src/test/scala/directives/QueryDirectivesSuite.scala create mode 100644 modules/core/src/test/scala/directives/SchemaDirectivesSuite.scala diff --git a/demo/src/main/scala/demo/starwars/StarWarsMapping.scala b/demo/src/main/scala/demo/starwars/StarWarsMapping.scala index ec58a0bb..fb01d120 100644 --- a/demo/src/main/scala/demo/starwars/StarWarsMapping.scala +++ b/demo/src/main/scala/demo/starwars/StarWarsMapping.scala @@ -73,22 +73,22 @@ trait StarWarsMapping[F[_]] extends GenericMapping[F] { self: StarWarsData[F] => ) // #elaborator - override val selectElaborator = new SelectElaborator(Map( - QueryType -> { - // The hero selector takes an Episode argument and yields a single value. We use the - // Unique operator to pick out the target using the FieldEquals predicate. - case Select("hero", List(Binding("episode", TypedEnumValue(e))), child) => - Episode.values.find(_.toString == e.name).map { episode => - Select("hero", Nil, Unique(Filter(Eql(CharacterType / "id", Const(hero(episode).id)), child))).success - }.getOrElse(Result.failure(s"Unknown episode '${e.name}'")) - - // The character, human and droid selectors all take a single ID argument and yield a - // single value (if any) or null. We use the Unique operator to pick out the target - // using the FieldEquals predicate. - case Select(f@("character" | "human" | "droid"), List(Binding("id", IDValue(id))), child) => - Select(f, Nil, Unique(Filter(Eql(CharacterType / "id", Const(id)), child))).success - } - )) + override val selectElaborator = SelectElaborator { + // The hero selector takes an Episode argument and yields a single value. We transform + // the nested child to use the Filter and Unique operators to pick out the target using + // the Eql predicate. + case (QueryType, "hero", List(Binding("episode", EnumValue(e)))) => + for { + episode <- Elab.liftR(Episode.values.find(_.toString == e).toResult(s"Unknown episode '$e'")) + _ <- Elab.transformChild(child => Unique(Filter(Eql(CharacterType / "id", Const(hero(episode).id)), child))) + } yield () + + // The character, human and droid selectors all take a single ID argument and yield a + // single value (if any) or null. We transform the nested child to use the Unique and + // Filter operators to pick out the target using the Eql predicate. + case (QueryType, "character" | "human" | "droid", List(Binding("id", IDValue(id)))) => + Elab.transformChild(child => Unique(Filter(Eql(CharacterType / "id", Const(id)), child))) + } // #elaborator } diff --git a/demo/src/main/scala/demo/world/WorldMapping.scala b/demo/src/main/scala/demo/world/WorldMapping.scala index 6ea68eae..6e4058d7 100644 --- a/demo/src/main/scala/demo/world/WorldMapping.scala +++ b/demo/src/main/scala/demo/world/WorldMapping.scala @@ -171,70 +171,73 @@ trait WorldMapping[F[_]] extends DoobieMapping[F] { ) // #elaborator - override val selectElaborator = new SelectElaborator(Map( - - QueryType -> { - - case Select("country", List(Binding("code", StringValue(code))), child) => - Select("country", Nil, Unique(Filter(Eql(CountryType / "code", Const(code)), child))).success - - case Select("city", List(Binding("id", IntValue(id))), child) => - Select("city", Nil, Unique(Filter(Eql(CityType / "id", Const(id)), child))).success - - case Select("countries", List(Binding("limit", IntValue(num)), Binding("offset", IntValue(off)), Binding("minPopulation", IntValue(min)), Binding("byPopulation", BooleanValue(byPop))), child) => - def limit(query: Query): Query = - if (num < 1) query - else Limit(num, query) - - def offset(query: Query): Query = - if (off < 1) query - else Offset(off, query) - - def order(query: Query): Query = { - if (byPop) - OrderBy(OrderSelections(List(OrderSelection[Int](CountryType / "population"))), query) - else if (num > 0 || off > 0) - OrderBy(OrderSelections(List(OrderSelection[String](CountryType / "code"))), query) - else query - } - - def filter(query: Query): Query = - if (min == 0) query - else Filter(GtEql(CountryType / "population", Const(min)), query) - - Select("countries", Nil, limit(offset(order(filter(child))))).success - - case Select("cities", List(Binding("namePattern", StringValue(namePattern))), child) => - if (namePattern == "%") - Select("cities", Nil, child).success - else - Select("cities", Nil, Filter(Like(CityType / "name", namePattern, true), child)).success - - case Select("language", List(Binding("language", StringValue(language))), child) => - Select("language", Nil, Unique(Filter(Eql(LanguageType / "language", Const(language)), child))).success - - case Select("search", List(Binding("minPopulation", IntValue(min)), Binding("indepSince", IntValue(year))), child) => - Select("search", Nil, - Filter( - And( - Not(Lt(CountryType / "population", Const(min))), - Not(Lt(CountryType / "indepyear", Const(Option(year)))) - ), - child - ) - ).success - - case Select("search2", List(Binding("indep", BooleanValue(indep)), Binding("limit", IntValue(num))), child) => - Select("search2", Nil, Limit(num, Filter(IsNull[Int](CountryType / "indepyear", isNull = !indep), child))).success - }, - CountryType -> { - case Select("numCities", List(Binding("namePattern", AbsentValue)), Empty) => - Count("numCities", Select("cities", Nil, Select("name", Nil, Empty))).success - - case Select("numCities", List(Binding("namePattern", StringValue(namePattern))), Empty) => - Count("numCities", Select("cities", Nil, Filter(Like(CityType / "name", namePattern, true), Select("name", Nil, Empty)))).success - } - )) + override val selectElaborator = SelectElaborator { + case (QueryType, "country", List(Binding("code", StringValue(code)))) => + Elab.transformChild(child => Unique(Filter(Eql(CountryType / "code", Const(code)), child))) + + case (QueryType, "city", List(Binding("id", IntValue(id)))) => + Elab.transformChild(child => Unique(Filter(Eql(CityType / "id", Const(id)), child))) + + case ( + QueryType, "countries", + List( + Binding("limit", IntValue(num)), + Binding("offset", IntValue(off)), + Binding("minPopulation", IntValue(min)), + Binding("byPopulation", BooleanValue(byPop)) + ) + ) => + def limit(query: Query): Query = + if (num < 1) query + else Limit(num, query) + + def offset(query: Query): Query = + if (off < 1) query + else Offset(off, query) + + def order(query: Query): Query = { + if (byPop) + OrderBy(OrderSelections(List(OrderSelection[Int](CountryType / "population"))), query) + else if (num > 0 || off > 0) + OrderBy(OrderSelections(List(OrderSelection[String](CountryType / "code"))), query) + else query + } + + def filter(query: Query): Query = + if (min == 0) query + else Filter(GtEql(CountryType / "population", Const(min)), query) + + Elab.transformChild(child => limit(offset(order(filter(child))))) + + case (QueryType, "cities", List(Binding("namePattern", StringValue(namePattern)))) => + if (namePattern == "%") + Elab.unit + else + Elab.transformChild(child => Filter(Like(CityType / "name", namePattern, true), child)) + + case (QueryType, "language", List(Binding("language", StringValue(language)))) => + Elab.transformChild(child => Unique(Filter(Eql(LanguageType / "language", Const(language)), child))) + + case (QueryType, "search", List(Binding("minPopulation", IntValue(min)), Binding("indepSince", IntValue(year)))) => + Elab.transformChild(child => + Filter( + And( + Not(Lt(CountryType / "population", Const(min))), + Not(Lt(CountryType / "indepyear", Const(Option(year)))) + ), + child + ) + ) + + case (QueryType, "search2", List(Binding("indep", BooleanValue(indep)), Binding("limit", IntValue(num)))) => + Elab.transformChild(child => Limit(num, Filter(IsNull[Int](CountryType / "indepyear", isNull = !indep), child))) + + case (CountryType, "numCities", List(Binding("namePattern", AbsentValue))) => + Elab.transformChild(_ => Count(Select("cities", Select("name")))) + + case (CountryType, "numCities", List(Binding("namePattern", StringValue(namePattern)))) => + Elab.transformChild(_ => Count(Select("cities", Filter(Like(CityType / "name", namePattern, true), Select("name"))))) + } // #elaborator } diff --git a/docs/src/main/paradox/tutorial/db-backed-model.md b/docs/src/main/paradox/tutorial/db-backed-model.md index 9d25b389..d2fae591 100644 --- a/docs/src/main/paradox/tutorial/db-backed-model.md +++ b/docs/src/main/paradox/tutorial/db-backed-model.md @@ -1,8 +1,7 @@ # DB Backed Model -In this tutorial we are going to implement GraphQL API of countries and cities of the world using Grackle backed by -a database model, i.e. provide mapping for Grackle to read data from PostgreSQL and return it as result of -GraphQL queries. +In this tutorial we are going to implement a GraphQL API for countries and cities of the world using Grackle backed by +a database, ie. provide a mapping for Grackle to read data from PostgreSQL and return it as result of GraphQL queries. ## Running the demo @@ -75,43 +74,45 @@ Grackle represents schemas as a Scala value of type `Schema` which can be constr ## Database mapping -The API is backed by mapping to database tables. Grackle contains ready to use integration -with [doobie](https://tpolecat.github.io/doobie/) for accessing SQL database via JDBC -and with [Skunk](https://tpolecat.github.io/skunk/) for accessing PostgreSQL via its native API. In this example -we will use doobie. +The API is backed by mapping to database tables. Grackle contains ready to use integration with +[doobie](https://tpolecat.github.io/doobie/) for accessing SQL database via JDBC and with +[Skunk](https://tpolecat.github.io/skunk/) for accessing PostgreSQL via its native API. In this example we will use +doobie. Let's start with defining what tables and columns are available in the database model, @@snip [WorldMapping.scala](/demo/src/main/scala/demo/world/WorldMapping.scala) { #db_tables } -For each column we need to provide its name and doobie codec. We should also mark if value is nullable. +For each column we need to provide its name and doobie codec of type `Meta`. We should also mark if the value is +nullable. -We define each query as SQL query, as below, +We define the top-level GraphQL fields as `SqlObject` mappings, @@snip [WorldMapping.scala](/demo/src/main/scala/demo/world/WorldMapping.scala) { #root } -Now, we need to map each type from GraphQL schema using available columns from database, +Now, we need to map each type from the GraphQL schema using columns from the database, @@snip [WorldMapping.scala](/demo/src/main/scala/demo/world/WorldMapping.scala) { #type_mappings } -Each GraphQL must contain key. It can contain fields from one table, but it can also contain nested types which -are translated to SQL joins using provided conditions. `Join(country.code, city.countrycode)` means joining country -and city tables where `code` in the country table is the same as `countrycode` in the city table. +Each GraphQL type mapping must contain a key. It can contain fields from one table, but it can also contain nested +types which are translated to SQL joins using the provided conditions. `Join(country.code, city.countrycode)` means +joining country and city tables where `code` in the country table is the same as `countrycode` in the city table. ## The query compiler and elaborator -Similar as in [in-memory model]((in-memory-model.html#the-query-compiler-and-elaborator)), we need to define elaborator -to transform query algebra terms into the form that can be then used as source of SQL queries, +Similarly to the [in-memory model]((in-memory-model.html#the-query-compiler-and-elaborator)), we need to define an +elaborator to transform query algebra terms into a form that can be then used to translate query algebra terms to SQL +queries, @@snip [WorldMapping.scala](/demo/src/main/scala/demo/world/WorldMapping.scala) { #elaborator } ## Putting it all together -To expose GraphQL API with http4s we will use `GraphQLService` and `DemoServer` -from [in-memory example](in-memory-model.html#the-service). +To expose GraphQL API with http4s we will use the `GraphQLService` and `DemoServer` +from the [in-memory example](in-memory-model.html#the-service). -We will use [testcontainers](https://github.com/testcontainers/testcontainers-scala) to run PostgreSQL database -in the background. The final main method, which starts PostgreSQL database in docker container, creates database +We use [testcontainers](https://github.com/testcontainers/testcontainers-scala) to run a PostgreSQL database +in the background. The final main method, which starts a dockerized PostgreSQL database, creates the database schema, writes initial data and exposes GraphQL API for in-memory and db-backend models is below, @@snip [main.scala](/demo/src/main/scala/demo/Main.scala) { #main } diff --git a/docs/src/main/paradox/tutorial/in-memory-model.md b/docs/src/main/paradox/tutorial/in-memory-model.md index 897c98bd..37ab8e44 100644 --- a/docs/src/main/paradox/tutorial/in-memory-model.md +++ b/docs/src/main/paradox/tutorial/in-memory-model.md @@ -205,28 +205,24 @@ simplifies or expands the term to bring it into a form which can be executed dir Grackle's query algebra consists of the following elements, ```scala -sealed trait Query { - case class Select(name: String, args: List[Binding], child: Query = Empty) extends Query - case class Group(queries: List[Query]) extends Query - case class Unique(child: Query) extends Query - case class Filter(pred: Predicate, child: Query) extends Query - case class Component[F[_]](mapping: Mapping[F], join: (Query, Cursor) => Result[Query], child: Query) extends Query - case class Effect[F[_]](handler: EffectHandler[F], child: Query) extends Query - case class Introspect(schema: Schema, child: Query) extends Query - case class Environment(env: Env, child: Query) extends Query - case class Wrap(name: String, child: Query) extends Query - case class Rename(name: String, child: Query) extends Query - case class UntypedNarrow(tpnme: String, child: Query) extends Query - case class Narrow(subtpe: TypeRef, child: Query) extends Query - case class Skip(sense: Boolean, cond: Value, child: Query) extends Query - case class Limit(num: Int, child: Query) extends Query - case class Offset(num: Int, child: Query) extends Query - case class OrderBy(selections: OrderSelections, child: Query) extends Query - case class Count(name: String, child: Query) extends Query - case class TransformCursor(f: Cursor => Result[Cursor], child: Query) extends Query - case object Skipped extends Query - case object Empty extends Query -} +case class UntypedSelect(name: String, alias: Option[String], args: List[Binding], dirs: List[Directive], child: Query) +case class Select(name: String, alias: Option[String], child: Query) +case class Group(queries: List[Query]) +case class Unique(child: Query) +case class Filter(pred: Predicate, child: Query) +case class Component[F[_]](mapping: Mapping[F], join: (Query, Cursor) => Result[Query], child: Query) +case class Effect[F[_]](handler: EffectHandler[F], child: Query) +case class Introspect(schema: Schema, child: Query) +case class Environment(env: Env, child: Query) +case class UntypedFragmentSpread(name: String, directives: List[Directive]) +case class UntypedInlineFragment(tpnme: Option[String], directives: List[Directive], child: Query) +case class Narrow(subtpe: TypeRef, child: Query) +case class Limit(num: Int, child: Query) +case class Offset(num: Int, child: Query) +case class OrderBy(selections: OrderSelections, child: Query) +case class Count(child: Query) +case class TransformCursor(f: Cursor => Result[Cursor], child: Query) +case object Empty ``` A simple query like this, @@ -242,8 +238,8 @@ query { is first translated into a term in the query algebra of the form, ```scala -Select("character", List(IntBinding("id", 1000)), - Select("name", Nil) +UntypedSelect("character", None, List(IntBinding("id", 1000)), Nil, + UntypedSelect("name", None, Nil, Nil, Empty) ) ``` @@ -260,23 +256,23 @@ the model) is specific to this model, so we have to provide that semantic via so Extracting out the case for the `character` selector, ```scala -case Select(f@("character"), List(Binding("id", IDValue(id))), child) => - Select(f, Nil, Unique(Filter(Eql(CharacterType / "id", Const(id)), child))).success - +case (QueryType, "character", List(Binding("id", IDValue(id)))) => + Elab.transformChild(child => Unique(Filter(Eql(CharacterType / "id", Const(id)), child))) ``` -we can see that this transforms the previous term as follows, +the previous term is transformed as follows as follows, ```scala -Select("character", Nil, - Unique(Eql(CharacterType / "id"), Const("1000")), Select("name", Nil)) +Select("character", None, + Unique(Eql(CharacterType / "id"), Const("1000")), Select("name", None, Empty)) ) ``` -Here the argument to the `character` selector has been translated into a predicate which refines the root data of the -model to the single element which satisfies it via `Unique`. The remainder of the query (`Select("name", Nil)`) is -then within the scope of that constraint. We have eliminated something with model-specific semantics (`character(id: -1000)`) in favour of something universal which can be interpreted directly against the model. +Here the original `UntypedSelect` terms have been converted to typed `Select` terms with the argument to the +`character` selector translated into a predicate which refines the root data of the model to the single element which +satisfies it via `Unique`. The remainder of the query (`Select("name", None, Nil)`) is then within the scope of that +constraint. We have eliminated something with model-specific semantics (`character(id: 1000)`) in favour of something +universal which can be interpreted directly against the model. ## The query interpreter and cursor @@ -293,7 +289,7 @@ For the Star Wars model the root definitions are of the following form, @@snip [StarWarsData.scala](/demo/src/main/scala/demo/starwars/StarWarsMapping.scala) { #root } -The first argument of the `GenericRoot` constructor correspond to the top-level selection of the query (see the +The first argument of the `GenericField` constructor corresponds to the top-level selection of the query (see the schema above) and the second argument is the initial model value for which a `Cursor` will be derived. When the query is executed, navigation will start with that `Cursor` and the corresponding GraphQL type. @@ -337,6 +333,15 @@ object Main extends IOApp { DemoServer.stream[IO](starWarsGraphQLRoutes).compile.drain } } +object Main extends IOApp { + def run(args: List[String]): IO[ExitCode] = { + val starWarsGraphQLRoutes = GraphQLService.routes[IO]( + "starwars", + GraphQLService.fromMapping(new StarWarsMapping[IO] with StarWarsData[IO]) + ) + DemoServer.stream[IO](starWarsGraphQLRoutes).compile.drain + } +} ``` @@snip [main.scala](/demo/src/main/scala/demo/DemoServer.scala) { #server } diff --git a/modules/circe/src/main/scala/circemapping.scala b/modules/circe/src/main/scala/circemapping.scala index 0012b367..8214829e 100644 --- a/modules/circe/src/main/scala/circemapping.scala +++ b/modules/circe/src/main/scala/circemapping.scala @@ -8,13 +8,13 @@ import scala.collection.Factory import cats.MonadThrow import cats.implicits._ -import fs2.Stream +import fs2.Stream import io.circe.Json import io.circe.Encoder import org.tpolecat.sourcepos.SourcePos import syntax._ -import Cursor.{Context, DeferredCursor, Env} +import Cursor.DeferredCursor import ScalarType._ abstract class CirceMapping[F[_]](implicit val M: MonadThrow[F]) extends Mapping[F] with CirceMappingLike[F] @@ -23,19 +23,19 @@ trait CirceMappingLike[F[_]] extends Mapping[F] { // Syntax to allow Circe-specific root effects implicit class CirceMappingRootEffectSyntax(self: RootEffect.type) { - def computeJson(fieldName: String)(effect: (Query, Path, Env) => F[Result[Json]])(implicit pos: SourcePos): RootEffect = - self.computeCursor(fieldName)((q, p, e) => effect(q, p, e).map(_.map(circeCursor(p, e, _)))) - - def computeEncodable[A](fieldName: String)(effect: (Query, Path, Env) => F[Result[A]])(implicit pos: SourcePos, enc: Encoder[A]): RootEffect = - computeJson(fieldName)((q, p, e) => effect(q, p, e).map(_.map(enc(_)))) + def computeJson(fieldName: String)(effect: (Path, Env) => F[Result[Json]])(implicit pos: SourcePos): RootEffect = + self.computeCursor(fieldName)((p, e) => effect(p, e).map(_.map(circeCursor(p, e, _)))) + + def computeEncodable[A](fieldName: String)(effect: (Path, Env) => F[Result[A]])(implicit pos: SourcePos, enc: Encoder[A]): RootEffect = + computeJson(fieldName)((p, e) => effect(p, e).map(_.map(enc(_)))) } implicit class CirceMappingRootStreamSyntax(self: RootStream.type) { - def computeJson(fieldName: String)(effect: (Query, Path, Env) => Stream[F, Result[Json]])(implicit pos: SourcePos): RootStream = - self.computeCursor(fieldName)((q, p, e) => effect(q, p, e).map(_.map(circeCursor(p, e, _)))) + def computeJson(fieldName: String)(effect: (Path, Env) => Stream[F, Result[Json]])(implicit pos: SourcePos): RootStream = + self.computeCursor(fieldName)((p, e) => effect(p, e).map(_.map(circeCursor(p, e, _)))) - def computeEncodable[A](fieldName: String)(effect: (Query, Path, Env) => Stream[F, Result[A]])(implicit pos: SourcePos, enc: Encoder[A]): RootStream = - computeJson(fieldName)((q, p, e) => effect(q, p, e).map(_.map(enc(_)))) + def computeEncodable[A](fieldName: String)(effect: (Path, Env) => Stream[F, Result[A]])(implicit pos: SourcePos, enc: Encoder[A]): RootStream = + computeJson(fieldName)((p, e) => effect(p, e).map(_.map(enc(_)))) } def circeCursor(path: Path, env: Env, value: Json): Cursor = diff --git a/modules/circe/src/test/scala/CirceData.scala b/modules/circe/src/test/scala/CirceData.scala index 74047c45..fd2c3d49 100644 --- a/modules/circe/src/test/scala/CirceData.scala +++ b/modules/circe/src/test/scala/CirceData.scala @@ -113,10 +113,8 @@ object TestCirceMapping extends CirceMapping[IO] { } yield i+1).value } - override val selectElaborator = new SelectElaborator(Map( - RootType -> { - case Select("numChildren", Nil, Empty) => - Count("numChildren", Select("children", Nil, Empty)).success - } - )) + override val selectElaborator = SelectElaborator { + case (RootType, "numChildren", Nil) => + Elab.transformChild(_ => Count(Select("children"))) + } } diff --git a/modules/circe/src/test/scala/CirceEffectData.scala b/modules/circe/src/test/scala/CirceEffectData.scala index 15c1f145..84576970 100644 --- a/modules/circe/src/test/scala/CirceEffectData.scala +++ b/modules/circe/src/test/scala/CirceEffectData.scala @@ -44,7 +44,7 @@ class TestCirceEffectMapping[F[_]: Sync](ref: SignallingRef[F, Int]) extends Cir List( // Compute a CirceCursor - RootEffect.computeCursor("foo")((_, p, e) => + RootEffect.computeCursor("foo")((p, e) => ref.update(_+1).as( Result(circeCursor(p, e, Json.obj( @@ -56,7 +56,7 @@ class TestCirceEffectMapping[F[_]: Sync](ref: SignallingRef[F, Int]) extends Cir ), // Compute a Json, let the implementation handle the cursor - RootEffect.computeJson("bar")((_, _, _) => + RootEffect.computeJson("bar")((_, _) => ref.update(_+1).as( Result(Json.obj( "n" -> Json.fromInt(42), @@ -66,14 +66,14 @@ class TestCirceEffectMapping[F[_]: Sync](ref: SignallingRef[F, Int]) extends Cir ), // Compute an encodable value, let the implementation handle json and the cursor - RootEffect.computeEncodable("baz")((_, _, _) => + RootEffect.computeEncodable("baz")((_, _) => ref.update(_+1).as( Result(Struct(44, "hee")) ) ), // Compute a CirceCursor focussed on the root - RootEffect.computeCursor("qux")((_, p, e) => + RootEffect.computeCursor("qux")((p, e) => ref.update(_+1).as( Result(circeCursor(Path(p.rootTpe), e, Json.obj( diff --git a/modules/circe/src/test/scala/CirceSuite.scala b/modules/circe/src/test/scala/CirceSuite.scala index ea503eeb..21858342 100644 --- a/modules/circe/src/test/scala/CirceSuite.scala +++ b/modules/circe/src/test/scala/CirceSuite.scala @@ -294,7 +294,7 @@ final class CirceSuite extends CatsEffectSuite { { "errors" : [ { - "message" : "Unknown field 'hidden' in select" + "message" : "No field 'hidden' for type Root" } ] } diff --git a/modules/core/src/main/scala/ast.scala b/modules/core/src/main/scala/ast.scala index 0757c555..52d4e2f8 100644 --- a/modules/core/src/main/scala/ast.scala +++ b/modules/core/src/main/scala/ast.scala @@ -12,11 +12,11 @@ object Ast { sealed trait ExecutableDefinition extends Definition sealed trait TypeSystemDefinition extends Definition - sealed trait OperationType + sealed abstract class OperationType(val name: String) object OperationType { - case object Query extends OperationType - case object Mutation extends OperationType - case object Subscription extends OperationType + case object Query extends OperationType("query") + case object Mutation extends OperationType("mutation") + case object Subscription extends OperationType("subscription") } sealed trait OperationDefinition extends ExecutableDefinition @@ -75,7 +75,8 @@ object Ast { name: Name, tpe: Type, defaultValue: Option[Value], - ) + directives: List[Directive] + ) sealed trait Value object Value { @@ -104,7 +105,8 @@ object Ast { case class RootOperationTypeDefinition( operationType: OperationType, - tpe: Type.Named + tpe: Type.Named, + directives: List[Directive] ) sealed trait TypeDefinition extends TypeSystemDefinition with Product with Serializable { diff --git a/modules/core/src/main/scala/compiler.scala b/modules/core/src/main/scala/compiler.scala index dc0d3d9b..d6ef3d3b 100644 --- a/modules/core/src/main/scala/compiler.scala +++ b/modules/core/src/main/scala/compiler.scala @@ -4,10 +4,12 @@ package edu.gemini.grackle import scala.annotation.tailrec +import scala.reflect.ClassTag -import cats.parse.{LocationMap, Parser} +import cats.data.StateT import cats.implicits._ import io.circe.Json +import org.tpolecat.typename.{ TypeName, typeName } import syntax._ import Query._, Predicate._, Value._, UntypedOperation._ @@ -18,289 +20,150 @@ import ScalarType._ * GraphQL query parser */ object QueryParser { - import Ast.{ Type => _, Value => _, _ }, OperationDefinition._, Selection._ + import Ast.{ Directive => _, Type => _, Value => _, _ }, OperationDefinition._, Selection._ /** - * Parse a query String to a query algebra term. + * Parse a String to query algebra operations and fragments. * - * Yields a Query value on the right and accumulates errors on the left. + * GraphQL errors and warnings are accumulated in the result. */ - def parseText(text: String, name: Option[String] = None): Result[UntypedOperation] = { - def toResult[T](pr: Either[Parser.Error, T]): Result[T] = - Result.fromEither(pr.leftMap { e => - val lm = LocationMap(text) - lm.toLineCol(e.failedAtOffset) match { - case Some((row, col)) => - lm.getLine(row) match { - case Some(line) => - s"""Parse error at line $row column $col - |$line - |${List.fill(col)(" ").mkString}^""".stripMargin - case None => "Malformed query" //This is probably a bug in Cats Parse as it has given us the (row, col) index - } - case None => "Truncated query" - } - }) + def parseText(text: String): Result[(List[UntypedOperation], List[UntypedFragment])] = + for { + doc <- GraphQLParser.toResult(text, GraphQLParser.Document.parseAll(text)) + res <- parseDocument(doc) + _ <- Result.failure("At least one operation required").whenA(res._1.isEmpty) + } yield res + + /** + * Parse a document AST to query algebra operations and fragments. + * + * GraphQL errors and warnings are accumulated in the result. + */ + def parseDocument(doc: Document): Result[(List[UntypedOperation], List[UntypedFragment])] = { + val ops0 = doc.collect { case op: OperationDefinition => op } + val fragments0 = doc.collect { case frag: FragmentDefinition => frag } for { - doc <- toResult(GraphQLParser.Document.parseAll(text)) - query <- parseDocument(doc, name) - } yield query + ops <- ops0.traverse { + case op: Operation => parseOperation(op) + case qs: QueryShorthand => parseQueryShorthand(qs) + } + frags <- fragments0.traverse { frag => + val tpnme = frag.typeCondition.name + for { + sels <- parseSelections(frag.selectionSet) + dirs <- parseDirectives(frag.directives) + } yield UntypedFragment(frag.name.value, tpnme, dirs, sels) + } + } yield (ops, frags) } - def parseDocument(doc: Document, name: Option[String]): Result[UntypedOperation] = { - val ops = doc.collect { case op: OperationDefinition => op } - val fragments = doc.collect { case frag: FragmentDefinition => (frag.name.value, frag) }.toMap - - (ops, name) match { - case (Nil, _) => Result.failure("At least one operation required") - case (List(op: Operation), None) => parseOperation(op, fragments) - case (List(qs: QueryShorthand), None) => parseQueryShorthand(qs, fragments) - case (_, None) => - Result.failure("Operation name required to select unique operation") - case (ops, _) if ops.exists { case _: QueryShorthand => true ; case _ => false } => - Result.failure("Query shorthand cannot be combined with multiple operations") - case (ops, Some(name)) => - ops.filter { case Operation(_, Some(Name(`name`)), _, _, _) => true ; case _ => false } match { - case List(op: Operation) => parseOperation(op, fragments) - case Nil => - Result.failure(s"No operation named '$name'") - case _ => - Result.failure(s"Multiple operations named '$name'") - } + /** + * Parse an operation AST to a query algebra operation. + * + * GraphQL errors and warnings are accumulated in the result. + */ + def parseOperation(op: Operation): Result[UntypedOperation] = { + val Operation(opType, name, vds, dirs0, sels) = op + for { + vs <- parseVariableDefinitions(vds) + q <- parseSelections(sels) + dirs <- parseDirectives(dirs0) + } yield { + val name0 = name.map(_.value) + opType match { + case OperationType.Query => UntypedQuery(name0, q, vs, dirs) + case OperationType.Mutation => UntypedMutation(name0, q, vs, dirs) + case OperationType.Subscription => UntypedSubscription(name0, q, vs, dirs) + } } } - def parseOperation(op: Operation, fragments: Map[String, FragmentDefinition]): Result[UntypedOperation] = { - val Operation(opType, _, vds, _, sels) = op - val q = parseSelections(sels, None, fragments) - val vs = vds.map { - case VariableDefinition(nme, tpe, _) => UntypedVarDef(nme.value, tpe, None) + /** + * Parse variable definition ASTs to query algebra variable definitions. + * + * GraphQL errors and warnings are accumulated in the result. + */ + def parseVariableDefinitions(vds: List[VariableDefinition]): Result[List[UntypedVarDef]] = + vds.traverse { + case VariableDefinition(Name(nme), tpe, dv0, dirs0) => + for { + dv <- dv0.traverse(SchemaParser.parseValue) + dirs <- parseDirectives(dirs0) + } yield UntypedVarDef(nme, tpe, dv, dirs) } - q.map(q => - opType match { - case OperationType.Query => UntypedQuery(q, vs) - case OperationType.Mutation => UntypedMutation(q, vs) - case OperationType.Subscription => UntypedSubscription(q, vs) - } - ) - } - def parseQueryShorthand(qs: QueryShorthand, fragments: Map[String, FragmentDefinition]): Result[UntypedOperation] = - parseSelections(qs.selectionSet, None, fragments).map(q => UntypedQuery(q, Nil)) + /** + * Parse a query shorthand AST to query algebra operation. + * + * GraphQL errors and warnings are accumulated in the result. + */ + def parseQueryShorthand(qs: QueryShorthand): Result[UntypedOperation] = + parseSelections(qs.selectionSet).map(q => UntypedQuery(None, q, Nil, Nil)) - def parseSelections(sels: List[Selection], typeCondition: Option[String], fragments: Map[String, FragmentDefinition]): Result[Query] = - sels.traverse(parseSelection(_, typeCondition, fragments)).map { sels0 => + /** + * Parse selection ASTs to query algebra terms. + * + * GraphQL errors and warnings are accumulated in the result + */ + def parseSelections(sels: List[Selection]): Result[Query] = + sels.traverse(parseSelection).map { sels0 => if (sels0.sizeCompare(1) == 0) sels0.head else Group(sels0) } - def parseSelection(sel: Selection, typeCondition: Option[String], fragments: Map[String, FragmentDefinition]): Result[Query] = sel match { + /** + * Parse a selection AST to a query algebra term. + * + * GraphQL errors and warnings are accumulated in the result. + */ + def parseSelection(sel: Selection): Result[Query] = sel match { case Field(alias, name, args, directives, sels) => for { args0 <- parseArgs(args) - sels0 <- parseSelections(sels, None, fragments) - skip <- parseSkipInclude(directives) + sels0 <- parseSelections(sels) + dirs <- parseDirectives(directives) } yield { - val sel0 = - if (sels.isEmpty) Select(name.value, args0, Empty) - else Select(name.value, args0, sels0) - val sel1 = alias match { - case Some(Name(nme)) => Rename(nme, sel0) - case None => sel0 - } - val sel2 = typeCondition match { - case Some(tpnme) => UntypedNarrow(tpnme, sel1) - case _ => sel1 - } - val sel3 = skip match { - case Some((si, value)) => Skip(si, value, sel2) - case _ => sel2 - } - sel3 + val nme = name.value + val alias0 = alias.map(_.value).flatMap(n => if (n == nme) None else Some(n)) + if (sels.isEmpty) UntypedSelect(nme, alias0, args0, dirs, Empty) + else UntypedSelect(nme, alias0, args0, dirs, sels0) } case FragmentSpread(Name(name), directives) => for { - frag <- fragments.get(name).toResult(s"Undefined fragment '$name'") - skip <- parseSkipInclude(directives) - sels0 <- parseSelections(frag.selectionSet, Some(frag.typeCondition.name), fragments) - } yield { - val sels = skip match { - case Some((si, value)) => Skip(si, value, sels0) - case _ => sels0 - } - sels - } + dirs <- parseDirectives(directives) + } yield UntypedFragmentSpread(name, dirs) - case InlineFragment(Some(Ast.Type.Named(Name(tpnme))), directives, sels) => + case InlineFragment(typeCondition, directives, sels) => for { - skip <- parseSkipInclude(directives) - sels0 <- parseSelections(sels, Some(tpnme), fragments) - } yield { - val sels = skip match { - case Some((si, value)) => Skip(si, value, sels0) - case _ => sels0 - } - sels - } - - case _ => - Result.failure("Field or fragment spread required") + dirs <- parseDirectives(directives) + sels0 <- parseSelections(sels) + } yield UntypedInlineFragment(typeCondition.map(_.name), dirs, sels0) } - def parseSkipInclude(directives: List[Directive]): Result[Option[(Boolean, Value)]] = - directives.collect { case dir@Directive(Name("skip"|"include"), _) => dir } match { - case Nil => None.success - case Directive(Name(si), List((Name("if"), value))) :: Nil => parseValue(value).map(v => Some((si == "skip", v))) - case Directive(Name(si), _) :: Nil => Result.failure(s"$si must have a single Boolean 'if' argument") - case _ => Result.failure(s"Only a single skip/include allowed at a given location") - } + /** + * Parse directive ASTs to query algebra directives. + * + * GraphQL errors and warnings are accumulated in the result. + */ + def parseDirectives(directives: List[Ast.Directive]): Result[List[Directive]] = + directives.traverse(SchemaParser.mkDirective) + /** + * Parse argument ASTs to query algebra bindings. + * + * GraphQL errors and warnings are accumulated in the result. + */ def parseArgs(args: List[(Name, Ast.Value)]): Result[List[Binding]] = args.traverse((parseArg _).tupled) + /** + * Parse an argument AST to a query algebra binding. + * + * GraphQL errors and warnings are accumulated in the result. + */ def parseArg(name: Name, value: Ast.Value): Result[Binding] = - parseValue(value).map(v => Binding(name.value, v)) - - def parseValue(value: Ast.Value): Result[Value] = { - value match { - case Ast.Value.IntValue(i) => IntValue(i).success - case Ast.Value.FloatValue(d) => FloatValue(d).success - case Ast.Value.StringValue(s) => StringValue(s).success - case Ast.Value.BooleanValue(b) => BooleanValue(b).success - case Ast.Value.EnumValue(e) => UntypedEnumValue(e.value).success - case Ast.Value.Variable(v) => UntypedVariableValue(v.value).success - case Ast.Value.NullValue => NullValue.success - case Ast.Value.ListValue(vs) => vs.traverse(parseValue).map(ListValue(_)) - case Ast.Value.ObjectValue(fs) => - fs.traverse { case (name, value) => - parseValue(value).map(v => (name.value, v)) - }.map(ObjectValue(_)) - } - } -} - -object QueryMinimizer { - import Ast._ - - def minimizeText(text: String): Either[String, String] = { - for { - doc <- GraphQLParser.Document.parseAll(text).leftMap(_.expected.toList.mkString(",")) - } yield minimizeDocument(doc) - } - - def minimizeDocument(doc: Document): String = { - import OperationDefinition._ - import OperationType._ - import Selection._ - import Value._ - - def renderDefinition(defn: Definition): String = - defn match { - case e: ExecutableDefinition => renderExecutableDefinition(e) - case _ => "" - } - - def renderExecutableDefinition(ex: ExecutableDefinition): String = - ex match { - case op: OperationDefinition => renderOperationDefinition(op) - case frag: FragmentDefinition => renderFragmentDefinition(frag) - } - - def renderOperationDefinition(op: OperationDefinition): String = - op match { - case qs: QueryShorthand => renderSelectionSet(qs.selectionSet) - case op: Operation => renderOperation(op) - } - - def renderOperation(op: Operation): String = - renderOperationType(op.operationType) + - op.name.map(nme => s" ${nme.value}").getOrElse("") + - renderVariableDefns(op.variables)+ - renderDirectives(op.directives)+ - renderSelectionSet(op.selectionSet) - - def renderOperationType(op: OperationType): String = - op match { - case Query => "query" - case Mutation => "mutation" - case Subscription => "subscription" - } - - def renderDirectives(dirs: List[Directive]): String = - dirs.map { case Directive(name, args) => s"@${name.value}${renderArguments(args)}" }.mkString - - def renderVariableDefns(vars: List[VariableDefinition]): String = - vars match { - case Nil => "" - case _ => - vars.map { - case VariableDefinition(name, tpe, default) => - s"$$${name.value}:${tpe.name}${default.map(v => s"=${renderValue(v)}").getOrElse("")}" - }.mkString("(", ",", ")") - } - - def renderSelectionSet(sels: List[Selection]): String = - sels match { - case Nil => "" - case _ => sels.map(renderSelection).mkString("{", ",", "}") - } - - def renderSelection(sel: Selection): String = - sel match { - case f: Field => renderField(f) - case s: FragmentSpread => renderFragmentSpread(s) - case i: InlineFragment => renderInlineFragment(i) - } - - def renderField(f: Field) = { - f.alias.map(a => s"${a.value}:").getOrElse("")+ - f.name.value+ - renderArguments(f.arguments)+ - renderDirectives(f.directives)+ - renderSelectionSet(f.selectionSet) - } - - def renderArguments(args: List[(Name, Value)]): String = - args match { - case Nil => "" - case _ => args.map { case (n, v) => s"${n.value}:${renderValue(v)}" }.mkString("(", ",", ")") - } - - def renderInputObject(args: List[(Name, Value)]): String = - args match { - case Nil => "" - case _ => args.map { case (n, v) => s"${n.value}:${renderValue(v)}" }.mkString("{", ",", "}") - } - - def renderTypeCondition(tpe: Type): String = - s"on ${tpe.name}" - - def renderFragmentDefinition(frag: FragmentDefinition): String = - s"fragment ${frag.name.value} ${renderTypeCondition(frag.typeCondition)}${renderDirectives(frag.directives)}${renderSelectionSet(frag.selectionSet)}" - - def renderFragmentSpread(spread: FragmentSpread): String = - s"...${spread.name.value}${renderDirectives(spread.directives)}" - - def renderInlineFragment(frag: InlineFragment): String = - s"...${frag.typeCondition.map(renderTypeCondition).getOrElse("")}${renderDirectives(frag.directives)}${renderSelectionSet(frag.selectionSet)}" - - def renderValue(v: Value): String = - v match { - case Variable(name) => s"$$${name.value}" - case IntValue(value) => value.toString - case FloatValue(value) => value.toString - case StringValue(value) => s""""$value"""" - case BooleanValue(value) => value.toString - case NullValue => "null" - case EnumValue(name) => name.value - case ListValue(values) => values.map(renderValue).mkString("[", ",", "]") - case ObjectValue(fields) => renderInputObject(fields) - } - - doc.map(renderDefinition).mkString(",") - } - + SchemaParser.parseValue(value).map(v => Binding(name.value, v)) } /** @@ -317,30 +180,82 @@ class QueryCompiler(schema: Schema, phases: List[Phase]) { * Compiles the GraphQL query `text` to a query algebra term which * can be directly executed. * - * Any errors are accumulated on the left. + * GraphQL errors and warnings are accumulated in the result. */ - def compile(text: String, name: Option[String] = None, untypedVars: Option[Json] = None, introspectionLevel: IntrospectionLevel = Full): Result[Operation] = - QueryParser.parseText(text, name).flatMap(compileUntyped(_, untypedVars, introspectionLevel)) - - def compileUntyped(parsed: UntypedOperation, untypedVars: Option[Json] = None, introspectionLevel: IntrospectionLevel = Full): Result[Operation] = { + def compile(text: String, name: Option[String] = None, untypedVars: Option[Json] = None, introspectionLevel: IntrospectionLevel = Full, env: Env = Env.empty): Result[Operation] = + QueryParser.parseText(text).flatMap { case (ops, frags) => + (ops, name) match { + case (Nil, _) => + Result.failure("At least one operation required") + case (List(op), None) => + compileOperation(op, untypedVars, frags, introspectionLevel, env) + case (_, None) => + Result.failure("Operation name required to select unique operation") + case (ops, _) if ops.exists(_.name.isEmpty) => + Result.failure("Query shorthand cannot be combined with multiple operations") + case (ops, name) => + ops.filter(_.name == name) match { + case List(op) => + compileOperation(op, untypedVars, frags, introspectionLevel, env) + case Nil => + Result.failure(s"No operation named '$name'") + case _ => + Result.failure(s"Multiple operations named '$name'") + } + } + } + /** + * Compiles the provided operation AST to a query algebra term + * which can be directly executed. + * + * GraphQL errors and warnings are accumulated in the result. + */ + def compileOperation(op: UntypedOperation, untypedVars: Option[Json], frags: List[UntypedFragment], introspectionLevel: IntrospectionLevel = Full, env: Env = Env.empty): Result[Operation] = { val allPhases = - IntrospectionElaborator(introspectionLevel).toList ++ (VariablesAndSkipElaborator :: phases) + IntrospectionElaborator(introspectionLevel).toList ++ (VariablesSkipAndFragmentElaborator :: phases) for { - varDefs <- compileVarDefs(parsed.variables) + varDefs <- compileVarDefs(op.variables) vars <- compileVars(varDefs, untypedVars) - rootTpe <- parsed.rootTpe(schema) - query <- allPhases.foldLeftM(parsed.query) { (acc, phase) => phase.transform(acc, vars, schema, rootTpe) } - } yield Operation(query, rootTpe) + _ <- Directive.validateDirectivesForQuery(schema, op, frags, vars) + rootTpe <- op.rootTpe(schema) + res <- ( + for { + query <- allPhases.foldLeftM(op.query) { (acc, phase) => phase.transform(acc) } + } yield Operation(query, rootTpe, op.directives) + ).runA( + ElabState( + None, + schema, + Context(rootTpe), + vars, + frags.map(f => (f.name, f)).toMap, + op.query, + env, + List.empty, + Elab.pure + ) + ) + } yield res } + /** + * Compiles variable definition ASTs to variable definitions for the target schema. + * + * GraphQL errors and warnings are accumulated in the result. + */ def compileVarDefs(untypedVarDefs: UntypedVarDefs): Result[VarDefs] = untypedVarDefs.traverse { - case UntypedVarDef(name, untypedTpe, default) => - compileType(untypedTpe).map(tpe => InputValue(name, None, tpe, default)) + case UntypedVarDef(name, untypedTpe, default, dirs) => + compileType(untypedTpe).map(tpe => InputValue(name, None, tpe, default, dirs)) } + /** + * Compiles raw query variables to variables for the target schema. + * + * GraphQL errors and warnings are accumulated in the result. + */ def compileVars(varDefs: VarDefs, untypedVars: Option[Json]): Result[Vars] = untypedVars match { case None => Map.empty.success @@ -349,17 +264,22 @@ class QueryCompiler(schema: Schema, phases: List[Phase]) { case None => Result.failure(s"Variables must be represented as a Json object") case Some(obj) => - varDefs.traverse(iv => checkVarValue(iv, obj(iv.name)).map(v => (iv.name, (iv.tpe, v)))).map(_.toMap) + varDefs.traverse(iv => checkVarValue(iv, obj(iv.name), "variable values").map(v => (iv.name, (iv.tpe, v)))).map(_.toMap) } } + /** + * Compiles a type AST to a type in the target schema. + * + * GraphQL errors and warnings are accumulated in the result. + */ def compileType(tpe: Ast.Type): Result[Type] = { def loop(tpe: Ast.Type, nonNull: Boolean): Result[Type] = tpe match { case Ast.Type.NonNull(Left(named)) => loop(named, true) case Ast.Type.NonNull(Right(list)) => loop(list, true) case Ast.Type.List(elem) => loop(elem, false).map(e => if (nonNull) ListType(e) else NullableType(ListType(e))) case Ast.Type.Named(name) => schema.definition(name.value) match { - case None => Result.failure(s"Undefined typed '${name.value}'") + case None => Result.failure(s"Undefined type '${name.value}'") case Some(tpe) => (if (nonNull) tpe else NullableType(tpe)).success } } @@ -377,77 +297,264 @@ object QueryCompiler { import IntrospectionLevel._ + /** + * Elaboration monad. + * + * Supports threading of state through the elaboration of a query. Provides, + * + access to the schema, context, variables and fragments of a query. + * + ability to transform the children of Selects to supply semantics for field arguments. + * + ability to add contextual data to the resulting query both to support propagation of + * context to the elaboration of children, and to to drive run time behaviour. + * + ability to add selects for additional attributes to the resulting query. + * + ability to test existence and properties of neighbour nodes of the node being + * elaborated. + * + ability to report errors and warnings during elaboration. + */ + type Elab[T] = StateT[Result, ElabState, T] + object Elab { + def unit: Elab[Unit] = StateT.pure(()) + def pure[T](t: T): Elab[T] = StateT.pure(t) + def liftR[T](rt: Result[T]): Elab[T] = StateT.liftF(rt) + + /** The scheam of the query being elaborated */ + def schema: Elab[Schema] = StateT.inspect(_.schema) + /** The context of the node currently being elaborated */ + def context: Elab[Context] = StateT.inspect(_.context) + /** The variables of the query being elaborated */ + def vars: Elab[Vars] = StateT.inspect(_.vars) + /** The fragments of the query being elaborated */ + def fragments: Elab[Map[String, UntypedFragment]] = StateT.inspect(_.fragments) + /** The fragment with the supplied name, if defined, failing otherwise */ + def fragment(nme: String): Elab[UntypedFragment] = + StateT.inspectF(_.fragments.get(nme).toResult(s"Fragment '$nme' is not defined")) + /** `true` if the node currently being elaborated has a child with the supplied name */ + def hasField(name: String): Elab[Boolean] = StateT.inspect(_.hasField(name)) + /** The alias, if any, of the child with the supplied name */ + def fieldAlias(name: String): Elab[Option[String]] = StateT.inspect(_.fieldAlias(name)) + /** `true` if the node currently being elaborated has a sibling with the supplied name */ + def hasSibling(name: String): Elab[Boolean] = StateT.inspect(_.hasSibling(name)) + /** The result name of the node currently being elaborated */ + def resultName: Elab[Option[String]] = StateT.inspect(_.resultName) + + /** Binds the supplied value to the supplied name in the elaboration environment */ + def env(nme: String, value: Any): Elab[Unit] = env(List(nme -> value)) + /** Binds the supplied names and values in the elaboration environment */ + def env(kv: (String, Any), kvs: (String, Any)*): Elab[Unit] = env(kv +: kvs.toSeq) + /** Binds the supplied names and values in the elaboration environment */ + def env(kvs: Seq[(String, Any)]): Elab[Unit] = StateT.modify(_.env(kvs)) + /** Adds all the bindings of the supplied environment to the elaboration environment */ + def env(other: Env): Elab[Unit] = StateT.modify(_.env(other)) + /** The value bound to the supplied name in the elaboration environment, if any */ + def env[T: ClassTag](nme: String): Elab[Option[T]] = StateT.inspect(_.env[T](nme)) + /** The value bound to the supplied name in the elaboration environment, if any, failing otherwise */ + def envE[T: ClassTag: TypeName](nme: String): Elab[T] = + env(nme).flatMap(v => Elab.liftR(v.toResultOrError(s"Key '$nme' of type ${typeName[T]} was not found in $this"))) + /** The subset of the elaboration environment defined directly at this node */ + def localEnv: Elab[Env] = StateT.inspect(_.localEnv) + + /** Applies the supplied transformation to the child of the node currently being elaborated */ + def transformChild(f: Query => Elab[Query]): Elab[Unit] = StateT.modify(_.addChildTransform(f)) + /** Applies the supplied transformation to the child of the node currently being elaborated */ + def transformChild(f: Query => Query)(implicit dummy: DummyImplicit): Elab[Unit] = transformChild(q => Elab.pure(f(q))) + /** Applies the supplied transformation to the child of the node currently being elaborated */ + def transformChild(f: Query => Result[Query])(implicit dummy1: DummyImplicit, dummy2: DummyImplicit): Elab[Unit] = transformChild(q => Elab.liftR(f(q))) + /** The transformation to be applied to the child of the node currently being elaborated */ + def transform: Elab[Query => Elab[Query]] = StateT.inspect(_.childTransform) + /** Add the supplied attributed and corresponding query, if any, to the query being elaborated */ + def addAttribute(name: String, query: Query = Empty): Elab[Unit] = StateT.modify(_.addAttribute(name, query)) + /** The attributes which have been added to the query being elaborated */ + def attributes: Elab[List[(String, Query)]] = StateT.inspect(_.attributes) + + /** Report the supplied GraphQL warning during elaboration */ + def warning(msg: String): Elab[Unit] = StateT(s => Result.warning[(ElabState, Unit)](msg, (s, ()))) + /** Report the supplied GraphQL warning during elaboration */ + def warning(err: Problem): Elab[Unit] = StateT(s => Result.warning[(ElabState, Unit)](err, (s, ()))) + /** Report the supplied GraphQL error during elaboration */ + def failure[T](msg: String): Elab[T] = StateT(_ => Result.failure[(ElabState, T)](msg)) + /** Report the supplied GraphQL error during elaboration */ + def failure[T](err: Problem): Elab[T] = StateT(_ => Result.failure[(ElabState, T)](err)) + /** Report the supplied internal error during elaboration */ + def internalError[T](msg: String): Elab[T] = StateT(_ => Result.internalError[(ElabState, T)](msg)) + /** Report the supplied internal error during elaboration */ + def internalError[T](err: Throwable): Elab[T] = StateT(_ => Result.internalError[(ElabState, T)](err)) + + /** Save the current elaboration state */ + def push: Elab[Unit] = StateT.modify(_.push) + /** Save the current elaboration state and switch to the supplied context and query */ + def push(context: Context, query: Query): Elab[Unit] = StateT.modify(_.push(context, query)) + /** Save the current elaboration state and switch to the supplied schema, context and query */ + def push(schema: Schema, context: Context, query: Query): Elab[Unit] = StateT.modify(_.push(schema, context, query)) + /** Restore the previous elaboration state */ + def pop: Elab[Unit] = StateT.modifyF(s => s.parent.toResultOrError("Cannot pop root state")) + } + + /** + * The state managed by the elaboration monad. + */ + case class ElabState( + parent: Option[ElabState], + schema: Schema, + context: Context, + vars: Vars, + fragments: Map[String, UntypedFragment], + query: Query, + localEnv: Env, + attributes: List[(String, Query)], + childTransform: Query => Elab[Query] + ) { + def hasField(fieldName: String): Boolean = Query.hasField(query, fieldName) + def fieldAlias(fieldName: String): Option[String] = Query.fieldAlias(query, fieldName) + def hasSibling(fieldName: String): Boolean = parent.exists(s => Query.hasField(s.query, fieldName)) + def resultName: Option[String] = Query.ungroup(query).headOption.flatMap(Query.resultName) + def env(kvs: Seq[(String, Any)]): ElabState = copy(localEnv = localEnv.add(kvs: _*)) + def env(other: Env): ElabState = copy(localEnv = localEnv.add(other)) + def env[T: ClassTag](nme: String): Option[T] = localEnv.get(nme).orElse(parent.flatMap(_.env(nme))) + def addAttribute(name: String, query: Query = Empty): ElabState = copy(attributes = (name, query) :: attributes) + def addChildTransform(f: Query => Elab[Query]): ElabState = copy(childTransform = childTransform.andThen(_.flatMap(f))) + def push: ElabState = copy(parent = Some(this), localEnv = Env.empty, attributes = Nil, childTransform = Elab.pure) + def push(context: Context, query: Query): ElabState = + copy(parent = Some(this), context = context, query = query, localEnv = Env.empty, attributes = Nil, childTransform = Elab.pure) + def push(schema: Schema, context: Context, query: Query): ElabState = + copy(parent = Some(this), schema = schema, context = context, query = query, localEnv = Env.empty, attributes = Nil, childTransform = Elab.pure) + } + /** A QueryCompiler phase. */ trait Phase { /** - * Transform the supplied query algebra term `query` with expected type - * `tpe`. + * Transform the supplied query algebra term `query`. */ - def transform(query: Query, vars: Vars, schema: Schema, tpe: Type): Result[Query] = + def transform(query: Query): Elab[Query] = query match { - case s@Select(fieldName, _, child) => - (for { - obj <- tpe.underlyingObject - childTpe <- obj.field(fieldName) - } yield { - val isLeaf = childTpe.isUnderlyingLeaf - if (isLeaf && child != Empty) - Result.failure(s"Leaf field '$fieldName' of $obj must have an empty subselection set") - else if (!isLeaf && child == Empty) - Result.failure(s"Non-leaf field '$fieldName' of $obj must have a non-empty subselection set") - else - transform(child, vars, schema, childTpe).map(ec => s.copy(child = ec)) - }).getOrElse(Result.failure(s"Unknown field '$fieldName' in select")) - - case UntypedNarrow(tpnme, child) => - (for { - subtpe <- schema.definition(tpnme) - } yield { - transform(child, vars, schema, subtpe).map { ec => - if (tpe.underlyingObject.map(_ <:< subtpe).getOrElse(false)) ec else Narrow(schema.ref(tpnme), ec) - } - }).getOrElse(Result.failure(s"Unknown type '$tpnme' in type condition")) + case s@UntypedSelect(fieldName, alias, _, _, child) => + transformSelect(fieldName, alias, child).map(ec => s.copy(child = ec)) - case i@Introspect(_, child) if tpe =:= schema.queryType => - transform(child, vars, Introspection.schema, Introspection.schema.queryType).map(ec => i.copy(child = ec)) + case s@Select(fieldName, alias, child) => + transformSelect(fieldName, alias, child).map(ec => s.copy(child = ec)) + + case n@Narrow(subtpe, child) => + for { + c <- Elab.context + _ <- Elab.push(c.asType(subtpe), child) + ec <- transform(child) + _ <- Elab.pop + } yield n.copy(child = ec) + + case f@UntypedFragmentSpread(_, _) => Elab.pure(f) + case i@UntypedInlineFragment(None, _, child) => + transform(child).map(ec => i.copy(child = ec)) + case i@UntypedInlineFragment(Some(tpnme), _, child) => + for { + s <- Elab.schema + c <- Elab.context + subtpe <- Elab.liftR(Result.fromOption(s.definition(tpnme), s"Unknown type '$tpnme' in type condition")) + _ <- Elab.push(c.asType(subtpe), child) + ec <- transform(child) + _ <- Elab.pop + } yield i.copy(child = ec) case i@Introspect(_, child) => - val typenameTpe = ObjectType(s"__Typename", None, List(Field("__typename", None, Nil, StringType, false, None)), Nil) - transform(child, vars, Introspection.schema, typenameTpe).map(ec => i.copy(child = ec)) - - case n@Narrow(subtpe, child) => transform(child, vars, schema, subtpe).map(ec => n.copy(child = ec)) - case w@Wrap(_, child) => transform(child, vars, schema, tpe).map(ec => w.copy(child = ec)) - case r@Rename(_, child) => transform(child, vars, schema, tpe).map(ec => r.copy(child = ec)) - case c@Count(_, child) => transform(child, vars, schema, tpe).map(ec => c.copy(child = ec)) - case g@Group(children) => children.traverse(q => transform(q, vars, schema, tpe)).map(eqs => g.copy(queries = eqs)) - case u@Unique(child) => transform(child, vars, schema, tpe.nonNull.list).map(ec => u.copy(child = ec)) - case f@Filter(_, child) => tpe.item.toResult(s"Filter of non-List type $tpe").flatMap(item => transform(child, vars, schema, item).map(ec => f.copy(child = ec))) - case c@Component(_, _, child) => transform(child, vars, schema, tpe).map(ec => c.copy(child = ec)) - case e@Effect(_, child) => transform(child, vars, schema, tpe).map(ec => e.copy(child = ec)) - case s@Skip(_, _, child) => transform(child, vars, schema, tpe).map(ec => s.copy(child = ec)) - case l@Limit(_, child) => transform(child, vars, schema, tpe).map(ec => l.copy(child = ec)) - case o@Offset(_, child) => transform(child, vars, schema, tpe).map(ec => o.copy(child = ec)) - case o@OrderBy(_, child) => transform(child, vars, schema, tpe).map(ec => o.copy(child = ec)) - case e@Environment(_, child) => transform(child, vars, schema, tpe).map(ec => e.copy(child = ec)) - case t@TransformCursor(_, child) => transform(child, vars, schema, tpe).map(ec => t.copy(child = ec)) - case Skipped => Skipped.success - case Empty => Empty.success + for { + s <- Elab.schema + c <- Elab.context + iTpe = if(c.tpe =:= s.queryType) Introspection.schema.queryType else TypenameType + _ <- Elab.push(Introspection.schema, c.asType(iTpe), child) + ec <- transform(child) + _ <- Elab.pop + } yield i.copy(child = ec) + + case u@Unique(child) => + for { + c <- Elab.context + _ <- Elab.push(c.asType(c.tpe.nonNull.list), child) + ec <- transform(child) + _ <- Elab.pop + } yield u.copy(child = ec) + + case f@Filter(_, child) => + for { + c <- Elab.context + item <- Elab.liftR(c.tpe.item.toResultOrError(s"Filter of non-List type ${c.tpe}")) + _ <- Elab.push(c.asType(item), child) + ec <- transform(child) + _ <- Elab.pop + } yield f.copy(child = ec) + + case n@Count(child) => + for { + c <- Elab.context + pc <- Elab.liftR(c.parent.toResultOrError(s"Count node has no parent")) + _ <- Elab.push(pc, child) + ec <- transform(child) + _ <- Elab.pop + } yield n.copy(child = ec) + + case g@Group(children) => + children.traverse { c => + for { + _ <- Elab.push + tc <- transform(c) + _ <- Elab.pop + } yield tc + }.map(eqs => g.copy(queries = eqs)) + + case c@Component(_, _, child) => transform(child).map(ec => c.copy(child = ec)) + case e@Effect(_, child) => transform(child).map(ec => e.copy(child = ec)) + case l@Limit(_, child) => transform(child).map(ec => l.copy(child = ec)) + case o@Offset(_, child) => transform(child).map(ec => o.copy(child = ec)) + case o@OrderBy(_, child) => transform(child).map(ec => o.copy(child = ec)) + case e@Environment(_, child) => transform(child).map(ec => e.copy(child = ec)) + case t@TransformCursor(_, child) => transform(child).map(ec => t.copy(child = ec)) + case Empty => Elab.pure(Empty) } + + def transformSelect(fieldName: String, alias: Option[String], child: Query): Elab[Query] = + for { + c <- Elab.context + _ <- validateSubselection(fieldName, child) + childCtx <- Elab.liftR(c.forField(fieldName, alias)) + _ <- Elab.push(childCtx, child) + ec <- transform(child) + _ <- Elab.pop + } yield ec + + def validateSubselection(fieldName: String, child: Query): Elab[Unit] = + for { + c <- Elab.context + obj <- Elab.liftR(c.tpe.underlyingObject.toResultOrError(s"Expected object type, found ${c.tpe}")) + childCtx <- Elab.liftR(c.forField(fieldName, None)) + tpe = childCtx.tpe + _ <- { + val isLeaf = tpe.isUnderlyingLeaf + if (isLeaf && child != Empty) + Elab.failure(s"Leaf field '$fieldName' of $obj must have an empty subselection set") + else if (!isLeaf && child == Empty) + Elab.failure(s"Non-leaf field '$fieldName' of $obj must have a non-empty subselection set") + else + Elab.pure(()) + } + } yield () + + val TypenameType = ObjectType(s"__Typename", None, List(Field("__typename", None, Nil, StringType, Nil)), Nil, Nil) } + /** + * A phase which elaborates GraphQL introspection queries into the query algrebra. + */ class IntrospectionElaborator(level: IntrospectionLevel) extends Phase { - override def transform(query: Query, vars: Vars, schema: Schema, tpe: Type): Result[Query] = + override def transform(query: Query): Elab[Query] = query match { - case s@PossiblyRenamedSelect(Select(fieldName @ ("__typename" | "__schema" | "__type"), _, _), _) => + case s@UntypedSelect(fieldName @ ("__typename" | "__schema" | "__type"), _, _, _, _) => (fieldName, level) match { case ("__typename", Disabled) => - Result.failure("Introspection is disabled") + Elab.failure("Introspection is disabled") case ("__schema" | "__type", TypenameOnly | Disabled) => - Result.failure("Introspection is disabled") + Elab.failure("Introspection is disabled") case _ => - Introspect(schema, s).success + for { + schema <- Elab.schema + } yield Introspect(schema, s) } - case _ => super.transform(query, vars, schema, tpe) + case _ => super.transform(query) } } @@ -459,62 +566,123 @@ object QueryCompiler { } } - object VariablesAndSkipElaborator extends Phase { - override def transform(query: Query, vars: Vars, schema: Schema, tpe: Type): Result[Query] = + /** + * A phase which elaborates variables, directives, fragment spreads + * and inline fragments. + * + * 1. Query variable values are substituted for all variable + * references. + * + * 2. `skip` and `include` directives are handled during this phase + * and the guarded subqueries are retained or removed as + * appropriate. + * + * 3. Fragment spread and inline fragments are expanded. + * + * 4. types narrowing coercions by resolving the target type + * against the schema. + * + * 5. verifies that leaves have an empty subselection set and that + * structured types have a non-empty subselection set. + */ + object VariablesSkipAndFragmentElaborator extends Phase { + override def transform(query: Query): Elab[Query] = query match { case Group(children) => - children.traverse(q => transform(q, vars, schema, tpe)).map { eqs => - eqs.filterNot(_ == Skipped) match { - case Nil => Skipped + children.traverse(q => transform(q)).map { eqs => + eqs.filterNot(_ == Empty) match { + case Nil => Empty case eq :: Nil => eq case eqs => Group(eqs) } } - case Select(fieldName, args, child) => - tpe.withUnderlyingField(fieldName) { childTpe => + case sel@UntypedSelect(fieldName, alias, args, dirs, child) => + isSkipped(dirs).ifM( + Elab.pure(Empty), for { - elaboratedChild <- transform(child, vars, schema, childTpe) - elaboratedArgs <- args.traverse(elaborateBinding(vars)) - } yield Select(fieldName, elaboratedArgs, elaboratedChild) - } - - case Skip(skip, cond, child) => - for { - c <- extractCond(vars, cond) - elaboratedChild <- if(c == skip) Skipped.success else transform(child, vars, schema, tpe) - } yield elaboratedChild - - case _ => super.transform(query, vars, schema, tpe) + _ <- validateSubselection(fieldName, child) + s <- Elab.schema + c <- Elab.context + childCtx <- Elab.liftR(c.forField(fieldName, alias)) + vars <- Elab.vars + eArgs <- args.traverse(elaborateBinding(_, vars)) + eDirs <- Elab.liftR(Directive.elaborateDirectives(s, dirs, vars)) + _ <- Elab.push(childCtx, child) + ec <- transform(child) + _ <- Elab.pop + } yield sel.copy(args = eArgs, directives = eDirs, child = ec) + ) + + case UntypedFragmentSpread(nme, dirs) => + isSkipped(dirs).ifM( + Elab.pure(Empty), + for { + s <- Elab.schema + c <- Elab.context + f <- Elab.fragment(nme) + ctpe <- Elab.liftR(c.tpe.underlyingObject.toResultOrError(s"Expected object type, found ${c.tpe}")) + subtpe <- Elab.liftR(s.definition(f.tpnme).toResult(s"Unknown type '${f.tpnme}' in type condition of fragment '$nme'")) + _ <- Elab.failure(s"Fragment '$nme' is not compatible with type '${c.tpe}'").whenA(!(subtpe <:< ctpe) && !(ctpe <:< subtpe)) + _ <- Elab.push(c.asType(subtpe), f.child) + ec <- transform(f.child) + _ <- Elab.pop + } yield + if (ctpe <:< subtpe) ec + else Narrow(s.ref(subtpe.name), ec) + ) + + case UntypedInlineFragment(tpnme0, dirs, child) => + isSkipped(dirs).ifM( + Elab.pure(Empty), + for { + s <- Elab.schema + c <- Elab.context + ctpe <- Elab.liftR(c.tpe.underlyingObject.toResultOrError(s"Expected object type, found ${c.tpe}")) + subtpe <- tpnme0 match { + case None => + Elab.pure(ctpe) + case Some(tpnme) => + Elab.liftR(s.definition(tpnme).toResult(s"Unknown type '$tpnme' in type condition inline fragment")) + } + _ <- Elab.failure(s"Inline fragment with type condition '$subtpe' is not compatible with type '$ctpe'").whenA(!(subtpe <:< ctpe) && !(ctpe <:< subtpe)) + _ <- Elab.push(c.asType(subtpe), child) + ec <- transform(child) + _ <- Elab.pop + } yield + if (ctpe <:< subtpe) ec + else Narrow(s.ref(subtpe.name), ec) + ) + + case _ => super.transform(query) } - def elaborateBinding(vars: Vars)(b: Binding): Result[Binding] = - elaborateValue(vars)(b.value).map(ev => b.copy(value = ev)) + def elaborateBinding(b: Binding, vars: Vars): Elab[Binding] = + Elab.liftR(Value.elaborateValue(b.value, vars).map(ev => b.copy(value = ev))) - def elaborateValue(vars: Vars)(value: Value): Result[Value] = - value match { - case UntypedVariableValue(varName) => - vars.get(varName) match { - case Some((_, value)) => value.success - case None => Result.failure(s"Undefined variable '$varName'") - } - case ObjectValue(fields) => - val (keys, values) = fields.unzip - values.traverse(elaborateValue(vars)).map(evs => ObjectValue(keys.zip(evs))) - case ListValue(elems) => elems.traverse(elaborateValue(vars)).map(ListValue.apply) - case other => other.success + def isSkipped(dirs: List[Directive]): Elab[Boolean] = + dirs.filter(d => d.name == "skip" || d.name == "include") match { + case Nil => Elab.pure(false) + case List(Directive(nme, List(Binding("if", value)))) => + for { + c <- extractCond(value) + } yield (nme == "skip" && c) || (nme == "include" && !c) + case List(Directive(nme, _)) => Elab.failure(s"Directive '$nme' must have a single Boolean 'if' argument") + case _ => Elab.failure("skip/include directives must be unique") } - - def extractCond(vars: Vars, value: Value): Result[Boolean] = + def extractCond(value: Value): Elab[Boolean] = value match { - case UntypedVariableValue(varName) => - vars.get(varName) match { - case Some((tpe, BooleanValue(value))) if tpe.nonNull =:= BooleanType => value.success - case Some((_, _)) => Result.failure(s"Argument of skip/include must be boolean") - case None => Result.failure(s"Undefined variable '$varName'") - } - case BooleanValue(value) => value.success - case _ => Result.failure(s"Argument of skip/include must be boolean") + case VariableRef(varName) => + for { + v <- Elab.vars + tv <- Elab.liftR(Result.fromOption(v.get(varName), s"Undefined variable '$varName'")) + b <- tv match { + case (tpe, BooleanValue(value)) if tpe.nonNull =:= BooleanType => Elab.pure(value) + case _ => Elab.failure(s"Argument of skip/include must be boolean") + } + } yield b + case BooleanValue(value) => Elab.pure(value) + case _ => Elab.failure(s"Argument of skip/include must be boolean") } } @@ -535,108 +703,112 @@ object QueryCompiler { * * 2. eliminates Select arguments by delegating to a model-specific * `PartialFunction` which is responsible for translating `Select` - * nodes into a form which is directly interpretable, replacing - * them with a `Filter` or `Unique` node with a `Predicate` which - * is parameterized by the arguments, eg. + * nodes into a form which is directly interpretable, for example, + * replacing them with a `Filter` or `Unique` node with a + * `Predicate` which is parameterized by the arguments, ie., * * ``` - * Select("character", List(IDBinding("id", "1000")), child) + * UntypedSelect("character", None, List(IDBinding("id", "1000")), Nil, child) * ``` * might be translated to, * ``` - * Filter(FieldEquals("id", "1000"), child) + * Select("character, None, Filter(FieldEquals("id", "1000"), child)) * ``` - * - * 3. types narrowing coercions by resolving the target type - * against the schema. - * - * 4. verifies that leaves have an empty subselection set and that - * structured types have a non-empty subselection set. - * - * 5. eliminates Skipped nodes. + * 3. GraphQL introspection query field arguments are elaborated. */ - class SelectElaborator(mapping: Map[TypeRef, PartialFunction[Select, Result[Query]]]) extends Phase { - override def transform(query: Query, vars: Vars, schema: Schema, tpe: Type): Result[Query] = + trait SelectElaborator extends Phase { + override def transform(query: Query): Elab[Query] = query match { - case Select(fieldName, args, child) => - tpe.withUnderlyingField(fieldName) { childTpe => - val mapping0 = if (schema eq Introspection.schema) introspectionMapping else mapping - val elaborator: Select => Result[Query] = - (for { - obj <- tpe.underlyingObject - ref <- schema.ref(obj) - e <- mapping0.get(ref) - } yield (s: Select) => if (e.isDefinedAt(s)) e(s) else s.success).getOrElse((s: Select) => s.success) - - val obj = tpe.underlyingObject - val isLeaf = childTpe.isUnderlyingLeaf - if (isLeaf && child != Empty) - Result.failure(s"Leaf field '$fieldName' of $obj must have an empty subselection set") - else if (!isLeaf && child == Empty) - Result.failure(s"Non-leaf field '$fieldName' of $obj must have a non-empty subselection set") - else - for { - elaboratedChild <- transform(child, vars, schema, childTpe) - elaboratedArgs <- elaborateArgs(tpe, fieldName, args) - elaborated <- elaborator(Select(fieldName, elaboratedArgs, elaboratedChild)) - } yield elaborated + case sel@UntypedSelect(fieldName, resultName, args, dirs, child) => + for { + c <- Elab.context + s <- Elab.schema + childCtx <- Elab.liftR(c.forField(fieldName, resultName)) + obj <- Elab.liftR(c.tpe.underlyingObject.toResultOrError(s"Expected object type, found ${c.tpe}")) + field <- obj match { + case twf: TypeWithFields => + Elab.liftR(twf.fieldInfo(fieldName).toResult(s"No field '$fieldName' for type ${obj.underlying}")) + case _ => Elab.failure(s"Type $obj is not an object or interface type") + } + eArgs <- Elab.liftR(elaborateFieldArgs(obj, field, args)) + ref <- Elab.liftR(s.ref(obj).orElse(introspectionRef(obj)).toResultOrError(s"Type $obj not found in schema")) + _ <- if (s eq Introspection.schema) elaborateIntrospection(ref, fieldName, eArgs) + else select(ref, fieldName, eArgs, dirs) + elab <- Elab.transform + env <- Elab.localEnv + attrs <- Elab.attributes + _ <- Elab.push(childCtx, child) + ec <- transform(child) + _ <- Elab.pop + e2 <- elab(ec) + } yield { + val e1 = Select(sel.name, sel.alias, e2) + val e0 = + if(attrs.isEmpty) e1 + else Group((e1 :: attrs.map { case (nme, child) => Select(nme, child) }).flatMap(Query.ungroup)) + + if (env.isEmpty) e0 + else Environment(env, e0) } - case r: Rename => - super.transform(query, vars, schema, tpe).map(_ match { - case Rename(nme, Environment(e, child)) => Environment(e, Rename(nme, child)) - case Rename(nme, Group(queries)) => - val Some((baseName, _)) = Query.rootName(r): @unchecked - val renamed = - queries.map { - case s@Select(`baseName`, _, _) => Rename(nme, s) - case c@Count(`baseName`, _) => Rename(nme, c) - case other => other - } - Group(renamed) - case q => q - }) + case _ => super.transform(query) + } + + def select(ref: TypeRef, name: String, args: List[Binding], directives: List[Directive]): Elab[Unit] + + val QueryTypeRef = Introspection.schema.ref("Query") + val TypeTypeRef = Introspection.schema.ref("__Type") + val FieldTypeRef = Introspection.schema.ref("__Field") + val EnumValueTypeRef = Introspection.schema.ref("__EnumValue") + + def introspectionRef(tpe: Type): Option[TypeRef] = + Introspection.schema.ref(tpe).orElse(tpe.asNamed.flatMap( + _.name match { + case "__Typename" => Some(Introspection.schema.ref("__Typename")) + case _ => None + } + )) + + def elaborateIntrospection(ref: TypeRef, name: String, args: List[Binding]): Elab[Unit] = + (ref, name, args) match { + case (QueryTypeRef, "__type", List(Binding("name", StringValue(name)))) => + Elab.transformChild(child => Unique(Filter(Eql(TypeTypeRef / "name", Const(Option(name))), child))) - case Skipped => Empty.success + case (TypeTypeRef, "fields", List(Binding("includeDeprecated", BooleanValue(include)))) => + Elab.transformChild(child => if (include) child else Filter(Eql(FieldTypeRef / "isDeprecated", Const(false)), child)) + case (TypeTypeRef, "enumValues", List(Binding("includeDeprecated", BooleanValue(include)))) => + Elab.transformChild(child => if (include) child else Filter(Eql(EnumValueTypeRef / "isDeprecated", Const(false)), child)) - case _ => super.transform(query, vars, schema, tpe) + case _ => + Elab.unit } - def elaborateArgs(tpe: Type, fieldName: String, args: List[Binding]): Result[List[Binding]] = - tpe.underlyingObject match { - case Some(twf: TypeWithFields) => - twf.fieldInfo(fieldName) match { - case Some(field) => - val infos = field.args - val unknownArgs = args.filterNot(arg => infos.exists(_.name == arg.name)) - if (unknownArgs.nonEmpty) - Result.failure(s"Unknown argument(s) ${unknownArgs.map(s => s"'${s.name}'").mkString("", ", ", "")} in field $fieldName of type ${twf.name}") - else { - val argMap = args.groupMapReduce(_.name)(_.value)((x, _) => x) - infos.traverse(info => checkValue(info, argMap.get(info.name)).map(v => Binding(info.name, v))) - } - case _ => Result.failure(s"No field '$fieldName' in type $tpe") - } - case _ => Result.failure(s"Type $tpe is not an object or interface type") + def elaborateFieldArgs(tpe: NamedType, field: Field, args: List[Binding]): Result[List[Binding]] = { + val infos = field.args + val unknownArgs = args.filterNot(arg => infos.exists(_.name == arg.name)) + if (unknownArgs.nonEmpty) + Result.failure(s"Unknown argument(s) ${unknownArgs.map(s => s"'${s.name}'").mkString("", ", ", "")} in field ${field.name} of type ${tpe.name}") + else { + val argMap = args.groupMapReduce(_.name)(_.value)((x, _) => x) + infos.traverse(info => checkValue(info, argMap.get(info.name), s"field '${field.name}' of type '$tpe'").map(v => Binding(info.name, v))) } + } } - val introspectionMapping: Map[TypeRef, PartialFunction[Select, Result[Query]]] = { - val TypeType = Introspection.schema.ref("__Type") - val FieldType = Introspection.schema.ref("__Field") - val EnumValueType = Introspection.schema.ref("__EnumValue") - Map( - Introspection.schema.ref("Query") -> { - case sel@Select("__type", List(Binding("name", StringValue(name))), _) => - sel.eliminateArgs(child => Unique(Filter(Eql(TypeType / "name", Const(Option(name))), child))).success - }, - Introspection.schema.ref("__Type") -> { - case sel@Select("fields", List(Binding("includeDeprecated", BooleanValue(include))), _) => - sel.eliminateArgs(child => if (include) child else Filter(Eql(FieldType / "isDeprecated", Const(false)), child)).success - case sel@Select("enumValues", List(Binding("includeDeprecated", BooleanValue(include))), _) => - sel.eliminateArgs(child => if (include) child else Filter(Eql(EnumValueType / "isDeprecated", Const(false)), child)).success + object SelectElaborator { + /** + * Construct a `SelectElaborator` given a partial function which is called for each + * Select` node in the query. + */ + def apply(sel: PartialFunction[(TypeRef, String, List[Binding]), Elab[Unit]]): SelectElaborator = + new SelectElaborator { + def select(ref: TypeRef, name: String, args: List[Binding], directives: List[Directive]): Elab[Unit] = + if(sel.isDefinedAt((ref, name, args))) sel((ref, name, args)) + else Elab.unit } - ) + + /** A select elaborator which discards all field arguments */ + def identity: SelectElaborator = SelectElaborator(_ => Elab.unit) } /** @@ -644,19 +816,18 @@ object QueryCompiler { * composed mappings. * * This phase transforms the input query by assigning subtrees to component - * mappings as specified by the supplied `mapping`. + * mappings as specified by the supplied `cmapping`. * - * The mapping has `Type` and field name pairs as keys and component id and + * The mapping has `Type` and field name pairs as keys and mapping and * join function pairs as values. When the traversal of the input query * visits a `Select` node with type `Type.field name` it will replace the * `Select` with a `Component` node comprising, * - * 1. the component id of the interpreter which will be responsible for - * evaluating the subquery. + * 1. the mapping which will be responsible for evaluating the subquery. * 2. A join function which will be called during interpretation with, * - * i) the cursor at that point in evaluation. - * ii) The deferred subquery. + * i) The deferred subquery. + * ii) the cursor at that point in evaluation. * * This join function is responsible for computing the continuation * query which will be evaluated by the responsible interpreter. @@ -666,24 +837,27 @@ object QueryCompiler { * from the parent query. */ class ComponentElaborator[F[_]] private (cmapping: Map[(Type, String), (Mapping[F], (Query, Cursor) => Result[Query])]) extends Phase { - override def transform(query: Query, vars: Vars, schema: Schema, tpe: Type): Result[Query] = + override def transform(query: Query): Elab[Query] = query match { - case PossiblyRenamedSelect(Select(fieldName, args, child), resultName) => - (for { - obj <- tpe.underlyingObject - childTpe = obj.field(fieldName).getOrElse(ScalarType.AttributeType) - } yield { - transform(child, vars, schema, childTpe).map { elaboratedChild => - schema.ref(obj).flatMap(ref => cmapping.get((ref, fieldName))) match { - case Some((mapping, join)) => - Wrap(resultName, Component(mapping, join, PossiblyRenamedSelect(Select(fieldName, args, elaboratedChild), resultName))) - case None => - PossiblyRenamedSelect(Select(fieldName, args, elaboratedChild), resultName) - } + case s@Select(fieldName, resultName, child) => + for { + c <- Elab.context + obj <- Elab.liftR(c.tpe.underlyingObject.toResultOrError(s"Type ${c.tpe} is not an object or interface type")) + childCtx = c.forFieldOrAttribute(fieldName, resultName) + _ <- Elab.push(childCtx, child) + ec <- transform(child) + _ <- Elab.pop + schema <- Elab.schema + ref = schema.ref(obj.name) + } yield + cmapping.get((ref, fieldName)) match { + case Some((component, join)) => + Component(component, join, s.copy(child = ec)) + case None => + s.copy(child = ec) } - }).getOrElse(Result.failure(s"Type $tpe has no field '$fieldName'")) - case _ => super.transform(query, vars, schema, tpe) + case _ => super.transform(query) } } @@ -696,25 +870,44 @@ object QueryCompiler { new ComponentElaborator(mappings.map(m => ((m.tpe, m.fieldName), (m.mapping, m.join))).toMap) } - class EffectElaborator[F[_]] private (cmapping: Map[(Type, String), EffectHandler[F]]) extends Phase { - override def transform(query: Query, vars: Vars, schema: Schema, tpe: Type): Result[Query] = + /** + * A compiler phase which partitions a query for execution which may invoke + * multiple effect handlers. + * + * This phase transforms the input query by assigning subtrees to effect + * handlers as specified by the supplied `emapping`. + * + * The mapping has `Type` and field name pairs as keys and effect handlers + * as values. When the traversal of the input query visits a `Select` node + * with type `Type.field name` it will replace the + * `Select` with an `Effect` node comprising, + * + * 1. the effect handler which will be responsible for running the effect + * and evaluating the subquery against its result. + * 2. the subquery which will be evaluated by the effect handler. + */ + class EffectElaborator[F[_]] private (emapping: Map[(Type, String), EffectHandler[F]]) extends Phase { + override def transform(query: Query): Elab[Query] = query match { - case PossiblyRenamedSelect(Select(fieldName, args, child), resultName) => - (for { - obj <- tpe.underlyingObject - childTpe = obj.field(fieldName).getOrElse(ScalarType.AttributeType) - } yield { - transform(child, vars, schema, childTpe).map { elaboratedChild => - schema.ref(obj).flatMap(ref => cmapping.get((ref, fieldName))) match { - case Some(handler) => - Wrap(resultName, Effect(handler, PossiblyRenamedSelect(Select(fieldName, args, elaboratedChild), resultName))) - case None => - PossiblyRenamedSelect(Select(fieldName, args, elaboratedChild), resultName) - } + case s@Select(fieldName, resultName, child) => + for { + c <- Elab.context + obj <- Elab.liftR(c.tpe.underlyingObject.toResultOrError(s"Type ${c.tpe} is not an object or interface type")) + childCtx = c.forFieldOrAttribute(fieldName, resultName) + _ <- Elab.push(childCtx, child) + ec <- transform(child) + _ <- Elab.pop + schema <- Elab.schema + ref = schema.ref(obj.name) + } yield + emapping.get((ref, fieldName)) match { + case Some(handler) => + Effect(handler, s.copy(child = ec)) + case None => + s.copy(child = ec) } - }).getOrElse(Result.failure(s"Type $tpe has no field '$fieldName'")) - case _ => super.transform(query, vars, schema, tpe) + case _ => super.transform(query) } } @@ -725,53 +918,58 @@ object QueryCompiler { new EffectElaborator(mappings.map(m => ((m.tpe, m.fieldName), m.handler)).toMap) } + /** + * A compiler phase which estimates the size of a query and applies width + * and depth limits. + */ class QuerySizeValidator(maxDepth: Int, maxWidth: Int) extends Phase { - override def transform(query: Query, vars: Vars, schema: Schema, tpe: Type): Result[Query] = - querySize(query) match { - case (depth, _) if depth > maxDepth => Result.failure(s"Query is too deep: depth is $depth levels, maximum is $maxDepth") - case (_, width) if width > maxWidth => Result.failure(s"Query is too wide: width is $width leaves, maximum is $maxWidth") - case (depth, width) if depth > maxDepth && width > maxWidth => Result.failure(s"Query is too complex: width/depth is $width/$depth leaves/levels, maximum is $maxWidth/$maxDepth") - case (_, _) => query.success + override def transform(query: Query): Elab[Query] = + Elab.fragments.flatMap { frags => + querySize(query, frags) match { + case (depth, _) if depth > maxDepth => Elab.failure(s"Query is too deep: depth is $depth levels, maximum is $maxDepth") + case (_, width) if width > maxWidth => Elab.failure(s"Query is too wide: width is $width leaves, maximum is $maxWidth") + case (depth, width) if depth > maxDepth && width > maxWidth => Elab.failure(s"Query is too complex: width/depth is $width/$depth leaves/levels, maximum is $maxWidth/$maxDepth") + case (_, _) => Elab.pure(query) + } } - def querySize(query: Query): (Int, Int) = { - def handleGroupedQueries(childQueries: List[Query], depth: Int, width: Int): (Int, Int) = { - val fragmentQueries = childQueries.diff(childQueries.collect { case n: Narrow => n }) - val childSizes = - if (fragmentQueries.isEmpty) childQueries.map(gq => loop(gq, depth, width, true)) - else childQueries.map(gq => loop(gq, depth + 1, width, true)) - - val childDepths = (childSizes.map(size => size._1)).max - val childWidths = childSizes.map(_._2).sum - (childDepths, childWidths) + def querySize(query: Query, frags: Map[String, UntypedFragment]): (Int, Int) = { + def handleGroup(g: Group, depth: Int, width: Int): (Int, Int) = { + val dws = Query.ungroup(g).map(loop(_, depth, width)) + val (depths, widths) = dws.unzip + (depths.max, widths.sum) } + @tailrec - def loop(q: Query, depth: Int, width: Int, group: Boolean): (Int, Int) = + def loop(q: Query, depth: Int, width: Int): (Int, Int) = q match { - case Select(_, _, Empty) => if (group) (depth, width + 1) else (depth + 1, width + 1) - case Count(_, _) => if (group) (depth, width + 1) else (depth + 1, width + 1) - case Select(_, _, child) => if (group) loop(child, depth, width, false) else loop(child, depth + 1, width, false) - case Group(queries) => handleGroupedQueries(queries, depth, width) - case Component(_, _, child) => loop(child, depth, width, false) - case Effect(_, child) => loop(child, depth, width, false) - case Environment(_, child) => loop(child, depth, width, false) + case UntypedSelect(_, _, _, _, Empty) => (depth + 1, width + 1) + case Select(_, _, Empty) => (depth + 1, width + 1) + case Count(_) => (depth + 1, width + 1) + case UntypedSelect(_, _, _, _, child) => loop(child, depth + 1, width) + case Select(_, _, child) => loop(child, depth + 1, width) + case g: Group => handleGroup(g, depth, width) + case Component(_, _, child) => loop(child, depth, width) + case Effect(_, child) => loop(child, depth, width) + case Environment(_, child) => loop(child, depth, width) case Empty => (depth, width) - case Filter(_, child) => loop(child, depth, width, false) + case Filter(_, child) => loop(child, depth, width) case Introspect(_, _) => (depth, width) - case Limit(_, child) => loop(child, depth, width, false) - case Offset(_, child) => loop(child, depth, width, false) - case Narrow(_, child) => loop(child, depth, width, true) - case OrderBy(_, child) => loop(child, depth, width, false) - case Rename(_, child) => loop(child, depth, width, false) - case Skip(_, _, child) => loop(child, depth, width, false) - case Skipped => (depth, width) - case TransformCursor(_, child) => loop(child, depth, width, false) - case Unique(child) => loop(child, depth, width, false) - case UntypedNarrow(_, child) => loop(child, depth, width, false) - case Wrap(_, child) => loop(child, depth, width, false) + case Limit(_, child) => loop(child, depth, width) + case Offset(_, child) => loop(child, depth, width) + case Narrow(_, child) => loop(child, depth, width) + case OrderBy(_, child) => loop(child, depth, width) + case TransformCursor(_, child) => loop(child, depth, width) + case Unique(child) => loop(child, depth, width) + case UntypedFragmentSpread(nme, _) => + frags.get(nme) match { + case Some(frag) => loop(frag.child, depth, width) + case None => (depth, width) + } + case UntypedInlineFragment(_, _, child) => loop(child, depth, width) } - loop(query, 0, 0, false) + loop(query, 0, 0) } } } diff --git a/modules/core/src/main/scala/cursor.scala b/modules/core/src/main/scala/cursor.scala index f16968d5..74e86d23 100644 --- a/modules/core/src/main/scala/cursor.scala +++ b/modules/core/src/main/scala/cursor.scala @@ -8,10 +8,9 @@ import scala.reflect.{classTag, ClassTag} import cats.implicits._ import io.circe.Json -import org.tpolecat.typename.{ TypeName, typeName } +import org.tpolecat.typename.{TypeName, typeName} import syntax._ -import Cursor.{ cast, Context, Env } /** * Indicates a position within an abstract data model during the interpretation @@ -51,14 +50,15 @@ trait Cursor { /** Yields the cumulative environment defined at this `Cursor`. */ def fullEnv: Env = parent.map(_.fullEnv).getOrElse(Env.empty).add(env) + /** Does the environment at this `Cursor` contain a value for the supplied key? */ def envContains(nme: String): Boolean = env.contains(nme) || parent.map(_.envContains(nme)).getOrElse(false) /** * Yield the value at this `Cursor` as a value of type `T` if possible, * an error or the left hand side otherwise. */ - def as[T: ClassTag]: Result[T] = - cast[T](focus).toResultOrError(s"Expected value of type ${classTag[T]} for focus of type $tpe at path $path, found $focus") + def as[T: ClassTag: TypeName]: Result[T] = + classTag[T].unapply(focus).toResultOrError(s"Expected value of type ${typeName[T]} for focus of type $tpe at path $path, found $focus") /** Is the value at this `Cursor` of a scalar or enum type? */ def isLeaf: Boolean @@ -139,7 +139,7 @@ trait Cursor { * Yield the value of the field `fieldName` of this `Cursor` as a value of * type `T` if possible, an error or the left hand side otherwise. */ - def fieldAs[T: ClassTag](fieldName: String): Result[T] = + def fieldAs[T: ClassTag : TypeName](fieldName: String): Result[T] = field(fieldName, None).flatMap(_.as[T]) /** True if this cursor is nullable and null, false otherwise. */ @@ -254,78 +254,6 @@ trait Cursor { } object Cursor { - /** - * Context represents a position in the output tree in terms of, - * 1) the path through the schema to the position - * 2) the path through the schema with query aliases applied - * 3) the type of the element at the position - */ - case class Context( - rootTpe: Type, - path: List[String], - resultPath: List[String], - typePath: List[Type] - ) { - lazy val tpe: Type = typePath.headOption.getOrElse(rootTpe) - - def asType(tpe: Type): Context = { - typePath match { - case Nil => copy(rootTpe = tpe) - case _ :: tl => copy(typePath = tpe :: tl) - } - } - - def isRoot: Boolean = path.isEmpty - - def parent: Option[Context] = - if(path.isEmpty) None - else Some(copy(path = path.tail, resultPath = resultPath.tail, typePath = typePath.tail)) - - def forField(fieldName: String, resultName: String): Result[Context] = - tpe.underlyingField(fieldName).map { fieldTpe => - copy(path = fieldName :: path, resultPath = resultName :: resultPath, typePath = fieldTpe :: typePath) - }.toResultOrError(s"No field '$fieldName' for type $tpe") - - def forField(fieldName: String, resultName: Option[String]): Result[Context] = - tpe.underlyingField(fieldName).map { fieldTpe => - copy(path = fieldName :: path, resultPath = resultName.getOrElse(fieldName) :: resultPath, typePath = fieldTpe :: typePath) - }.toResultOrError(s"No field '$fieldName' for type $tpe") - - def forPath(path1: List[String]): Result[Context] = - path1 match { - case Nil => this.success - case hd :: tl => forField(hd, hd).flatMap(_.forPath(tl)) - } - - def forFieldOrAttribute(fieldName: String, resultName: Option[String]): Context = { - val fieldTpe = tpe.underlyingField(fieldName).getOrElse(ScalarType.AttributeType) - copy(path = fieldName :: path, resultPath = resultName.getOrElse(fieldName) :: resultPath, typePath = fieldTpe :: typePath) - } - - override def equals(other: Any): Boolean = - other match { - case Context(oRootTpe, oPath, oResultPath, _) => - rootTpe =:= oRootTpe && resultPath == oResultPath && path == oPath - case _ => false - } - - override def hashCode(): Int = resultPath.hashCode - } - - object Context { - def apply(rootTpe: Type, fieldName: String, resultName: Option[String]): Option[Context] = { - for { - tpe <- rootTpe.underlyingField(fieldName) - } yield new Context(rootTpe, List(fieldName), List(resultName.getOrElse(fieldName)), List(tpe)) - } - - def apply(rootTpe: Type): Context = Context(rootTpe, Nil, Nil, Nil) - - def apply(path: Path): Result[Context] = - path.path.foldLeftM(Context(path.rootTpe, Nil, Nil, Nil)) { case (acc, elem) => - acc.forField(elem, None) - } - } def flatten(c: Cursor): Result[List[Cursor]] = if(c.isList) c.asList.flatMap(flatten) @@ -335,66 +263,6 @@ object Cursor { def flatten(cs: List[Cursor]): Result[List[Cursor]] = cs.flatTraverse(flatten) - sealed trait Env { - def add[T](items: (String, T)*): Env - def add(env: Env): Env - def contains(name: String): Boolean - def get[T: ClassTag](name: String): Option[T] - - def getR[A: ClassTag: TypeName](name: String): Result[A] = - get[A](name).toResultOrError(s"Key '$name' of type ${typeName[A]} was not found in $this") - - def addFromQuery(query: Query): Env = - query match { - case Query.Environment(childEnv, child) => - add(childEnv).addFromQuery(child) - case _ => this - } - } - - object Env { - def empty: Env = EmptyEnv - - def apply[T](items: (String, T)*): Env = NonEmptyEnv(Map(items: _*)) - - case object EmptyEnv extends Env { - def add[T](items: (String, T)*): Env = NonEmptyEnv(Map(items: _*)) - def add(env: Env): Env = env - def contains(name: String): Boolean = false - def get[T: ClassTag](name: String): Option[T] = None - } - - case class NonEmptyEnv(elems: Map[String, Any]) extends Env { - def add[T](items: (String, T)*): Env = NonEmptyEnv(elems++items) - def add(other: Env): Env = other match { - case EmptyEnv => this - case NonEmptyEnv(elems0) => NonEmptyEnv(elems++elems0) - } - def contains(name: String): Boolean = elems.contains(name) - def get[T: ClassTag](name: String): Option[T] = - elems.get(name).flatMap(cast[T]) - } - } - - private def cast[T: ClassTag](x: Any): Option[T] = { - val clazz = classTag[T].runtimeClass - if ( - clazz.isInstance(x) || - (clazz == classOf[Int] && x.isInstanceOf[java.lang.Integer]) || - (clazz == classOf[Boolean] && x.isInstanceOf[java.lang.Boolean]) || - (clazz == classOf[Long] && x.isInstanceOf[java.lang.Long]) || - (clazz == classOf[Double] && x.isInstanceOf[java.lang.Double]) || - (clazz == classOf[Byte] && x.isInstanceOf[java.lang.Byte]) || - (clazz == classOf[Short] && x.isInstanceOf[java.lang.Short]) || - (clazz == classOf[Char] && x.isInstanceOf[java.lang.Character]) || - (clazz == classOf[Float] && x.isInstanceOf[java.lang.Float]) || - (clazz == classOf[Unit] && x.isInstanceOf[scala.runtime.BoxedUnit]) - ) - Some(x.asInstanceOf[T]) - else - None - } - /** Abstract `Cursor` providing default implementation of most methods. */ abstract class AbstractCursor extends Cursor { def isLeaf: Boolean = false @@ -429,7 +297,7 @@ object Cursor { def hasField(fieldName: String): Boolean = false def field(fieldName: String, resultName: Option[String]): Result[Cursor] = - Result.internalError(s"No field '$fieldName' for type $tpe") + Result.internalError(s"No field '$fieldName' for type ${tpe.underlying}") } /** Proxy `Cursor` which delegates most methods to an underlying `Cursor`.. */ @@ -489,6 +357,20 @@ object Cursor { factory.fromSpecific(newElems).success } + /** Proxy `Cursor` which always yields a `NullCursor` for fields of the underlying cursor */ + case class NullFieldCursor(underlying: Cursor) extends ProxyCursor(underlying) { + override def withEnv(env: Env): Cursor = new NullFieldCursor(underlying.withEnv(env)) + override def field(fieldName: String, resultName: Option[String]): Result[Cursor] = + underlying.field(fieldName, resultName).map(NullCursor(_)) + } + + /** Proxy `Cursor` which always yields null */ + case class NullCursor(underlying: Cursor) extends ProxyCursor(underlying) { + override def withEnv(env: Env): Cursor = new NullCursor(underlying.withEnv(env)) + override def isDefined: Result[Boolean] = false.success + override def asNullable: Result[Option[Cursor]] = None.success + } + case class DeferredCursor(context: Context, parent: Option[Cursor], env: Env, deferredPath: List[String], mkCursor: (Context, Cursor) => Result[Cursor]) extends AbstractCursor { def focus: Any = Result.internalError(s"Empty cursor has no focus") def withEnv(env0: Env): DeferredCursor = copy(env = env.add(env0)) @@ -496,7 +378,7 @@ object Cursor { override def hasField(fieldName: String): Boolean = fieldName == deferredPath.head override def field(fieldName: String, resultName: Option[String]): Result[Cursor] = - if(fieldName != deferredPath.head) Result.internalError(s"No field '$fieldName' for type $tpe") + if(fieldName != deferredPath.head) Result.internalError(s"No field '$fieldName' for type ${tpe.underlying}") else for { fieldContext <- context.forField(fieldName, resultName) @@ -510,3 +392,123 @@ object Cursor { DeferredCursor(Context(path.rootTpe), None, Env.empty, path.path, mkCursor) } } + +/** + * Context represents a position in the output tree in terms of, + * 1) the path through the schema to the position + * 2) the path through the schema with query aliases applied + * 3) the type of the element at the position + */ +case class Context( + rootTpe: Type, + path: List[String], + resultPath: List[String], + typePath: List[Type] +) { + lazy val tpe: Type = typePath.headOption.getOrElse(rootTpe) + + def asType(tpe: Type): Context = { + typePath match { + case Nil => copy(rootTpe = tpe) + case _ :: tl => copy(typePath = tpe :: tl) + } + } + + def isRoot: Boolean = path.isEmpty + + def parent: Option[Context] = + if(path.isEmpty) None + else Some(copy(path = path.tail, resultPath = resultPath.tail, typePath = typePath.tail)) + + def forField(fieldName: String, resultName: String): Result[Context] = + tpe.underlyingField(fieldName).map { fieldTpe => + copy(path = fieldName :: path, resultPath = resultName :: resultPath, typePath = fieldTpe :: typePath) + }.toResult(s"No field '$fieldName' for type ${tpe.underlying}") + + def forField(fieldName: String, resultName: Option[String]): Result[Context] = + tpe.underlyingField(fieldName).map { fieldTpe => + copy(path = fieldName :: path, resultPath = resultName.getOrElse(fieldName) :: resultPath, typePath = fieldTpe :: typePath) + }.toResult(s"No field '$fieldName' for type ${tpe.underlying}") + + def forPath(path1: List[String]): Result[Context] = + path1 match { + case Nil => this.success + case hd :: tl => forField(hd, hd).flatMap(_.forPath(tl)) + } + + def forFieldOrAttribute(fieldName: String, resultName: Option[String]): Context = { + val fieldTpe = tpe.underlyingField(fieldName).getOrElse(ScalarType.AttributeType) + copy(path = fieldName :: path, resultPath = resultName.getOrElse(fieldName) :: resultPath, typePath = fieldTpe :: typePath) + } + + override def equals(other: Any): Boolean = + other match { + case Context(oRootTpe, oPath, oResultPath, _) => + rootTpe =:= oRootTpe && resultPath == oResultPath && path == oPath + case _ => false + } + + override def hashCode(): Int = resultPath.hashCode +} + +object Context { + def apply(rootTpe: Type, fieldName: String, resultName: Option[String]): Option[Context] = { + for { + tpe <- rootTpe.underlyingField(fieldName) + } yield new Context(rootTpe, List(fieldName), List(resultName.getOrElse(fieldName)), List(tpe)) + } + + def apply(rootTpe: Type): Context = Context(rootTpe, Nil, Nil, Nil) + + def apply(path: Path): Result[Context] = + path.path.foldLeftM(Context(path.rootTpe, Nil, Nil, Nil)) { case (acc, elem) => + acc.forField(elem, None) + } +} + +/** + * An environment for elaboration or execution of a GraphQL query. + */ +sealed trait Env { + def add[T](items: (String, T)*): Env + def add(env: Env): Env + def contains(name: String): Boolean + def get[T: ClassTag](name: String): Option[T] + def isEmpty: Boolean + + def getR[A: ClassTag: TypeName](name: String): Result[A] = + get[A](name).toResultOrError(s"Key '$name' of type ${typeName[A]} was not found in $this") + + def addFromQuery(query: Query): Env = + query match { + case Query.Environment(childEnv, child) => + add(childEnv).addFromQuery(child) + case _ => this + } +} + +object Env { + def empty: Env = EmptyEnv + + def apply[T](items: (String, T)*): Env = NonEmptyEnv(Map(items: _*)) + + case object EmptyEnv extends Env { + def add[T](items: (String, T)*): Env = NonEmptyEnv(Map(items: _*)) + def add(env: Env): Env = env + def contains(name: String): Boolean = false + def get[T: ClassTag](name: String): Option[T] = None + def isEmpty: Boolean = true + } + + case class NonEmptyEnv(elems: Map[String, Any]) extends Env { + def add[T](items: (String, T)*): Env = NonEmptyEnv(elems++items) + def add(other: Env): Env = other match { + case EmptyEnv => this + case NonEmptyEnv(elems0) => NonEmptyEnv(elems++elems0) + } + def contains(name: String): Boolean = elems.contains(name) + def get[T: ClassTag](name: String): Option[T] = + elems.get(name).flatMap(classTag[T].unapply) + def isEmpty: Boolean = false + } +} diff --git a/modules/core/src/main/scala/introspection.scala b/modules/core/src/main/scala/introspection.scala index 15becbdf..b1c2494c 100644 --- a/modules/core/src/main/scala/introspection.scala +++ b/modules/core/src/main/scala/introspection.scala @@ -122,12 +122,40 @@ object Introspection { val __EnumValueType = schema.ref("__EnumValue") val __DirectiveType = schema.ref("__Directive") val __TypeKindType = schema.ref("__TypeKind") + val __DirectiveLocationType = schema.ref("__DirectiveLocation") object TypeKind extends Enumeration { val SCALAR, OBJECT, INTERFACE, UNION, ENUM, INPUT_OBJECT, LIST, NON_NULL = Value implicit val typeKindEncoder: Encoder[Value] = Encoder[String].contramap(_.toString) } + implicit val directiveLocationEncoder: Encoder[Ast.DirectiveLocation] = { + import Ast.DirectiveLocation._ + + Encoder[String].contramap { + case QUERY => "QUERY" + case MUTATION => "MUTATION" + case SUBSCRIPTION => "SUBSCRIPTION" + case FIELD => "FIELD" + case FRAGMENT_DEFINITION => "FRAGMENT_DEFINITION" + case FRAGMENT_SPREAD => "FRAGMENT_SPREAD" + case INLINE_FRAGMENT => "INLINE_FRAGMENT" + case VARIABLE_DEFINITION => "VARIABLE_DEFINITION" + + case SCHEMA => "SCHEMA" + case SCALAR => "SCALAR" + case OBJECT => "OBJECT" + case FIELD_DEFINITION => "FIELD_DEFINITION" + case ARGUMENT_DEFINITION => "ARGUMENT_DEFINITION" + case INTERFACE => "INTERFACE" + case UNION => "UNION" + case ENUM => "ENUM" + case ENUM_VALUE => "ENUM_VALUE" + case INPUT_OBJECT => "INPUT_OBJECT" + case INPUT_FIELD_DEFINITION => "INPUT_FIELD_DEFINITION" + } + } + case class NonNullType(tpe: Type) val flipNullityDealias: PartialFunction[Type, Any] = { @@ -243,7 +271,7 @@ object Introspection { ValueField("defaultValue", _.defaultValue.map(SchemaRenderer.renderValue)) ) ), - ValueObjectMapping[EnumValue]( + ValueObjectMapping[EnumValueDefinition]( tpe = __EnumValueType, fieldMappings = List( @@ -253,7 +281,7 @@ object Introspection { ValueField("deprecationReason", _.deprecationReason) ) ), - ValueObjectMapping[Directive]( + ValueObjectMapping[DirectiveDef]( tpe = __DirectiveType, fieldMappings = List( @@ -264,7 +292,8 @@ object Introspection { ValueField("isRepeatable", _.isRepeatable) ) ), - LeafMapping[TypeKind.Value](__TypeKindType) + LeafMapping[TypeKind.Value](__TypeKindType), + LeafMapping[Ast.DirectiveLocation](__DirectiveLocationType) ) } } diff --git a/modules/core/src/main/scala/mapping.scala b/modules/core/src/main/scala/mapping.scala index 1293b1d4..355bfbd3 100644 --- a/modules/core/src/main/scala/mapping.scala +++ b/modules/core/src/main/scala/mapping.scala @@ -4,6 +4,7 @@ package edu.gemini.grackle import scala.collection.Factory +import scala.reflect.ClassTag import cats.MonadThrow import cats.data.Chain @@ -15,52 +16,43 @@ import org.tpolecat.sourcepos.SourcePos import org.tpolecat.typename._ import syntax._ -import Cursor.{AbstractCursor, Context, Env} -import Query.{EffectHandler, Select} -import QueryCompiler.{ComponentElaborator, EffectElaborator, SelectElaborator, IntrospectionLevel} +import Cursor.{AbstractCursor, ProxyCursor} +import Query.EffectHandler +import QueryCompiler.{ComponentElaborator, EffectElaborator, IntrospectionLevel, SelectElaborator} import QueryInterpreter.ProtoJson import IntrospectionLevel._ -trait QueryExecutor[F[_], T] { outer => - def compileAndRun(text: String, name: Option[String] = None, untypedVars: Option[Json] = None, introspectionLevel: IntrospectionLevel = Full, env: Env = Env.empty)( - implicit sc: Compiler[F,F] - ): F[T] = - compileAndRunOne(text, name, untypedVars, introspectionLevel, env) - - def compileAndRunAll(text: String, name: Option[String] = None, untypedVars: Option[Json] = None, introspectionLevel: IntrospectionLevel = Full, env: Env = Env.empty): Stream[F,T] - - def compileAndRunOne(text: String, name: Option[String] = None, untypedVars: Option[Json] = None, introspectionLevel: IntrospectionLevel = Full, env: Env = Env.empty)( - implicit sc: Compiler[F,F] - ): F[T] -} - -abstract class Mapping[F[_]] extends QueryExecutor[F, Json] { +/** + * Represents a mapping between a GraphQL schema and an underlying abstract data source. + */ +abstract class Mapping[F[_]] { implicit val M: MonadThrow[F] val schema: Schema val typeMappings: List[TypeMapping] - def run(query: Query, rootTpe: Type, env: Env): Stream[F,Json] = - interpreter.run(query, rootTpe, env) - - def run(op: Operation, env: Env = Env.empty): Stream[F,Json] = - run(op.query, op.rootTpe, env) - - def compileAndRunOne(text: String, name: Option[String] = None, untypedVars: Option[Json] = None, introspectionLevel: IntrospectionLevel = Full, env: Env = Env.empty)( + /** + * Compile and run a single GraphQL query or mutation. + * + * Yields a JSON response containing the result of the query or mutation. + */ + def compileAndRun(text: String, name: Option[String] = None, untypedVars: Option[Json] = None, introspectionLevel: IntrospectionLevel = Full, env: Env = Env.empty)( implicit sc: Compiler[F,F] ): F[Json] = - compileAndRunAll(text, name, untypedVars, introspectionLevel, env).compile.toList.flatMap { + compileAndRunSubscription(text, name, untypedVars, introspectionLevel, env).compile.toList.flatMap { case List(j) => j.pure[F] case Nil => M.raiseError(new IllegalStateException("Result stream was empty.")) case js => M.raiseError(new IllegalStateException(s"Result stream contained ${js.length} results; expected exactly one.")) } - def compileAndRunAll(text: String, name: Option[String] = None, untypedVars: Option[Json] = None, introspectionLevel: IntrospectionLevel = Full, env: Env = Env.empty): Stream[F,Json] = - compiler.compile(text, name, untypedVars, introspectionLevel) match { - case Result.Success(operation) => - run(operation.query, operation.rootTpe, env) - case invalid => - Stream.eval(mkInvalidResponse(invalid)) - } + /** + * Compile and run a GraphQL subscription. + * + * Yields a stream of JSON responses containing the results of the subscription. + */ + def compileAndRunSubscription(text: String, name: Option[String] = None, untypedVars: Option[Json] = None, introspectionLevel: IntrospectionLevel = Full, env: Env = Env.empty): Stream[F,Json] = { + val compiled = compiler.compile(text, name, untypedVars, introspectionLevel, env) + Stream.eval(compiled.pure[F]).flatMap(_.flatTraverse(op => interpreter.run(op.query, op.rootTpe, env))).evalMap(mkResponse) + } /** Combine and execute multiple queries. * @@ -80,7 +72,7 @@ abstract class Mapping[F[_]] extends QueryExecutor[F, Json] { * method to implement their specific combinging logic. */ def combineAndRun(queries: List[(Query, Cursor)]): F[Result[List[ProtoJson]]] = - queries.map { case (q, c) => (q, schema.queryType, c) }.traverse((interpreter.runRootEffects _).tupled).map(ProtoJson.combineResults) + queries.map { case (q, c) => (q, schema.queryType, c) }.traverse((interpreter.runOneShot _).tupled).map(ProtoJson.combineResults) /** Yields a `Cursor` focused on the top level operation type of the query */ def defaultRootCursor(query: Query, tpe: Type, parentCursor: Option[Cursor]): F[Result[(Query, Cursor)]] = @@ -127,6 +119,7 @@ abstract class Mapping[F[_]] extends QueryExecutor[F, Json] { } } + /** Yields the `TypeMapping` associated with the provided type, if any. */ def typeMapping(tpe: NamedType): Option[TypeMapping] = typeMappingIndex.get(tpe.name) @@ -136,6 +129,7 @@ abstract class Mapping[F[_]] extends QueryExecutor[F, Json] { val validator: MappingValidator = MappingValidator(this) + /** Yields the `ObjectMapping` associated with the provided context, if any. */ def objectMapping(context: Context): Option[ObjectMapping] = context.tpe.underlyingObject.flatMap { obj => obj.asNamed.flatMap(typeMapping) match { @@ -147,6 +141,7 @@ abstract class Mapping[F[_]] extends QueryExecutor[F, Json] { } } + /** Yields the `FieldMapping` associated with `fieldName` in `context`, if any. */ def fieldMapping(context: Context, fieldName: String): Option[FieldMapping] = objectMapping(context).flatMap(_.fieldMapping(fieldName)).orElse { context.tpe.underlyingObject match { @@ -168,6 +163,7 @@ abstract class Mapping[F[_]] extends QueryExecutor[F, Json] { case rs: RootStream => rs } + /** Yields the `LeafMapping` associated with the provided type, if any. */ def leafMapping[T](tpe: Type): Option[LeafMapping[T]] = typeMappings.collectFirst { case lm@LeafMapping(tpe0, _) if tpe0 =:= tpe => lm.asInstanceOf[LeafMapping[T]] @@ -182,6 +178,7 @@ abstract class Mapping[F[_]] extends QueryExecutor[F, Json] { case tpe => leafMapping(tpe).isDefined } + /** Yields the `Encoder` associated with the provided type, if any. */ def encoderForLeaf(tpe: Type): Option[Encoder[Any]] = encoderMemo.get(tpe.dealias) @@ -299,26 +296,41 @@ abstract class Mapping[F[_]] extends QueryExecutor[F, Json] { def apply(fieldName: String)(effect: (Query, Path, Env) => F[Result[(Query, Cursor)]])(implicit pos: SourcePos, di: DummyImplicit): RootEffect = new RootEffect(fieldName, effect) + /** + * Yields a `RootEffect` which performs an initial effect which leaves the query and default root cursor + * unchanged. + */ + def computeUnit(fieldName: String)(effect: Env => F[Result[Unit]])(implicit pos: SourcePos): RootEffect = + new RootEffect( + fieldName, + (query, path, env) => + (for { + _ <- ResultT(effect(env)) + qc <- ResultT(defaultRootCursor(query, path.rootTpe, None)) + } yield qc.map(_.withEnv(env))).value + ) + /** * Yields a `RootEffect` which performs an initial effect and yields an effect-specific root cursor. */ - def computeCursor(fieldName: String)(effect: (Query, Path, Env) => F[Result[Cursor]])(implicit pos: SourcePos): RootEffect = + def computeCursor(fieldName: String)(effect: (Path, Env) => F[Result[Cursor]])(implicit pos: SourcePos): RootEffect = new RootEffect( fieldName, - (query, path, env) => effect(query, path, env).map(_.map(c => (query, c))) + (query, path, env) => effect(path, env).map(_.map(c => (query, c))) ) /** * Yields a `RootEffect` which performs an initial effect and yields an effect-specific query * which is executed with respect to the default root cursor for the corresponding `Mapping`. */ - def computeQuery(fieldName: String)(effect: (Query, Path, Env) => F[Result[Query]])(implicit pos: SourcePos): RootEffect = + def computeChild(fieldName: String)(effect: (Query, Path, Env) => F[Result[Query]])(implicit pos: SourcePos): RootEffect = new RootEffect( fieldName, (query, path, env) => (for { - q <- ResultT(effect(query, path, env)) - qc <- ResultT(defaultRootCursor(q, path.rootTpe, None)) + child <- ResultT(Query.extractChild(query).toResultOrError("Root query has unexpected shape").pure[F]) + q <- ResultT(effect(child, path, env).map(_.flatMap(Query.substChild(query, _).toResultOrError("Root query has unexpected shape")))) + qc <- ResultT(defaultRootCursor(q, path.rootTpe, None)) } yield qc.map(_.withEnv(env))).value ) } @@ -356,10 +368,10 @@ abstract class Mapping[F[_]] extends QueryExecutor[F, Json] { * * This form of effect is typically used to implement GraphQL subscriptions. */ - def computeCursor(fieldName: String)(effect: (Query, Path, Env) => Stream[F, Result[Cursor]])(implicit pos: SourcePos): RootStream = + def computeCursor(fieldName: String)(effect: (Path, Env) => Stream[F, Result[Cursor]])(implicit pos: SourcePos): RootStream = new RootStream( fieldName, - (query, path, env) => effect(query, path, env).map(_.map(c => (query, c))) + (query, path, env) => effect(path, env).map(_.map(c => (query, c))) ) /** @@ -369,18 +381,20 @@ abstract class Mapping[F[_]] extends QueryExecutor[F, Json] { * * This form of effect is typically used to implement GraphQL subscriptions. */ - def computeQuery(fieldName: String)(effect: (Query, Path, Env) => Stream[F, Result[Query]])(implicit pos: SourcePos): RootStream = + def computeChild(fieldName: String)(effect: (Query, Path, Env) => Stream[F, Result[Query]])(implicit pos: SourcePos): RootStream = new RootStream( fieldName, (query, path, env) => - effect(query, path, env).flatMap(rq => - Stream.eval( - (for { - q <- ResultT(rq.pure[F]) - qc <- ResultT(defaultRootCursor(q, path.rootTpe, None)) - } yield qc.map(_.withEnv(env))).value + Query.extractChild(query).fold(Stream.emit[F, Result[(Query, Cursor)]](Result.internalError("Root query has unexpected shape"))) { child => + effect(child, path, env).flatMap(child0 => + Stream.eval( + (for { + q <- ResultT(child0.flatMap(Query.substChild(query, _).toResultOrError("Root query has unexpected shape")).pure[F]) + qc <- ResultT(defaultRootCursor(q, path.rootTpe, None)) + } yield qc.map(_.withEnv(env))).value + ) ) - ) + } ) } @@ -422,7 +436,7 @@ abstract class Mapping[F[_]] extends QueryExecutor[F, Json] { def withParent(tpe: Type): Delegate = this } - val selectElaborator: SelectElaborator = new SelectElaborator(Map.empty[TypeRef, PartialFunction[Select, Result[Query]]]) + val selectElaborator: SelectElaborator = SelectElaborator.identity lazy val componentElaborator = { val componentMappings = @@ -525,6 +539,23 @@ abstract class Mapping[F[_]] extends QueryExecutor[F, Json] { Result.failure(s"Cannot select field '$fieldName' from leaf type $tpe") } + /** + * Proxy `Cursor` which applies a function to the focus of an underlying `LeafCursor`. + */ + case class FieldTransformCursor[T : ClassTag : TypeName](underlying: Cursor, f: T => Result[T]) extends ProxyCursor(underlying) { + override def withEnv(env: Env): Cursor = new FieldTransformCursor(underlying.withEnv(env), f) + override def field(fieldName: String, resultName: Option[String]): Result[Cursor] = + underlying.field(fieldName, resultName).flatMap { + case l: LeafCursor => + for { + focus <- l.as[T] + ffocus <- f(focus) + } yield l.copy(focus = ffocus) + case _ => + Result.internalError(s"Expected leaf cursor for field $fieldName") + } + } + /** * Construct a GraphQL response from the possibly absent result `data` * and a collection of errors. @@ -546,16 +577,6 @@ abstract class Mapping[F[_]] extends QueryExecutor[F, Json] { case Result.InternalError(err) => M.raiseError(err) case _ => mkResponse(result.toOption, result.toProblems).pure[F] } - - /** - * Construct a GraphQL error response from a `Result`, ignoring any - * right hand side in `result`. - */ - def mkInvalidResponse(result: Result[Operation]): F[Json] = - result match { - case Result.InternalError(err) => M.raiseError(err) - case _ => mkResponse(None, result.toProblems).pure[F] - } } abstract class ComposedMapping[F[_]](implicit val M: MonadThrow[F]) extends Mapping[F] { diff --git a/modules/core/src/main/scala/mappingvalidator.scala b/modules/core/src/main/scala/mappingvalidator.scala index 93cebb9f..6af717ed 100644 --- a/modules/core/src/main/scala/mappingvalidator.scala +++ b/modules/core/src/main/scala/mappingvalidator.scala @@ -173,7 +173,7 @@ trait MappingValidator { protected def validateLeafMapping(lm: LeafMapping[_]): Chain[Failure] = lm.tpe.dealias match { - case ScalarType(_, _)|(_: EnumType)|(_: ListType) => + case (_: ScalarType)|(_: EnumType)|(_: ListType) => Chain.empty // these are valid on construction. Nothing to do. case _ => Chain(InapplicableGraphQLType(lm, "Leaf Type")) } diff --git a/modules/core/src/main/scala/minimizer.scala b/modules/core/src/main/scala/minimizer.scala new file mode 100644 index 00000000..8eec9ab2 --- /dev/null +++ b/modules/core/src/main/scala/minimizer.scala @@ -0,0 +1,128 @@ +// Copyright (c) 2016-2020 Association of Universities for Research in Astronomy, Inc. (AURA) +// For license information see LICENSE or https://opensource.org/licenses/BSD-3-Clause + +package edu.gemini.grackle + +import cats.implicits._ + +object QueryMinimizer { + import Ast._ + + def minimizeText(text: String): Either[String, String] = { + for { + doc <- GraphQLParser.Document.parseAll(text).leftMap(_.expected.toList.mkString(",")) + } yield minimizeDocument(doc) + } + + def minimizeDocument(doc: Document): String = { + import OperationDefinition._ + import OperationType._ + import Selection._ + import Value._ + + def renderDefinition(defn: Definition): String = + defn match { + case e: ExecutableDefinition => renderExecutableDefinition(e) + case _ => "" + } + + def renderExecutableDefinition(ex: ExecutableDefinition): String = + ex match { + case op: OperationDefinition => renderOperationDefinition(op) + case frag: FragmentDefinition => renderFragmentDefinition(frag) + } + + def renderOperationDefinition(op: OperationDefinition): String = + op match { + case qs: QueryShorthand => renderSelectionSet(qs.selectionSet) + case op: Operation => renderOperation(op) + } + + def renderOperation(op: Operation): String = + renderOperationType(op.operationType) + + op.name.map(nme => s" ${nme.value}").getOrElse("") + + renderVariableDefns(op.variables)+ + renderDirectives(op.directives)+ + renderSelectionSet(op.selectionSet) + + def renderOperationType(op: OperationType): String = + op match { + case Query => "query" + case Mutation => "mutation" + case Subscription => "subscription" + } + + def renderDirectives(dirs: List[Directive]): String = + dirs.map { case Directive(name, args) => s"@${name.value}${renderArguments(args)}" }.mkString("") + + def renderVariableDefns(vars: List[VariableDefinition]): String = + vars match { + case Nil => "" + case _ => + vars.map { + case VariableDefinition(name, tpe, default, dirs) => + s"$$${name.value}:${tpe.name}${default.map(v => s"=${renderValue(v)}").getOrElse("")}${renderDirectives(dirs)}" + }.mkString("(", ",", ")") + } + + def renderSelectionSet(sels: List[Selection]): String = + sels match { + case Nil => "" + case _ => sels.map(renderSelection).mkString("{", ",", "}") + } + + def renderSelection(sel: Selection): String = + sel match { + case f: Field => renderField(f) + case s: FragmentSpread => renderFragmentSpread(s) + case i: InlineFragment => renderInlineFragment(i) + } + + def renderField(f: Field) = { + f.alias.map(a => s"${a.value}:").getOrElse("")+ + f.name.value+ + renderArguments(f.arguments)+ + renderDirectives(f.directives)+ + renderSelectionSet(f.selectionSet) + } + + def renderArguments(args: List[(Name, Value)]): String = + args match { + case Nil => "" + case _ => args.map { case (n, v) => s"${n.value}:${renderValue(v)}" }.mkString("(", ",", ")") + } + + def renderInputObject(args: List[(Name, Value)]): String = + args match { + case Nil => "" + case _ => args.map { case (n, v) => s"${n.value}:${renderValue(v)}" }.mkString("{", ",", "}") + } + + def renderTypeCondition(tpe: Type): String = + s"on ${tpe.name}" + + def renderFragmentDefinition(frag: FragmentDefinition): String = + s"fragment ${frag.name.value} ${renderTypeCondition(frag.typeCondition)}${renderDirectives(frag.directives)}${renderSelectionSet(frag.selectionSet)}" + + def renderFragmentSpread(spread: FragmentSpread): String = + s"...${spread.name.value}${renderDirectives(spread.directives)}" + + def renderInlineFragment(frag: InlineFragment): String = + s"...${frag.typeCondition.map(renderTypeCondition).getOrElse("")}${renderDirectives(frag.directives)}${renderSelectionSet(frag.selectionSet)}" + + def renderValue(v: Value): String = + v match { + case Variable(name) => s"$$${name.value}" + case IntValue(value) => value.toString + case FloatValue(value) => value.toString + case StringValue(value) => s""""$value"""" + case BooleanValue(value) => value.toString + case NullValue => "null" + case EnumValue(name) => name.value + case ListValue(values) => values.map(renderValue).mkString("[", ",", "]") + case ObjectValue(fields) => renderInputObject(fields) + } + + doc.map(renderDefinition).mkString(",") + } +} diff --git a/modules/core/src/main/scala/operation.scala b/modules/core/src/main/scala/operation.scala index 0ddeb056..c846f214 100644 --- a/modules/core/src/main/scala/operation.scala +++ b/modules/core/src/main/scala/operation.scala @@ -7,20 +7,40 @@ import syntax._ import Query._ sealed trait UntypedOperation { + val name: Option[String] val query: Query val variables: UntypedVarDefs + val directives: List[Directive] def rootTpe(schema: Schema): Result[NamedType] = this match { - case UntypedOperation.UntypedQuery(_, _) => schema.queryType.success - case UntypedOperation.UntypedMutation(_, _) => schema.mutationType.toResult("No mutation type defined in this schema.") - case UntypedOperation.UntypedSubscription(_, _) => schema.subscriptionType.toResult("No subscription type defined in this schema.") + case _: UntypedOperation.UntypedQuery => schema.queryType.success + case _: UntypedOperation.UntypedMutation => schema.mutationType.toResult("No mutation type defined in this schema.") + case _: UntypedOperation.UntypedSubscription => schema.subscriptionType.toResult("No subscription type defined in this schema.") } } object UntypedOperation { - case class UntypedQuery(query: Query, variables: UntypedVarDefs) extends UntypedOperation - case class UntypedMutation(query: Query, variables: UntypedVarDefs) extends UntypedOperation - case class UntypedSubscription(query: Query, variables: UntypedVarDefs) extends UntypedOperation + case class UntypedQuery( + name: Option[String], + query: Query, + variables: UntypedVarDefs, + directives: List[Directive] + ) extends UntypedOperation + case class UntypedMutation( + name: Option[String], + query: Query, + variables: UntypedVarDefs, + directives: List[Directive] + ) extends UntypedOperation + case class UntypedSubscription( + name: Option[String], + query: Query, + variables: UntypedVarDefs, + directives: List[Directive] + ) extends UntypedOperation } -case class Operation(query: Query, rootTpe: NamedType) - +case class Operation( + query: Query, + rootTpe: NamedType, + directives: List[Directive] +) diff --git a/modules/core/src/main/scala/parser.scala b/modules/core/src/main/scala/parser.scala index 6be0737a..20483d72 100644 --- a/modules/core/src/main/scala/parser.scala +++ b/modules/core/src/main/scala/parser.scala @@ -3,7 +3,7 @@ package edu.gemini.grackle -import cats.parse.{Parser, Parser0} +import cats.parse.{LocationMap, Parser, Parser0} import cats.parse.Parser._ import cats.parse.Numbers._ import cats.parse.Rfc5234.{cr, crlf, digit, hexdig, lf} @@ -70,8 +70,8 @@ object GraphQLParser { def directiveDefinition(desc: Option[Ast.Value.StringValue]): Parser[Ast.DirectiveDefinition] = ((keyword("directive") *> keyword("@") *> Name) ~ - ArgumentsDefinition ~ (keyword("repeatable").? <* keyword("on")) ~ DirectiveLocations).map { - case (((name, args), rpt), locs) => Ast.DirectiveDefinition(name, desc.map(_.value), args, rpt.isDefined, locs) + ArgumentsDefinition.? ~ (keyword("repeatable").? <* keyword("on")) ~ DirectiveLocations).map { + case (((name, args), rpt), locs) => Ast.DirectiveDefinition(name, desc.map(_.value), args.getOrElse(Nil), rpt.isDefined, locs) } SchemaDefinition | @@ -81,8 +81,8 @@ object GraphQLParser { } lazy val RootOperationTypeDefinition: Parser[Ast.RootOperationTypeDefinition] = - (OperationType, keyword(":"), NamedType).mapN { - case (optpe, _, tpe) => Ast.RootOperationTypeDefinition(optpe, tpe) + (OperationType ~ keyword(":") ~ NamedType ~ Directives).map { + case (((optpe, _), tpe), dirs) => Ast.RootOperationTypeDefinition(optpe, tpe, dirs) } @@ -260,8 +260,8 @@ object GraphQLParser { parens(VariableDefinition.rep0) lazy val VariableDefinition: Parser[Ast.VariableDefinition] = - ((Variable <* keyword(":")) ~ Type ~ DefaultValue.?).map { - case ((v, tpe), dv) => Ast.VariableDefinition(v.name, tpe, dv) + ((Variable <* keyword(":")) ~ Type ~ DefaultValue.? ~ Directives.?).map { + case (((v, tpe), dv), dirs) => Ast.VariableDefinition(v.name, tpe, dv, dirs.getOrElse(Nil)) } lazy val Variable: Parser[Ast.Value.Variable] = @@ -305,6 +305,22 @@ object GraphQLParser { case (h, t) => Ast.Name((h :: t).mkString) } } + + def toResult[T](text: String, pr: Either[Parser.Error, T]): Result[T] = + Result.fromEither(pr.leftMap { e => + val lm = LocationMap(text) + lm.toLineCol(e.failedAtOffset) match { + case Some((row, col)) => + lm.getLine(row) match { + case Some(line) => + s"""Parse error at line $row column $col + |$line + |${List.fill(col)(" ").mkString}^""".stripMargin + case None => "Malformed query" //This is probably a bug in Cats Parse as it has given us the (row, col) index + } + case None => "Truncated query" + } + }) } object CommentedText { diff --git a/modules/core/src/main/scala/problem.scala b/modules/core/src/main/scala/problem.scala index 0d47408a..a568d8bd 100644 --- a/modules/core/src/main/scala/problem.scala +++ b/modules/core/src/main/scala/problem.scala @@ -14,7 +14,6 @@ final case class Problem( path: List[String] = Nil, extension: Option[JsonObject] = None, ) { - override def toString = { lazy val pathText: String = diff --git a/modules/core/src/main/scala/query.scala b/modules/core/src/main/scala/query.scala index 2141abdd..7c4762e2 100644 --- a/modules/core/src/main/scala/query.scala +++ b/modules/core/src/main/scala/query.scala @@ -9,7 +9,6 @@ import cats.implicits._ import cats.kernel.Order import syntax._ -import Cursor.Env import Query._ /** GraphQL query Algebra */ @@ -27,14 +26,37 @@ sealed trait Query { } object Query { - /** Select field `name` given arguments `args` and continue with `child` */ - case class Select(name: String, args: List[Binding], child: Query = Empty) extends Query { - def eliminateArgs(elim: Query => Query): Query = copy(args = Nil, child = elim(child)) + /** Select field `name` possibly aliased, and continue with `child` */ + case class Select(name: String, alias: Option[String], child: Query) extends Query { + def resultName: String = alias.getOrElse(name) def render = { + val rname = s"${alias.map(a => s"$a:").getOrElse("")}$name" + val rchild = if(child == Empty) "" else s" { ${child.render} }" + s"$rname$rchild" + } + } + + object Select { + def apply(name: String): Select = + new Select(name, None, Empty) + + def apply(name: String, alias: Option[String]): Select = + new Select(name, alias, Empty) + + def apply(name: String, child: Query): Select = + new Select(name, None, child) + } + + /** Precursor of a `Select` node, containing uncompiled field arguments and directives. */ + case class UntypedSelect(name: String, alias: Option[String], args: List[Binding], directives: List[Directive], child: Query) extends Query { + def resultName: String = alias.getOrElse(name) + + def render = { + val rname = s"${alias.map(a => s"$a:").getOrElse("")}$name" val rargs = if(args.isEmpty) "" else s"(${args.map(_.render).mkString(", ")})" val rchild = if(child == Empty) "" else s" { ${child.render} }" - s"$name$rargs$rchild" + s"$rname$rargs$rchild" } } @@ -84,31 +106,22 @@ object Query { def render = s"" } - /** - * Wraps the result of `child` as a field named `name` of an enclosing object. - */ - case class Wrap(name: String, child: Query) extends Query { - def render = { - val rchild = if(child == Empty) "" else s" { ${child.render} }" - s"[$name]$rchild" - } - } - - /** - * Rename the topmost field of `sel` to `name`. + /** Representation of a fragment spread prior to comilation. + * + * During compilation this node will be replaced by its definition, guarded by a `Narrow` + * corresponding to the type condition of the fragment. */ - case class Rename(name: String, child: Query) extends Query { - def render = s"" + case class UntypedFragmentSpread(name: String, directives: List[Directive]) extends Query { + def render = s"" } - /** - * Untyped precursor of `Narrow`. + /** Representation of an inline fragment prior to comilation. * - * Trees of this type will be replaced by a corresponding `Narrow` by - * `SelectElaborator`. + * During compilation this node will be replaced by its child, guarded by a `Narrow` + * corresponding to the type condition of the fragment, if any. */ - case class UntypedNarrow(tpnme: String, child: Query) extends Query { - def render = s"" + case class UntypedInlineFragment(tpnme: Option[String], directives: List[Directive], child: Query) extends Query { + def render = s" s"on $tc ").getOrElse("")} ${child.render}>" } /** @@ -118,11 +131,6 @@ object Query { def render = s"" } - /** Skips/includes the continuation `child` depending on the value of `cond` */ - case class Skip(sense: Boolean, cond: Value, child: Query) extends Query { - def render = s"" - } - /** Limits the results of list-producing continuation `child` to `num` elements */ case class Limit(num: Int, child: Query) extends Query { def render = s"" @@ -181,9 +189,9 @@ object Query { def subst(term: Term[T]): OrderSelection[T] = copy(term = term) } - /** Computes the number of top-level elements of `child` as field `name` */ - case class Count(name: String, child: Query) extends Query { - def render = s"$name:count { ${child.render} }" + /** Computes the number of top-level elements of `child` */ + case class Count(child: Query) extends Query { + def render = s"count { ${child.render} }" } /** @@ -194,11 +202,6 @@ object Query { def render = s"" } - /** A placeholder for a skipped node */ - case object Skipped extends Query { - def render = "" - } - /** The terminal query */ case object Empty extends Query { def render = "" @@ -212,44 +215,17 @@ object Query { type VarDefs = List[InputValue] type Vars = Map[String, (Type, Value)] - case class UntypedVarDef(name: String, tpe: Ast.Type, default: Option[Value]) - - /** - * Extractor for nested Rename/Select patterns in the query algebra - * - * PossiblyRenamedSelect is an extractor/constructor for a Select node - * possibly wrapped in a Rename node so that they can be handled together - * conveniently. - */ - object PossiblyRenamedSelect { - def apply(sel: Select, resultName: String): Query = sel match { - case Select(`resultName`, _, _) => sel - case _ => Rename(resultName, sel) - } + /** Precursor of a variable definition before compilation */ + case class UntypedVarDef(name: String, tpe: Ast.Type, default: Option[Value], directives: List[Directive]) - def apply(sel: Select, resultName: Option[String]): Query = resultName match { - case Some(resultName) => Rename(resultName, sel) - case None => sel - } - - def unapply(q: Query): Option[(Select, String)] = - q match { - case Rename(name, sel: Select) => Some((sel, name)) - case sel: Select => Some((sel, sel.name)) - case _ => None - } - } + /** Precursor of a fragment definition before compilation */ + case class UntypedFragment(name: String, tpnme: String, directives: List[Directive], child: Query) def renameRoot(q: Query, rootName: String): Option[Query] = q match { - case Rename(_, sel@Select(`rootName`, _, _)) => Some(sel) - case r@Rename(`rootName`, _) => Some(r) - case Rename(_, sel: Select) => Some(Rename(rootName, sel)) - case sel@Select(`rootName`, _, _) => Some(sel) - case sel: Select => Some(Rename(rootName, sel)) - case w@Wrap(`rootName`, _) => Some(w) - case w: Wrap => Some(w.copy(name = rootName)) - case e@Environment(_, child) => renameRoot(child, rootName).map(rc => e.copy(child = rc)) - case t@TransformCursor(_, child) => renameRoot(child, rootName).map(rc => t.copy(child = rc)) + case sel: Select if sel.resultName == rootName => Some(sel) + case sel: Select => Some(sel.copy(alias = Some(rootName))) + case e@Environment(_, child) => renameRoot(child, rootName).map(rc => e.copy(child = rc)) + case t@TransformCursor(_, child) => renameRoot(child, rootName).map(rc => t.copy(child = rc)) case _ => None } @@ -258,19 +234,42 @@ object Query { * if it is unique, `None` otherwise. */ def rootName(q: Query): Option[(String, Option[String])] = { - def loop(q: Query, alias: Option[String]): Option[(String, Option[String])] = + def loop(q: Query): Option[(String, Option[String])] = q match { - case Select(name, _, _) => Some((name, alias)) - case Wrap(name, _) => Some((name, alias)) - case Count(name, _) => Some((name, alias)) - case Rename(name, child) => loop(child, alias.orElse(Some(name))) - case Environment(_, child) => loop(child, alias) - case TransformCursor(_, child) => loop(child, alias) - case _ => None + case UntypedSelect(name, alias, _, _, _) => Some((name, alias)) + case Select(name, alias, _) => Some((name, alias)) + case Environment(_, child) => loop(child) + case TransformCursor(_, child) => loop(child) + case _ => None + } + loop(q) + } + + /** + * Computes the possibly aliased result name of the supplied query + * if it is unique, `None` otherwise. + */ + def resultName(q: Query): Option[String] = { + def loop(q: Query): Option[String] = + q match { + case UntypedSelect(name, alias, _, _, _) => Some(alias.getOrElse(name)) + case Select(name, alias, _) => Some(alias.getOrElse(name)) + case Environment(_, child) => loop(child) + case TransformCursor(_, child) => loop(child) + case _ => None } - loop(q, None) + loop(q) } + /** + * Renames the root of `target` to match `source` if possible. + */ + def alignResultName(source: Query, target: Query): Option[Query] = + for { + nme <- resultName(source) + res <- renameRoot(target, nme) + } yield res + /** * Yields a list of the top level queries of the supplied, possibly * grouped query. @@ -282,15 +281,13 @@ object Query { } /** - * Returns the top-level field selections of the supplied query. + * Yields the top-level field selections of the supplied query. */ def children(q: Query): List[Query] = { def loop(q: Query): List[Query] = q match { + case UntypedSelect(_, _, _, _, child) => ungroup(child) case Select(_, _, child) => ungroup(child) - case Wrap(_, child) => ungroup(child) - case Count(_, child) => ungroup(child) - case Rename(_, child) => loop(child) case Environment(_, child) => loop(child) case TransformCursor(_, child) => loop(child) case _ => Nil @@ -298,6 +295,39 @@ object Query { loop(q) } + /** + * Yields the top-level field selection of the supplied Query + * if it is unique, `None` otherwise. + */ + def extractChild(query: Query): Option[Query] = { + def loop(q: Query): Option[Query] = + q match { + case UntypedSelect(_, _, _, _, child) => Some(child) + case Select(_, _, child) => Some(child) + case Environment(_, child) => loop(child) + case TransformCursor(_, child) => loop(child) + case _ => None + } + loop(query) + } + + /** + * Yields the supplied query with its the top-level field selection + * of the supplied replaced with `newChild` if it is unique, `None` + * otherwise. + */ + def substChild(query: Query, newChild: Query): Option[Query] = { + def loop(q: Query): Option[Query] = + q match { + case u: UntypedSelect => Some(u.copy(child = newChild)) + case s: Select => Some(s.copy(child = newChild)) + case e@Environment(_, child) => loop(child).map(c => e.copy(child = c)) + case t@TransformCursor(_, child) => loop(child).map(c => t.copy(child = c)) + case _ => None + } + loop(query) + } + /** * True if `fieldName` is a top-level selection of the supplied query, * false otherwise. @@ -305,8 +335,8 @@ object Query { def hasField(query: Query, fieldName: String): Boolean = { def loop(q: Query): Boolean = ungroup(q).exists { + case UntypedSelect(`fieldName`, _, _, _, _) => true case Select(`fieldName`, _, _) => true - case Rename(_, child) => loop(child) case Environment(_, child) => loop(child) case TransformCursor(_, child) => loop(child) case _ => false @@ -319,30 +349,40 @@ object Query { * the supplied query. */ def fieldAlias(query: Query, fieldName: String): Option[String] = { - def loop(q: Query, alias: Option[String]): Option[String] = + def loop(q: Query): Option[String] = ungroup(q).collectFirstSome { - case Select(`fieldName`, _, _) => alias - case Wrap(`fieldName`, _) => alias - case Count(`fieldName`, _) => alias - case Rename(alias, child) => loop(child, Some(alias)) - case Environment(_, child) => loop(child, alias) - case TransformCursor(_, child) => loop(child, alias) + case UntypedSelect(`fieldName`, alias, _, _, _) => alias + case Select(`fieldName`, alias, _) => alias + case Environment(_, child) => loop(child) + case TransformCursor(_, child) => loop(child) case _ => None } - loop(query, None) + loop(query) + } + + /** + * Tranform the children of `query` using the supplied function. + */ + def mapFields(query: Query)(f: Query => Query): Query = { + def loop(q: Query): Query = + q match { + case Group(qs) => Group(qs.map(loop)) + case s: Select => f(s) + case e@Environment(_, child) => e.copy(child = loop(child)) + case t@TransformCursor(_, child) => t.copy(child = loop(child)) + case other => other + } + loop(query) } /** * Tranform the children of `query` using the supplied function. */ - def mapFields(query: Query)(f: Query => Result[Query]): Result[Query] = { + def mapFieldsR(query: Query)(f: Query => Result[Query]): Result[Query] = { def loop(q: Query): Result[Query] = q match { case Group(qs) => qs.traverse(loop).map(Group(_)) case s: Select => f(s) - case w: Wrap => f(w) - case c: Count => f(c) - case r@Rename(_, child) => loop(child).map(ec => r.copy(child = ec)) case e@Environment(_, child) => loop(child).map(ec => e.copy(child = ec)) case t@TransformCursor(_, child) => loop(child).map(ec => t.copy(child = ec)) case other => other.success @@ -426,11 +466,11 @@ object Query { case Nil => Nil case paths => val oneElemPaths = paths.filter(_.sizeCompare(1) == 0).distinct - val oneElemQueries: List[Query] = oneElemPaths.map(p => Select(p.head, Nil, Empty)) + val oneElemQueries: List[Query] = oneElemPaths.map(p => Select(p.head)) val multiElemPaths = paths.filter(_.length > 1).distinct val grouped: List[Query] = multiElemPaths.groupBy(_.head).toList.map { case (fieldName, suffixes) => - Select(fieldName, Nil, mergeQueries(mkPathQuery(suffixes.map(_.tail).filterNot(_.isEmpty)))) + Select(fieldName, mergeQueries(mkPathQuery(suffixes.map(_.tail).filterNot(_.isEmpty)))) } oneElemQueries ++ grouped } @@ -453,15 +493,14 @@ object Query { } val flattened = flattenLevel(qs) - val (selects, rest) = flattened.partition { case PossiblyRenamedSelect(_, _) => true ; case _ => false } + val (selects, rest) = flattened.partitionMap { case sel: Select => Left(sel) ; case other => Right(other) } val mergedSelects = - selects.groupBy { case PossiblyRenamedSelect(Select(fieldName, _, _), resultName) => (fieldName, resultName) ; case _ => throw new MatchError("Impossible") }.values.map { rsels => - val PossiblyRenamedSelect(Select(fieldName, _, _), resultName) = rsels.head : @unchecked - val sels = rsels.collect { case PossiblyRenamedSelect(sel, _) => sel } + selects.groupBy { case Select(fieldName, resultName, _) => (fieldName, resultName) }.values.map { sels => + val Select(fieldName, resultName, _) = sels.head : @unchecked val children = sels.map(_.child) val merged = mergeQueries(children) - PossiblyRenamedSelect(Select(fieldName, Nil, merged), resultName) + Select(fieldName, resultName, merged) } Group(rest ++ mergedSelects) diff --git a/modules/core/src/main/scala/queryinterpreter.scala b/modules/core/src/main/scala/queryinterpreter.scala index 74937fee..1f15599a 100644 --- a/modules/core/src/main/scala/queryinterpreter.scala +++ b/modules/core/src/main/scala/queryinterpreter.scala @@ -14,39 +14,28 @@ import fs2.Stream import io.circe.Json import syntax._ -import Cursor.{Context, Env, ListTransformCursor} +import Cursor.ListTransformCursor import Query._ import QueryInterpreter.ProtoJson import ProtoJson._ class QueryInterpreter[F[_]](mapping: Mapping[F]) { - import mapping.{mkResponse, M, RootCursor, RootEffect, RootStream} + import mapping.{M, RootCursor, RootEffect, RootStream} /** Interpret `query` with expected type `rootTpe`. * * The query is fully interpreted, including deferred or staged * components. * - * The resulting Json value should include standard GraphQL error - * information in the case of failure. + * GraphQL errors are accumulated in the result. */ - def run(query: Query, rootTpe: Type, env: Env): Stream[F,Json] = - runRoot(query, rootTpe, env).evalMap(mkResponse) - - /** Interpret `query` with expected type `rootTpe`. - * - * The query is fully interpreted, including deferred or staged - * components. - * - * Errors are accumulated on the `Left` of the result. - */ - def runRoot(query: Query, rootTpe: Type, env: Env): Stream[F,Result[Json]] = { + def run(query: Query, rootTpe: Type, env: Env): Stream[F, Result[Json]] = { val rootCursor = RootCursor(Context(rootTpe), None, env) val mergedResults = if(mapping.schema.subscriptionType.map(_ =:= rootTpe).getOrElse(false)) - runRootStream(query, rootTpe, rootCursor) + runSubscription(query, rootTpe, rootCursor) else - Stream.eval(runRootEffects(query, rootTpe, rootCursor)) + Stream.eval(runOneShot(query, rootTpe, rootCursor)) (for { pvalue <- ResultT(mergedResults) @@ -54,7 +43,10 @@ class QueryInterpreter[F[_]](mapping: Mapping[F]) { } yield value).value } - def runRootStream(query: Query, rootTpe: Type, rootCursor: Cursor): Stream[F, Result[ProtoJson]] = + /** + * Run a subscription query yielding a stream of results. + */ + def runSubscription(query: Query, rootTpe: Type, rootCursor: Cursor): Stream[F, Result[ProtoJson]] = ungroup(query) match { case Nil => Result(ProtoJson.fromJson(Json.Null)).pure[Stream[F, *]] case List(root) => @@ -65,13 +57,16 @@ class QueryInterpreter[F[_]](mapping: Mapping[F]) { effect(root, rootTpe / fieldName, rootCursor.fullEnv.addFromQuery(root)).map(_.flatMap { // TODO Rework in terms of cursor case (q, c) => runValue(q, rootTpe, c) }) - ).getOrElse(Result.failure("EffectMapping required for subscriptions").pure[Stream[F, *]]) + ).getOrElse(Result.internalError("EffectMapping required for subscriptions").pure[Stream[F, *]]) case _ => - Result.failure("Only one root selection permitted for subscriptions").pure[Stream[F, *]] + Result.internalError("Only one root selection permitted for subscriptions").pure[Stream[F, *]] } - def runRootEffects(query: Query, rootTpe: Type, rootCursor: Cursor): F[Result[ProtoJson]] = { + /** + * Run a non-subscription query yielding a single result. + */ + def runOneShot(query: Query, rootTpe: Type, rootCursor: Cursor): F[Result[ProtoJson]] = { case class PureQuery(query: Query) case class EffectfulQuery(query: Query, rootEffect: RootEffect) @@ -83,7 +78,7 @@ class QueryInterpreter[F[_]](mapping: Mapping[F]) { } if(hasRootStream) - Result.failure("RootStream only permitted in subscriptions").pure[F].widen + Result.internalError("RootStream only permitted in subscriptions").pure[F].widen else { val (effectfulQueries, pureQueries) = ungrouped.partitionMap { query => (for { @@ -195,7 +190,7 @@ class QueryInterpreter[F[_]](mapping: Mapping[F]) { fields <- runFields(child, tp1, c) } yield fields - case (Introspect(schema, PossiblyRenamedSelect(Select("__typename", Nil, Empty), resultName)), tpe: NamedType) => + case (Introspect(schema, s@Select("__typename", _, Empty)), tpe: NamedType) => (tpe match { case o: ObjectType => Some(o.name) case i: InterfaceType => @@ -209,49 +204,48 @@ class QueryInterpreter[F[_]](mapping: Mapping[F]) { case _ => None }) match { case Some(name) => - List((resultName, ProtoJson.fromJson(Json.fromString(name)))).success + List((s.resultName, ProtoJson.fromJson(Json.fromString(name)))).success case None => Result.failure(s"'__typename' cannot be applied to non-selectable type '$tpe'") } - case (PossiblyRenamedSelect(sel, resultName), NullableType(tpe)) => + case (sel: Select, NullableType(tpe)) => cursor.asNullable.sequence.map { rc => for { c <- rc fields <- runFields(sel, tpe, c) } yield fields - }.getOrElse(List((resultName, ProtoJson.fromJson(Json.Null))).success) + }.getOrElse(List((sel.resultName, ProtoJson.fromJson(Json.Null))).success) + + case (sel@Select(_, _, Count(Select(countName, _, _))), _) => + def size(c: Cursor): Result[Int] = + if (c.isList) c.asList(Iterator).map(_.size) + else 1.success + + for { + c0 <- cursor.field(countName, None) + count <- if (c0.isNullable) c0.asNullable.flatMap(_.map(size).getOrElse(0.success)) + else size(c0) + } yield List((sel.resultName, ProtoJson.fromJson(Json.fromInt(count)))) - case (PossiblyRenamedSelect(Select(fieldName, _, child), resultName), _) => + case (sel@Select(fieldName, resultName, child), _) => val fieldTpe = tpe.field(fieldName).getOrElse(ScalarType.AttributeType) for { - c <- cursor.field(fieldName, Some(resultName)) + c <- cursor.field(fieldName, resultName) value <- runValue(child, fieldTpe, c) - } yield List((resultName, value)) + } yield List((sel.resultName, value)) - case (Rename(resultName, Wrap(_, child)), tpe) => - runFields(Wrap(resultName, child), tpe, cursor) + case (c@Component(_, _, cont), _) => + for { + componentName <- resultName(cont).toResultOrError("Join continuation has unexpected shape") + value <- runValue(c, tpe, cursor) + } yield List((componentName, ProtoJson.select(value, componentName))) - case (Wrap(fieldName, child), tpe) => + case (e@Effect(_, cont), _) => for { - value <- runValue(child, tpe, cursor) - } yield List((fieldName, value)) - - case (Rename(resultName, Count(_, child)), tpe) => - runFields(Count(resultName, child), tpe, cursor) - - case (Count(fieldName, Select(countName, _, _)), _) => - cursor.field(countName, None).flatMap { c0 => - if (c0.isNullable) - c0.asNullable.flatMap { - case None => 0.success - case Some(c1) => - if (c1.isList) c1.asList(Iterator).map(_.size) - else 1.success - } - else if (c0.isList) c0.asList(Iterator).map(_.size) - else 1.success - }.map { value => List((fieldName, ProtoJson.fromJson(Json.fromInt(value)))) } + effectName <- resultName(cont).toResultOrError("Effect continuation has unexpected shape") + value <- runValue(e, tpe, cursor) + } yield List((effectName, value)) case (Group(siblings), _) => siblings.flatTraverse(query => runFields(query, tpe, cursor)) @@ -369,17 +363,11 @@ class QueryInterpreter[F[_]](mapping: Mapping[F]) { if (!cursorCompatible(tpe, cursor.tpe)) Result.internalError(s"Mismatched query and cursor type in runValue: $tpe ${cursor.tpe}") else { - def mkResult[T](ot: Option[T]): Result[T] = ot match { - case Some(t) => t.success - case None => Result.internalError(s"Join continuation has unexpected shape") - } - (query, tpe.dealias) match { case (Environment(childEnv: Env, child: Query), tpe) => runValue(child, tpe, cursor.withEnv(childEnv)) - case (Wrap(_, Component(_, _, _)), ListType(tpe)) => - // Keep the wrapper with the component when going under the list + case (Component(_, _, _), ListType(tpe)) => cursor.asList(Iterator) match { case Result.Success(ic) => val builder = Vector.newBuilder[ProtoJson] @@ -392,32 +380,33 @@ class QueryInterpreter[F[_]](mapping: Mapping[F]) { } } ProtoJson.fromValues(builder.result()).success + case Result.Warning(ps, _) => Result.Failure(ps) case fail@Result.Failure(_) => fail case err@Result.InternalError(_) => err - case Result.Warning(ps, _) => Result.Failure(ps) } - case (Wrap(fieldName, child), _) => - for { - pvalue <- runValue(child, tpe, cursor) - } yield ProtoJson.fromFields(List((fieldName, pvalue))) - - case (Component(mapping, join, PossiblyRenamedSelect(child, resultName)), _) => + case (Component(mapping, join, child), _) => join(child, cursor).flatMap { case Group(conts) => - conts.traverse { case cont => - for { - componentName <- mkResult(rootName(cont).map(nme => nme._2.getOrElse(nme._1))) - } yield - ProtoJson.select( - ProtoJson.component(mapping, cont, cursor), - componentName - ) - }.map(ProtoJson.fromValues) + for { + childName <- resultName(child).toResultOrError("Join child has unexpected shape") + elems <- conts.traverse { case cont => + for { + componentName <- resultName(cont).toResultOrError("Join continuation has unexpected shape") + } yield + ProtoJson.select( + ProtoJson.component(mapping, cont, cursor), + componentName + ) + } + } yield + ProtoJson.fromFields( + List(childName -> ProtoJson.fromValues(elems)) + ) case cont => for { - renamedCont <- mkResult(renameRoot(cont, resultName)) + renamedCont <- alignResultName(child, cont).toResultOrError("Join continuation has unexpected shape") } yield ProtoJson.component(mapping, renamedCont, cursor) } diff --git a/modules/core/src/main/scala/result.scala b/modules/core/src/main/scala/result.scala index 9c2b68ca..14f0f746 100644 --- a/modules/core/src/main/scala/result.scala +++ b/modules/core/src/main/scala/result.scala @@ -37,7 +37,13 @@ sealed trait Result[+T] { def flatMap[U](f: T => Result[U]): Result[U] = this match { case Result.Success(value) => f(value) - case Result.Warning(problems, value) => f(value).withProblems(problems) + case Result.Warning(problems, value) => + f(value) match { + case Result.Success(fv) => Result.Warning(problems, fv) + case Result.Warning(fps, fv) => Result.Warning(problems ++ fps, fv) + case Result.Failure(fps) => Result.Failure(problems ++ fps) + case other@Result.InternalError(_) => other + } case other@Result.Failure(_) => other case other@Result.InternalError(_) => other } @@ -154,11 +160,16 @@ object Result extends ResultInstances { final case class Failure(problems: NonEmptyChain[Problem]) extends Result[Nothing] final case class InternalError(error: Throwable) extends Result[Nothing] - val unit: Result[Unit] = - apply(()) - + /** Yields a Success with the given value. */ def apply[A](a: A): Result[A] = Success(a) + /** Yields a Success with the given value. */ + def pure[A](a: A): Result[A] = Success(a) + + /** Yields a Success with unit value. */ + val unit: Result[Unit] = pure(()) + + /** Yields a Success with the given value. */ def success[A](a: A): Result[A] = Success(a) def warning[A](warning: Problem, value: A): Result[A] = @@ -269,8 +280,8 @@ trait ResultInstances extends ResultInstances0 { fn(a) match { case err@Result.InternalError(_) => err case Result.Success(a) => loop(Result.Warning(ps, a)) - case Result.Failure(ps0) => Result.Failure(ps0 ++ ps) - case Result.Warning(ps0, a) => loop(Result.Warning(ps0 ++ ps, a)) + case Result.Failure(ps0) => Result.Failure(ps ++ ps0) + case Result.Warning(ps0, a) => loop(Result.Warning(ps ++ ps0, a)) } } loop(fn(a)) @@ -295,7 +306,13 @@ trait ResultInstances extends ResultInstances0 { def ap[A, B](ff: Result[A => B])(fa: Result[A]): Result[B] = fa match { case err@Result.InternalError(_) => err - case fail@Result.Failure(_) => fail + case fail@Result.Failure(ps) => + ff match { + case err@Result.InternalError(_) => err + case Result.Success(_) => fail + case Result.Failure(ps0) => Result.Failure(ps0 ++ ps) + case Result.Warning(ps0, _) => Result.Failure(ps0 ++ ps) + } case Result.Success(a) => ff match { case err@Result.InternalError(_) => err diff --git a/modules/core/src/main/scala/schema.scala b/modules/core/src/main/scala/schema.scala index 4a9a04b2..08d3120e 100644 --- a/modules/core/src/main/scala/schema.scala +++ b/modules/core/src/main/scala/schema.scala @@ -3,14 +3,15 @@ package edu.gemini.grackle -import cats.parse.Parser import cats.implicits._ import io.circe.Json import org.tpolecat.sourcepos.SourcePos import syntax._ -import Ast.{InterfaceTypeDefinition, ObjectTypeDefinition, TypeDefinition, UnionTypeDefinition} +import Ast.{DirectiveLocation, InterfaceTypeDefinition, ObjectTypeDefinition, TypeDefinition, UnionTypeDefinition} +import Query._ import ScalarType._ +import UntypedOperation._ import Value._ /** @@ -26,7 +27,7 @@ trait Schema { def types: List[NamedType] /** The directives defined by this `Schema`. */ - def directives: List[Directive] + def directives: List[DirectiveDef] /** A reference by name to a type defined by this `Schema`. * @@ -59,7 +60,7 @@ trait Schema { */ def defaultSchemaType: NamedType = { def mkRootDef(fieldName: String)(tpe: NamedType): Field = - Field(fieldName, None, Nil, tpe, false, None) + Field(fieldName, None, Nil, tpe, Nil) ObjectType( name = "Schema", @@ -70,7 +71,8 @@ trait Schema { definition("Mutation").map(mkRootDef("mutation")), definition("Subscription").map(mkRootDef("subscription")) ).flatten, - interfaces = Nil + interfaces = Nil, + directives = Nil ) } @@ -98,13 +100,13 @@ trait Schema { def schemaType: NamedType = definition("Schema").getOrElse(defaultSchemaType) /** The type of queries defined by this `Schema`*/ - def queryType: NamedType = schemaType.field("query").flatMap(_.asNamed).get + def queryType: NamedType = schemaType.field("query").flatMap(_.nonNull.asNamed).get /** The type of mutations defined by this `Schema`*/ - def mutationType: Option[NamedType] = schemaType.field("mutation").flatMap(_.asNamed) + def mutationType: Option[NamedType] = schemaType.field("mutation").flatMap(_.nonNull.asNamed) /** The type of subscriptions defined by this `Schema`*/ - def subscriptionType: Option[NamedType] = schemaType.field("subscription").flatMap(_.asNamed) + def subscriptionType: Option[NamedType] = schemaType.field("subscription").flatMap(_.nonNull.asNamed) /** True if the supplied type is one of the Query, Mutation or Subscription root types, false otherwise */ def isRootType(tpe: Type): Boolean = @@ -143,9 +145,9 @@ sealed trait Type extends Product { def <:<(other: Type): Boolean = (this.dealias, other.dealias) match { case (tp1, tp2) if tp1 == tp2 => true - case (tp1, UnionType(_, _, members)) => members.exists(tp1 <:< _.dealias) - case (ObjectType(_, _, _, interfaces), tp2) => interfaces.exists(_ <:< tp2) - case (InterfaceType(_, _, _, interfaces), tp2) => interfaces.exists(_ <:< tp2) + case (tp1, UnionType(_, _, members, _)) => members.exists(tp1 <:< _.dealias) + case (ObjectType(_, _, _, interfaces, _), tp2) => interfaces.exists(_ <:< tp2) + case (InterfaceType(_, _, _, interfaces, _), tp2) => interfaces.exists(_ <:< tp2) case (NullableType(tp1), NullableType(tp2)) => tp1 <:< tp2 case (tp1, NullableType(tp2)) => tp1 <:< tp2 case (ListType(tp1), ListType(tp2)) => tp1 <:< tp2 @@ -166,8 +168,8 @@ sealed trait Type extends Product { def field(fieldName: String): Option[Type] = this match { case NullableType(tpe) => tpe.field(fieldName) case TypeRef(_, _) if exists => dealias.field(fieldName) - case ObjectType(_, _, fields, _) => fields.find(_.name == fieldName).map(_.tpe) - case InterfaceType(_, _, fields, _) => fields.find(_.name == fieldName).map(_.tpe) + case ObjectType(_, _, fields, _, _) => fields.find(_.name == fieldName).map(_.tpe) + case InterfaceType(_, _, fields, _, _) => fields.find(_.name == fieldName).map(_.tpe) case _ => None } @@ -175,20 +177,25 @@ sealed trait Type extends Product { def hasField(fieldName: String): Boolean = field(fieldName).isDefined + /** Yields the definition of `fieldName` in this type if it exists, `None` otherwise. */ + def fieldInfo(fieldName: String): Option[Field] = this match { + case NullableType(tpe) => tpe.fieldInfo(fieldName) + case ListType(tpe) => tpe.fieldInfo(fieldName) + case _: TypeRef => dealias.fieldInfo(fieldName) + case _ => None + } + /** * `true` if this type has a field named `fieldName` which is undefined in * some interface it implements */ def variantField(fieldName: String): Boolean = underlyingObject match { - case Some(ObjectType(_, _, _, interfaces)) => + case Some(ObjectType(_, _, _, interfaces, _)) => hasField(fieldName) && interfaces.exists(!_.hasField(fieldName)) case _ => false } - def withField[T](fieldName: String)(body: Type => Result[T]): Result[T] = - field(fieldName).map(body).getOrElse(Result.failure(s"Unknown field '$fieldName' in '$this'")) - /** * Yield the type of the field at the end of the path `fns` starting * from this type, or `None` if there is no such field. @@ -198,9 +205,9 @@ sealed trait Type extends Product { case (_, ListType(tpe)) => tpe.path(fns) case (_, NullableType(tpe)) => tpe.path(fns) case (_, TypeRef(_, _)) => dealias.path(fns) - case (fieldName :: rest, ObjectType(_, _, fields, _)) => + case (fieldName :: rest, ObjectType(_, _, fields, _, _)) => fields.find(_.name == fieldName).flatMap(_.tpe.path(rest)) - case (fieldName :: rest, InterfaceType(_, _, fields, _)) => + case (fieldName :: rest, InterfaceType(_, _, fields, _, _)) => fields.find(_.name == fieldName).flatMap(_.tpe.path(rest)) case _ => None } @@ -217,9 +224,9 @@ sealed trait Type extends Product { case (_, _: ListType) => true case (_, NullableType(tpe)) => tpe.pathIsList(fns) case (_, TypeRef(_, _)) => dealias.pathIsList(fns) - case (fieldName :: rest, ObjectType(_, _, fields, _)) => + case (fieldName :: rest, ObjectType(_, _, fields, _, _)) => fields.find(_.name == fieldName).map(_.tpe.pathIsList(rest)).getOrElse(false) - case (fieldName :: rest, InterfaceType(_, _, fields, _)) => + case (fieldName :: rest, InterfaceType(_, _, fields, _, _)) => fields.find(_.name == fieldName).map(_.tpe.pathIsList(rest)).getOrElse(false) case _ => false } @@ -236,9 +243,9 @@ sealed trait Type extends Product { case (_, ListType(tpe)) => tpe.pathIsNullable(fns) case (_, _: NullableType) => true case (_, TypeRef(_, _)) => dealias.pathIsNullable(fns) - case (fieldName :: rest, ObjectType(_, _, fields, _)) => + case (fieldName :: rest, ObjectType(_, _, fields, _, _)) => fields.find(_.name == fieldName).map(_.tpe.pathIsNullable(rest)).getOrElse(false) - case (fieldName :: rest, InterfaceType(_, _, fields, _)) => + case (fieldName :: rest, InterfaceType(_, _, fields, _, _)) => fields.find(_.name == fieldName).map(_.tpe.pathIsNullable(rest)).getOrElse(false) case _ => false } @@ -312,7 +319,7 @@ sealed trait Type extends Product { * non-object type which isn't further reducible is reached, in which * case yield `None`. */ - def underlyingObject: Option[Type] = this match { + def underlyingObject: Option[NamedType] = this match { case NullableType(tpe) => tpe.underlyingObject case ListType(tpe) => tpe.underlyingObject case _: TypeRef => dealias.underlyingObject @@ -335,14 +342,11 @@ sealed trait Type extends Product { case NullableType(tpe) => tpe.underlyingField(fieldName) case ListType(tpe) => tpe.underlyingField(fieldName) case TypeRef(_, _) => dealias.underlyingField(fieldName) - case ObjectType(_, _, fields, _) => fields.find(_.name == fieldName).map(_.tpe) - case InterfaceType(_, _, fields, _) => fields.find(_.name == fieldName).map(_.tpe) + case ObjectType(_, _, fields, _, _) => fields.find(_.name == fieldName).map(_.tpe) + case InterfaceType(_, _, fields, _, _) => fields.find(_.name == fieldName).map(_.tpe) case _ => None } - def withUnderlyingField[T](fieldName: String)(body: Type => Result[T]): Result[T] = - underlyingObject.toResult(s"$this is not an object or interface type").flatMap(_.withField(fieldName)(body)) - /** Is this type a leaf type? * * `true` if after stripping of aliases the underlying type a scalar or an @@ -419,6 +423,7 @@ sealed trait Type extends Product { def /(pathElement: String): Path = Path.from(this) / pathElement + def directives: List[Directive] } // Move all below into object Type? @@ -439,6 +444,8 @@ sealed trait NamedType extends Type { def description: Option[String] + def directives: List[Directive] + override def toString: String = name } @@ -452,6 +459,7 @@ case class TypeRef(schema: Schema, name: String) extends NamedType { def description: Option[String] = dealias.description + def directives: List[Directive] = dealias.directives } /** @@ -461,7 +469,8 @@ case class TypeRef(schema: Schema, name: String) extends NamedType { */ case class ScalarType( name: String, - description: Option[String] + description: Option[String], + directives: List[Directive] ) extends Type with NamedType { import ScalarType._ @@ -496,7 +505,8 @@ object ScalarType { |Response formats that support a 32‐bit integer or a number type should use that |type to represent this scalar. """.stripMargin.trim - ) + ), + directives = Nil ) val FloatType = ScalarType( name = "Float", @@ -506,7 +516,8 @@ object ScalarType { |specified by IEEE 754. Response formats that support an appropriate |double‐precision number type should use that type to represent this scalar. """.stripMargin.trim - ) + ), + directives = Nil ) val StringType = ScalarType( name = "String", @@ -516,7 +527,8 @@ object ScalarType { |sequences. The String type is most often used by GraphQL to represent free‐form |human‐readable text. """.stripMargin.trim - ) + ), + directives = Nil ) val BooleanType = ScalarType( name = "Boolean", @@ -526,7 +538,8 @@ object ScalarType { |built‐in boolean type if supported; otherwise, they should use their |representation of the integers 1 and 0. """.stripMargin.trim - ) + ), + directives = Nil ) val IDType = ScalarType( @@ -537,12 +550,14 @@ object ScalarType { |object or as the key for a cache. The ID type is serialized in the same way as a |String; however, it is not intended to be human‐readable. """.stripMargin.trim - ) + ), + directives = Nil ) val AttributeType = ScalarType( name = "InternalAttribute", - description = None + description = None, + directives = Nil ) } @@ -555,7 +570,7 @@ sealed trait TypeWithFields extends NamedType { def fields: List[Field] def interfaces: List[NamedType] - def fieldInfo(name: String): Option[Field] = fields.find(_.name == name) + override def fieldInfo(name: String): Option[Field] = fields.find(_.name == name) } /** @@ -568,7 +583,8 @@ case class InterfaceType( name: String, description: Option[String], fields: List[Field], - interfaces: List[NamedType] + interfaces: List[NamedType], + directives: List[Directive] ) extends Type with TypeWithFields { override def isInterface: Boolean = true } @@ -582,7 +598,8 @@ case class ObjectType( name: String, description: Option[String], fields: List[Field], - interfaces: List[NamedType] + interfaces: List[NamedType], + directives: List[Directive] ) extends Type with TypeWithFields /** @@ -595,7 +612,8 @@ case class ObjectType( case class UnionType( name: String, description: Option[String], - members: List[NamedType] + members: List[NamedType], + directives: List[Directive] ) extends Type with NamedType { override def isUnion: Boolean = true override def toString: String = members.mkString("|") @@ -609,11 +627,13 @@ case class UnionType( case class EnumType( name: String, description: Option[String], - enumValues: List[EnumValue] + enumValues: List[EnumValueDefinition], + directives: List[Directive] ) extends Type with NamedType { def hasValue(name: String): Boolean = enumValues.exists(_.name == name) - def value(name: String): Option[EnumValue] = enumValues.find(_.name == name) + def value(name: String): Option[EnumValue] = valueDefinition(name).map(_ => EnumValue(name)) + def valueDefinition(name: String): Option[EnumValueDefinition] = enumValues.find(_.name == name) } /** @@ -621,12 +641,20 @@ case class EnumType( * * @see https://facebook.github.io/graphql/draft/#sec-The-__EnumValue-Type */ -case class EnumValue( +case class EnumValueDefinition( name: String, description: Option[String], - isDeprecated: Boolean = false, - deprecationReason: Option[String] = None -) + directives: List[Directive] +) { + def deprecatedDirective: Option[Directive] = + directives.find(_.name == "deprecated") + def isDeprecated: Boolean = deprecatedDirective.isDefined + def deprecationReason: Option[String] = + for { + dir <- deprecatedDirective + reason <- dir.args.collectFirst { case Binding("reason", StringValue(reason)) => reason } + } yield reason +} /** * Input objects are composite types used as inputs into queries defined as a list of named input @@ -637,7 +665,8 @@ case class EnumValue( case class InputObjectType( name: String, description: Option[String], - inputFields: List[InputValue] + inputFields: List[InputValue], + directives: List[Directive] ) extends Type with NamedType { def inputFieldInfo(name: String): Option[InputValue] = inputFields.find(_.name == name) } @@ -651,6 +680,7 @@ case class InputObjectType( case class ListType( ofType: Type ) extends Type { + def directives: List[Directive] = Nil override def toString: String = s"[$ofType]" } @@ -664,6 +694,7 @@ case class ListType( case class NullableType( ofType: Type ) extends Type { + def directives: List[Directive] = Nil override def toString: String = s"$ofType?" } @@ -677,9 +708,17 @@ case class Field( description: Option[String], args: List[InputValue], tpe: Type, - isDeprecated: Boolean, - deprecationReason: Option[String] -) + directives: List[Directive] +) { + def deprecatedDirective: Option[Directive] = + directives.find(_.name == "deprecated") + def isDeprecated: Boolean = deprecatedDirective.isDefined + def deprecationReason: Option[String] = + for { + dir <- deprecatedDirective + reason <- dir.args.collect { case Binding("reason", StringValue(reason)) => reason }.headOption + } yield reason +} /** * @param defaultValue a String encoding (using the GraphQL language) of the default value used by @@ -689,7 +728,8 @@ case class InputValue( name: String, description: Option[String], tpe: Type, - defaultValue: Option[Value] + defaultValue: Option[Value], + directives: List[Directive] ) sealed trait Value @@ -706,16 +746,14 @@ object Value { case class IDValue(value: String) extends Value - case class UntypedEnumValue(name: String) extends Value - - case class TypedEnumValue(value: EnumValue) extends Value - - case class UntypedVariableValue(name: String) extends Value + case class EnumValue(name: String) extends Value case class ListValue(elems: List[Value]) extends Value case class ObjectValue(fields: List[(String, Value)]) extends Value + case class VariableRef(name: String) extends Value + case object NullValue extends Value case object AbsentValue extends Value @@ -734,7 +772,34 @@ object Value { } } - def checkValue(iv: InputValue, value: Option[Value]): Result[Value] = + /** + * Elaborate a value by replacing variable references with their values. + */ + def elaborateValue(value: Value, vars: Vars): Result[Value] = { + def loop(value: Value): Result[Value] = + value match { + case VariableRef(varName) => + Result.fromOption(vars.get(varName).map(_._2), s"Undefined variable '$varName'") + case ObjectValue(fields) => + val (keys, values) = fields.unzip + values.traverse(loop).map(evs => ObjectValue(keys.zip(evs))) + case ListValue(elems) => elems.traverse(loop).map(ListValue.apply) + case other => Result(other) + } + loop(value) + } + + /** + * Resolve a value against its definition. + * + * + Absent and null values are defaulted if the InputValue provides a default. + * + Absent and null values are checked against the nullability of the InputValue. + * + Enum values are checked against the possible values of the EnumType. + * + Primitive values are converted to custom Scalars or IDs where appropriate. + * + The elements of list values are checked against their element type. + * + The fields of input object values are checked against their field definitions. + */ + def checkValue(iv: InputValue, value: Option[Value], location: String): Result[Value] = (iv.tpe.dealias, value) match { case (_, None) if iv.defaultValue.isDefined => iv.defaultValue.get.success @@ -745,7 +810,7 @@ object Value { case (_: NullableType, Some(NullValue)) => NullValue.success case (NullableType(tpe), Some(_)) => - checkValue(iv.copy(tpe = tpe), value) + checkValue(iv.copy(tpe = tpe), value, location) case (IntType, Some(value: IntValue)) => value.success case (FloatType, Some(value: FloatValue)) => @@ -756,13 +821,13 @@ object Value { value.success // Custom Scalars - case (s @ ScalarType(_, _), Some(value: IntValue)) if !s.isBuiltIn => + case (s: ScalarType, Some(value: IntValue)) if !s.isBuiltIn => value.success - case (s @ ScalarType(_, _), Some(value: FloatValue)) if !s.isBuiltIn => + case (s: ScalarType, Some(value: FloatValue)) if !s.isBuiltIn => value.success - case (s @ ScalarType(_, _), Some(value: StringValue)) if !s.isBuiltIn => + case (s: ScalarType, Some(value: StringValue)) if !s.isBuiltIn => value.success - case (s @ ScalarType(_, _), Some(value: BooleanValue)) if !s.isBuiltIn => + case (s: ScalarType, Some(value: BooleanValue)) if !s.isBuiltIn => value.success case (IDType, Some(value: IDValue)) => @@ -771,27 +836,34 @@ object Value { IDValue(s).success case (IDType, Some(IntValue(i))) => IDValue(i.toString).success - case (_: EnumType, Some(value: TypedEnumValue)) => + case (e: EnumType, Some(value@EnumValue(name))) if e.hasValue(name) => value.success - case (e: EnumType, Some(UntypedEnumValue(name))) if e.hasValue(name) => - TypedEnumValue(e.value(name).get).success case (ListType(tpe), Some(ListValue(arr))) => arr.traverse { elem => - checkValue(iv.copy(tpe = tpe, defaultValue = None), Some(elem)) + checkValue(iv.copy(tpe = tpe, defaultValue = None), Some(elem), location) }.map(ListValue.apply) - case (InputObjectType(nme, _, ivs), Some(ObjectValue(fs))) => + case (InputObjectType(nme, _, ivs, _), Some(ObjectValue(fs))) => val obj = fs.toMap val unknownFields = fs.map(_._1).filterNot(f => ivs.exists(_.name == f)) if (unknownFields.nonEmpty) - Result.failure(s"Unknown field(s) ${unknownFields.map(s => s"'$s'").mkString("", ", ", "")} in input object value of type ${nme}") + Result.failure(s"Unknown field(s) ${unknownFields.map(s => s"'$s'").mkString("", ", ", "")} for input object value of type ${nme} in $location") else - ivs.traverse(iv => checkValue(iv, obj.get(iv.name)).map(v => (iv.name, v))).map(ObjectValue.apply) - case (_: ScalarType, Some(value)) => value.success - case (tpe, Some(value)) => Result.failure(s"Expected $tpe found '$value' for '${iv.name}'") - case (tpe, None) => Result.failure(s"Value of type $tpe required for '${iv.name}'") + ivs.traverse(iv => checkValue(iv, obj.get(iv.name), location).map(v => (iv.name, v))).map(ObjectValue.apply) + case (tpe, Some(value)) => Result.failure(s"Expected $tpe found '${SchemaRenderer.renderValue(value)}' for '${iv.name}' in $location") + case (tpe, None) => Result.failure(s"Value of type $tpe required for '${iv.name}' in $location") } - def checkVarValue(iv: InputValue, value: Option[Json]): Result[Value] = { + /** + * Resolve a Json variable value against its definition. + * + * + Absent and null values are defaulted if the InputValue provides a default. + * + Absent and null values are checked against the nullability of the InputValue. + * + Enum values are checked against the possible values of the EnumType. + * + Primitive values are converted to custom Scalars or IDs where appropriate. + * + The elements of list values are checked against their element type. + * + The fields of input object values are checked against their field definitions. + */ + def checkVarValue(iv: InputValue, value: Option[Json], location: String): Result[Value] = { import JsonExtractor._ (iv.tpe.dealias, value) match { @@ -802,7 +874,7 @@ object Value { case (_: NullableType, Some(jsonNull(_))) => NullValue.success case (NullableType(tpe), Some(_)) => - checkVarValue(iv.copy(tpe = tpe), value) + checkVarValue(iv.copy(tpe = tpe), value, location) case (IntType, Some(jsonInt(value))) => IntValue(value).success case (FloatType, Some(jsonDouble(value))) => @@ -815,32 +887,31 @@ object Value { IDValue(value.toString).success // Custom scalars - case (s @ ScalarType(_, _), Some(jsonInt(value))) if !s.isBuiltIn => + case (s: ScalarType, Some(jsonInt(value))) if !s.isBuiltIn => IntValue(value).success - case (s @ ScalarType(_, _), Some(jsonDouble(value))) if !s.isBuiltIn => + case (s: ScalarType, Some(jsonDouble(value))) if !s.isBuiltIn => FloatValue(value).success - case (s @ ScalarType(_, _), Some(jsonString(value))) if !s.isBuiltIn => + case (s: ScalarType, Some(jsonString(value))) if !s.isBuiltIn => StringValue(value).success - case (s @ ScalarType(_, _), Some(jsonBoolean(value))) if !s.isBuiltIn => + case (s: ScalarType, Some(jsonBoolean(value))) if !s.isBuiltIn => BooleanValue(value).success case (IDType, Some(jsonString(value))) => IDValue(value).success case (e: EnumType, Some(jsonString(name))) if e.hasValue(name) => - TypedEnumValue(e.value(name).get).success + EnumValue(name).success case (ListType(tpe), Some(jsonArray(arr))) => arr.traverse { elem => - checkVarValue(iv.copy(tpe = tpe, defaultValue = None), Some(elem)) + checkVarValue(iv.copy(tpe = tpe, defaultValue = None), Some(elem), location) }.map(vs => ListValue(vs.toList)) - case (InputObjectType(nme, _, ivs), Some(jsonObject(obj))) => + case (InputObjectType(nme, _, ivs, _), Some(jsonObject(obj))) => val unknownFields = obj.keys.filterNot(f => ivs.exists(_.name == f)) if (unknownFields.nonEmpty) - Result.failure(s"Unknown field(s) ${unknownFields.map(s => s"'$s'").mkString("", ", ", "")} in input object value of type ${nme}") + Result.failure(s"Unknown field(s) ${unknownFields.map(s => s"'$s'").mkString("", ", ", "")} in input object value of type ${nme} in $location") else - ivs.traverse(iv => checkVarValue(iv, obj(iv.name)).map(v => (iv.name, v))).map(ObjectValue.apply) - case (_: ScalarType, Some(jsonString(value))) => StringValue(value).success - case (tpe, Some(value)) => Result.failure(s"Expected $tpe found '$value' for '${iv.name}'") - case (tpe, None) => Result.failure(s"Value of type $tpe required for '${iv.name}'") + ivs.traverse(iv => checkVarValue(iv, obj(iv.name), location).map(v => (iv.name, v))).map(ObjectValue.apply) + case (tpe, Some(value)) => Result.failure(s"Expected $tpe found '$value' for '${iv.name}' in $location") + case (tpe, None) => Result.failure(s"Value of type $tpe required for '${iv.name}' in $location") } } } @@ -850,36 +921,240 @@ object Value { * * @see https://facebook.github.io/graphql/draft/#sec-The-__Directive-Type */ -case class Directive( +case class DirectiveDef( name: String, description: Option[String], - locations: List[Ast.DirectiveLocation], args: List[InputValue], - isRepeatable: Boolean + isRepeatable: Boolean, + locations: List[DirectiveLocation] +) + +object DirectiveDef { + val Skip: DirectiveDef = + DirectiveDef( + "skip", + Some( + """|The @skip directive may be provided for fields, fragment spreads, and inline + |fragments, and allows for conditional exclusion during execution as described + |by the if argument. + """.stripMargin.trim + ), + List(InputValue("if", Some("Skipped with true."), BooleanType, None, Nil)), + false, + List(DirectiveLocation.FIELD, DirectiveLocation.FRAGMENT_SPREAD, DirectiveLocation.INLINE_FRAGMENT) + ) + + val Include: DirectiveDef = + DirectiveDef( + "include", + Some( + """|The @include directive may be provided for fields, fragment spreads, and inline + |fragments, and allows for conditional inclusion during execution as described + |by the if argument. + """.stripMargin.trim + ), + List(InputValue("if", Some("Included when true."), BooleanType, None, Nil)), + false, + List(DirectiveLocation.FIELD, DirectiveLocation.FRAGMENT_SPREAD, DirectiveLocation.INLINE_FRAGMENT) + ) + + val Deprecated: DirectiveDef = + DirectiveDef( + "deprecated", + Some( + """|The @deprecated directive is used within the type system definition language + |to indicate deprecated portions of a GraphQL service’s schema, such as deprecated + |fields on a type or deprecated enum values. + """.stripMargin.trim + ), + List(InputValue("reason", Some("Explains why this element was deprecated, usually also including a suggestion for how to access supported similar data. Formatted using the Markdown syntax, as specified by [CommonMark](https://commonmark.org/)."), NullableType(StringType), Some(StringValue("No longer supported")), Nil)), + false, + List(DirectiveLocation.FIELD_DEFINITION, DirectiveLocation.ENUM_VALUE) + ) + + val builtIns: List[DirectiveDef] = + List(Skip, Include, Deprecated) +} + +case class Directive( + name: String, + args: List[Binding] ) +object Directive { + def validateDirectivesForSchema(schema: Schema): List[Problem] = { + def validateTypeDirectives(tpe: NamedType): List[Problem] = + tpe match { + case o: ObjectType => + validateDirectives(schema, Ast.DirectiveLocation.OBJECT, o.directives, Map.empty) ++ + o.fields.flatMap(validateFieldDirectives) + case i: InterfaceType => + validateDirectives(schema, Ast.DirectiveLocation.INTERFACE, i.directives, Map.empty) ++ + i.fields.flatMap(validateFieldDirectives) + case u: UnionType => + validateDirectives(schema, Ast.DirectiveLocation.UNION, u.directives, Map.empty) + case e: EnumType => + validateDirectives(schema, Ast.DirectiveLocation.ENUM, e.directives, Map.empty) ++ + e.enumValues.flatMap(v => validateDirectives(schema, Ast.DirectiveLocation.ENUM_VALUE, v.directives, Map.empty)) + case s: ScalarType => + validateDirectives(schema, Ast.DirectiveLocation.SCALAR, s.directives, Map.empty) + case i: InputObjectType => + validateDirectives(schema, Ast.DirectiveLocation.INPUT_OBJECT, i.directives, Map.empty) ++ + i.inputFields.flatMap(f => validateDirectives(schema, Ast.DirectiveLocation.INPUT_FIELD_DEFINITION, f.directives, Map.empty)) + case _ => Nil + } + + def validateFieldDirectives(field: Field): List[Problem] = + validateDirectives(schema, Ast.DirectiveLocation.FIELD_DEFINITION, field.directives, Map.empty) ++ + field.args.flatMap(a => validateDirectives(schema, Ast.DirectiveLocation.ARGUMENT_DEFINITION, a.directives, Map.empty)) + + validateDirectives(schema, Ast.DirectiveLocation.SCHEMA, schema.schemaType.directives, Map.empty) ++ + (schema.schemaType match { + case twf: TypeWithFields => twf.fields.flatMap(validateFieldDirectives) + case _ => Nil + }) ++ + schema.types.flatMap(validateTypeDirectives) + } + + def validateDirectivesForQuery(schema: Schema, op: UntypedOperation, frags: List[UntypedFragment], vars: Vars): Result[Unit] = { + def queryWarnings(query: Query): List[Problem] = { + def loop(query: Query): List[Problem] = + query match { + case UntypedSelect(_, _, _, dirs, child) => + validateDirectives(schema, Ast.DirectiveLocation.FIELD, dirs, vars) ++ loop(child) + case UntypedFragmentSpread(_, dirs) => + validateDirectives(schema, Ast.DirectiveLocation.FRAGMENT_SPREAD, dirs, vars) + case UntypedInlineFragment(_, dirs, child) => + validateDirectives(schema, Ast.DirectiveLocation.INLINE_FRAGMENT, dirs, vars) ++ loop(child) + case Select(_, _, child) => loop(child) + case Group(children) => children.flatMap(loop) + case Narrow(_, child) => loop(child) + case Unique(child) => loop(child) + case Filter(_, child) => loop(child) + case Limit(_, child) => loop(child) + case Offset(_, child) => loop(child) + case OrderBy(_, child) => loop(child) + case Introspect(_, child) => loop(child) + case Environment(_, child) => loop(child) + case Component(_, _, child) => loop(child) + case Effect(_, child) => loop(child) + case TransformCursor(_, child) => loop(child) + case Count(_) => Nil + case Empty => Nil + } + + loop(query) + } + + def operationWarnings(op: UntypedOperation): List[Problem] = { + lazy val opLocation = op match { + case _: UntypedQuery => Ast.DirectiveLocation.QUERY + case _: UntypedMutation => Ast.DirectiveLocation.MUTATION + case _: UntypedSubscription => Ast.DirectiveLocation.SUBSCRIPTION + } + + val varWarnings = op.variables.flatMap(v => validateDirectives(schema, Ast.DirectiveLocation.VARIABLE_DEFINITION, v.directives, vars)) + val opWarnings = validateDirectives(schema, opLocation, op.directives, vars) + val childWarnings = queryWarnings(op.query) + varWarnings ++ opWarnings ++ childWarnings + } + + def fragmentWarnings(frag: UntypedFragment): List[Problem] = { + val defnWarnings = validateDirectives(schema, Ast.DirectiveLocation.FRAGMENT_DEFINITION, frag.directives, vars) + val childWarnings = queryWarnings(frag.child) + defnWarnings ++ childWarnings + } + + val opWarnings = operationWarnings(op) + val fragWarnings = frags.flatMap(fragmentWarnings) + + Result.fromProblems(opWarnings ++ fragWarnings) + } + + def validateDirectiveOccurrences(schema: Schema, location: Ast.DirectiveLocation, directives: List[Directive]): List[Problem] = { + val (locationProblems, repetitionProblems) = + directives.foldLeft((List.empty[Problem], List.empty[Problem])) { case ((locs, reps), directive) => + val nme = directive.name + schema.directives.find(_.name == nme) match { + case None => (Problem(s"Undefined directive '$nme'") :: locs, reps) + case Some(defn) => + val locs0 = + if (defn.locations.exists(_ == location)) locs + else Problem(s"Directive '$nme' is not allowed on $location") :: locs + + val reps0 = + if (!defn.isRepeatable && directives.count(_.name == nme) > 1) + Problem(s"Directive '$nme' may not occur more than once") :: reps + else reps + + (locs0, reps0) + } + } + + locationProblems.reverse ++ repetitionProblems.reverse.distinct + } + + def validateDirectives(schema: Schema, location: Ast.DirectiveLocation, directives: List[Directive], vars: Vars): List[Problem] = { + val occurrenceProblems = validateDirectiveOccurrences(schema, location, directives) + val argProblems = + directives.flatMap { directive => + val nme = directive.name + schema.directives.find(_.name == nme) match { + case None => List(Problem(s"Undefined directive '$nme'")) + case Some(defn) => + val infos = defn.args + val unknownArgs = directive.args.filterNot(arg => infos.exists(_.name == arg.name)) + if (unknownArgs.nonEmpty) + List(Problem(s"Unknown argument(s) ${unknownArgs.map(s => s"'${s.name}'").mkString("", ", ", "")} in directive $nme")) + else { + val argMap = directive.args.groupMapReduce(_.name)(_.value)((x, _) => x) + infos.traverse { info => + for { + value <- argMap.get(info.name).traverse(Value.elaborateValue(_, vars)) + _ <- checkValue(info, value, s"directive ${defn.name}") + } yield () + }.toProblems.toList + } + } + } + + occurrenceProblems ++ argProblems + } + + def elaborateDirectives(schema: Schema, directives: List[Directive], vars: Vars): Result[List[Directive]] = + directives.traverse { directive => + val nme = directive.name + schema.directives.find(_.name == nme) match { + case None => Result.failure(s"Undefined directive '$nme'") + case Some(defn) => + val argMap = directive.args.groupMapReduce(_.name)(_.value)((x, _) => x) + defn.args.traverse { info => + for { + value0 <- argMap.get(info.name).traverse(Value.elaborateValue(_, vars)) + value1 <- checkValue(info, value0, s"directive ${defn.name}") + } yield Binding(info.name, value1) + }.map(eArgs => directive.copy(args = eArgs)) + } + } +} + /** * GraphQL schema parser */ object SchemaParser { - import Ast.{Directive => DefinedDirective, Type => _, Value => _, _} - import OperationType._ + import Ast.{Directive => _, EnumValueDefinition => _, Type => _, Value => _, _} /** * Parse a query String to a query algebra term. * * Yields a Query value on the right and accumulates errors on the left. */ - def parseText(text: String)(implicit pos: SourcePos): Result[Schema] = { - def toResult[T](pr: Either[Parser.Error, T]): Result[T] = - Result.fromEither(pr.leftMap(_.expected.toList.mkString(","))) - + def parseText(text: String)(implicit pos: SourcePos): Result[Schema] = for { - doc <- toResult(GraphQLParser.Document.parseAll(text)) + doc <- GraphQLParser.toResult(text, GraphQLParser.Document.parseAll(text)) query <- parseDocument(doc) } yield query - } def parseDocument(doc: Document)(implicit sourcePos: SourcePos): Result[Schema] = { object schema extends Schema { @@ -889,48 +1164,45 @@ object SchemaParser { override def schemaType: NamedType = schemaType1.getOrElse(super.schemaType) - var directives: List[Directive] = Nil + var directives: List[DirectiveDef] = Nil - def complete(types0: List[NamedType], schemaType0: Option[NamedType], directives0: List[Directive]): Unit = { + def complete(types0: List[NamedType], schemaType0: Option[NamedType], directives0: List[DirectiveDef]): Unit = { types = types0 schemaType1 = schemaType0 - directives = directives0 + directives = directives0 ++ DirectiveDef.builtIns } } - val defns: List[TypeDefinition] = doc.collect { case tpe: TypeDefinition => tpe } + val typeDefns: List[TypeDefinition] = doc.collect { case tpe: TypeDefinition => tpe } + val dirDefns: List[DirectiveDefinition] = doc.collect { case dir: DirectiveDefinition => dir } for { - types <- mkTypeDefs(schema, defns) + types <- mkTypeDefs(schema, typeDefns) + directives <- mkDirectiveDefs(schema, dirDefns) schemaType <- mkSchemaType(schema, doc) - _ = schema.complete(types, schemaType, Nil) - _ <- SchemaValidator.validateSchema(schema, defns) + _ = schema.complete(types, schemaType, directives) + _ <- Result.fromProblems(SchemaValidator.validateSchema(schema, typeDefns)) } yield schema } // explicit Schema type, if any def mkSchemaType(schema: Schema, doc: Document): Result[Option[NamedType]] = { - def mkRootOperationType(rootTpe: RootOperationTypeDefinition): Result[(OperationType, Type)] = { - val RootOperationTypeDefinition(optype, tpe) = rootTpe - mkType(schema)(tpe).flatMap { - case NullableType(nt: NamedType) => (optype, nt).success - case other => Result.failure(s"Root operation types must be named types, found $other") - } + def mkRootOperationType(rootTpe: RootOperationTypeDefinition): Result[Field] = { + val RootOperationTypeDefinition(optype, tpe, dirs0) = rootTpe + for { + dirs <- dirs0.traverse(mkDirective) + tpe <- mkType(schema)(tpe) + _ <- Result.failure(s"Root operation types must be named types, found '$tpe'").whenA(!tpe.nonNull.isNamed) + } yield Field(optype.name, None, Nil, tpe, dirs) } - def build(query: Type, mutation: Option[Type], subscription: Option[Type]): NamedType = { - def mkRootDef(fieldName: String)(tpe: Type): Field = - Field(fieldName, None, Nil, tpe, false, None) - + def build(dirs: List[Directive], ops: List[Field]): NamedType = { + val query = ops.find(_.name == "query").getOrElse(Field("query", None, Nil, defaultQueryType, Nil)) ObjectType( name = "Schema", description = None, - fields = - mkRootDef("query")(query) :: - List( - mutation.map(mkRootDef("mutation")), - subscription.map(mkRootDef("subscription")) - ).flatten, - interfaces = Nil + fields = query :: List(ops.find(_.name == "mutation"), ops.find(_.name == "subscription")).flatten, + interfaces = Nil, + directives = dirs ) } @@ -939,11 +1211,11 @@ object SchemaParser { val defns = doc.collect { case schema: SchemaDefinition => schema } defns match { case Nil => None.success - case SchemaDefinition(rootTpes, _) :: Nil => - rootTpes.traverse(mkRootOperationType).map { ops0 => - val ops = ops0.toMap - Some(build(ops.get(Query).getOrElse(defaultQueryType), ops.get(Mutation), ops.get(Subscription))) - } + case SchemaDefinition(rootOpTpes, dirs0) :: Nil => + for { + ops <- rootOpTpes.traverse(mkRootOperationType) + dirs <- dirs0.traverse(mkDirective) + } yield Some(build(dirs, ops)) case _ => Result.failure("At most one schema definition permitted") } @@ -958,49 +1230,64 @@ object SchemaParser { case ScalarTypeDefinition(Name("String"), _, _) => StringType.success case ScalarTypeDefinition(Name("Boolean"), _, _) => BooleanType.success case ScalarTypeDefinition(Name("ID"), _, _) => IDType.success - case ScalarTypeDefinition(Name(nme), desc, _) => ScalarType(nme, desc).success - case ObjectTypeDefinition(Name(nme), desc, fields0, ifs0, _) => + case ScalarTypeDefinition(Name(nme), desc, dirs0) => + for { + dirs <- dirs0.traverse(mkDirective) + } yield ScalarType(nme, desc, dirs) + case ObjectTypeDefinition(Name(nme), desc, fields0, ifs0, dirs0) => if (fields0.isEmpty) Result.failure(s"object type $nme must define at least one field") else for { fields <- fields0.traverse(mkField(schema)) - ifs = ifs0.map { case Ast.Type.Named(Name(nme)) => schema.ref(nme) } - } yield ObjectType(nme, desc, fields, ifs) - case InterfaceTypeDefinition(Name(nme), desc, fields0, ifs0, _) => + ifs = ifs0.map { case Ast.Type.Named(Name(nme)) => schema.ref(nme) } + dirs <- dirs0.traverse(mkDirective) + } yield ObjectType(nme, desc, fields, ifs, dirs) + case InterfaceTypeDefinition(Name(nme), desc, fields0, ifs0, dirs0) => if (fields0.isEmpty) Result.failure(s"interface type $nme must define at least one field") else for { fields <- fields0.traverse(mkField(schema)) - ifs = ifs0.map { case Ast.Type.Named(Name(nme)) => schema.ref(nme) } - } yield InterfaceType(nme, desc, fields, ifs) - case UnionTypeDefinition(Name(nme), desc, _, members0) => + ifs = ifs0.map { case Ast.Type.Named(Name(nme)) => schema.ref(nme) } + dirs <- dirs0.traverse(mkDirective) + } yield InterfaceType(nme, desc, fields, ifs, dirs) + case UnionTypeDefinition(Name(nme), desc, dirs0, members0) => if (members0.isEmpty) Result.failure(s"union type $nme must define at least one member") else { - val members = members0.map { case Ast.Type.Named(Name(nme)) => schema.ref(nme) } - UnionType(nme, desc, members).success + for { + dirs <- dirs0.traverse(mkDirective) + members = members0.map { case Ast.Type.Named(Name(nme)) => schema.ref(nme) } + } yield UnionType(nme, desc, members, dirs) } - case EnumTypeDefinition(Name(nme), desc, _, values0) => + case EnumTypeDefinition(Name(nme), desc, dirs0, values0) => if (values0.isEmpty) Result.failure(s"enum type $nme must define at least one enum value") else for { values <- values0.traverse(mkEnumValue) - } yield EnumType(nme, desc, values) - case InputObjectTypeDefinition(Name(nme), desc, fields0, _) => + dirs <- dirs0.traverse(mkDirective) + } yield EnumType(nme, desc, values, dirs) + case InputObjectTypeDefinition(Name(nme), desc, fields0, dirs0) => if (fields0.isEmpty) Result.failure(s"input object type $nme must define at least one input field") else for { fields <- fields0.traverse(mkInputValue(schema)) - } yield InputObjectType(nme, desc, fields) + dirs <- dirs0.traverse(mkDirective) + } yield InputObjectType(nme, desc, fields, dirs) + } + + def mkDirective(d: Ast.Directive): Result[Directive] = { + val Ast.Directive(Name(nme), args) = d + args.traverse { + case (Name(nme), value) => parseValue(value).map(Binding(nme, _)) + }.map(Directive(nme, _)) } def mkField(schema: Schema)(f: FieldDefinition): Result[Field] = { - val FieldDefinition(Name(nme), desc, args0, tpe0, dirs) = f + val FieldDefinition(Name(nme), desc, args0, tpe0, dirs0) = f for { args <- args0.traverse(mkInputValue(schema)) - tpe <- mkType(schema)(tpe0) - deprecation <- parseDeprecated(dirs) - (isDeprecated, reason) = deprecation - } yield Field(nme, desc, args, tpe, isDeprecated, reason) + tpe <- mkType(schema)(tpe0) + dirs <- dirs0.traverse(mkDirective) + } yield Field(nme, desc, args, tpe, dirs) } def mkType(schema: Schema)(tpe: Ast.Type): Result[Type] = { @@ -1018,46 +1305,46 @@ object SchemaParser { loop(tpe, true) } + def mkDirectiveDefs(schema: Schema, defns: List[DirectiveDefinition]): Result[List[DirectiveDef]] = + defns.traverse(mkDirectiveDef(schema)) + + def mkDirectiveDef(schema: Schema)(dd: DirectiveDefinition): Result[DirectiveDef] = { + val DirectiveDefinition(Name(nme), desc, args0, repeatable, locations) = dd + for { + args <- args0.traverse(mkInputValue(schema)) + } yield DirectiveDef(nme, desc, args, repeatable, locations) + } + def mkInputValue(schema: Schema)(f: InputValueDefinition): Result[InputValue] = { - val InputValueDefinition(Name(nme), desc, tpe0, default0, _) = f + val InputValueDefinition(Name(nme), desc, tpe0, default0, dirs0) = f for { tpe <- mkType(schema)(tpe0) dflt <- default0.traverse(parseValue) - } yield InputValue(nme, desc, tpe, dflt) + dirs <- dirs0.traverse(mkDirective) + } yield InputValue(nme, desc, tpe, dflt, dirs) } - def mkEnumValue(e: EnumValueDefinition): Result[EnumValue] = { - val EnumValueDefinition(Name(nme), desc, dirs) = e + def mkEnumValue(e: Ast.EnumValueDefinition): Result[EnumValueDefinition] = { + val Ast.EnumValueDefinition(Name(nme), desc, dirs0) = e for { - deprecation <- parseDeprecated(dirs) - (isDeprecated, reason) = deprecation - } yield EnumValue(nme, desc, isDeprecated, reason) + dirs <- dirs0.traverse(mkDirective) + } yield EnumValueDefinition(nme, desc, dirs) } - def parseDeprecated(directives: List[DefinedDirective]): Result[(Boolean, Option[String])] = - directives.collect { case dir@DefinedDirective(Name("deprecated"), _) => dir } match { - case Nil => (false, None).success - case DefinedDirective(_, List((Name("reason"), Ast.Value.StringValue(reason)))) :: Nil => (true, Some(reason)).success - case DefinedDirective(_, Nil) :: Nil => (true, Some("No longer supported")).success - case DefinedDirective(_, _) :: Nil => Result.failure(s"deprecated must have a single String 'reason' argument, or no arguments") - case _ => Result.failure(s"Only a single deprecated allowed at a given location") - } - - // Share with Query parser def parseValue(value: Ast.Value): Result[Value] = { value match { case Ast.Value.IntValue(i) => IntValue(i).success case Ast.Value.FloatValue(d) => FloatValue(d).success case Ast.Value.StringValue(s) => StringValue(s).success case Ast.Value.BooleanValue(b) => BooleanValue(b).success - case Ast.Value.EnumValue(e) => UntypedEnumValue(e.value).success - case Ast.Value.Variable(v) => UntypedVariableValue(v.value).success + case Ast.Value.EnumValue(e) => EnumValue(e.value).success + case Ast.Value.Variable(v) => VariableRef(v.value).success case Ast.Value.NullValue => NullValue.success - case Ast.Value.ListValue(vs) => vs.traverse(parseValue).map(ListValue.apply) + case Ast.Value.ListValue(vs) => vs.traverse(parseValue).map(ListValue(_)) case Ast.Value.ObjectValue(fs) => fs.traverse { case (name, value) => parseValue(value).map(v => (name.value, v)) - }.map(ObjectValue.apply) + }.map(ObjectValue(_)) } } } @@ -1065,13 +1352,14 @@ object SchemaParser { object SchemaValidator { import SchemaRenderer.renderType - def validateSchema(schema: Schema, defns: List[TypeDefinition]): Result[Unit] = - validateReferences(schema, defns) *> - validateUniqueDefns(schema) *> - validateUniqueEnumValues(schema) *> - validateImplementations(schema) + def validateSchema(schema: Schema, defns: List[TypeDefinition]): List[Problem] = + validateReferences(schema, defns) ++ + validateUniqueDefns(schema) ++ + validateUniqueEnumValues(schema) ++ + validateImplementations(schema) ++ + Directive.validateDirectivesForSchema(schema) - def validateReferences(schema: Schema, defns: List[TypeDefinition]): Result[Unit] = { + def validateReferences(schema: Schema, defns: List[TypeDefinition]): List[Problem] = { def underlyingName(tpe: Ast.Type): String = tpe match { case Ast.Type.List(tpe) => underlyingName(tpe) @@ -1095,41 +1383,33 @@ object SchemaValidator { val defaultTypes = List(StringType, IntType, FloatType, BooleanType, IDType) val typeNames = (defaultTypes ++ schema.types).map(_.name).toSet - val problems = - referencedTypes(defns).collect { - case tpe if !typeNames.contains(tpe) => Problem(s"Reference to undefined type '$tpe'") - } - - Result.fromProblems(problems) + referencedTypes(defns).collect { + case tpe if !typeNames.contains(tpe) => Problem(s"Reference to undefined type '$tpe'") + } } - def validateUniqueDefns(schema: Schema): Result[Unit] = { + def validateUniqueDefns(schema: Schema): List[Problem] = { val dupes = schema.types.groupBy(_.name).collect { case (nme, tpes) if tpes.length > 1 => nme }.toSet - val problems = schema.types.map(_.name).distinct.collect { + schema.types.map(_.name).distinct.collect { case nme if dupes.contains(nme) => Problem(s"Duplicate definition of type '$nme' found") } - - Result.fromProblems(problems) } - def validateUniqueEnumValues(schema: Schema): Result[Unit] = { + def validateUniqueEnumValues(schema: Schema): List[Problem] = { val enums = schema.types.collect { case e: EnumType => e } - val problems = - enums.flatMap { e => - val duplicateValues = e.enumValues.groupBy(_.name).collect { case (nme, vs) if vs.length > 1 => nme }.toList - duplicateValues.map(dupe => Problem(s"Duplicate definition of enum value '$dupe' for Enum type '${e.name}'")) - } - - Result.fromProblems(problems) + enums.flatMap { e => + val duplicateValues = e.enumValues.groupBy(_.name).collect { case (nme, vs) if vs.length > 1 => nme }.toList + duplicateValues.map(dupe => Problem(s"Duplicate definition of enum value '$dupe' for Enum type '${e.name}'")) + } } - def validateImplementations(schema: Schema): Result[Unit] = { + def validateImplementations(schema: Schema): List[Problem] = { def validateImplementor(impl: TypeWithFields): List[Problem] = { import impl.{name, fields, interfaces} @@ -1163,7 +1443,7 @@ object SchemaValidator { } val impls = schema.types.collect { case impl: TypeWithFields => impl } - Result.fromProblems(impls.flatMap(validateImplementor)) + impls.flatMap(validateImplementor) } } @@ -1179,54 +1459,92 @@ object SchemaRenderer { schema.subscriptionType.map(mkRootDef("subscription")) ).flatten - val schemaDefn = - if (fields.sizeCompare(1) == 0 && schema.queryType =:= schema.ref("Query")) "" - else fields.mkString("schema {\n ", "\n ", "\n}\n") + val schemaDefn = { + val dirs0 = schema.schemaType.directives + if (fields.sizeCompare(1) == 0 && schema.queryType =:= schema.ref("Query") && dirs0.isEmpty) "" + else { + val dirs = renderDirectives(dirs0) + fields.mkString(s"schema$dirs {\n ", "\n ", "\n}\n") + } + } + + val dirDefns = { + val nonBuiltInDefns = + schema.directives.filter { + case DirectiveDef("skip"|"include"|"deprecated", _, _, _, _) => false + case _ => true + } + + if(nonBuiltInDefns.isEmpty) "" + else "\n"+nonBuiltInDefns.map(renderDirectiveDefn).mkString("\n")+"\n" + } schemaDefn ++ - schema.types.map(renderTypeDefn).mkString("\n") + schema.types.map(renderTypeDefn).mkString("\n") ++ + dirDefns + } + + def renderDescription(desc: Option[String]): String = + desc match { + case None => "" + case Some(desc) => s""""$desc"\n""" + } + + def renderDirectives(dirs: List[Directive]): String = + if (dirs.isEmpty) "" else dirs.map(renderDirective).mkString(" ", " ", "") + + def renderDirective(d: Directive): String = { + val Directive(name, args0) = d + val args = if(args0.isEmpty) "" else args0.map { case Binding(nme, v) => s"$nme: ${renderValue(v)}" }.mkString("(", ", ", ")") + s"@$name$args" } def renderTypeDefn(tpe: NamedType): String = { def renderField(f: Field): String = { - val Field(nme, _, args, tpe, isDeprecated, reason) = f - val dep = renderDeprecation(isDeprecated, reason) + val Field(nme, _, args, tpe, dirs0) = f + val dirs = renderDirectives(dirs0) if (args.isEmpty) - s"$nme: ${renderType(tpe)}" + dep + s"$nme: ${renderType(tpe)}$dirs" else - s"$nme(${args.map(renderInputValue).mkString(", ")}): ${renderType(tpe)}" + dep + s"$nme(${args.map(renderInputValue).mkString(", ")}): ${renderType(tpe)}$dirs" } tpe match { case tr: TypeRef => renderTypeDefn(tr.dealias) - case ScalarType(nme, _) => - s"""scalar $nme""" + case ScalarType(nme, _, dirs0) => + val dirs = renderDirectives(dirs0) + s"""scalar $nme$dirs""" - case ObjectType(nme, _, fields, ifs0) => + case ObjectType(nme, _, fields, ifs0, dirs0) => val ifs = if (ifs0.isEmpty) "" else " implements " + ifs0.map(_.name).mkString("&") + val dirs = renderDirectives(dirs0) - s"""|type $nme$ifs { + s"""|type $nme$ifs$dirs { | ${fields.map(renderField).mkString("\n ")} |}""".stripMargin - case InterfaceType(nme, _, fields, ifs0) => + case InterfaceType(nme, _, fields, ifs0, dirs0) => val ifs = if (ifs0.isEmpty) "" else " implements " + ifs0.map(_.name).mkString("&") + val dirs = renderDirectives(dirs0) - s"""|interface $nme$ifs { + s"""|interface $nme$ifs$dirs { | ${fields.map(renderField).mkString("\n ")} |}""".stripMargin - case UnionType(nme, _, members) => - s"""union $nme = ${members.map(_.name).mkString(" | ")}""" + case UnionType(nme, _, members, dirs0) => + val dirs = renderDirectives(dirs0) + s"""union $nme$dirs = ${members.map(_.name).mkString(" | ")}""" - case EnumType(nme, _, values) => - s"""|enum $nme { - | ${values.map(renderEnumValue).mkString("\n ")} + case EnumType(nme, _, values, dirs0) => + val dirs = renderDirectives(dirs0) + s"""|enum $nme$dirs { + | ${values.map(renderEnumValueDefinition).mkString("\n ")} |}""".stripMargin - case InputObjectType(nme, _, fields) => - s"""|input $nme { + case InputObjectType(nme, _, fields, dirs0) => + val dirs = renderDirectives(dirs0) + s"""|input $nme$dirs { | ${fields.map(renderInputValue).mkString("\n ")} |}""".stripMargin } @@ -1246,15 +1564,26 @@ object SchemaRenderer { loop(tpe, false) } - def renderEnumValue(v: EnumValue): String = { - val EnumValue(nme, _, isDeprecated, reason) = v - s"$nme" + renderDeprecation(isDeprecated, reason) + def renderDirectiveDefn(directive: DirectiveDef): String = { + val DirectiveDef(nme, desc, args, repeatable, locations) = directive + val rpt = if (repeatable) " repeatable" else "" + if (args.isEmpty) + s"${renderDescription(desc)}directive @$nme$rpt on ${locations.mkString("|")}" + else + s"${renderDescription(desc)}directive @$nme(${args.map(renderInputValue).mkString(", ")})$rpt on ${locations.mkString("|")}" + } + + def renderEnumValueDefinition(v: EnumValueDefinition): String = { + val EnumValueDefinition(nme, _, dirs0) = v + val dirs = renderDirectives(dirs0) + s"$nme$dirs" } def renderInputValue(iv: InputValue): String = { - val InputValue(nme, _, tpe, default) = iv + val InputValue(nme, _, tpe, default, dirs0) = iv + val dirs = renderDirectives(dirs0) val df = default.map(v => s" = ${renderValue(v)}").getOrElse("") - s"$nme: ${renderType(tpe)}$df" + s"$nme: ${renderType(tpe)}$df$dirs" } def renderValue(value: Value): String = value match { @@ -1263,7 +1592,7 @@ object SchemaRenderer { case StringValue(s) => s""""$s"""" case BooleanValue(b) => b.toString case IDValue(i) => s""""$i"""" - case TypedEnumValue(e) => e.name + case EnumValue(e) => e case ListValue(elems) => elems.map(renderValue).mkString("[", ", ", "]") case ObjectValue(fields) => fields.map { @@ -1271,7 +1600,4 @@ object SchemaRenderer { }.mkString("{", ", ", "}") case _ => "null" } - - def renderDeprecation(isDeprecated: Boolean, reason: Option[String]): String = - if (isDeprecated) " @deprecated" + reason.fold("")(r => "(reason: \"" + r + "\")") else "" } diff --git a/modules/core/src/main/scala/valuemapping.scala b/modules/core/src/main/scala/valuemapping.scala index 9072a838..dac566bd 100644 --- a/modules/core/src/main/scala/valuemapping.scala +++ b/modules/core/src/main/scala/valuemapping.scala @@ -11,7 +11,7 @@ import io.circe.Json import org.tpolecat.sourcepos.SourcePos import syntax._ -import Cursor.{Context, DeferredCursor, Env} +import Cursor.{DeferredCursor} abstract class ValueMapping[F[_]](implicit val M: MonadThrow[F]) extends Mapping[F] with ValueMappingLike[F] diff --git a/modules/core/src/test/scala/arb/AstArb.scala b/modules/core/src/test/scala/arb/AstArb.scala index e108084b..43f861db 100644 --- a/modules/core/src/test/scala/arb/AstArb.scala +++ b/modules/core/src/test/scala/arb/AstArb.scala @@ -93,7 +93,8 @@ trait AstArb { variable <- arbitrary[Name] tpe <- arbitrary[Type] defaultValue <- option(genValueForType(tpe)) - } yield VariableDefinition(variable, tpe, defaultValue) + directives <- shortListOf(arbitrary[Directive]) + } yield VariableDefinition(variable, tpe, defaultValue, directives) } implicit lazy val arbSelectionFragmentSpread: Arbitrary[Selection.FragmentSpread] = diff --git a/modules/core/src/test/scala/compiler/AttributesSuite.scala b/modules/core/src/test/scala/compiler/AttributesSuite.scala index 071eb331..fd5632e6 100644 --- a/modules/core/src/test/scala/compiler/AttributesSuite.scala +++ b/modules/core/src/test/scala/compiler/AttributesSuite.scala @@ -61,7 +61,7 @@ final class AttributesSuite extends CatsEffectSuite { { "errors" : [ { - "message" : "Unknown field 'tagCountVA' in select" + "message" : "No field 'tagCountVA' for type Item" } ] } @@ -86,7 +86,7 @@ final class AttributesSuite extends CatsEffectSuite { { "errors" : [ { - "message" : "Unknown field 'tagCountCA' in select" + "message" : "No field 'tagCountCA' for type Item" } ] } diff --git a/modules/core/src/test/scala/compiler/CascadeSuite.scala b/modules/core/src/test/scala/compiler/CascadeSuite.scala new file mode 100644 index 00000000..16f27547 --- /dev/null +++ b/modules/core/src/test/scala/compiler/CascadeSuite.scala @@ -0,0 +1,751 @@ +// Copyright (c) 2016-2020 Association of Universities for Research in Astronomy, Inc. (AURA) +// For license information see LICENSE or https://opensource.org/licenses/BSD-3-Clause + +package compiler + +import PartialFunction.condOpt + +import cats.effect.IO +import io.circe.literal._ +import munit.CatsEffectSuite + +import edu.gemini.grackle._ +import edu.gemini.grackle.syntax._ +import Query._ +import QueryCompiler._ +import Value._ + +final class CascadeSuite extends CatsEffectSuite { + test("elaboration of simple query") { + val query = """ + query { + foo(filter: { foo: "foo", fooBar: 23 }, limit: 10) { + cascaded { + foo + bar + fooBar + limit + } + } + } + """ + + val expected = + Environment( + Env.NonEmptyEnv(Map("filter" -> CascadeMapping.CascadedFilter(Some("foo"), None, Some(23), Some(10)))), + Select("foo", + Select("cascaded", + Group( + List( + Select("foo"), + Select("bar"), + Select("fooBar"), + Select("limit") + ) + ) + ) + ) + ) + + val res = CascadeMapping.compiler.compile(query) + + assertEquals(res.map(_.query), Result.Success(expected)) + } + + test("simple query") { + val query = """ + query { + foo(filter: { foo: "foo", fooBar: 23 }, limit: 10) { + cascaded { + foo + bar + fooBar + limit + } + } + } + """ + + val expected = json""" + { + "data" : { + "foo" : { + "cascaded" : { + "foo" : "foo", + "bar" : null, + "fooBar" : 23, + "limit" : 10 + } + } + } + } + """ + + val res = CascadeMapping.compileAndRun(query) + + //res.flatMap(IO.println) *> + assertIO(res, expected) + } + + test("simple cascade (1)") { + val query = """ + query { + foo(filter: { foo: "foo", fooBar: 23 }, limit: 10) { + cascaded { foo bar fooBar limit } + bar { + cascaded { foo bar fooBar limit } + } + } + } + """ + + val expected = json""" + { + "data" : { + "foo" : { + "cascaded" : { + "foo" : "foo", + "bar" : null, + "fooBar" : 23, + "limit" : 10 + }, + "bar" : { + "cascaded" : { + "foo" : "foo", + "bar" : null, + "fooBar" : 23, + "limit" : null + } + } + } + } + } + """ + + val res = CascadeMapping.compileAndRun(query) + + //res.flatMap(IO.println) *> + assertIO(res, expected) + } + + test("simple cascade (2)") { + val query = """ + query { + foo(filter: { foo: "foo", fooBar: 23 }, limit: 10) { + cascaded { foo bar fooBar limit } + bar { + cascaded { foo bar fooBar limit } + foo { + cascaded { foo bar fooBar limit } + } + } + } + } + """ + + val expected = json""" + { + "data" : { + "foo" : { + "cascaded" : { + "foo" : "foo", + "bar" : null, + "fooBar" : 23, + "limit" : 10 + }, + "bar" : { + "cascaded" : { + "foo" : "foo", + "bar" : null, + "fooBar" : 23, + "limit" : null + }, + "foo" : { + "cascaded" : { + "foo" : "foo", + "bar" : null, + "fooBar" : 23, + "limit" : null + } + } + } + } + } + } + """ + + val res = CascadeMapping.compileAndRun(query) + + //res.flatMap(IO.println) *> + assertIO(res, expected) + } + + test("cascade with override (1)") { + val query = """ + query { + foo(filter: { foo: "foo", fooBar: 23 }, limit: 10) { + cascaded { foo bar fooBar limit } + bar1:bar(filter: { bar: true, fooBar: 13 }, limit: 5) { + cascaded { foo bar fooBar limit } + foo { + cascaded { foo bar fooBar limit } + } + } + bar2:bar(filter: { bar: false, fooBar: 11 }, limit: 7) { + cascaded { foo bar fooBar limit } + foo { + cascaded { foo bar fooBar limit } + } + } + bar3:bar { + cascaded { foo bar fooBar limit } + foo { + cascaded { foo bar fooBar limit } + } + } + } + } + """ + + val expected = json""" + { + "data" : { + "foo" : { + "cascaded" : { + "foo" : "foo", + "bar" : null, + "fooBar" : 23, + "limit" : 10 + }, + "bar1" : { + "cascaded" : { + "foo" : "foo", + "bar" : true, + "fooBar" : 13, + "limit" : 5 + }, + "foo" : { + "cascaded" : { + "foo" : "foo", + "bar" : true, + "fooBar" : 13, + "limit" : null + } + } + }, + "bar2" : { + "cascaded" : { + "foo" : "foo", + "bar" : false, + "fooBar" : 11, + "limit" : 7 + }, + "foo" : { + "cascaded" : { + "foo" : "foo", + "bar" : false, + "fooBar" : 11, + "limit" : null + } + } + }, + "bar3" : { + "cascaded" : { + "foo" : "foo", + "bar" : null, + "fooBar" : 23, + "limit" : null + }, + "foo" : { + "cascaded" : { + "foo" : "foo", + "bar" : null, + "fooBar" : 23, + "limit" : null + } + } + } + } + } + } + """ + + val res = CascadeMapping.compileAndRun(query) + + //res.flatMap(IO.println) *> + assertIO(res, expected) + } + + test("cascade with override (2)") { + val query = """ + query { + foo(filter: { foo: "foo", fooBar: 23 }, limit: 10) { + cascaded { foo bar fooBar limit } + bar1:bar(filter: { bar: true, fooBar: 13 }, limit: 5) { + cascaded { foo bar fooBar limit } + foo(filter: { foo: "foo1" }, limit: 3) { + cascaded { foo bar fooBar limit } + } + } + bar2:bar(filter: { bar: false, fooBar: 11 }, limit: 7) { + cascaded { foo bar fooBar limit } + foo(filter: { foo: "foo2" }, limit: 5) { + cascaded { foo bar fooBar limit } + } + } + bar3:bar { + cascaded { foo bar fooBar limit } + foo(filter: { foo: "foo3" }, limit: 2) { + cascaded { foo bar fooBar limit } + } + } + } + } + """ + + val expected = json""" + { + "data" : { + "foo" : { + "cascaded" : { + "foo" : "foo", + "bar" : null, + "fooBar" : 23, + "limit" : 10 + }, + "bar1" : { + "cascaded" : { + "foo" : "foo", + "bar" : true, + "fooBar" : 13, + "limit" : 5 + }, + "foo" : { + "cascaded" : { + "foo" : "foo1", + "bar" : true, + "fooBar" : 13, + "limit" : 3 + } + } + }, + "bar2" : { + "cascaded" : { + "foo" : "foo", + "bar" : false, + "fooBar" : 11, + "limit" : 7 + }, + "foo" : { + "cascaded" : { + "foo" : "foo2", + "bar" : false, + "fooBar" : 11, + "limit" : 5 + } + } + }, + "bar3" : { + "cascaded" : { + "foo" : "foo", + "bar" : null, + "fooBar" : 23, + "limit" : null + }, + "foo" : { + "cascaded" : { + "foo" : "foo3", + "bar" : null, + "fooBar" : 23, + "limit" : 2 + } + } + } + } + } + } + """ + + val res = CascadeMapping.compileAndRun(query) + + //res.flatMap(IO.println) *> + assertIO(res, expected) + } + + test("cascade with reset (1)") { + val query = """ + query { + foo(filter: { foo: "foo", fooBar: 23 }, limit: 10) { + cascaded { foo bar fooBar limit } + reset { + bar(filter: { bar: true, fooBar: 13 }, limit: 5) { + cascaded { foo bar fooBar limit } + foo { + cascaded { foo bar fooBar limit } + } + } + } + bar1:bar(filter: null, limit: 5) { + cascaded { foo bar fooBar limit } + foo { + cascaded { foo bar fooBar limit } + } + } + bar2:bar(filter: { fooBar: null }, limit: 5) { + cascaded { foo bar fooBar limit } + foo { + cascaded { foo bar fooBar limit } + } + } + bar3:bar(limit: 5) { + cascaded { foo bar fooBar limit } + foo(filter: { foo : null }, limit: 11) { + cascaded { foo bar fooBar limit } + } + } + } + } + """ + + val expected = json""" + { + "data" : { + "foo" : { + "cascaded" : { + "foo" : "foo", + "bar" : null, + "fooBar" : 23, + "limit" : 10 + }, + "reset" : { + "bar" : { + "cascaded" : { + "foo" : null, + "bar" : true, + "fooBar" : 13, + "limit" : 5 + }, + "foo" : { + "cascaded" : { + "foo" : null, + "bar" : true, + "fooBar" : 13, + "limit" : null + } + } + } + }, + "bar1" : { + "cascaded" : { + "foo" : null, + "bar" : null, + "fooBar" : null, + "limit" : 5 + }, + "foo" : { + "cascaded" : { + "foo" : null, + "bar" : null, + "fooBar" : null, + "limit" : null + } + } + }, + "bar2" : { + "cascaded" : { + "foo" : "foo", + "bar" : null, + "fooBar" : null, + "limit" : 5 + }, + "foo" : { + "cascaded" : { + "foo" : "foo", + "bar" : null, + "fooBar" : null, + "limit" : null + } + } + }, + "bar3" : { + "cascaded" : { + "foo" : "foo", + "bar" : null, + "fooBar" : 23, + "limit" : 5 + }, + "foo" : { + "cascaded" : { + "foo" : null, + "bar" : null, + "fooBar" : 23, + "limit" : 11 + } + } + } + } + } + } + """ + + val res = CascadeMapping.compileAndRun(query) + + //res.flatMap(IO.println) *> + assertIO(res, expected) + } + + test("repeated cascade with overrides and resets") { + val query = """ + query { + foo(filter: { foo: "foo" }) { + cascaded { foo } + bar { + foo { + cascaded { foo } + bar { + foo(filter: { foo: "foo2" }) { + cascaded { foo } + bar { + foo(filter: { foo: null }) { + cascaded { foo } + bar { + foo(filter: { foo: "foo3" }) { + cascaded { foo } + reset { + cascaded { foo } + } + } + } + } + } + } + } + } + } + } + } + """ + + val expected = json""" + { + "data" : { + "foo" : { + "cascaded" : { + "foo" : "foo" + }, + "bar" : { + "foo" : { + "cascaded" : { + "foo" : "foo" + }, + "bar" : { + "foo" : { + "cascaded" : { + "foo" : "foo2" + }, + "bar" : { + "foo" : { + "cascaded" : { + "foo" : null + }, + "bar" : { + "foo" : { + "cascaded" : { + "foo" : "foo3" + }, + "reset" : { + "cascaded" : { + "foo" : null + } + } + } + } + } + } + } + } + } + } + } + } + } + """ + + val res = CascadeMapping.compileAndRun(query) + + //res.flatMap(IO.println) *> + assertIO(res, expected) + } +} + +object CascadeMapping extends ValueMapping[IO] { + val schema = + schema""" + type Query { + foo(filter: FooFilter, limit: Int): Foo + } + type Foo { + bar(filter: BarFilter, limit: Int): Bar + reset: Foo! + cascaded: FilterValue! + } + type Bar { + cascaded: FilterValue! + reset: Bar! + foo(filter: FooFilter, limit: Int): Foo + } + input FooFilter { + foo: String + fooBar: Int + } + input BarFilter { + bar: Boolean + fooBar: Int + } + type FilterValue { + foo: String + bar: Boolean + fooBar: Int + limit: Int + } + """ + + val QueryType = schema.ref("Query") + val FooType = schema.ref("Foo") + val BarType = schema.ref("Bar") + val FilterValueType = schema.ref("FilterValue") + + override val typeMappings = + List( + ValueObjectMapping[Unit]( + tpe = QueryType, + fieldMappings = + List( + ValueField("foo", _ => Some(())) + ) + ), + ValueObjectMapping[Unit]( + tpe = FooType, + fieldMappings = + List( + ValueField("cascaded", identity), + ValueField("reset", identity), + ValueField("bar", _ => Some(())) + ) + ), + ValueObjectMapping[Unit]( + tpe = BarType, + fieldMappings = + List( + ValueField("cascaded", identity), + ValueField("reset", identity), + ValueField("foo", _ => Some(())) + ) + ), + ValueObjectMapping[CascadedFilter]( + tpe = FilterValueType, + fieldMappings = + List( + CursorField("foo", getFilterValue(_).map(_.foo)), + CursorField("bar", getFilterValue(_).map(_.bar)), + CursorField("fooBar", getFilterValue(_).map(_.fooBar)), + CursorField("limit", getFilterValue(_).map(_.limit)) + ) + ) + ) + + def getFilterValue(c: Cursor): Result[CascadedFilter] = + c.envR[CascadedFilter]("filter") + + type Tri[T] = Either[Unit, Option[T]] + def triToOption[T](t: Tri[T]): Option[T] = + t match { + case Right(t) => t + case Left(()) => None + } + + abstract class TriValue[T](matchValue: Value => Option[T]) { + def unapply(v: Value): Option[Tri[T]] = + v match { + case NullValue => Some(Right(None)) + case AbsentValue => Some(Left(())) + case _ => matchValue(v).map(t => Right(Some(t))) + } + } + + object TriString extends TriValue[String](condOpt(_) { case StringValue(s) => s }) + object TriInt extends TriValue[Int](condOpt(_) { case IntValue(i) => i }) + object TriBoolean extends TriValue[Boolean](condOpt(_) { case BooleanValue(b) => b }) + + case class CascadedFilter(foo: Option[String], bar: Option[Boolean], fooBar: Option[Int], limit: Option[Int]) { + def combine[T](current: Option[T], next: Tri[T]): Option[T] = + next match { + case Left(()) => current + case Right(None) => None + case Right(Some(t)) => Some(t) + } + + def cascadeFoo(foo0: Tri[String], fooBar0: Tri[Int]): CascadedFilter = + copy( + foo = combine(foo, foo0), + fooBar = combine(fooBar, fooBar0), + limit = None + ) + + def cascadeBar(bar0: Tri[Boolean], fooBar0: Tri[Int]): CascadedFilter = + copy( + bar = combine(bar, bar0), + fooBar = combine(fooBar, fooBar0), + limit = None + ) + + def withLimit(limit0: Tri[Int]): CascadedFilter = + copy(limit = triToOption(limit0)) + } + + object CascadedFilter { + def empty: CascadedFilter = + CascadedFilter(None, None, None, None) + } + + object FooFilter { + def unapply(v: Value): Option[CascadedFilter => CascadedFilter] = + v match { + case ObjectValue(List(("foo", TriString(foo)), ("fooBar", TriInt(fooBar)))) => + Some(_.cascadeFoo(foo, fooBar)) + case NullValue => Some(_ => CascadedFilter.empty) + case AbsentValue => Some(identity) + case _ => None + } + } + + object BarFilter { + def unapply(v: Value): Option[CascadedFilter => CascadedFilter] = + v match { + case ObjectValue(List(("bar", TriBoolean(bar)), ("fooBar", TriInt(fooBar)))) => + Some(_.cascadeBar(bar, fooBar)) + case NullValue => Some(_ => CascadedFilter.empty) + case AbsentValue => Some(identity) + case _ => None + } + } + + override val selectElaborator = SelectElaborator { + case (QueryType | BarType, "foo", List(Binding("filter", FooFilter(filter)), Binding("limit", TriInt(limit)))) => + for { + current0 <- Elab.env[CascadedFilter]("filter") + current = current0.getOrElse(CascadedFilter.empty) + _ <- Elab.env("filter", filter(current).withLimit(limit)) + } yield () + + case (FooType, "bar", List(Binding("filter", BarFilter(filter)), Binding("limit", TriInt(limit)))) => + for { + current0 <- Elab.env[CascadedFilter]("filter") + current = current0.getOrElse(CascadedFilter.empty) + _ <- Elab.env("filter", filter(current).withLimit(limit)) + } yield () + + case (FooType | BarType, "reset", Nil) => + Elab.env("filter", CascadedFilter.empty) + } +} diff --git a/modules/core/src/test/scala/compiler/CompilerSuite.scala b/modules/core/src/test/scala/compiler/CompilerSuite.scala index 31030755..6dde77f5 100644 --- a/modules/core/src/test/scala/compiler/CompilerSuite.scala +++ b/modules/core/src/test/scala/compiler/CompilerSuite.scala @@ -9,7 +9,8 @@ import munit.CatsEffectSuite import edu.gemini.grackle._ import edu.gemini.grackle.syntax._ -import Query._, Predicate._, Value._, UntypedOperation._ +import Query._ +import Predicate._, Value._, UntypedOperation._ import QueryCompiler._, ComponentElaborator.TrivialJoin final class CompilerSuite extends CatsEffectSuite { @@ -23,12 +24,12 @@ final class CompilerSuite extends CatsEffectSuite { """ val expected = - Select("character", List(Binding("id", StringValue("1000"))), - Select("name", Nil) + UntypedSelect("character", None, List(Binding("id", StringValue("1000"))), Nil, + UntypedSelect("name", None, Nil, Nil, Empty) ) - val res = QueryParser.parseText(query) - assertEquals(res, Result.Success(UntypedQuery(expected, Nil))) + val res = QueryParser.parseText(query).map(_._1) + assertEquals(res, Result.Success(List(UntypedQuery(None, expected, Nil, Nil)))) } test("simple mutation") { @@ -43,14 +44,14 @@ final class CompilerSuite extends CatsEffectSuite { """ val expected = - Select("update_character", List(Binding("id", StringValue("1000")), Binding("name", StringValue("Luke"))), - Select("character", Nil, - Select("name", Nil) + UntypedSelect("update_character", None, List(Binding("id", StringValue("1000")), Binding("name", StringValue("Luke"))), Nil, + UntypedSelect("character", None, Nil, Nil, + UntypedSelect("name", None, Nil, Nil, Empty) ) ) - val res = QueryParser.parseText(query) - assertEquals(res, Result.Success(UntypedMutation(expected, Nil))) + val res = QueryParser.parseText(query).map(_._1) + assertEquals(res, Result.Success(List(UntypedMutation(None, expected, Nil, Nil)))) } test("simple subscription") { @@ -63,12 +64,12 @@ final class CompilerSuite extends CatsEffectSuite { """ val expected = - Select("character", List(Binding("id", StringValue("1000"))), - Select("name", Nil) + UntypedSelect("character", None, List(Binding("id", StringValue("1000"))), Nil, + UntypedSelect("name", None, Nil, Nil, Empty) ) - val res = QueryParser.parseText(query) - assertEquals(res, Result.Success(UntypedSubscription(expected, Nil))) + val res = QueryParser.parseText(query).map(_._1) + assertEquals(res, Result.Success(List(UntypedSubscription(None, expected, Nil, Nil)))) } test("simple nested query") { @@ -84,17 +85,17 @@ final class CompilerSuite extends CatsEffectSuite { """ val expected = - Select( - "character", List(Binding("id", StringValue("1000"))), - Select("name", Nil) ~ - Select( - "friends", Nil, - Select("name", Nil) + UntypedSelect( + "character", None, List(Binding("id", StringValue("1000"))), Nil, + UntypedSelect("name", None, Nil, Nil, Empty) ~ + UntypedSelect( + "friends", None, Nil, Nil, + UntypedSelect("name", None, Nil, Nil, Empty) ) ) - val res = QueryParser.parseText(query) - assertEquals(res, Result.Success(UntypedQuery(expected, Nil))) + val res = QueryParser.parseText(query).map(_._1) + assertEquals(res, Result.Success(List(UntypedQuery(None, expected, Nil, Nil)))) } test("shorthand query") { @@ -113,19 +114,19 @@ final class CompilerSuite extends CatsEffectSuite { """ val expected = - Select( - "hero", List(Binding("episode", UntypedEnumValue("NEWHOPE"))), - Select("name", Nil, Empty) ~ - Select("friends", Nil, - Select("name", Nil, Empty) ~ - Select("friends", Nil, - Select("name", Nil, Empty) + UntypedSelect( + "hero", None, List(Binding("episode", EnumValue("NEWHOPE"))), Nil, + UntypedSelect("name", None, Nil, Nil, Empty) ~ + UntypedSelect("friends", None, Nil, Nil, + UntypedSelect("name", None, Nil, Nil, Empty) ~ + UntypedSelect("friends", None, Nil, Nil, + UntypedSelect("name", None, Nil, Nil, Empty) ) ) ) - val res = QueryParser.parseText(query) - assertEquals(res, Result.Success(UntypedQuery(expected, Nil))) + val res = QueryParser.parseText(query).map(_._1) + assertEquals(res, Result.Success(List(UntypedQuery(None, expected, Nil, Nil)))) } test("field alias") { @@ -141,17 +142,17 @@ final class CompilerSuite extends CatsEffectSuite { """ val expected = - Select("user", List(Binding("id", IntValue(4))), + UntypedSelect("user", None, List(Binding("id", IntValue(4))), Nil, Group(List( - Select("id", Nil, Empty), - Select("name", Nil, Empty), - Rename("smallPic", Select("profilePic", List(Binding("size", IntValue(64))), Empty)), - Rename("bigPic", Select("profilePic", List(Binding("size", IntValue(1024))), Empty)) + UntypedSelect("id", None, Nil, Nil, Empty), + UntypedSelect("name", None, Nil, Nil, Empty), + UntypedSelect("profilePic", Some("smallPic"), List(Binding("size", IntValue(64))), Nil, Empty), + UntypedSelect("profilePic", Some("bigPic"), List(Binding("size", IntValue(1024))), Nil, Empty) )) ) - val res = QueryParser.parseText(query) - assertEquals(res, Result.Success(UntypedQuery(expected, Nil))) + val res = QueryParser.parseText(query).map(_._1) + assertEquals(res, Result.Success(List(UntypedQuery(None, expected, Nil, Nil)))) } test("introspection query") { @@ -172,15 +173,15 @@ final class CompilerSuite extends CatsEffectSuite { """ val expected = - Select( - "__schema", Nil, - Select("queryType", Nil, Select("name", Nil, Empty)) ~ - Select("mutationType", Nil, Select("name", Nil, Empty)) ~ - Select("subscriptionType", Nil, Select("name", Nil, Empty)) + UntypedSelect( + "__schema", None, Nil, Nil, + UntypedSelect("queryType", None, Nil, Nil, UntypedSelect("name", None, Nil, Nil, Empty)) ~ + UntypedSelect("mutationType", None, Nil, Nil, UntypedSelect("name", None, Nil, Nil, Empty)) ~ + UntypedSelect("subscriptionType", None, Nil, Nil, UntypedSelect("name", None, Nil, Nil, Empty)) ) - val res = QueryParser.parseText(query) - assertEquals(res, Result.Success(UntypedQuery(expected, Nil))) + val res = QueryParser.parseText(query).map(_._1) + assertEquals(res, Result.Success(List(UntypedQuery(Some("IntrospectionQuery"), expected, Nil, Nil)))) } test("simple selector elaborated query") { @@ -197,13 +198,13 @@ final class CompilerSuite extends CatsEffectSuite { val expected = Select( - "character", Nil, + "character", None, Unique( Filter(Eql(AtomicMapping.CharacterType / "id", Const("1000")), - Select("name", Nil) ~ + Select("name") ~ Select( - "friends", Nil, - Select("name", Nil) + "friends", + Select("name") ) ) ) @@ -237,7 +238,7 @@ final class CompilerSuite extends CatsEffectSuite { } """ - val expected = Problem("Unknown field 'foo' in select") + val expected = Problem("No field 'foo' for type Character") val res = AtomicMapping.compiler.compile(query) @@ -316,23 +317,17 @@ final class CompilerSuite extends CatsEffectSuite { """ val expected = - Wrap("componenta", - Component(ComponentA, TrivialJoin, - Select("componenta", Nil, - Select("fielda1", Nil) ~ - Select("fielda2", Nil, - Wrap("componentb", - Component(ComponentB, TrivialJoin, - Select("componentb", Nil, - Select("fieldb1", Nil) ~ - Select("fieldb2", Nil, - Wrap("componentc", - Component(ComponentC, TrivialJoin, - Select("componentc", Nil, - Select("fieldc1", Nil) - ) - ) - ) + Component(ComponentA, TrivialJoin, + Select("componenta", + Select("fielda1") ~ + Select("fielda2", + Component(ComponentB, TrivialJoin, + Select("componentb", + Select("fieldb1") ~ + Select("fieldb2", + Component(ComponentC, TrivialJoin, + Select("componentc", + Select("fieldc1") ) ) ) @@ -407,12 +402,10 @@ object AtomicMapping extends TestMapping { val QueryType = schema.ref("Query") val CharacterType = schema.ref("Character") - override val selectElaborator = new SelectElaborator(Map( - QueryType -> { - case Select("character", List(Binding("id", StringValue(id))), child) => - Select("character", Nil, Unique(Filter(Eql(CharacterType / "id", Const(id)), child))).success - } - )) + override val selectElaborator = SelectElaborator { + case (QueryType, "character", List(Binding("id", StringValue(id)))) => + Elab.transformChild(child => Unique(Filter(Eql(CharacterType / "id", Const(id)), child))) + } } trait DummyComponent extends TestMapping { val schema = schema"type Query { dummy: Int }" diff --git a/modules/core/src/test/scala/compiler/DirectivesSuite.scala b/modules/core/src/test/scala/compiler/DirectivesSuite.scala new file mode 100644 index 00000000..d6713e85 --- /dev/null +++ b/modules/core/src/test/scala/compiler/DirectivesSuite.scala @@ -0,0 +1,319 @@ +// Copyright (c) 2016-2020 Association of Universities for Research in Astronomy, Inc. (AURA) +// For license information see LICENSE or https://opensource.org/licenses/BSD-3-Clause + +package compiler + +import munit.CatsEffectSuite + +import edu.gemini.grackle._ +import edu.gemini.grackle.syntax._ +import Ast.DirectiveLocation._ +import Query._ + +final class DirectivesSuite extends CatsEffectSuite { + def testDirectiveDefs(s: Schema): List[DirectiveDef] = + s.directives.filter { + case DirectiveDef("skip"|"include"|"deprecated", _, _, _, _) => false + case _ => true + } + + test("Simple directive definition") { + val expected = DirectiveDef("foo", None, Nil, false, List(FIELD)) + val schema = + Schema(""" + type Query { + foo: Int + } + + directive @foo on FIELD + """) + + assertEquals(schema.map(testDirectiveDefs), List(expected).success) + } + + test("Directive definition with description") { + val expected = DirectiveDef("foo", Some("A directive"), Nil, false, List(FIELD)) + val schema = + Schema(""" + type Query { + foo: Int + } + + "A directive" + directive @foo on FIELD + """) + + assertEquals(schema.map(testDirectiveDefs), List(expected).success) + } + + test("Directive definition with multiple locations (1)") { + val expected = DirectiveDef("foo", None, Nil, false, List(FIELD, FRAGMENT_SPREAD, INLINE_FRAGMENT)) + val schema = + Schema(""" + type Query { + foo: Int + } + + directive @foo on FIELD | FRAGMENT_SPREAD | INLINE_FRAGMENT + """) + + assertEquals(schema.map(testDirectiveDefs), List(expected).success) + } + + test("Directive definition with multiple locations (2)") { + val expected = DirectiveDef("foo", None, Nil, false, List(FIELD, FRAGMENT_SPREAD, INLINE_FRAGMENT)) + val schema = + Schema(""" + type Query { + foo: Int + } + + directive @foo on | FIELD | FRAGMENT_SPREAD | INLINE_FRAGMENT + """) + + assertEquals(schema.map(testDirectiveDefs), List(expected).success) + } + + test("Directive definition with repeatable") { + val expected = DirectiveDef("foo", None, Nil, true, List(FIELD)) + val schema = + Schema(""" + type Query { + foo: Int + } + + directive @foo repeatable on FIELD + """) + + assertEquals(schema.map(testDirectiveDefs), List(expected).success) + } + + test("Directive definition with arguments (1)") { + val expected = + DirectiveDef( + "foo", + None, + List(InputValue("arg", None, ScalarType.StringType, None, Nil)), + false, + List(FIELD) + ) + + val schema = + Schema(""" + type Query { + foo: Int + } + + directive @foo(arg: String!) on FIELD + """) + + assertEquals(schema.map(testDirectiveDefs), List(expected).success) + } + + test("Directive definition with arguments (2)") { + val expected = + DirectiveDef( + "foo", + None, + List( + InputValue("arg0", None, ScalarType.StringType, None, Nil), + InputValue("arg1", None, NullableType(ScalarType.IntType), None, Nil) + ), + false, + List(FIELD) + ) + + val schema = + Schema(""" + type Query { + foo: Int + } + + directive @foo(arg0: String!, arg1: Int) on FIELD + """) + + assertEquals(schema.map(testDirectiveDefs), List(expected).success) + } + + test("Schema with directives") { + val schema = + """|schema @foo { + | query: Query + |} + |scalar Scalar @foo + |interface Interface @foo { + | field(e: Enum, i: Input): Int @foo + |} + |type Object implements Interface @foo { + | field(e: Enum, i: Input): Int @foo + |} + |union Union @foo = Object + |enum Enum @foo { + | VALUE @foo + |} + |input Input @foo { + | field: Int @foo + |} + |directive @foo on SCHEMA|SCALAR|OBJECT|FIELD_DEFINITION|ARGUMENT_DEFINITION|INTERFACE|UNION|ENUM|ENUM_VALUE|INPUT_OBJECT|INPUT_FIELD_DEFINITION + |""".stripMargin + + val res = SchemaParser.parseText(schema) + val ser = res.map(_.toString) + + assertEquals(ser, schema.success) + } + + test("Query with directive") { + val expected = + Operation( + UntypedSelect("foo", None, Nil, List(Directive("dir", Nil)), + UntypedSelect("id", None, Nil, List(Directive("dir", Nil)), Empty) + ), + DirectiveMapping.QueryType, + List(Directive("dir", Nil)) + ) + + val query = + """|query @dir { + | foo @dir { + | id @dir + | } + |} + |""".stripMargin + + val res = DirectiveMapping.compiler.compile(query) + + assertEquals(res, expected.success) + } + + test("Mutation with directive") { + val expected = + Operation( + UntypedSelect("foo", None, Nil, List(Directive("dir", Nil)), + UntypedSelect("id", None, Nil, List(Directive("dir", Nil)), Empty) + ), + DirectiveMapping.MutationType, + List(Directive("dir", Nil)) + ) + + val query = + """|mutation @dir { + | foo @dir { + | id @dir + | } + |} + |""".stripMargin + + val res = DirectiveMapping.compiler.compile(query) + + assertEquals(res, expected.success) + } + + test("Subscription with directive") { + val expected = + Operation( + UntypedSelect("foo", None, Nil, List(Directive("dir", Nil)), + UntypedSelect("id", None, Nil, List(Directive("dir", Nil)), Empty) + ), + DirectiveMapping.SubscriptionType, + List(Directive("dir", Nil)) + ) + + val query = + """|subscription @dir { + | foo @dir { + | id @dir + | } + |} + |""".stripMargin + + val res = DirectiveMapping.compiler.compile(query) + + assertEquals(res, expected.success) + } + + test("Fragment with directive") { // TOD: will need new elaborator to expose fragment directives + val expected = + Operation( + UntypedSelect("foo", None, Nil, Nil, + Narrow(DirectiveMapping.BazType, + UntypedSelect("baz", None, Nil, List(Directive("dir", Nil)), Empty) + ) + ), + DirectiveMapping.QueryType, + Nil + ) + + val query = + """|query { + | foo { + | ... Frag @dir + | } + |} + |fragment Frag on Baz @dir { + | baz @dir + |} + |""".stripMargin + + val res = DirectiveMapping.compiler.compile(query) + + assertEquals(res, expected.success) + } + + test("Inline fragment with directive") { // TOD: will need new elaborator to expose fragment directives + val expected = + Operation( + UntypedSelect("foo", None, Nil, Nil, + Narrow(DirectiveMapping.BazType, + UntypedSelect("baz", None, Nil, List(Directive("dir", Nil)), Empty) + ) + ), + DirectiveMapping.QueryType, + Nil + ) + + val query = + """|query { + | foo { + | ... on Baz @dir { + | baz @dir + | } + | } + |} + |""".stripMargin + + val res = DirectiveMapping.compiler.compile(query) + + assertEquals(res, expected.success) + } +} + +object DirectiveMapping extends TestMapping { + val schema = + schema""" + type Query { + foo: Bar + } + type Mutation { + foo: Bar + } + type Subscription { + foo: Bar + } + interface Bar { + id: ID + } + type Baz implements Bar { + id: ID + baz: Int + } + directive @dir on QUERY|MUTATION|SUBSCRIPTION|FIELD|FRAGMENT_DEFINITION|FRAGMENT_SPREAD|INLINE_FRAGMENT + """ + + val QueryType = schema.queryType + val MutationType = schema.mutationType.get + val SubscriptionType = schema.subscriptionType.get + val BazType = schema.ref("Baz") + + override val selectElaborator = PreserveArgsElaborator +} diff --git a/modules/core/src/test/scala/compiler/EnvironmentSuite.scala b/modules/core/src/test/scala/compiler/EnvironmentSuite.scala index 48ee862e..552f17fd 100644 --- a/modules/core/src/test/scala/compiler/EnvironmentSuite.scala +++ b/modules/core/src/test/scala/compiler/EnvironmentSuite.scala @@ -9,8 +9,8 @@ import munit.CatsEffectSuite import edu.gemini.grackle._ import edu.gemini.grackle.syntax._ -import Cursor.Env -import Query._, Value._ +import Query._ +import Value._ import QueryCompiler._ object EnvironmentMapping extends ValueMapping[IO] { @@ -76,20 +76,14 @@ object EnvironmentMapping extends ValueMapping[IO] { ).toResult(s"Missing argument") } - override val selectElaborator = new SelectElaborator(Map( - QueryType -> { - case Select("nestedSum", List(Binding("x", IntValue(x)), Binding("y", IntValue(y))), child) => - Environment(Env("x" -> x, "y" -> y), Select("nestedSum", Nil, child)).success - }, - NestedType -> { - case Select("sum", List(Binding("x", IntValue(x)), Binding("y", IntValue(y))), child) => - Environment(Env("x" -> x, "y" -> y), Select("sum", Nil, child)).success - }, - NestedSumType -> { - case Select("nestedSum", List(Binding("x", IntValue(x)), Binding("y", IntValue(y))), child) => - Environment(Env("x" -> x, "y" -> y), Select("nestedSum", Nil, child)).success - } - )) + override val selectElaborator = SelectElaborator { + case (QueryType, "nestedSum", List(Binding("x", IntValue(x)), Binding("y", IntValue(y)))) => + Elab.env("x" -> x, "y" -> y) + case (NestedType, "sum", List(Binding("x", IntValue(x)), Binding("y", IntValue(y)))) => + Elab.env("x" -> x, "y" -> y) + case (NestedSumType, "nestedSum", List(Binding("x", IntValue(x)), Binding("y", IntValue(y)))) => + Elab.env("x" -> x, "y" -> y) + } } final class EnvironmentSuite extends CatsEffectSuite { diff --git a/modules/core/src/test/scala/compiler/FragmentSuite.scala b/modules/core/src/test/scala/compiler/FragmentSuite.scala index 296c2bee..455ddaff 100644 --- a/modules/core/src/test/scala/compiler/FragmentSuite.scala +++ b/modules/core/src/test/scala/compiler/FragmentSuite.scala @@ -5,15 +5,22 @@ package compiler import cats.effect.IO import cats.implicits._ +import io.circe.Json import io.circe.literal._ import munit.CatsEffectSuite import edu.gemini.grackle._ import edu.gemini.grackle.syntax._ -import Query._, Predicate._, Value._ +import Query._ +import Predicate._, Value._ import QueryCompiler._ final class FragmentSuite extends CatsEffectSuite { + def runOperation(op: Result[Operation]): IO[List[Json]] = { + val op0 = op.toOption.get + FragmentMapping.interpreter.run(op0.query, op0.rootTpe, Env.empty).evalMap(FragmentMapping.mkResponse).compile.toList + } + test("simple fragment query") { val query = """ query withFragments { @@ -35,22 +42,22 @@ final class FragmentSuite extends CatsEffectSuite { """ val expected = - Select("user", Nil, + Select("user", Unique( Filter(Eql(FragmentMapping.UserType / "id", Const("1")), Group(List( - Select("friends", Nil, + Select("friends", Group(List( - Select("id", Nil, Empty), - Select("name", Nil, Empty), - Select("profilePic", Nil, Empty) + Select("id"), + Select("name"), + Select("profilePic") )) ), - Select("mutualFriends", Nil, + Select("mutualFriends", Group(List( - Select("id", Nil, Empty), - Select("name", Nil, Empty), - Select("profilePic", Nil, Empty) + Select("id"), + Select("name"), + Select("profilePic") )) ) )) @@ -93,9 +100,9 @@ final class FragmentSuite extends CatsEffectSuite { val compiled = FragmentMapping.compiler.compile(query) - assert(compiled.map(_.query) == Result.Success(expected)) + assertEquals(compiled.map(_.query), Result.Success(expected)) - val res = FragmentMapping.run(compiled.toOption.get).compile.toList + val res = runOperation(compiled) assertIO(res, List(expectedResult)) } @@ -125,22 +132,22 @@ final class FragmentSuite extends CatsEffectSuite { """ val expected = - Select("user", Nil, + Select("user", Unique( Filter(Eql(FragmentMapping.UserType / "id", Const("1")), Group(List( - Select("friends", Nil, + Select("friends", Group(List( - Select("id", Nil, Empty), - Select("name", Nil, Empty), - Select("profilePic", Nil, Empty) + Select("id"), + Select("name"), + Select("profilePic") )) ), - Select("mutualFriends", Nil, + Select("mutualFriends", Group(List( - Select("id", Nil, Empty), - Select("name", Nil, Empty), - Select("profilePic", Nil, Empty) + Select("id"), + Select("name"), + Select("profilePic") )) ) )) @@ -183,9 +190,9 @@ final class FragmentSuite extends CatsEffectSuite { val compiled = FragmentMapping.compiler.compile(query) - assert(compiled.map(_.query) == Result.Success(expected)) + assertEquals(compiled.map(_.query), Result.Success(expected)) - val res = FragmentMapping.run(compiled.toOption.get).compile.toList + val res = runOperation(compiled) assertIO(res, List(expectedResult)) } @@ -215,12 +222,11 @@ final class FragmentSuite extends CatsEffectSuite { val expected = Select("profiles", - Nil, Group(List( - Select("id", Nil, Empty), - Introspect(FragmentMapping.schema, Select("__typename", Nil, Empty)), - Narrow(User, Select("name", Nil, Empty)), - Narrow(Page, Select("title", Nil, Empty)) + Select("id"), + Introspect(FragmentMapping.schema, Select("__typename")), + Narrow(User, Select("name")), + Narrow(Page, Select("title")) )) ) @@ -260,9 +266,9 @@ final class FragmentSuite extends CatsEffectSuite { val compiled = FragmentMapping.compiler.compile(query) - assert(compiled.map(_.query) == Result.Success(expected)) + assertEquals(compiled.map(_.query), Result.Success(expected)) - val res = FragmentMapping.run(compiled.toOption.get).compile.toList + val res = runOperation(compiled) assertIO(res, List(expectedResult)) } @@ -286,11 +292,11 @@ final class FragmentSuite extends CatsEffectSuite { val Page = FragmentMapping.schema.ref("Page") val expected = - Select("profiles", Nil, + Select("profiles", Group(List( - Select("id", Nil, Empty), - Narrow(User, Select("name", Nil, Empty)), - Narrow(Page, Select("title", Nil, Empty)) + Select("id"), + Narrow(User, Select("name")), + Narrow(Page, Select("title")) )) ) @@ -325,9 +331,9 @@ final class FragmentSuite extends CatsEffectSuite { val compiled = FragmentMapping.compiler.compile(query) - assert(compiled.map(_.query) == Result.Success(expected)) + assertEquals(compiled.map(_.query), Result.Success(expected)) - val res = FragmentMapping.run(compiled.toOption.get).compile.toList + val res = runOperation(compiled) assertIO(res, List(expectedResult)) } @@ -367,43 +373,56 @@ final class FragmentSuite extends CatsEffectSuite { val expected = Group(List( - Select("user", Nil, + Select("user", Unique( Filter(Eql(FragmentMapping.UserType / "id", Const("1")), - Select("favourite", Nil, + Select("favourite", Group(List( - Introspect(FragmentMapping.schema, Select("__typename", Nil, Empty)), - Group(List( - Narrow(User, Select("id", Nil, Empty)), - Narrow(User, Select("name", Nil, Empty)))), - Group(List( - Narrow(Page, Select("id", Nil, Empty)), - Narrow(Page, Select("title", Nil, Empty)) - )) + Introspect(FragmentMapping.schema, Select("__typename")), + Narrow( + User, + Group(List( + Select("id"), + Select("name") + )) + ), + Narrow( + Page, + Group(List( + Select("id"), + Select("title") + )) + ) )) ) ) ) ), - Rename("page", Select("user", Nil, + Select("user", Some("page"), Unique( Filter(Eql(FragmentMapping.PageType / "id", Const("2")), - Select("favourite", Nil, + Select("favourite", Group(List( - Introspect(FragmentMapping.schema, Select("__typename", Nil, Empty)), - Group(List( - Narrow(User, Select("id", Nil, Empty)), - Narrow(User, Select("name", Nil, Empty)) - )), - Group(List( - Narrow(Page, Select("id", Nil, Empty)), - Narrow(Page, Select("title", Nil, Empty)) - )) + Introspect(FragmentMapping.schema, Select("__typename")), + Narrow( + User, + Group(List( + Select("id"), + Select("name") + )) + ), + Narrow( + Page, + Group(List( + Select("id"), + Select("title") + )) + ) )) ) ) ) - )) + ) )) val expectedResult = json""" @@ -429,9 +448,9 @@ final class FragmentSuite extends CatsEffectSuite { val compiled = FragmentMapping.compiler.compile(query) - assert(compiled.map(_.query) == Result.Success(expected)) + assertEquals(compiled.map(_.query), Result.Success(expected)) - val res = FragmentMapping.run(compiled.toOption.get).compile.toList + val res = runOperation(compiled) assertIO(res, List(expectedResult)) } @@ -452,11 +471,11 @@ final class FragmentSuite extends CatsEffectSuite { """ val expected = - Select("user", Nil, + Select("user", Unique( Filter(Eql(FragmentMapping.UserType / "id", Const("1")), - Select("friends", Nil, - Select("id", Nil, Empty) + Select("friends", + Select("id") ) ) ) @@ -481,9 +500,9 @@ final class FragmentSuite extends CatsEffectSuite { val compiled = FragmentMapping.compiler.compile(query) - assert(compiled.map(_.query) == Result.Success(expected)) + assertEquals(compiled.map(_.query), Result.Success(expected)) - val res = FragmentMapping.run(compiled.toOption.get).compile.toList + val res = runOperation(compiled) assertIO(res, List(expectedResult)) } @@ -581,7 +600,7 @@ final class FragmentSuite extends CatsEffectSuite { { "errors" : [ { - "message" : "Unknown field 'name' in select" + "message" : "No field 'name' for type Profile" } ] } @@ -693,12 +712,8 @@ object FragmentMapping extends ValueMapping[IO] { ) ) - override val selectElaborator = new SelectElaborator(Map( - QueryType -> { - case Select("user", List(Binding("id", IDValue(id))), child) => - Select("user", Nil, Unique(Filter(Eql(FragmentMapping.UserType / "id", Const(id)), child))).success - case sel@Select("profiles", _, _) => - sel.success - } - )) + override val selectElaborator = SelectElaborator { + case (QueryType, "user", List(Binding("id", IDValue(id)))) => + Elab.transformChild(child => Unique(Filter(Eql(FragmentMapping.UserType / "id", Const(id)), child))) + } } diff --git a/modules/core/src/test/scala/compiler/InputValuesSuite.scala b/modules/core/src/test/scala/compiler/InputValuesSuite.scala index 26247fdb..a859a69d 100644 --- a/modules/core/src/test/scala/compiler/InputValuesSuite.scala +++ b/modules/core/src/test/scala/compiler/InputValuesSuite.scala @@ -8,8 +8,8 @@ import munit.CatsEffectSuite import edu.gemini.grackle._ import edu.gemini.grackle.syntax._ -import Query._, Value._ -import QueryCompiler._ +import Query._ +import Value._ final class InputValuesSuite extends CatsEffectSuite { test("null value") { @@ -29,9 +29,9 @@ final class InputValuesSuite extends CatsEffectSuite { val expected = Group(List( - Select("field", List(Binding("arg", AbsentValue)), Select("subfield", Nil, Empty)), - Select("field", List(Binding("arg", NullValue)), Select("subfield", Nil, Empty)), - Select("field", List(Binding("arg", IntValue(23))), Select("subfield", Nil, Empty)) + UntypedSelect("field", None, List(Binding("arg", AbsentValue)), Nil, UntypedSelect("subfield", None, Nil, Nil, Empty)), + UntypedSelect("field", None, List(Binding("arg", NullValue)), Nil, UntypedSelect("subfield", None, Nil, Nil, Empty)), + UntypedSelect("field", None, List(Binding("arg", IntValue(23))), Nil, UntypedSelect("subfield", None, Nil, Nil, Empty)) )) val compiled = InputValuesMapping.compiler.compile(query, None) @@ -53,11 +53,11 @@ final class InputValuesSuite extends CatsEffectSuite { val expected = Group(List( - Select("listField", List(Binding("arg", ListValue(Nil))), - Select("subfield", Nil, Empty) + UntypedSelect("listField", None, List(Binding("arg", ListValue(Nil))), Nil, + UntypedSelect("subfield", None, Nil, Nil, Empty) ), - Select("listField", List(Binding("arg", ListValue(List(StringValue("foo"), StringValue("bar"))))), - Select("subfield", Nil, Empty) + UntypedSelect("listField", None, List(Binding("arg", ListValue(List(StringValue("foo"), StringValue("bar"))))), Nil, + UntypedSelect("subfield", None, Nil, Nil, Empty) ) )) @@ -76,7 +76,7 @@ final class InputValuesSuite extends CatsEffectSuite { """ val expected = - Select("objectField", + UntypedSelect("objectField", None, List(Binding("arg", ObjectValue(List( ("foo", IntValue(23)), @@ -86,7 +86,8 @@ final class InputValuesSuite extends CatsEffectSuite { ("nullable", AbsentValue) )) )), - Select("subfield", Nil, Empty) + Nil, + UntypedSelect("subfield", None, Nil, Nil, Empty) ) val compiled = InputValuesMapping.compiler.compile(query, None) @@ -103,7 +104,7 @@ final class InputValuesSuite extends CatsEffectSuite { } """ - val expected = Problem("Unknown field(s) 'wibble' in input object value of type InObj") + val expected = Problem("Unknown field(s) 'wibble' for input object value of type InObj in field 'objectField' of type 'Query'") val compiled = InputValuesMapping.compiler.compile(query, None) //println(compiled) @@ -131,9 +132,5 @@ object InputValuesMapping extends TestMapping { } """ - val QueryType = schema.ref("Query") - - override val selectElaborator = new SelectElaborator(Map( - QueryType -> PartialFunction.empty - )) + override val selectElaborator = PreserveArgsElaborator } diff --git a/modules/core/src/test/scala/compiler/PredicatesSuite.scala b/modules/core/src/test/scala/compiler/PredicatesSuite.scala index d6633877..4183215a 100644 --- a/modules/core/src/test/scala/compiler/PredicatesSuite.scala +++ b/modules/core/src/test/scala/compiler/PredicatesSuite.scala @@ -4,13 +4,13 @@ package compiler import cats.effect.IO -import cats.implicits._ import io.circe.literal._ import munit.CatsEffectSuite import edu.gemini.grackle._ import edu.gemini.grackle.syntax._ -import Query._, Predicate._, Value._ +import Query._ +import Predicate._, Value._ import QueryCompiler._ object ItemData { @@ -69,18 +69,16 @@ object ItemMapping extends ValueMapping[IO] { def tagCount(c: Cursor): Result[Int] = c.fieldAs[List[String]]("tags").map(_.size) - override val selectElaborator = new SelectElaborator(Map( - QueryType -> { - case Select("itemByTag", List(Binding("tag", IDValue(tag))), child) => - Select("itemByTag", Nil, Filter(Contains(ItemType / "tags", Const(tag)), child)).success - case Select("itemByTagCount", List(Binding("count", IntValue(count))), child) => - Select("itemByTagCount", Nil, Filter(Eql(ItemType / "tagCount", Const(count)), child)).success - case Select("itemByTagCountVA", List(Binding("count", IntValue(count))), child) => - Select("itemByTagCountVA", Nil, Filter(Eql(ItemType / "tagCountVA", Const(count)), child)).success - case Select("itemByTagCountCA", List(Binding("count", IntValue(count))), child) => - Select("itemByTagCountCA", Nil, Filter(Eql(ItemType / "tagCountCA", Const(count)), child)).success - } - )) + override val selectElaborator = SelectElaborator { + case (QueryType, "itemByTag", List(Binding("tag", IDValue(tag)))) => + Elab.transformChild(child => Filter(Contains(ItemType / "tags", Const(tag)), child)) + case (QueryType, "itemByTagCount", List(Binding("count", IntValue(count)))) => + Elab.transformChild(child => Filter(Eql(ItemType / "tagCount", Const(count)), child)) + case (QueryType, "itemByTagCountVA", List(Binding("count", IntValue(count)))) => + Elab.transformChild(child => Filter(Eql(ItemType / "tagCountVA", Const(count)), child)) + case (QueryType, "itemByTagCountCA", List(Binding("count", IntValue(count)))) => + Elab.transformChild(child => Filter(Eql(ItemType / "tagCountCA", Const(count)), child)) + } } final class PredicatesSuite extends CatsEffectSuite { diff --git a/modules/core/src/test/scala/compiler/PreserveArgsElaborator.scala b/modules/core/src/test/scala/compiler/PreserveArgsElaborator.scala new file mode 100644 index 00000000..16da664d --- /dev/null +++ b/modules/core/src/test/scala/compiler/PreserveArgsElaborator.scala @@ -0,0 +1,42 @@ +// Copyright (c) 2016-2020 Association of Universities for Research in Astronomy, Inc. (AURA) +// For license information see LICENSE or https://opensource.org/licenses/BSD-3-Clause + +package compiler + +import edu.gemini.grackle._ +import Query._ +import QueryCompiler._ + +object PreserveArgsElaborator extends SelectElaborator { + case class Preserved(args: List[Binding], directives: List[Directive]) + + def subst(query: Query, fieldName: String, preserved: Preserved): Query = { + def loop(query: Query): Query = + query match { + case Select(`fieldName`, alias, child) => + UntypedSelect(fieldName, alias, preserved.args, directives = preserved.directives, child) + case Environment(env, child) if env.contains("preserved") => loop(child) + case e@Environment(_, child) => e.copy(child = loop(child)) + case g: Group => g.copy(queries = g.queries.map(loop)) + case t@TransformCursor(_, child) => t.copy(child = loop(child)) + case other => other + } + + loop(query) + } + + override def transform(query: Query): Elab[Query] = { + query match { + case UntypedSelect(fieldName, _, _, _, _) => + for { + t <- super.transform(query) + preserved <- Elab.envE[Preserved]("preserved") + } yield subst(t, fieldName, preserved) + + case other => super.transform(other) + } + } + + def select(ref: TypeRef, name: String, args: List[Binding], directives: List[Directive]): Elab[Unit] = + Elab.env("preserved", Preserved(args, directives)) +} diff --git a/modules/core/src/test/scala/compiler/QuerySizeSuite.scala b/modules/core/src/test/scala/compiler/QuerySizeSuite.scala index 818c085e..139f560c 100644 --- a/modules/core/src/test/scala/compiler/QuerySizeSuite.scala +++ b/modules/core/src/test/scala/compiler/QuerySizeSuite.scala @@ -21,7 +21,7 @@ class QuerySizeSuite extends CatsEffectSuite { """ val compiledQuery = StarWarsMapping.compiler.compile(query).toOption.get.query - val res = StarWarsMapping.querySizeValidator.querySize(compiledQuery) + val res = StarWarsMapping.querySizeValidator.querySize(compiledQuery, Map.empty) assertEquals(res, ((2,1))) } @@ -37,7 +37,7 @@ class QuerySizeSuite extends CatsEffectSuite { """ val compiledQuery = StarWarsMapping.compiler.compile(query).toOption.get.query - val res = StarWarsMapping.querySizeValidator.querySize(compiledQuery) + val res = StarWarsMapping.querySizeValidator.querySize(compiledQuery, Map.empty) assertEquals(res, ((2,2))) } @@ -54,7 +54,7 @@ class QuerySizeSuite extends CatsEffectSuite { """ val compiledQuery = StarWarsMapping.compiler.compile(query).toOption.get.query - val res = StarWarsMapping.querySizeValidator.querySize(compiledQuery) + val res = StarWarsMapping.querySizeValidator.querySize(compiledQuery, Map.empty) assertEquals(res, ((3,1))) } @@ -75,7 +75,7 @@ class QuerySizeSuite extends CatsEffectSuite { """ val compiledQuery = StarWarsMapping.compiler.compile(query).toOption.get.query - val res = StarWarsMapping.querySizeValidator.querySize(compiledQuery) + val res = StarWarsMapping.querySizeValidator.querySize(compiledQuery, Map.empty) assertEquals(res, ((4,3))) } @@ -91,7 +91,7 @@ class QuerySizeSuite extends CatsEffectSuite { """ val compiledQuery = StarWarsMapping.compiler.compile(query).toOption.get.query - val res = StarWarsMapping.querySizeValidator.querySize(compiledQuery) + val res = StarWarsMapping.querySizeValidator.querySize(compiledQuery, Map.empty) assertEquals(res, ((2,1))) } @@ -108,7 +108,7 @@ class QuerySizeSuite extends CatsEffectSuite { """ val compiledQuery = StarWarsMapping.compiler.compile(query).toOption.get.query - val res = StarWarsMapping.querySizeValidator.querySize(compiledQuery) + val res = StarWarsMapping.querySizeValidator.querySize(compiledQuery, Map.empty) assertEquals(res, ((2,1))) } @@ -133,7 +133,7 @@ class QuerySizeSuite extends CatsEffectSuite { """ val compiledQuery = StarWarsMapping.compiler.compile(query).toOption.get.query - val res = StarWarsMapping.querySizeValidator.querySize(compiledQuery) + val res = StarWarsMapping.querySizeValidator.querySize(compiledQuery, Map.empty) assertEquals(res, ((3,5))) } @@ -149,7 +149,7 @@ class QuerySizeSuite extends CatsEffectSuite { """ val compiledQuery = StarWarsMapping.compiler.compile(query).toOption.get.query - val res = StarWarsMapping.querySizeValidator.querySize(compiledQuery) + val res = StarWarsMapping.querySizeValidator.querySize(compiledQuery, Map.empty) assert(res._2 == 2) } @@ -172,7 +172,7 @@ class QuerySizeSuite extends CatsEffectSuite { """ val compiledQuery = StarWarsMapping.compiler.compile(query).toOption.get.query - val res = StarWarsMapping.querySizeValidator.querySize(compiledQuery) + val res = StarWarsMapping.querySizeValidator.querySize(compiledQuery, Map.empty) assert(res._2 == 5) } diff --git a/modules/core/src/test/scala/compiler/ScalarsSuite.scala b/modules/core/src/test/scala/compiler/ScalarsSuite.scala index 9a46b8d9..1faf34d6 100644 --- a/modules/core/src/test/scala/compiler/ScalarsSuite.scala +++ b/modules/core/src/test/scala/compiler/ScalarsSuite.scala @@ -15,7 +15,8 @@ import munit.CatsEffectSuite import edu.gemini.grackle._ import edu.gemini.grackle.syntax._ -import Query._, Predicate._, Value._ +import Query._ +import Predicate._, Value._ import QueryCompiler._ import io.circe.Encoder @@ -168,8 +169,8 @@ object MovieMapping extends ValueMapping[IO] { } object GenreValue { - def unapply(e: TypedEnumValue): Option[Genre] = - Genre.fromString(e.value.name) + def unapply(e: EnumValue): Option[Genre] = + Genre.fromString(e.name) } object DateValue { @@ -192,48 +193,46 @@ object MovieMapping extends ValueMapping[IO] { Try(Duration.parse(s.value)).toOption } - override val selectElaborator = new SelectElaborator(Map( - QueryType -> { - case Select("movieById", List(Binding("id", UUIDValue(id))), child) => - Select("movieById", Nil, Unique(Filter(Eql(MovieType / "id", Const(id)), child))).success - case Select("moviesByGenre", List(Binding("genre", GenreValue(genre))), child) => - Select("moviesByGenre", Nil, Filter(Eql(MovieType / "genre", Const(genre)), child)).success - case Select("moviesReleasedBetween", List(Binding("from", DateValue(from)), Binding("to", DateValue(to))), child) => - Select("moviesReleasedBetween", Nil, - Filter( - And( - Not(Lt(MovieType / "releaseDate", Const(from))), - Lt(MovieType / "releaseDate", Const(to)) - ), - child - ) - ).success - case Select("moviesLongerThan", List(Binding("duration", IntervalValue(duration))), child) => - Select("moviesLongerThan", Nil, - Filter( - Not(Lt(MovieType / "duration", Const(duration))), - child - ) - ).success - case Select("moviesShownLaterThan", List(Binding("time", TimeValue(time))), child) => - Select("moviesShownLaterThan", Nil, - Filter( - Not(Lt(MovieType / "showTime", Const(time))), - child - ) - ).success - case Select("moviesShownBetween", List(Binding("from", DateTimeValue(from)), Binding("to", DateTimeValue(to))), child) => - Select("moviesShownBetween", Nil, - Filter( - And( - Not(Lt(MovieType / "nextShowing", Const(from))), - Lt(MovieType / "nextShowing", Const(to)) - ), - child - ) - ).success - } - )) + override val selectElaborator = SelectElaborator { + case (QueryType, "movieById", List(Binding("id", UUIDValue(id)))) => + Elab.transformChild(child => Unique(Filter(Eql(MovieType / "id", Const(id)), child))) + case (QueryType, "moviesByGenre", List(Binding("genre", GenreValue(genre)))) => + Elab.transformChild(child => Filter(Eql(MovieType / "genre", Const(genre)), child)) + case (QueryType, "moviesReleasedBetween", List(Binding("from", DateValue(from)), Binding("to", DateValue(to)))) => + Elab.transformChild(child => + Filter( + And( + Not(Lt(MovieType / "releaseDate", Const(from))), + Lt(MovieType / "releaseDate", Const(to)) + ), + child + ) + ) + case (QueryType, "moviesLongerThan", List(Binding("duration", IntervalValue(duration)))) => + Elab.transformChild(child => + Filter( + Not(Lt(MovieType / "duration", Const(duration))), + child + ) + ) + case (QueryType, "moviesShownLaterThan", List(Binding("time", TimeValue(time)))) => + Elab.transformChild(child => + Filter( + Not(Lt(MovieType / "showTime", Const(time))), + child + ) + ) + case (QueryType, "moviesShownBetween", List(Binding("from", DateTimeValue(from)), Binding("to", DateTimeValue(to)))) => + Elab.transformChild(child => + Filter( + And( + Not(Lt(MovieType / "nextShowing", Const(from))), + Lt(MovieType / "nextShowing", Const(to)) + ), + child + ) + ) + } } final class ScalarsSuite extends CatsEffectSuite { diff --git a/modules/core/src/test/scala/compiler/SkipIncludeSuite.scala b/modules/core/src/test/scala/compiler/SkipIncludeSuite.scala index e33d272a..796fb565 100644 --- a/modules/core/src/test/scala/compiler/SkipIncludeSuite.scala +++ b/modules/core/src/test/scala/compiler/SkipIncludeSuite.scala @@ -38,8 +38,8 @@ final class SkipIncludeSuite extends CatsEffectSuite { val expected = Group(List( - Rename("b", Select("field", Nil, Select("subfieldB", Nil, Empty))), - Rename("c", Select("field", Nil, Select("subfieldA", Nil, Empty))) + Select("field", Some("b"), Select("subfieldB")), + Select("field", Some("c"), Select("subfieldA")) )) val compiled = SkipIncludeMapping.compiler.compile(query, untypedVars = Some(variables)) @@ -79,24 +79,20 @@ final class SkipIncludeSuite extends CatsEffectSuite { val expected = Group(List( - Rename("a", Select("field", Nil, Empty)), - Rename("b", - Select("field", Nil, - Group(List( - Select("subfieldA", Nil, Empty), - Select("subfieldB", Nil, Empty) - )) - ) + Select("field", Some("a")), + Select("field", Some("b"), + Group(List( + Select("subfieldA"), + Select("subfieldB") + )) ), - Rename("c", - Select("field", Nil, - Group(List( - Select("subfieldA", Nil, Empty), - Select("subfieldB", Nil, Empty) - )) - ) + Select("field", Some("c"), + Group(List( + Select("subfieldA"), + Select("subfieldB") + )) ), - Rename("d", Select("field", Nil, Empty)) + Select("field", Some("d")) )) val compiled = SkipIncludeMapping.compiler.compile(query, untypedVars = Some(variables)) @@ -128,10 +124,10 @@ final class SkipIncludeSuite extends CatsEffectSuite { """ val expected = - Select("field", Nil, + Select("field", Group(List( - Rename("b", Select("subfieldB", Nil, Empty)), - Rename("c", Select("subfieldA", Nil, Empty)) + Select("subfieldB", Some("b")), + Select("subfieldA", Some("c")) )) ) @@ -179,24 +175,20 @@ final class SkipIncludeSuite extends CatsEffectSuite { val expected = Group(List( - Rename("a", Select("field", Nil, Empty)), - Rename("b", - Select("field", Nil, - Group(List( - Select("subfieldA", Nil, Empty), - Select("subfieldB", Nil, Empty) - )) - ) + Select("field", Some("a")), + Select("field", Some("b"), + Group(List( + Select("subfieldA"), + Select("subfieldB") + )) ), - Rename("c", - Select("field", Nil, - Group(List( - Select("subfieldA", Nil, Empty), - Select("subfieldB", Nil, Empty) - )) - ) + Select("field", Some("c"), + Group(List( + Select("subfieldA"), + Select("subfieldB") + )) ), - Rename("d", Select("field", Nil, Empty)) + Select("field", Some("d")) )) val compiled = SkipIncludeMapping.compiler.compile(query, untypedVars = Some(variables)) @@ -226,10 +218,10 @@ final class SkipIncludeSuite extends CatsEffectSuite { """ val expected = - Select("field", Nil, + Select("field", Group(List( - Rename("b", Select("subfieldB", Nil, Empty)), - Rename("c", Select("subfieldA", Nil, Empty)) + Select("subfieldB", Some("b")), + Select("subfieldA", Some("c")) )) ) diff --git a/modules/core/src/test/scala/compiler/VariablesSuite.scala b/modules/core/src/test/scala/compiler/VariablesSuite.scala index 956828af..54c1b869 100644 --- a/modules/core/src/test/scala/compiler/VariablesSuite.scala +++ b/modules/core/src/test/scala/compiler/VariablesSuite.scala @@ -9,8 +9,8 @@ import munit.CatsEffectSuite import edu.gemini.grackle._ import edu.gemini.grackle.syntax._ -import Query._, Value._ -import QueryCompiler._ +import Query._ +import Value._ final class VariablesSuite extends CatsEffectSuite { test("simple variables query") { @@ -31,11 +31,11 @@ final class VariablesSuite extends CatsEffectSuite { """ val expected = - Select("user", List(Binding("id", IDValue("4"))), + UntypedSelect("user", None, List(Binding("id", IDValue("4"))), Nil, Group(List( - Select("id", Nil, Empty), - Select("name", Nil, Empty), - Select("profilePic", List(Binding("size", IntValue(60))), Empty) + UntypedSelect("id", None, Nil, Nil, Empty), + UntypedSelect("name", None, Nil, Nil, Empty), + UntypedSelect("profilePic", None, List(Binding("size", IntValue(60))), Nil, Empty) )) ) @@ -60,9 +60,10 @@ final class VariablesSuite extends CatsEffectSuite { """ val expected = - Select("users", + UntypedSelect("users", None, List(Binding("ids", ListValue(List(IDValue("1"), IDValue("2"), IDValue("3"))))), - Select("name", Nil, Empty) + Nil, + UntypedSelect("name", None, Nil, Nil, Empty) ) val compiled = VariablesMapping.compiler.compile(query, untypedVars = Some(variables)) @@ -86,9 +87,10 @@ final class VariablesSuite extends CatsEffectSuite { """ val expected = - Select("usersByType", - List(Binding("userType", TypedEnumValue(EnumValue("ADMIN", None, false, None)))), - Select("name", Nil, Empty) + UntypedSelect("usersByType", None, + List(Binding("userType", EnumValue("ADMIN"))), + Nil, + UntypedSelect("name", None, Nil, Nil, Empty) ) val compiled = VariablesMapping.compiler.compile(query, untypedVars = Some(variables)) @@ -112,9 +114,10 @@ final class VariablesSuite extends CatsEffectSuite { """ val expected = - Select("usersLoggedInByDate", + UntypedSelect("usersLoggedInByDate", None, List(Binding("date", StringValue("2021-02-22"))), - Select("name", Nil, Empty) + Nil, + UntypedSelect("name", None, Nil, Nil, Empty) ) val compiled = VariablesMapping.compiler.compile(query, untypedVars = Some(variables)) @@ -138,9 +141,10 @@ final class VariablesSuite extends CatsEffectSuite { """ val expected = - Select("queryWithBigDecimal", + UntypedSelect("queryWithBigDecimal", None, List(Binding("input", IntValue(2021))), - Select("name", Nil, Empty) + Nil, + UntypedSelect("name", None, Nil, Nil, Empty) ) val compiled = VariablesMapping.compiler.compile(query, untypedVars = Some(variables)) @@ -169,7 +173,7 @@ final class VariablesSuite extends CatsEffectSuite { """ val expected = - Select("search", + UntypedSelect("search", None, List(Binding("pattern", ObjectValue(List( ("name", StringValue("Foo")), @@ -179,9 +183,10 @@ final class VariablesSuite extends CatsEffectSuite { ("date", AbsentValue) )) )), + Nil, Group(List( - Select("name", Nil, Empty), - Select("id", Nil, Empty) + UntypedSelect("name", None, Nil, Nil, Empty), + UntypedSelect("id", None, Nil, Nil, Empty) )) ) @@ -211,7 +216,7 @@ final class VariablesSuite extends CatsEffectSuite { } """ - val expected = Problem("Unknown field(s) 'quux' in input object value of type Pattern") + val expected = Problem("Unknown field(s) 'quux' in input object value of type Pattern in variable values") val compiled = VariablesMapping.compiler.compile(query, untypedVars = Some(variables)) @@ -234,9 +239,10 @@ final class VariablesSuite extends CatsEffectSuite { """ val expected = - Select("users", + UntypedSelect("users", None, List(Binding("ids", ListValue(List(IDValue("1"), IDValue("2"), IDValue("3"))))), - Select("name", Nil, Empty) + Nil, + UntypedSelect("name", None, Nil, Nil, Empty) ) val compiled = VariablesMapping.compiler.compile(query, untypedVars = Some(variables)) @@ -261,7 +267,7 @@ final class VariablesSuite extends CatsEffectSuite { """ val expected = - Select("search", + UntypedSelect("search", None, List(Binding("pattern", ObjectValue(List( ("name", StringValue("Foo")), @@ -271,9 +277,10 @@ final class VariablesSuite extends CatsEffectSuite { ("date", AbsentValue) )) )), + Nil, Group(List( - Select("name", Nil, Empty), - Select("id", Nil, Empty) + UntypedSelect("name", None, Nil, Nil, Empty), + UntypedSelect("id", None, Nil, Nil, Empty) )) ) @@ -299,19 +306,20 @@ final class VariablesSuite extends CatsEffectSuite { """ val expected = - Select("search", + UntypedSelect("search", None, List(Binding("pattern", ObjectValue(List( ("name", StringValue("Foo")), ("age", IntValue(23)), ("id", IDValue("123")), - ("userType", TypedEnumValue(EnumValue("ADMIN", None, false, None))), + ("userType", EnumValue("ADMIN")), ("date", AbsentValue) )) )), + Nil, Group(List( - Select("name", Nil, Empty), - Select("id", Nil, Empty) + UntypedSelect("name", None, Nil, Nil, Empty), + UntypedSelect("id", None, Nil, Nil, Empty) )) ) @@ -337,7 +345,7 @@ final class VariablesSuite extends CatsEffectSuite { """ val expected = - Select("search", + UntypedSelect("search", None, List(Binding("pattern", ObjectValue(List( ("name", StringValue("Foo")), @@ -347,9 +355,10 @@ final class VariablesSuite extends CatsEffectSuite { ("date", StringValue("2021-02-22")) )) )), + Nil, Group(List( - Select("name", Nil, Empty), - Select("id", Nil, Empty) + UntypedSelect("name", None, Nil, Nil, Empty), + UntypedSelect("id", None, Nil, Nil, Empty) )) ) @@ -390,9 +399,5 @@ object VariablesMapping extends TestMapping { scalar BigDecimal """ - val QueryType = schema.ref("Query") - - override val selectElaborator = new SelectElaborator(Map( - QueryType -> PartialFunction.empty - )) + override val selectElaborator = PreserveArgsElaborator } diff --git a/modules/core/src/test/scala/composed/ComposedData.scala b/modules/core/src/test/scala/composed/ComposedData.scala index 2a951192..ebf9ebd5 100644 --- a/modules/core/src/test/scala/composed/ComposedData.scala +++ b/modules/core/src/test/scala/composed/ComposedData.scala @@ -8,7 +8,8 @@ import cats.implicits._ import edu.gemini.grackle._ import edu.gemini.grackle.syntax._ -import Query._, Predicate._, Value._ +import Query._ +import Predicate._, Value._ import QueryCompiler._ /* Currency component */ @@ -61,12 +62,10 @@ object CurrencyMapping extends ValueMapping[IO] { ) ) - override val selectElaborator = new SelectElaborator(Map( - QueryType -> { - case Select("fx", List(Binding("code", StringValue(code))), child) => - Select("fx", Nil, Unique(Filter(Eql(CurrencyType / "code", Const(code)), child))).success - } - )) + override val selectElaborator = SelectElaborator { + case (QueryType, "fx", List(Binding("code", StringValue(code)))) => + Elab.transformChild(child => Unique(Filter(Eql(CurrencyType / "code", Const(code)), child))) + } } /* Country component */ @@ -123,14 +122,10 @@ object CountryMapping extends ValueMapping[IO] { ) ) - override val selectElaborator = new SelectElaborator(Map( - QueryType -> { - case Select("country", List(Binding("code", StringValue(code))), child) => - Select("country", Nil, Unique(Filter(Eql(CurrencyMapping.CurrencyType / "code", Const(code)), child))).success - case Select("countries", _, child) => - Select("countries", Nil, child).success - } - )) + override val selectElaborator = SelectElaborator { + case (QueryType, "country", List(Binding("code", StringValue(code)))) => + Elab.transformChild(child => Unique(Filter(Eql(CurrencyMapping.CurrencyType / "code", Const(code)), child))) + } } /* Composition */ @@ -158,16 +153,12 @@ object ComposedMapping extends ComposedMapping[IO] { val CountryType = schema.ref("Country") val CurrencyType = schema.ref("Currency") - override val selectElaborator = new SelectElaborator(Map( - QueryType -> { - case Select("fx", List(Binding("code", StringValue(code))), child) => - Select("fx", Nil, Unique(Filter(Eql(CurrencyType / "code", Const(code)), child))).success - case Select("country", List(Binding("code", StringValue(code))), child) => - Select("country", Nil, Unique(Filter(Eql(CountryType / "code", Const(code)), child))).success - case Select("countries", _, child) => - Select("countries", Nil, child).success - } - )) + override val selectElaborator = SelectElaborator { + case (QueryType, "fx", List(Binding("code", StringValue(code)))) => + Elab.transformChild(child => Unique(Filter(Eql(CurrencyType / "code", Const(code)), child))) + case (QueryType, "country", List(Binding("code", StringValue(code)))) => + Elab.transformChild(child => Unique(Filter(Eql(CountryType / "code", Const(code)), child))) + } val typeMappings = List( @@ -192,7 +183,7 @@ object ComposedMapping extends ComposedMapping[IO] { def countryCurrencyJoin(q: Query, c: Cursor): Result[Query] = (c.focus, q) match { case (c: CountryData.Country, Select("currency", _, child)) => - Select("fx", Nil, Unique(Filter(Eql(CurrencyType / "code", Const(c.currencyCode)), child))).success + Select("fx", Unique(Filter(Eql(CurrencyType / "code", Const(c.currencyCode)), child))).success case _ => Result.internalError(s"Unexpected cursor focus type in countryCurrencyJoin") } diff --git a/modules/core/src/test/scala/composed/ComposedListSuite.scala b/modules/core/src/test/scala/composed/ComposedListSuite.scala index f7066549..86a92f07 100644 --- a/modules/core/src/test/scala/composed/ComposedListSuite.scala +++ b/modules/core/src/test/scala/composed/ComposedListSuite.scala @@ -10,7 +10,8 @@ import munit.CatsEffectSuite import edu.gemini.grackle._ import edu.gemini.grackle.syntax._ -import Query._, Predicate._, Value._ +import Query._ +import Predicate._, Value._ import QueryCompiler._ object CollectionData { @@ -62,12 +63,10 @@ object CollectionMapping extends ValueMapping[IO] { ) ) - override val selectElaborator = new SelectElaborator(Map( - QueryType -> { - case Select("collectionByName", List(Binding("name", StringValue(name))), child) => - Select("collectionByName", Nil, Unique(Filter(Eql(CollectionType / "name", Const(name)), child))).success - } - )) + override val selectElaborator = SelectElaborator { + case (QueryType, "collectionByName", List(Binding("name", StringValue(name)))) => + Elab.transformChild(child => Unique(Filter(Eql(CollectionType / "name", Const(name)), child))) + } } object ItemData { @@ -111,12 +110,10 @@ object ItemMapping extends ValueMapping[IO] { ) ) - override val selectElaborator = new SelectElaborator(Map( - QueryType -> { - case Select("itemById", List(Binding("id", IDValue(id))), child) => - Select("itemById", Nil, Unique(Filter(Eql(ItemType / "id", Const(id)), child))).success - } - )) + override val selectElaborator = SelectElaborator { + case (QueryType, "itemById", List(Binding("id", IDValue(id)))) => + Elab.transformChild(child => Unique(Filter(Eql(ItemType / "id", Const(id)), child))) + } } object ComposedListMapping extends ComposedMapping[IO] { @@ -164,19 +161,17 @@ object ComposedListMapping extends ComposedMapping[IO] { ) ) - override val selectElaborator = new SelectElaborator(Map( - QueryType -> { - case Select("itemById", List(Binding("id", IDValue(id))), child) => - Select("itemById", Nil, Unique(Filter(Eql(ItemType / "id", Const(id)), child))).success - case Select("collectionByName", List(Binding("name", StringValue(name))), child) => - Select("collectionByName", Nil, Unique(Filter(Eql(CollectionType / "name", Const(name)), child))).success - } - )) + override val selectElaborator = SelectElaborator { + case (QueryType, "itemById", List(Binding("id", IDValue(id)))) => + Elab.transformChild(child => Unique(Filter(Eql(ItemType / "id", Const(id)), child))) + case (QueryType, "collectionByName", List(Binding("name", StringValue(name)))) => + Elab.transformChild(child => Unique(Filter(Eql(CollectionType / "name", Const(name)), child))) + } def collectionItemJoin(q: Query, c: Cursor): Result[Query] = (c.focus, q) match { case (c: CollectionData.Collection, Select("items", _, child)) => - Group(c.itemIds.map(id => Select("itemById", Nil, Unique(Filter(Eql(ItemType / "id", Const(id)), child))))).success + Group(c.itemIds.map(id => Select("itemById", Unique(Filter(Eql(ItemType / "id", Const(id)), child))))).success case _ => Result.internalError(s"Unexpected cursor focus type in collectionItemJoin") } diff --git a/modules/core/src/test/scala/directives/DirectiveValidationSuite.scala b/modules/core/src/test/scala/directives/DirectiveValidationSuite.scala new file mode 100644 index 00000000..021fa0ed --- /dev/null +++ b/modules/core/src/test/scala/directives/DirectiveValidationSuite.scala @@ -0,0 +1,369 @@ +// Copyright (c) 2016-2020 Association of Universities for Research in Astronomy, Inc. (AURA) +// For license information see LICENSE or https://opensource.org/licenses/BSD-3-Clause + +package directives + +import cats.MonadThrow +import cats.data.Chain +import cats.effect.IO +import cats.implicits._ +import munit.CatsEffectSuite + +import compiler.PreserveArgsElaborator +import edu.gemini.grackle._ +import edu.gemini.grackle.syntax._ +import Query._ + +final class DirectiveValidationSuite extends CatsEffectSuite { + test("Schema with validly located directives") { + val schema = + Schema( + """ + schema @onSchema { + query: Query @onFieldDefinition + } + + type Query @onObject { + field: Interface @onFieldDefinition + } + + interface Interface @onInterface { + field: String @onFieldDefinition + } + + type Object1 implements Interface @onObject { + field: String @onFieldDefinition + fieldWithArg(arg: Input @onArgumentDefinition): String @onFieldDefinition + unionField: Union @onFieldDefinition + enumField: Enum @onFieldDefinition + scalarField: Scalar @onFieldDefinition + } + + type Object2 implements Interface @onObject { + field: String @onFieldDefinition + } + + union Union @onUnion = Object1 | Object2 + + enum Enum @onEnum { + FOO @onEnumValue + BAR @onEnumValue + } + + scalar Scalar @onScalar + + input Input @onInputObject { + field: String @onInputFieldDefinition + } + + directive @onSchema on SCHEMA + directive @onScalar on SCALAR + directive @onObject on OBJECT + directive @onFieldDefinition on FIELD_DEFINITION + directive @onArgumentDefinition on ARGUMENT_DEFINITION + directive @onInterface on INTERFACE + directive @onUnion on UNION + directive @onEnum on ENUM + directive @onEnumValue on ENUM_VALUE + directive @onInputObject on INPUT_OBJECT + directive @onInputFieldDefinition on INPUT_FIELD_DEFINITION + """ + ) + + assertEquals(schema.toProblems, Chain.empty) + } + + test("Schema with invalidly located directives") { + val schema = + Schema( + """ + schema @onFieldDefinition { + query: Query @onSchema + } + + type Query @onSchema { + field: Interface @onSchema + } + + interface Interface @onSchema { + field: String @onSchema + } + + type Object1 implements Interface @onSchema { + field: String @onSchema + fieldWithArg(arg: Input @onSchema): String @onSchema + unionField: Union @onSchema + enumField: Enum @onSchema + scalarField: Scalar @onSchema + } + + type Object2 implements Interface @onSchema { + field: String @onSchema + } + + union Union @onSchema = Object1 | Object2 + + enum Enum @onSchema { + FOO @onSchema + BAR @onSchema + } + + scalar Scalar @onSchema + + input Input @onSchema { + field: String @onSchema + } + + directive @onSchema on SCHEMA + directive @onFieldDefinition on FIELD_DEFINITION + """ + ) + + val problems = + Chain( + Problem("Directive 'onFieldDefinition' is not allowed on SCHEMA"), + Problem("Directive 'onSchema' is not allowed on FIELD_DEFINITION"), + Problem("Directive 'onSchema' is not allowed on OBJECT"), + Problem("Directive 'onSchema' is not allowed on FIELD_DEFINITION"), + Problem("Directive 'onSchema' is not allowed on INTERFACE"), + Problem("Directive 'onSchema' is not allowed on FIELD_DEFINITION"), + Problem("Directive 'onSchema' is not allowed on OBJECT"), + Problem("Directive 'onSchema' is not allowed on FIELD_DEFINITION"), + Problem("Directive 'onSchema' is not allowed on FIELD_DEFINITION"), + Problem("Directive 'onSchema' is not allowed on ARGUMENT_DEFINITION"), + Problem("Directive 'onSchema' is not allowed on FIELD_DEFINITION"), + Problem("Directive 'onSchema' is not allowed on FIELD_DEFINITION"), + Problem("Directive 'onSchema' is not allowed on FIELD_DEFINITION"), + Problem("Directive 'onSchema' is not allowed on OBJECT"), + Problem("Directive 'onSchema' is not allowed on FIELD_DEFINITION"), + Problem("Directive 'onSchema' is not allowed on UNION"), + Problem("Directive 'onSchema' is not allowed on ENUM"), + Problem("Directive 'onSchema' is not allowed on ENUM_VALUE"), + Problem("Directive 'onSchema' is not allowed on ENUM_VALUE"), + Problem("Directive 'onSchema' is not allowed on SCALAR"), + Problem("Directive 'onSchema' is not allowed on INPUT_OBJECT"), + Problem("Directive 'onSchema' is not allowed on INPUT_FIELD_DEFINITION") + ) + + assertEquals(schema.toProblems, problems) + } + + test("Schema with invalid directive arguments") { + val schema = + Schema( + """ + type Query { + f1: Int @withArg(i: 1) + f2: Int @withArg + f3: Int @withArg(i: "foo") + f4: Int @withArg(x: "foo") + f5: Int @withRequiredArg(i: 1) + f6: Int @withRequiredArg + f7: Int @withRequiredArg(i: "foo") + f8: Int @withRequiredArg(x: "foo") + f9: Int @deprecated(reason: "foo") + f10: Int @deprecated + f11: Int @deprecated(reason: 1) + f12: Int @deprecated(x: "foo") + } + + directive @withArg(i: Int) on FIELD_DEFINITION + directive @withRequiredArg(i: Int!) on FIELD_DEFINITION + """ + ) + + val problems = + Chain( + Problem("""Expected Int found '"foo"' for 'i' in directive withArg"""), + Problem("""Unknown argument(s) 'x' in directive withArg"""), + Problem("""Value of type Int required for 'i' in directive withRequiredArg"""), + Problem("""Expected Int found '"foo"' for 'i' in directive withRequiredArg"""), + Problem("""Unknown argument(s) 'x' in directive withRequiredArg"""), + Problem("""Expected String found '1' for 'reason' in directive deprecated"""), + Problem("""Unknown argument(s) 'x' in directive deprecated"""), + ) + + assertEquals(schema.toProblems, problems) + } + + test("Query with validly located directives") { + val expected = + List( + Operation( + UntypedSelect("foo", None, Nil, List(Directive("onField", Nil)), + Group( + List( + UntypedSelect("bar",None, Nil, List(Directive("onField", Nil)), Empty), + UntypedSelect("bar",None, Nil, List(Directive("onField", Nil)), Empty) + ) + ) + ), + ExecutableDirectiveMapping.QueryType, + List(Directive("onQuery",List())) + ), + Operation( + UntypedSelect("foo",None, Nil, List(Directive("onField", Nil)), Empty), + ExecutableDirectiveMapping.MutationType, + List(Directive("onMutation",List())) + ), + Operation( + UntypedSelect("foo",None, Nil, List(Directive("onField", Nil)), Empty), + ExecutableDirectiveMapping.SubscriptionType, + List(Directive("onSubscription",List())) + ) + ) + + val query = + """|query ($var: Boolean @onVariableDefinition) @onQuery { + | foo @onField { + | ...Frag @onFragmentSpread + | ... @onInlineFragment { + | bar @onField + | } + | } + |} + | + |mutation @onMutation { + | foo @onField + |} + | + |subscription @onSubscription { + | foo @onField + |} + | + |fragment Frag on Foo @onFragmentDefinition { + | bar @onField + |} + |""".stripMargin + + val res = ExecutableDirectiveMapping.compileAllOperations(query) + //println(res) + + assertEquals(res, expected.success) + } + + test("Query with invalidly located directives") { + val problems = + Chain( + Problem("Directive 'onField' is not allowed on VARIABLE_DEFINITION"), + Problem("Directive 'onField' is not allowed on QUERY"), + Problem("Directive 'onQuery' is not allowed on FIELD"), + Problem("Directive 'onField' is not allowed on FRAGMENT_SPREAD"), + Problem("Directive 'onField' is not allowed on INLINE_FRAGMENT"), + Problem("Directive 'onQuery' is not allowed on FIELD"), + Problem("Directive 'onField' is not allowed on FRAGMENT_DEFINITION"), + Problem("Directive 'onQuery' is not allowed on FIELD"), + Problem("Directive 'onField' is not allowed on MUTATION"), + Problem("Directive 'onQuery' is not allowed on FIELD"), + Problem("Directive 'onField' is not allowed on FRAGMENT_DEFINITION"), + Problem("Directive 'onQuery' is not allowed on FIELD"), + Problem("Directive 'onField' is not allowed on SUBSCRIPTION"), + Problem("Directive 'onQuery' is not allowed on FIELD"), + Problem("Directive 'onField' is not allowed on FRAGMENT_DEFINITION"), + Problem("Directive 'onQuery' is not allowed on FIELD") + ) + + val query = + """|query ($var: Boolean @onField) @onField { + | foo @onQuery { + | ...Frag @onField + | ... @onField { + | bar @onQuery + | } + | } + |} + | + |mutation @onField { + | foo @onQuery + |} + | + |subscription @onField { + | foo @onQuery + |} + | + |fragment Frag on Foo @onField { + | bar @onQuery + |} + |""".stripMargin + + val res = ExecutableDirectiveMapping.compileAllOperations(query) + //println(res) + + assertEquals(res.toProblems, problems) + } + + test("Query with invalid directive arguments") { + val problems = + Chain( + Problem("""Expected Int found '"foo"' for 'i' in directive withArg"""), + Problem("""Unknown argument(s) 'x' in directive withArg"""), + Problem("""Value of type Int required for 'i' in directive withRequiredArg"""), + Problem("""Expected Int found '"foo"' for 'i' in directive withRequiredArg"""), + Problem("""Unknown argument(s) 'x' in directive withRequiredArg""") + ) + + val query = + """|query { + | foo { + | b1: bar @withArg(i: 1) + | b2: bar @withArg + | b3: bar @withArg(i: "foo") + | b4: bar @withArg(x: "foo") + | b5: bar @withRequiredArg(i: 1) + | b6: bar @withRequiredArg + | b7: bar @withRequiredArg(i: "foo") + | b8: bar @withRequiredArg(x: "foo") + | } + |} + |""".stripMargin + + val res = ExecutableDirectiveMapping.compileAllOperations(query) + //println(res) + + assertEquals(res.toProblems, problems) + } +} + +object ExecutableDirectiveMapping extends Mapping[IO] { + val M: MonadThrow[IO] = MonadThrow[IO] + + val schema = + schema""" + type Query { + foo: Foo + } + type Mutation { + foo: String + } + type Subscription { + foo: String + } + type Foo { + bar: String + } + directive @onQuery on QUERY + directive @onMutation on MUTATION + directive @onSubscription on SUBSCRIPTION + directive @onField on FIELD + directive @onFragmentDefinition on FRAGMENT_DEFINITION + directive @onFragmentSpread on FRAGMENT_SPREAD + directive @onInlineFragment on INLINE_FRAGMENT + directive @onVariableDefinition on VARIABLE_DEFINITION + + directive @withArg(i: Int) on FIELD + directive @withRequiredArg(i: Int!) on FIELD + """ + + val QueryType = schema.queryType + val MutationType = schema.mutationType.get + val SubscriptionType = schema.subscriptionType.get + + val typeMappings: List[TypeMapping] = Nil + + override val selectElaborator = PreserveArgsElaborator + + def compileAllOperations(text: String): Result[List[Operation]] = + QueryParser.parseText(text).flatMap { + case (ops, frags) => ops.parTraverse(compiler.compileOperation(_, None, frags)) + } +} diff --git a/modules/core/src/test/scala/directives/QueryDirectivesSuite.scala b/modules/core/src/test/scala/directives/QueryDirectivesSuite.scala new file mode 100644 index 00000000..86d9e72d --- /dev/null +++ b/modules/core/src/test/scala/directives/QueryDirectivesSuite.scala @@ -0,0 +1,198 @@ +// Copyright (c) 2016-2020 Association of Universities for Research in Astronomy, Inc. (AURA) +// For license information see LICENSE or https://opensource.org/licenses/BSD-3-Clause + +package directives + +import cats.effect.IO +import cats.syntax.all._ +import io.circe.literal._ +import munit.CatsEffectSuite + +import edu.gemini.grackle._ +import edu.gemini.grackle.syntax._ +import Query._, QueryCompiler._ + +final class QueryDirectivesSuite extends CatsEffectSuite { + test("simple query") { + val query = """ + query { + user { + name + handle + age + } + } + """ + + val expected = json""" + { + "data" : { + "user" : { + "name" : "Mary", + "handle" : "mary", + "age" : 42 + } + } + } + """ + + val res = QueryDirectivesMapping.compileAndRun(query) + + //res.flatMap(IO.println) *> + assertIO(res, expected) + } + + test("query with directive (1)") { + val query = """ + query { + user { + name @upperCase + handle + age + } + } + """ + + val expected = json""" + { + "data" : { + "user" : { + "name" : "MARY", + "handle" : "mary", + "age" : 42 + } + } + } + """ + + val res = QueryDirectivesMapping.compileAndRun(query) + + //res.flatMap(IO.println) *> + assertIO(res, expected) + } + + test("query with directive (2)") { + val query = """ + query { + user { + name @upperCase + handle @upperCase + age + } + } + """ + + val expected = json""" + { + "data" : { + "user" : { + "name" : "MARY", + "handle" : "MARY", + "age" : 42 + } + } + } + """ + + val res = QueryDirectivesMapping.compileAndRun(query) + + //res.flatMap(IO.println) *> + assertIO(res, expected) + } + + test("query with directive (3)") { + val query = """ + query { + user { + name @upperCase + handle + age @upperCase + } + } + """ + + val expected = json""" + { + "errors" : [ + { + "message" : "'upperCase' directive may only be applied to fields of type String" + } + ], + "data" : { + "user" : { + "name" : "MARY", + "handle" : "mary", + "age" : 42 + } + } + } + """ + + val res = QueryDirectivesMapping.compileAndRun(query) + + //res.flatMap(IO.println) *> + assertIO(res, expected) + } +} + +object QueryDirectivesMapping extends ValueMapping[IO] { + val schema = + schema""" + type Query { + user: User! + } + type User { + name: String! + handle: String! + age: Int! + } + directive @upperCase on FIELD + """ + + val QueryType = schema.ref("Query") + val UserType = schema.ref("User") + + val typeMappings = + List( + ValueObjectMapping[Unit]( + tpe = QueryType, + fieldMappings = + List( + ValueField("user", _ => ()) + ) + ), + ValueObjectMapping[Unit]( + tpe = UserType, + fieldMappings = + List( + ValueField("name", _ => "Mary"), + ValueField("handle", _ => "mary"), + ValueField("age", _ => 42) + ) + ) + ) + + object upperCaseElaborator extends Phase { + override def transform(query: Query): Elab[Query] = + query match { + case UntypedSelect(nme, alias, _, directives, _) if directives.exists(_.name == "upperCase") => + for { + c <- Elab.context + fc <- Elab.liftR(c.forField(nme, alias)) + res <- if (fc.tpe =:= ScalarType.StringType) + super.transform(query).map(TransformCursor(toUpperCase, _)) + else + // We could make this fail the whole query by yielding Elab.failure here + Elab.warning(s"'upperCase' directive may only be applied to fields of type String") *> super.transform(query) + } yield res + case _ => + super.transform(query) + } + + def toUpperCase(c: Cursor): Result[Cursor] = + FieldTransformCursor[String](c, _.toUpperCase.success).success + } + + override def compilerPhases: List[QueryCompiler.Phase] = + List(upperCaseElaborator, selectElaborator, componentElaborator, effectElaborator) +} diff --git a/modules/core/src/test/scala/directives/SchemaDirectivesSuite.scala b/modules/core/src/test/scala/directives/SchemaDirectivesSuite.scala new file mode 100644 index 00000000..ce8a8fe1 --- /dev/null +++ b/modules/core/src/test/scala/directives/SchemaDirectivesSuite.scala @@ -0,0 +1,425 @@ +// Copyright (c) 2016-2020 Association of Universities for Research in Astronomy, Inc. (AURA) +// For license information see LICENSE or https://opensource.org/licenses/BSD-3-Clause + +package directives + +import cats.effect.IO +import io.circe.literal._ +import munit.CatsEffectSuite + +import edu.gemini.grackle._ +import edu.gemini.grackle.syntax._ +import Cursor._, Query._, QueryCompiler._, Value._ + +import SchemaDirectivesMapping.AuthStatus + +final class SchemaDirectivesSuite extends CatsEffectSuite { + test("No auth, success") { + val query = """ + query { + products + } + """ + + val expected = json""" + { + "data" : { + "products" : [ + "Cheese", + "Wine", + "Bread" + ] + } + } + """ + + val res = SchemaDirectivesMapping.compileAndRun(query) + + //res.flatMap(IO.println) *> + assertIO(res, expected) + } + + test("No auth, fail") { + val query = """ + query { + products + user { + name + email + } + } + """ + + val expected = json""" + { + "errors" : [ + { + "message" : "Unauthorized" + } + ] + } + """ + + val res = SchemaDirectivesMapping.compileAndRun(query) + + //res.flatMap(IO.println) *> + assertIO(res, expected) + } + + test("Authenticated user, success") { + val query = """ + query { + products + user { + name + } + } + """ + + val expected = json""" + { + "data" : { + "products" : [ + "Cheese", + "Wine", + "Bread" + ], + "user" : { + "name" : "Mary" + } + } + } + """ + + val res = SchemaDirectivesMapping.compileAndRun(query, env = Env("authStatus" -> AuthStatus("USER"))) + + //res.flatMap(IO.println) *> + assertIO(res, expected) + } + + test("Authenticated user, fail") { + val query = """ + query { + products + user { + name + email + } + } + """ + + val expected = json""" + { + "errors" : [ + { + "message" : "Unauthorized" + } + ] + } + """ + + val res = SchemaDirectivesMapping.compileAndRun(query, env = Env("authStatus" -> AuthStatus("USER"))) + + //res.flatMap(IO.println) *> + assertIO(res, expected) + } + + test("Authenticated admin, success") { + val query = """ + query { + products + user { + name + email + } + } + """ + + val expected = json""" + { + "data" : { + "products" : [ + "Cheese", + "Wine", + "Bread" + ], + "user" : { + "name" : "Mary", + "email" : "mary@example.com" + } + } + } + """ + + val res = SchemaDirectivesMapping.compileAndRun(query, env = Env("authStatus" -> AuthStatus("ADMIN"))) + + //res.flatMap(IO.println) *> + assertIO(res, expected) + } + + test("Authenticated user, warn with null") { + val query = """ + query { + products + user { + phone + } + } + """ + + val expected = json""" + { + "errors" : [ + { + "message" : "Unauthorized access to field 'phone' of type 'User'" + } + ], + "data" : { + "products" : [ + "Cheese", + "Wine", + "Bread" + ], + "user" : { + "phone" : null + } + } + } + """ + + val res = SchemaDirectivesMapping.compileAndRun(query, env = Env("authStatus" -> AuthStatus("USER"))) + + //res.flatMap(IO.println) *> + assertIO(res, expected) + } + + test("Authenticated admin, success with non-null") { + val query = """ + query { + products + user { + phone + } + } + """ + + val expected = json""" + { + "data" : { + "products" : [ + "Cheese", + "Wine", + "Bread" + ], + "user" : { + "phone" : "123456789" + } + } + } + """ + + val res = SchemaDirectivesMapping.compileAndRun(query, env = Env("authStatus" -> AuthStatus("ADMIN"))) + + //res.flatMap(IO.println) *> + assertIO(res, expected) + } + + test("Mutation, authenticated user, success") { + val query = """ + mutation { + needsBackup + } + """ + + val expected = json""" + { + "data" : { + "needsBackup" : true + } + } + """ + + val res = SchemaDirectivesMapping.compileAndRun(query, env = Env("authStatus" -> AuthStatus("USER"))) + + //res.flatMap(IO.println) *> + assertIO(res, expected) + } + + test("Mutation, no auth, fail") { + val query = """ + mutation { + needsBackup + } + """ + + val expected = json""" + { + "errors" : [ + { + "message" : "Unauthorized" + } + ] + } + """ + + val res = SchemaDirectivesMapping.compileAndRun(query) + + //res.flatMap(IO.println) *> + assertIO(res, expected) + } + + test("Mutation, authenticated admin, success") { + val query = """ + mutation { + backup + } + """ + + val expected = json""" + { + "data" : { + "backup" : true + } + } + """ + + val res = SchemaDirectivesMapping.compileAndRun(query, env = Env("authStatus" -> AuthStatus("ADMIN"))) + + //res.flatMap(IO.println) *> + assertIO(res, expected) + } + + test("Mutation, authenticated user, fail") { + val query = """ + mutation { + backup + } + """ + + val expected = json""" + { + "errors" : [ + { + "message" : "Unauthorized" + } + ] + } + """ + + val res = SchemaDirectivesMapping.compileAndRun(query, env = Env("authStatus" -> AuthStatus("USER"))) + + //res.flatMap(IO.println) *> + assertIO(res, expected) + } +} + +object SchemaDirectivesMapping extends ValueMapping[IO] { + val schema = + schema""" + type Query { + products: [String!]! + user: User! @authenticated + } + + type Mutation @authenticated { + needsBackup: Boolean! + backup: Boolean! @hasRole(requires: ADMIN) + } + + type User { + name: String! + email: String! @hasRole(requires: ADMIN) + phone: String @hasRole(requires: ADMIN) + } + + directive @authenticated on FIELD_DEFINITION | OBJECT + directive @hasRole(requires: Role = ADMIN) on FIELD_DEFINITION | OBJECT + + enum Role { + ADMIN + USER + } + """ + + val QueryType = schema.ref("Query") + val MutationType = schema.ref("Mutation") + val UserType = schema.ref("User") + + val typeMappings = + List( + ValueObjectMapping[Unit]( + tpe = QueryType, + fieldMappings = + List( + ValueField("products", _ => List("Cheese", "Wine", "Bread")), + ValueField("user", _ => ()) + ) + ), + ValueObjectMapping[Unit]( + tpe = MutationType, + fieldMappings = + List( + ValueField("needsBackup", _ => true), + ValueField("backup", _ => true) + ) + ), + ValueObjectMapping[Unit]( + tpe = UserType, + fieldMappings = + List( + ValueField("name", _ => "Mary"), + ValueField("email", _ => "mary@example.com"), + ValueField("phone", _ => Some("123456789")), + ) + ) + ) + + case class AuthStatus(role: String) + + object permissionsElaborator extends Phase { + override def transform(query: Query): Elab[Query] = { + def checkPermissions(c: Context, name: String, status: Option[AuthStatus], query: Query, nullAndWarn: Boolean): Elab[Query] = { + val dirs = c.tpe.directives ++ c.tpe.fieldInfo(name).map(_.directives).getOrElse(Nil) + val requiresAuth = dirs.exists(_.name == "authenticated") + val roles = + dirs.filter(_.name == "hasRole").flatMap(_.args.filter(_.name == "requires")).map(_.value).collect { + case EnumValue(role) => role + } + val requiresRole = + if (roles.contains("ADMIN")) Some(AuthStatus("ADMIN")) + else if (roles.contains("USER")) Some(AuthStatus("USER")) + else None + + (status, requiresAuth, requiresRole) match { + case (None, false, None) => Elab.pure(query) + case (Some(_), _, None) => Elab.pure(query) + case (Some(AuthStatus("ADMIN")), _, _) => Elab.pure(query) + case (Some(AuthStatus("USER")), _, Some(AuthStatus("USER"))) => + Elab.pure(query) + case _ => + if (!nullAndWarn) Elab.failure(s"Unauthorized") + else + for { + _ <- Elab.warning(s"Unauthorized access to field '$name' of type '${c.tpe}'") + } yield TransformCursor(NullFieldCursor(_).success, query) + } + } + + query match { + case s: UntypedSelect => + for { + c <- Elab.context + status <- Elab.env[AuthStatus]("authStatus") + query0 <- super.transform(query) + res <- checkPermissions(c, s.name, status, query0, s.name == "phone") + } yield res + + case _ => + super.transform(query) + } + } + } + + override def compilerPhases: List[QueryCompiler.Phase] = + List(permissionsElaborator, selectElaborator, componentElaborator, effectElaborator) +} diff --git a/modules/core/src/test/scala/effects/ValueEffectData.scala b/modules/core/src/test/scala/effects/ValueEffectData.scala index 3a04add3..d9adf8db 100644 --- a/modules/core/src/test/scala/effects/ValueEffectData.scala +++ b/modules/core/src/test/scala/effects/ValueEffectData.scala @@ -33,7 +33,7 @@ class ValueEffectMapping[F[_]: Sync](ref: SignallingRef[F, Int]) extends ValueMa fieldMappings = List( // Compute a ValueCursor - RootEffect.computeCursor("foo")((_, p, e) => + RootEffect.computeCursor("foo")((p, e) => ref.update(_+1).as( Result(valueCursor(p, e, Struct(42, "hi"))) ) diff --git a/modules/core/src/test/scala/introspection/IntrospectionSuite.scala b/modules/core/src/test/scala/introspection/IntrospectionSuite.scala index b5511fc7..077bb1b0 100644 --- a/modules/core/src/test/scala/introspection/IntrospectionSuite.scala +++ b/modules/core/src/test/scala/introspection/IntrospectionSuite.scala @@ -19,6 +19,11 @@ final class IntrospectionSuite extends CatsEffectSuite { case _ => name.startsWith("__") } + def standardDirectiveName(name: String): Boolean = name match { + case "skip" | "include" | "deprecated" => true + case _ => name.startsWith("__") + } + implicit class ACursorOps(self: ACursor) { def filterArray(f: Json => Boolean): ACursor = { def go(ac: ACursor, i: Int = 0): ACursor = { @@ -32,13 +37,16 @@ final class IntrospectionSuite extends CatsEffectSuite { } } - def stripStandardTypes(result: Json): Option[Json] = + def stripStandardElements(result: Json): Option[Json] = result .hcursor .downField("data") .downField("__schema") .downField("types") .filterArray(_.hcursor.downField("name").as[String].exists(!standardTypeName(_))) + .up + .downField("directives") + .filterArray(_.hcursor.downField("name").as[String].exists(!standardDirectiveName(_))) .top // .root doesn't work in 0.13 test("simple type query") { @@ -854,7 +862,7 @@ final class IntrospectionSuite extends CatsEffectSuite { } """ - val res = TestMapping.compileAndRun(query).map(stripStandardTypes) + val res = TestMapping.compileAndRun(query).map(stripStandardElements) assertIO(res, Some(expected)) } @@ -1183,7 +1191,7 @@ final class IntrospectionSuite extends CatsEffectSuite { } """ - val res = SmallMapping.compileAndRun(query).map(stripStandardTypes) + val res = SmallMapping.compileAndRun(query).map(stripStandardElements) assertIO(res, Some(expected)) } @@ -1311,7 +1319,7 @@ final class IntrospectionSuite extends CatsEffectSuite { { "errors" : [ { - "message" : "Unknown field '__type' in 'Query'" + "message" : "No field '__type' for type Query" } ] } diff --git a/modules/core/src/test/scala/parser/ParserSuite.scala b/modules/core/src/test/scala/parser/ParserSuite.scala index 0204bde2..04d08898 100644 --- a/modules/core/src/test/scala/parser/ParserSuite.scala +++ b/modules/core/src/test/scala/parser/ParserSuite.scala @@ -264,7 +264,71 @@ final class ParserSuite extends CatsEffectSuite { val expected = Operation(Query, Some(Name("getZuckProfile")), - List(VariableDefinition(Name("devicePicSize"), Named(Name("Int")), None)), + List(VariableDefinition(Name("devicePicSize"), Named(Name("Int")), None, Nil)), + Nil, + List( + Field(None, Name("user"), List((Name("id"), IntValue(4))), Nil, + List( + Field(None, Name("id"), Nil, Nil, Nil), + Field(None, Name("name"), Nil, Nil, Nil), + Field(None, Name("profilePic"), List((Name("size"), Variable(Name("devicePicSize")))), Nil, Nil) + ) + ) + ) + ) + + GraphQLParser.Document.parseAll(query).toOption match { + case Some(List(q)) => assertEquals(q, expected) + case _ => assert(false) + } + } + + test("variables with default value") { + val query = """ + query getZuckProfile($devicePicSize: Int = 10) { + user(id: 4) { + id + name + profilePic(size: $devicePicSize) + } + } + """ + + val expected = + Operation(Query, Some(Name("getZuckProfile")), + List(VariableDefinition(Name("devicePicSize"), Named(Name("Int")), Some(IntValue(10)), Nil)), + Nil, + List( + Field(None, Name("user"), List((Name("id"), IntValue(4))), Nil, + List( + Field(None, Name("id"), Nil, Nil, Nil), + Field(None, Name("name"), Nil, Nil, Nil), + Field(None, Name("profilePic"), List((Name("size"), Variable(Name("devicePicSize")))), Nil, Nil) + ) + ) + ) + ) + + GraphQLParser.Document.parseAll(query).toOption match { + case Some(List(q)) => assertEquals(q, expected) + case _ => assert(false) + } + } + + test("variables with directive") { + val query = """ + query getZuckProfile($devicePicSize: Int @dir) { + user(id: 4) { + id + name + profilePic(size: $devicePicSize) + } + } + """ + + val expected = + Operation(Query, Some(Name("getZuckProfile")), + List(VariableDefinition(Name("devicePicSize"), Named(Name("Int")), None, List(Directive(Name("dir"), Nil)))), Nil, List( Field(None, Name("user"), List((Name("id"), IntValue(4))), Nil, @@ -393,7 +457,7 @@ final class ParserSuite extends CatsEffectSuite { } - test("fragment with directive") { + test("fragment with standard directive") { val query = """ query frag($expanded: Boolean){ character(id: 1000) { @@ -409,7 +473,7 @@ final class ParserSuite extends CatsEffectSuite { Operation( Query, Some(Name("frag")), - List(VariableDefinition(Name("expanded"),Named(Name("Boolean")),None)), + List(VariableDefinition(Name("expanded"),Named(Name("Boolean")),None, Nil)), Nil, List( Field(None, Name("character"), List((Name("id"), IntValue(1000))), Nil, @@ -429,7 +493,44 @@ final class ParserSuite extends CatsEffectSuite { case Some(List(q)) => assertEquals(q, expected) case _ => assert(false) } + } + + test("fragment with custorm directive") { + val query = """ + query { + character(id: 1000) { + name + ... @dir { + age + } + } + } + """ + val expected = + Operation( + Query, + None, + Nil, + Nil, + List( + Field(None, Name("character"), List((Name("id"), IntValue(1000))), Nil, + List( + Field(None, Name("name"), Nil, Nil, Nil), + InlineFragment( + None, + List(Directive(Name("dir"), Nil)), + List(Field(None,Name("age"),List(),List(),List())) + ) + ) + ) + ) + ) + + GraphQLParser.Document.parseAll(query).toOption match { + case Some(List(q)) => assertEquals(q, expected) + case _ => assert(false) + } } test("value literals") { diff --git a/modules/core/src/test/scala/schema/SchemaSuite.scala b/modules/core/src/test/scala/schema/SchemaSuite.scala index d3d8a193..220c0326 100644 --- a/modules/core/src/test/scala/schema/SchemaSuite.scala +++ b/modules/core/src/test/scala/schema/SchemaSuite.scala @@ -13,16 +13,16 @@ final class SchemaSuite extends CatsEffectSuite { test("schema validation: undefined types: typo in the use of a Query result type") { val schema = Schema( - """ - type Query { - episodeById(id: String!): Episod - } + """ + type Query { + episodeById(id: String!): Episod + } - type Episode { - id: String! - } - """ - ) + type Episode { + id: String! + } + """ + ) schema match { case Result.Failure(ps) => assertEquals(ps.map(_.message), NonEmptyChain("Reference to undefined type 'Episod'")) @@ -85,7 +85,7 @@ final class SchemaSuite extends CatsEffectSuite { ) schema match { - case Result.Failure(ps) => assertEquals(ps.map(_.message), NonEmptyChain("Only a single deprecated allowed at a given location")) + case Result.Failure(ps) => assertEquals(ps.map(_.message), NonEmptyChain("Directive 'deprecated' may not occur more than once")) case unexpected => fail(s"This was unexpected: $unexpected") } } @@ -101,7 +101,7 @@ final class SchemaSuite extends CatsEffectSuite { ) schema match { - case Result.Failure(ps) => assertEquals(ps.map(_.message), NonEmptyChain("deprecated must have a single String 'reason' argument, or no arguments")) + case Result.Failure(ps) => assertEquals(ps.map(_.message), NonEmptyChain("Unknown argument(s) 'notareason' in directive deprecated")) case unexpected => fail(s"This was unexpected: $unexpected") } } @@ -133,7 +133,15 @@ final class SchemaSuite extends CatsEffectSuite { schema match { case Result.Failure(ps) => - assertEquals(ps.map(_.message), NonEmptyChain("Reference to undefined type 'Character'", "Reference to undefined type 'Contactable'")) + assertEquals( + ps.map(_.message), + NonEmptyChain( + "Reference to undefined type 'Character'", + "Reference to undefined type 'Contactable'", + "Non-interface type 'Character' declared as implemented by type 'Human'", + "Non-interface type 'Contactable' declared as implemented by type 'Human'" + ) + ) case unexpected => fail(s"This was unexpected: $unexpected") } } diff --git a/modules/core/src/test/scala/sdl/SDLSuite.scala b/modules/core/src/test/scala/sdl/SDLSuite.scala index e471de47..ed1bb336 100644 --- a/modules/core/src/test/scala/sdl/SDLSuite.scala +++ b/modules/core/src/test/scala/sdl/SDLSuite.scala @@ -23,9 +23,9 @@ final class SDLSuite extends CatsEffectSuite { List( SchemaDefinition( List( - RootOperationTypeDefinition(Query, Named(Name("MyQuery"))), - RootOperationTypeDefinition(Mutation, Named(Name("MyMutation"))), - RootOperationTypeDefinition(Subscription, Named(Name("MySubscription"))) + RootOperationTypeDefinition(Query, Named(Name("MyQuery")), Nil), + RootOperationTypeDefinition(Mutation, Named(Name("MyMutation")), Nil), + RootOperationTypeDefinition(Subscription, Named(Name("MySubscription")), Nil) ), Nil ) @@ -195,34 +195,18 @@ final class SDLSuite extends CatsEffectSuite { } test("parse directive definition") { - val schema = """ - "A directive" - directive @delegateField(name: String!) repeatable on OBJECT | INTERFACE | FIELD | FIELD_DEFINITION | ENUM | ENUM_VALUE - """ - - val expected = - List( - DirectiveDefinition( - Name("delegateField"), - Some("A directive"), - List( - InputValueDefinition(Name("name"), None, NonNull(Left(Named(Name("String")))), None, Nil) - ), - true, - List( - DirectiveLocation.OBJECT, - DirectiveLocation.INTERFACE, - DirectiveLocation.FIELD, - DirectiveLocation.FIELD_DEFINITION, - DirectiveLocation.ENUM, - DirectiveLocation.ENUM_VALUE - ) - ) - ) + val schema = + """|type Query { + | foo: Int + |} + |"A directive" + |directive @delegateField(name: String!) repeatable on OBJECT|INTERFACE|FIELD|FIELD_DEFINITION|ENUM|ENUM_VALUE + |""".stripMargin - val res = GraphQLParser.Document.parseAll(schema) + val res = SchemaParser.parseText(schema) + val ser = res.map(_.toString) - assertEquals(res, Right(expected)) + assertEquals(ser, schema.success) } test("deserialize schema (1)") { diff --git a/modules/core/src/test/scala/starwars/StarWarsData.scala b/modules/core/src/test/scala/starwars/StarWarsData.scala index a00caebb..acb1a700 100644 --- a/modules/core/src/test/scala/starwars/StarWarsData.scala +++ b/modules/core/src/test/scala/starwars/StarWarsData.scala @@ -10,7 +10,8 @@ import io.circe.Encoder import edu.gemini.grackle._ import edu.gemini.grackle.syntax._ -import Query._, Predicate._, Value._ +import Query._ +import Predicate._, Value._ import QueryCompiler._ object StarWarsData { @@ -202,27 +203,18 @@ object StarWarsMapping extends ValueMapping[IO] { LeafMapping[Episode.Value](EpisodeType) ) - val numberOfFriends: PartialFunction[Query, Result[Query]] = { - case Select("numberOfFriends", Nil, Empty) => - Count("numberOfFriends", Select("friends", Nil, Empty)).success + override val selectElaborator = SelectElaborator { + case (QueryType, "hero", List(Binding("episode", EnumValue(e)))) => + val episode = Episode.values.find(_.toString == e).get + Elab.transformChild(child => Unique(Filter(Eql(CharacterType / "id", Const(hero(episode).id)), child))) + case (QueryType, "character" | "human" | "droid", List(Binding("id", IDValue(id)))) => + Elab.transformChild(child => Unique(Filter(Eql(CharacterType / "id", Const(id)), child))) + case (QueryType, "characters", List(Binding("offset", IntValue(offset)), Binding("limit", IntValue(limit)))) => + Elab.transformChild(child => Limit(limit, Offset(offset, child))) + case (CharacterType | HumanType | DroidType, "numberOfFriends", _) => + Elab.transformChild(_ => Count(Select("friends"))) } - - override val selectElaborator = new SelectElaborator(Map( - QueryType -> { - case Select("hero", List(Binding("episode", TypedEnumValue(e))), child) => - val episode = Episode.values.find(_.toString == e.name).get - Select("hero", Nil, Unique(Filter(Eql(CharacterType / "id", Const(hero(episode).id)), child))).success - case Select(f@("character" | "human" | "droid"), List(Binding("id", IDValue(id))), child) => - Select(f, Nil, Unique(Filter(Eql(CharacterType / "id", Const(id)), child))).success - case Select("characters", List(Binding("offset", IntValue(offset)), Binding("limit", IntValue(limit))), child) => - Select("characters", Nil, Limit(limit, Offset(offset, child))).success - }, - CharacterType -> numberOfFriends, - HumanType -> numberOfFriends, - DroidType -> numberOfFriends - )) - val querySizeValidator = new QuerySizeValidator(5, 5) override def compilerPhases = super.compilerPhases :+ querySizeValidator diff --git a/modules/core/src/test/scala/subscription/SubscriptionSuite.scala b/modules/core/src/test/scala/subscription/SubscriptionSuite.scala index a28cc0aa..6db78f50 100644 --- a/modules/core/src/test/scala/subscription/SubscriptionSuite.scala +++ b/modules/core/src/test/scala/subscription/SubscriptionSuite.scala @@ -12,8 +12,9 @@ import io.circe.Json import io.circe.literal._ import munit.CatsEffectSuite -import edu.gemini.grackle.{Cursor, Query, QueryCompiler, Mapping, Result, Schema, Value, ValueMapping} +import edu.gemini.grackle._ import edu.gemini.grackle.syntax._ +import QueryCompiler._ final class SubscriptionSuite extends CatsEffectSuite { @@ -40,10 +41,10 @@ final class SubscriptionSuite extends CatsEffectSuite { val typeMappings: List[TypeMapping] = List( ObjectMapping(QueryType, List( - RootEffect.computeCursor("get")((_, path, env) => ref.get.map(n => Result(valueCursor(path, env, n)))) + RootEffect.computeCursor("get")((path, env) => ref.get.map(n => Result(valueCursor(path, env, n)))) )), ObjectMapping(MutationType, List( - RootEffect.computeCursor("put")( (_, path, env) => + RootEffect.computeCursor("put")( (path, env) => env.get[Int]("n") match { case None => Result.failure(s"Implementation error: `n: Int` not found in $env").pure[IO] case Some(n) => ref.set(n).map(_ => Result(valueCursor(path, env, n))) @@ -51,18 +52,16 @@ final class SubscriptionSuite extends CatsEffectSuite { ) )), ObjectMapping(SubscriptionType, List( - RootStream.computeCursor("watch")((_, path, env) => + RootStream.computeCursor("watch")((path, env) => ref.discrete.map(n => Result(valueCursor(path, env, n)))) )) ) - override val selectElaborator: QueryCompiler.SelectElaborator = - new QueryCompiler.SelectElaborator(Map( - MutationType -> { - case Query.Select("put", List(Query.Binding("n", Value.IntValue(n))), child) => - Result(Query.Environment(Cursor.Env("n" -> n), Query.Select("put", Nil, child))) - } - )) + override val selectElaborator: SelectElaborator = + SelectElaborator { + case (MutationType, "put", List(Query.Binding("n", Value.IntValue(n)))) => + Elab.env("n" -> n) + } } test("sanity check get") { @@ -70,7 +69,7 @@ final class SubscriptionSuite extends CatsEffectSuite { for { ref <- SignallingRef[IO, Int](0) map = mapping(ref) - r1 <- map.compileAndRunOne("query { get }") + r1 <- map.compileAndRun("query { get }") } yield r1 assertIO(prog, @@ -89,7 +88,7 @@ final class SubscriptionSuite extends CatsEffectSuite { for { ref <- SignallingRef[IO, Int](0) map = mapping(ref) - r1 <- map.compileAndRunOne("mutation { put(n: 42) }") + r1 <- map.compileAndRun("mutation { put(n: 42) }") } yield r1 assertIO(prog, @@ -109,9 +108,9 @@ final class SubscriptionSuite extends CatsEffectSuite { for { ref <- SignallingRef[IO, Int](0) map = mapping(ref) - r0 <- map.compileAndRunOne("query { get }") - r1 <- map.compileAndRunOne("mutation { put(n: 42) }") - r2 <- map.compileAndRunOne("query { get }") + r0 <- map.compileAndRun("query { get }") + r1 <- map.compileAndRun("mutation { put(n: 42) }") + r2 <- map.compileAndRun("query { get }") } yield (r0, r1, r2) assertIO(prog, (( @@ -146,13 +145,13 @@ final class SubscriptionSuite extends CatsEffectSuite { for { ref <- SignallingRef[IO, Int](0) map = mapping(ref) - fib <- map.compileAndRunAll("subscription { watch }").take(4).compile.toList.start + fib <- map.compileAndRunSubscription("subscription { watch }").take(4).compile.toList.start _ <- IO.sleep(100.milli) // this is the best we can do for now; I will try to improve in a followup - _ <- map.compileAndRunOne("mutation { put(n: 123) }") + _ <- map.compileAndRun("mutation { put(n: 123) }") _ <- IO.sleep(100.milli) - _ <- map.compileAndRunOne("mutation { put(n: 42) }") + _ <- map.compileAndRun("mutation { put(n: 42) }") _ <- IO.sleep(100.milli) - _ <- map.compileAndRunOne("mutation { put(n: 77) }") + _ <- map.compileAndRun("mutation { put(n: 77) }") _ <- IO.sleep(100.milli) out <- fib.join res <- out.embedNever diff --git a/modules/doobie-pg/src/test/scala/DoobieSuites.scala b/modules/doobie-pg/src/test/scala/DoobieSuites.scala index e4a22a22..9fb47579 100644 --- a/modules/doobie-pg/src/test/scala/DoobieSuites.scala +++ b/modules/doobie-pg/src/test/scala/DoobieSuites.scala @@ -6,9 +6,7 @@ package edu.gemini.grackle.doobie.test import cats.effect.IO import doobie.implicits._ import doobie.Meta -import io.circe.Json -import edu.gemini.grackle.QueryExecutor import edu.gemini.grackle.doobie.postgres.DoobieMonitor import edu.gemini.grackle.sql.SqlStatsMonitor @@ -21,12 +19,12 @@ final class ArrayJoinSuite extends DoobieDatabaseSuite with SqlArrayJoinSuite { final class CoalesceSuite extends DoobieDatabaseSuite with SqlCoalesceSuite { type Fragment = doobie.Fragment - def mapping: IO[(QueryExecutor[IO, Json], SqlStatsMonitor[IO,Fragment])] = + def mapping: IO[(Mapping[IO], SqlStatsMonitor[IO,Fragment])] = DoobieMonitor.statsMonitor[IO].map(mon => (new DoobieTestMapping(xa, mon) with SqlCoalesceMapping[IO], mon)) } final class ComposedWorldSuite extends DoobieDatabaseSuite with SqlComposedWorldSuite { - def mapping: IO[(CurrencyMapping[IO], QueryExecutor[IO, Json])] = + def mapping: IO[(CurrencyMapping[IO], Mapping[IO])] = for { currencyMapping <- CurrencyMapping[IO] } yield (currencyMapping, new SqlComposedMapping(new DoobieTestMapping(xa) with SqlWorldMapping[IO], currencyMapping)) @@ -126,7 +124,7 @@ final class MutationSuite extends DoobieDatabaseSuite with SqlMutationSuite { } final class NestedEffectsSuite extends DoobieDatabaseSuite with SqlNestedEffectsSuite { - def mapping: IO[(CurrencyService[IO], QueryExecutor[IO, Json])] = + def mapping: IO[(CurrencyService[IO], Mapping[IO])] = for { currencyService0 <- CurrencyService[IO] } yield { diff --git a/modules/generic/src/main/scala-2/genericmapping2.scala b/modules/generic/src/main/scala-2/genericmapping2.scala index b84faafa..22cb94a2 100644 --- a/modules/generic/src/main/scala-2/genericmapping2.scala +++ b/modules/generic/src/main/scala-2/genericmapping2.scala @@ -13,7 +13,7 @@ import shapeless.ops.coproduct.{LiftAll => CLiftAll} import shapeless.ops.record.Keys import syntax._ -import Cursor.{AbstractCursor, Context, Env} +import Cursor.AbstractCursor import ShapelessUtils._ trait ScalaVersionSpecificGenericMappingLike[F[_]] extends Mapping[F] { self: GenericMappingLike[F] => diff --git a/modules/generic/src/main/scala-3/genericmapping3.scala b/modules/generic/src/main/scala-3/genericmapping3.scala index 65a4896f..e6520c3d 100644 --- a/modules/generic/src/main/scala-3/genericmapping3.scala +++ b/modules/generic/src/main/scala-3/genericmapping3.scala @@ -8,7 +8,7 @@ import cats.implicits._ import shapeless3.deriving._ import syntax._ -import Cursor.{AbstractCursor, Context, Env} +import Cursor.AbstractCursor trait ScalaVersionSpecificGenericMappingLike[F[_]] extends Mapping[F] { self: GenericMappingLike[F] => trait MkObjectCursorBuilder[T] { diff --git a/modules/generic/src/main/scala/CursorBuilder.scala b/modules/generic/src/main/scala/CursorBuilder.scala index c0b880eb..33f6774f 100644 --- a/modules/generic/src/main/scala/CursorBuilder.scala +++ b/modules/generic/src/main/scala/CursorBuilder.scala @@ -10,7 +10,7 @@ import cats.implicits._ import io.circe.{Encoder, Json} import syntax._ -import Cursor.{AbstractCursor, Context, Env} +import Cursor.AbstractCursor trait CursorBuilder[T] { def tpe: Type diff --git a/modules/generic/src/main/scala/genericmapping.scala b/modules/generic/src/main/scala/genericmapping.scala index c40a1e2d..5bbad5b8 100644 --- a/modules/generic/src/main/scala/genericmapping.scala +++ b/modules/generic/src/main/scala/genericmapping.scala @@ -8,7 +8,7 @@ import cats.MonadThrow import org.tpolecat.sourcepos.SourcePos import syntax._ -import Cursor.{Context, DeferredCursor, Env} +import Cursor.DeferredCursor abstract class GenericMapping[F[_]](implicit val M: MonadThrow[F]) extends Mapping[F] with GenericMappingLike[F] diff --git a/modules/generic/src/test/scala/DerivationSuite.scala b/modules/generic/src/test/scala/DerivationSuite.scala index 05cc8d3b..28b46df4 100644 --- a/modules/generic/src/test/scala/DerivationSuite.scala +++ b/modules/generic/src/test/scala/DerivationSuite.scala @@ -13,7 +13,6 @@ import io.circe.literal._ import munit.CatsEffectSuite import edu.gemini.grackle.syntax._ -import Cursor.Context import Query._, Predicate._, Value._ import QueryCompiler._ import ScalarType._ @@ -203,25 +202,19 @@ object StarWarsMapping extends GenericMapping[IO] { ) ) - val numberOfFriends: PartialFunction[Query, Result[Query]] = { - case Select("numberOfFriends", Nil, Empty) => - Count("numberOfFriends", Select("friends", Nil, Empty)).success - } + override val selectElaborator = SelectElaborator { + case (QueryType, "hero", List(Binding("episode", EnumValue(e)))) => + for { + episode <- Elab.liftR(Episode.values.find(_.toString == e).toResult(s"Unknown episode '$e'")) + _ <- Elab.transformChild(child => Unique(Filter(Eql(CharacterType / "id", Const(hero(episode).id)), child))) + } yield () - override val selectElaborator = new SelectElaborator(Map( - QueryType -> { - case Select("hero", List(Binding("episode", TypedEnumValue(e))), child) => - Episode.values.find(_.toString == e.name).map { episode => - Select("hero", Nil, Unique(Filter(Eql(CharacterType / "id", Const(hero(episode).id)), child))).success - }.getOrElse(Result.failure(s"Unknown episode '${e.name}'")) - - case Select(f@("character" | "human" | "droid"), List(Binding("id", IDValue(id))), child) => - Select(f, Nil, Unique(Filter(Eql(CharacterType / "id", Const(id)), child))).success - }, - CharacterType -> numberOfFriends, - HumanType -> numberOfFriends, - DroidType -> numberOfFriends - )) + case (QueryType, "character" | "human" | "droid", List(Binding("id", IDValue(id)))) => + Elab.transformChild(child => Unique(Filter(Eql(CharacterType / "id", Const(id)), child))) + + case (CharacterType | HumanType | DroidType, "numberOfFriends", Nil) => + Elab.transformChild(_ => Count(Select("friends"))) + } } import StarWarsData._, StarWarsMapping._ diff --git a/modules/generic/src/test/scala/EffectsSuite.scala b/modules/generic/src/test/scala/EffectsSuite.scala index 3abad2f3..abb79ddb 100644 --- a/modules/generic/src/test/scala/EffectsSuite.scala +++ b/modules/generic/src/test/scala/EffectsSuite.scala @@ -42,7 +42,7 @@ class GenericEffectMapping[F[_]: Sync](ref: SignallingRef[F, Int]) extends Gener fieldMappings = List( // Compute a ValueCursor - RootEffect.computeCursor("foo")((_, p, e) => + RootEffect.computeCursor("foo")((p, e) => ref.update(_+1).as( genericCursor(p, e, Struct(42, "hi")) ) diff --git a/modules/generic/src/test/scala/RecursionSuite.scala b/modules/generic/src/test/scala/RecursionSuite.scala index 2186ed9b..d1ea7a15 100644 --- a/modules/generic/src/test/scala/RecursionSuite.scala +++ b/modules/generic/src/test/scala/RecursionSuite.scala @@ -82,12 +82,10 @@ object MutualRecursionMapping extends GenericMapping[IO] { ) ) - override val selectElaborator = new SelectElaborator(Map( - QueryType -> { - case Select(f@("programmeById" | "productionById"), List(Binding("id", IDValue(id))), child) => - Select(f, Nil, Unique(Filter(Eql(ProgrammeType / "id", Const(id)), child))).success - } - )) + override val selectElaborator = SelectElaborator { + case (QueryType, "programmeById" | "productionById", List(Binding("id", IDValue(id)))) => + Elab.transformChild(child => Unique(Filter(Eql(ProgrammeType / "id", Const(id)), child))) + } } final class RecursionSuite extends CatsEffectSuite { diff --git a/modules/generic/src/test/scala/ScalarsSuite.scala b/modules/generic/src/test/scala/ScalarsSuite.scala index 69415dab..10c664a0 100644 --- a/modules/generic/src/test/scala/ScalarsSuite.scala +++ b/modules/generic/src/test/scala/ScalarsSuite.scala @@ -154,8 +154,8 @@ object MovieMapping extends GenericMapping[IO] { } object GenreValue { - def unapply(e: TypedEnumValue): Option[Genre] = - Genre.fromString(e.value.name) + def unapply(e: EnumValue): Option[Genre] = + Genre.fromString(e.name) } object DateValue { @@ -178,48 +178,46 @@ object MovieMapping extends GenericMapping[IO] { Try(Duration.parse(s.value)).toOption } - override val selectElaborator = new SelectElaborator(Map( - QueryType -> { - case Select("movieById", List(Binding("id", UUIDValue(id))), child) => - Select("movieById", Nil, Unique(Filter(Eql(MovieType / "id", Const(id)), child))).success - case Select("moviesByGenre", List(Binding("genre", GenreValue(genre))), child) => - Select("moviesByGenre", Nil, Filter(Eql(MovieType / "genre", Const(genre)), child)).success - case Select("moviesReleasedBetween", List(Binding("from", DateValue(from)), Binding("to", DateValue(to))), child) => - Select("moviesReleasedBetween", Nil, - Filter( - And( - Not(Lt(MovieType / "releaseDate", Const(from))), - Lt(MovieType / "releaseDate", Const(to)) - ), - child - ) - ).success - case Select("moviesLongerThan", List(Binding("duration", IntervalValue(duration))), child) => - Select("moviesLongerThan", Nil, - Filter( - Not(Lt(MovieType / "duration", Const(duration))), - child - ) - ).success - case Select("moviesShownLaterThan", List(Binding("time", TimeValue(time))), child) => - Select("moviesShownLaterThan", Nil, - Filter( - Not(Lt(MovieType / "showTime", Const(time))), - child - ) - ).success - case Select("moviesShownBetween", List(Binding("from", DateTimeValue(from)), Binding("to", DateTimeValue(to))), child) => - Select("moviesShownBetween", Nil, - Filter( - And( - Not(Lt(MovieType / "nextShowing", Const(from))), - Lt(MovieType / "nextShowing", Const(to)) - ), - child - ) - ).success - } - )) + override val selectElaborator = SelectElaborator { + case (QueryType, "movieById", List(Binding("id", UUIDValue(id)))) => + Elab.transformChild(child => Unique(Filter(Eql(MovieType / "id", Const(id)), child))) + case (QueryType, "moviesByGenre", List(Binding("genre", GenreValue(genre)))) => + Elab.transformChild(child => Filter(Eql(MovieType / "genre", Const(genre)), child)) + case (QueryType, "moviesReleasedBetween", List(Binding("from", DateValue(from)), Binding("to", DateValue(to)))) => + Elab.transformChild(child => + Filter( + And( + Not(Lt(MovieType / "releaseDate", Const(from))), + Lt(MovieType / "releaseDate", Const(to)) + ), + child + ) + ) + case (QueryType, "moviesLongerThan", List(Binding("duration", IntervalValue(duration)))) => + Elab.transformChild(child => + Filter( + Not(Lt(MovieType / "duration", Const(duration))), + child + ) + ) + case (QueryType, "moviesShownLaterThan", List(Binding("time", TimeValue(time)))) => + Elab.transformChild(child => + Filter( + Not(Lt(MovieType / "showTime", Const(time))), + child + ) + ) + case (QueryType, "moviesShownBetween", List(Binding("from", DateTimeValue(from)), Binding("to", DateTimeValue(to)))) => + Elab.transformChild(child => + Filter( + And( + Not(Lt(MovieType / "nextShowing", Const(from))), + Lt(MovieType / "nextShowing", Const(to)) + ), + child + ) + ) + } } final class ScalarsSuite extends CatsEffectSuite { diff --git a/modules/skunk/js-jvm/src/test/scala/SkunkDatabaseSuite.scala b/modules/skunk/js-jvm/src/test/scala/SkunkDatabaseSuite.scala index 3dfe08cc..fead905c 100644 --- a/modules/skunk/js-jvm/src/test/scala/SkunkDatabaseSuite.scala +++ b/modules/skunk/js-jvm/src/test/scala/SkunkDatabaseSuite.scala @@ -62,7 +62,7 @@ trait SkunkDatabaseSuite extends SqlDatabaseSuite { def list(c: Codec): Codec = { val cc = c._1.asInstanceOf[_root_.skunk.Codec[Any]] - val ty = _root_.skunk.data.Type(s"_${cc.types.head.name}", cc.types) + val ty = _root_.skunk.data.Type(s"_${cc.types.head.name}", cc.types) val encode = (elem: Any) => cc.encode(elem).head.get val decode = (str: String) => cc.decode(0, List(Some(str))).left.map(_.message) (_root_.skunk.Codec.array(encode, decode, ty), false) diff --git a/modules/skunk/js-jvm/src/test/scala/SkunkSuites.scala b/modules/skunk/js-jvm/src/test/scala/SkunkSuites.scala index fd62d9c9..62fcc0f5 100644 --- a/modules/skunk/js-jvm/src/test/scala/SkunkSuites.scala +++ b/modules/skunk/js-jvm/src/test/scala/SkunkSuites.scala @@ -6,9 +6,7 @@ package edu.gemini.grackle.skunk.test import cats.effect.IO import skunk.codec.{all => codec} import skunk.implicits._ -import io.circe.Json -import edu.gemini.grackle.QueryExecutor import edu.gemini.grackle.skunk.SkunkMonitor import edu.gemini.grackle.sql.SqlStatsMonitor @@ -23,12 +21,12 @@ final class ArrayJoinSuite extends SkunkDatabaseSuite with SqlArrayJoinSuite { final class CoalesceSuite extends SkunkDatabaseSuite with SqlCoalesceSuite { type Fragment = skunk.AppliedFragment - def mapping: IO[(QueryExecutor[IO, Json], SqlStatsMonitor[IO,Fragment])] = + def mapping: IO[(Mapping[IO], SqlStatsMonitor[IO,Fragment])] = SkunkMonitor.statsMonitor[IO].map(mon => (new SkunkTestMapping(pool, mon) with SqlCoalesceMapping[IO], mon)) } final class ComposedWorldSuite extends SkunkDatabaseSuite with SqlComposedWorldSuite { - def mapping: IO[(CurrencyMapping[IO], QueryExecutor[IO, Json])] = + def mapping: IO[(CurrencyMapping[IO], Mapping[IO])] = for { currencyMapping <- CurrencyMapping[IO] } yield (currencyMapping, new SqlComposedMapping(new SkunkTestMapping(pool) with SqlWorldMapping[IO], currencyMapping)) @@ -131,7 +129,7 @@ final class MutationSuite extends SkunkDatabaseSuite with SqlMutationSuite { } final class NestedEffectsSuite extends SkunkDatabaseSuite with SqlNestedEffectsSuite { - def mapping: IO[(CurrencyService[IO], QueryExecutor[IO, Json])] = + def mapping: IO[(CurrencyService[IO], Mapping[IO])] = for { currencyService0 <- CurrencyService[IO] } yield { diff --git a/modules/skunk/js-jvm/src/test/scala/subscription/SubscriptionMapping.scala b/modules/skunk/js-jvm/src/test/scala/subscription/SubscriptionMapping.scala index a10c22da..b9c7d230 100644 --- a/modules/skunk/js-jvm/src/test/scala/subscription/SubscriptionMapping.scala +++ b/modules/skunk/js-jvm/src/test/scala/subscription/SubscriptionMapping.scala @@ -83,27 +83,20 @@ trait SubscriptionMapping[F[_]] extends SkunkMapping[F] { ObjectMapping( tpe = SubscriptionType, fieldMappings = List( - RootStream.computeQuery("channel")((query, _, _) => + RootStream.computeChild("channel")((child, _, _) => for { s <- fs2.Stream.resource(pool) id <- s.channel(id"city_channel").listen(256).map(_.value.toInt) - } yield - query match { - case s@Select(_, _, child0) => - Result(s.copy(child = Unique(Filter(Eql(CityType / "id", Const(id)), child0)))) - case _ => Result.internalError("Implementation error: expected Select") - } + } yield Unique(Filter(Eql(CityType / "id", Const(id)), child)).success ) ) ) ) - override val selectElaborator = new SelectElaborator(Map( - QueryType -> { - case Select("city", List(Binding("id", IntValue(id))), child) => - Select("city", Nil, Unique(Filter(Eql(CityType / "id", Const(id)), child))).success - }, - )) + override val selectElaborator = SelectElaborator { + case (QueryType, "city", List(Binding("id", IntValue(id)))) => + Elab.transformChild(child => Unique(Filter(Eql(CityType / "id", Const(id)), child))) + } } object SubscriptionMapping extends SkunkMappingCompanion { diff --git a/modules/skunk/js-jvm/src/test/scala/subscription/SubscriptionSuite.scala b/modules/skunk/js-jvm/src/test/scala/subscription/SubscriptionSuite.scala index 44dba90d..2233b9d8 100644 --- a/modules/skunk/js-jvm/src/test/scala/subscription/SubscriptionSuite.scala +++ b/modules/skunk/js-jvm/src/test/scala/subscription/SubscriptionSuite.scala @@ -13,7 +13,7 @@ import skunk.implicits._ import edu.gemini.grackle.skunk.test.SkunkDatabaseSuite -class SubscriptionSpec extends SkunkDatabaseSuite { +class SubscriptionSuite extends SkunkDatabaseSuite { lazy val mapping = SubscriptionMapping.mkMapping(pool) @@ -61,7 +61,7 @@ class SubscriptionSpec extends SkunkDatabaseSuite { for { // start a fiber that subscibes and takes the first two notifications - fi <- mapping.compileAndRunAll(query).take(2).compile.toList.start + fi <- mapping.compileAndRunSubscription(query).take(2).compile.toList.start // We're racing now, so wait a sec before starting notifications _ <- IO.sleep(1.second) diff --git a/modules/sql/shared/src/main/scala/SqlMapping.scala b/modules/sql/shared/src/main/scala/SqlMapping.scala index b397cead..51002ea5 100644 --- a/modules/sql/shared/src/main/scala/SqlMapping.scala +++ b/modules/sql/shared/src/main/scala/SqlMapping.scala @@ -15,7 +15,6 @@ import io.circe.Json import org.tpolecat.sourcepos.SourcePos import syntax._ -import Cursor.{Context, Env} import Predicate._ import Query._ import circe.CirceMappingLike @@ -625,23 +624,24 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self // Preserve OrderBy case o: OrderBy => o.copy(child = loop(o.child, context)) - case PossiblyRenamedSelect(s@Select(fieldName, _, _), resultName) => - val fieldContext = context.forField(fieldName, resultName).getOrElse(throw new SqlMappingException(s"No field '$fieldName' of type ${context.tpe}")) - PossiblyRenamedSelect(s.copy(child = loop(s.child, fieldContext)), resultName) - case Count(countName, _) => - if(context.tpe.underlying.hasField(countName)) Select(countName, Nil, Empty) + case s@Select(fieldName, _, Count(_)) => + if(context.tpe.underlying.hasField(fieldName)) s.copy(child = Empty) else Empty + case s@Select(fieldName, resultName, _) => + val fieldContext = context.forField(fieldName, resultName).getOrElse(throw new SqlMappingException(s"No field '$fieldName' of type ${context.tpe}")) + s.copy(child = loop(s.child, fieldContext)) + + case s@UntypedSelect(fieldName, resultName, _, _, _) => + val fieldContext = context.forField(fieldName, resultName).getOrElse(throw new SqlMappingException(s"No field '$fieldName' of type ${context.tpe}")) + s.copy(child = loop(s.child, fieldContext)) + case Group(queries) => Group(queries.map(q => loop(q, context)).filterNot(_ == Empty)) case u: Unique => u.copy(child = loop(u.child, context.asType(context.tpe.list))) case e: Environment => e.copy(child = loop(e.child, context)) - case w: Wrap => w.copy(child = loop(w.child, context)) - case r: Rename => r.copy(child = loop(r.child, context)) case t: TransformCursor => t.copy(child = loop(t.child, context)) - case u: UntypedNarrow => u.copy(child = loop(u.child, context)) case n@Narrow(subtpe, _) => n.copy(child = loop(n.child, context.asType(subtpe))) - case s: Skip => s.copy(child = loop(s.child, context)) - case other@(_: Component[_] | _: Effect[_] | Empty | _: Introspect | _: Select | Skipped) => other + case other@(_: Component[_] | _: Effect[_] | Empty | _: Introspect | _: Select | _: Count | _: UntypedFragmentSpread | _: UntypedInlineFragment) => other } Result.catchNonFatal { @@ -2863,7 +2863,7 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self q match { // Leaf or Json element: no subobjects - case PossiblyRenamedSelect(Select(fieldName, _, child), _) if child == Empty || isJsonb(context, fieldName) => + case Select(fieldName, _, child) if child == Empty || isJsonb(context, fieldName) => columnsForLeaf(context, fieldName).flatMap { case Nil => EmptySqlQuery.success case cols => @@ -2876,8 +2876,48 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self SqlSelect(context, Nil, parentTable, (cols ++ extraCols).distinct, extraJoins, Nil, Nil, None, None, Nil, true, false) } + case Select(fieldName, _, Count(child)) => + def childContext(q: Query): Result[Context] = + q match { + case Select(fieldName, resultName, _) => + context.forField(fieldName, resultName) + case FilterOrderByOffsetLimit(_, _, _, _, child) => + childContext(child) + case _ => Result.internalError(s"No context for count of ${child}") + } + + for { + fieldContext <- childContext(child) + countCol <- columnForAtomicField(context, fieldName) + sq <- loop(child, context, parentConstraints, exposeJoins) + parentTable <- parentTableForType(context) + res <- + sq match { + case sq: SqlSelect => + sq.joins match { + case hd :: tl => + val keyCols = keyColumnsForType(fieldContext) + val parentCols0 = hd.colsOf(parentTable) + val wheres = hd.on.map { case (p, c) => Eql(c.toTerm, p.toTerm) } + val ssq = sq.copy(table = hd.child, cols = SqlColumn.CountColumn(countCol.in(hd.child), keyCols.map(_.in(hd.child))) :: Nil, joins = tl, wheres = wheres) + val ssqCol = SqlColumn.SubqueryColumn(countCol, ssq) + SqlSelect(context, Nil, parentTable, (ssqCol :: parentCols0).distinct, Nil, Nil, Nil, None, None, Nil, true, false).success + case _ => + val keyCols = keyColumnsForType(fieldContext) + val countTable = sq.table + val ssq = sq.copy(cols = SqlColumn.CountColumn(countCol.in(countTable), keyCols.map(_.in(countTable))) :: Nil) + val ssqCol = SqlColumn.SubqueryColumn(countCol, ssq) + SqlSelect(context, Nil, parentTable, ssqCol :: Nil, Nil, Nil, Nil, None, None, Nil, true, false).success + } + case _ => + Result.internalError("Implementation restriction: cannot count an SQL union") + } + } yield res + + case _: Count => Result.internalError("Count node must be a child of a Select node") + // Non-leaf non-Json element: compile subobject queries - case PossiblyRenamedSelect(Select(fieldName, _, child), resultName) => + case s@Select(fieldName, resultName, child) => context.forField(fieldName, resultName).flatMap { fieldContext => if(schema.isRootType(context.tpe)) loop(child, fieldContext, Nil, false) else if(!isLocallyMapped(context, q)) EmptySqlQuery.success @@ -2887,7 +2927,7 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self val extraCols = keyCols ++ constraintCols for { parentTable <- parentTableForType(context) - parentConstraints0 <- parentConstraintsFromJoins(context, fieldName, resultName) + parentConstraints0 <- parentConstraintsFromJoins(context, fieldName, s.resultName) extraJoins <- parentConstraintsToSqlJoins(parentTable, parentConstraints) res <- loop(child, fieldContext, parentConstraints0, false).flatMap { sq => @@ -2903,7 +2943,7 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self } } - case Effect(_, PossiblyRenamedSelect(Select(fieldName, _, _), _)) => + case Effect(_, Select(fieldName, _, _)) => columnsForLeaf(context, fieldName).flatMap { case Nil => EmptySqlQuery.success case cols => @@ -2921,7 +2961,7 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self def loop(query: Query): Boolean = query match { case Empty => true - case PossiblyRenamedSelect(Select(_, _, Empty), _) => true + case Select(_, _, Empty) => true case Group(children) => children.forall(loop) case _ => false } @@ -3001,50 +3041,6 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self case Group(queries) => group(queries) - case Wrap(_, child) => - loop(child, context, parentConstraints, exposeJoins) - - case Count(countName, child) => - def childContext(q: Query): Result[Context] = - q match { - case PossiblyRenamedSelect(Select(fieldName, _, _), resultName) => - context.forField(fieldName, resultName) - case FilterOrderByOffsetLimit(_, _, _, _, child) => - childContext(child) - case _ => Result.internalError(s"No context for count of ${child}") - } - - for { - fieldContext <- childContext(child) - countCol <- columnForAtomicField(context, countName) - sq <- loop(child, context, parentConstraints, exposeJoins) - parentTable <- parentTableForType(context) - res <- - sq match { - case sq: SqlSelect => - sq.joins match { - case hd :: tl => - val keyCols = keyColumnsForType(fieldContext) - val parentCols0 = hd.colsOf(parentTable) - val wheres = hd.on.map { case (p, c) => Eql(c.toTerm, p.toTerm) } - val ssq = sq.copy(table = hd.child, cols = SqlColumn.CountColumn(countCol.in(hd.child), keyCols.map(_.in(hd.child))) :: Nil, joins = tl, wheres = wheres) - val ssqCol = SqlColumn.SubqueryColumn(countCol, ssq) - SqlSelect(context, Nil, parentTable, (ssqCol :: parentCols0).distinct, Nil, Nil, Nil, None, None, Nil, true, false).success - case _ => - val keyCols = keyColumnsForType(fieldContext) - val countTable = sq.table - val ssq = sq.copy(cols = SqlColumn.CountColumn(countCol.in(countTable), keyCols.map(_.in(countTable))) :: Nil) - val ssqCol = SqlColumn.SubqueryColumn(countCol, ssq) - SqlSelect(context, Nil, parentTable, ssqCol :: Nil, Nil, Nil, Nil, None, None, Nil, true, false).success - } - case _ => - Result.internalError("Implementation restriction: cannot count an SQL union") - } - } yield res - - case Rename(_, child) => - loop(child, context, parentConstraints, exposeJoins) - case Unique(child) => loop(child, context.asType(context.tpe.nonNull.list), parentConstraints, exposeJoins).map { case node => node.withContext(context, Nil, Nil) @@ -3120,7 +3116,7 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self case TransformCursor(_, child) => loop(child, context, parentConstraints, exposeJoins) - case Empty | Skipped | Query.Component(_, _, _) | Query.Effect(_, _) | (_: UntypedNarrow) | (_: Skip) | (_: Select) => + case Empty | Query.Component(_, _, _) | Query.Effect(_, _) | (_: UntypedSelect) | (_: UntypedFragmentSpread) | (_: UntypedInlineFragment) | (_: Select) => EmptySqlQuery.success } } diff --git a/modules/sql/shared/src/main/scala/SqlMappingValidator.scala b/modules/sql/shared/src/main/scala/SqlMappingValidator.scala index 9d16d806..c3ac1991 100644 --- a/modules/sql/shared/src/main/scala/SqlMappingValidator.scala +++ b/modules/sql/shared/src/main/scala/SqlMappingValidator.scala @@ -54,7 +54,7 @@ trait SqlMappingValidator extends MappingValidator { case NullableType(ScalarType.IDType) if columnRef.scalaTypeName == typeName[Option[String]] => Chain.empty case NullableType(ScalarType.IntType) if columnRef.scalaTypeName == typeName[Option[Int]] => Chain.empty - case tpe @ ScalarType(_, _) => + case tpe: ScalarType => typeMapping(tpe) match { case Some(lm: LeafMapping[_]) => if (lm.scalaTypeName == columnRef.scalaTypeName) Chain.empty diff --git a/modules/sql/shared/src/test/scala/SqlArrayJoinSuite.scala b/modules/sql/shared/src/test/scala/SqlArrayJoinSuite.scala index 6abc1d20..6a318a3f 100644 --- a/modules/sql/shared/src/test/scala/SqlArrayJoinSuite.scala +++ b/modules/sql/shared/src/test/scala/SqlArrayJoinSuite.scala @@ -5,14 +5,13 @@ package edu.gemini.grackle.sql.test import cats.effect.IO import edu.gemini.grackle._ -import io.circe.Json import io.circe.literal._ import munit.CatsEffectSuite import grackle.test.GraphQLResponseTests.assertWeaklyEqualIO trait SqlArrayJoinSuite extends CatsEffectSuite { - def mapping: QueryExecutor[IO, Json] + def mapping: Mapping[IO] test("base query") { val query = """ diff --git a/modules/sql/shared/src/test/scala/SqlCoalesceSuite.scala b/modules/sql/shared/src/test/scala/SqlCoalesceSuite.scala index 6029884b..a372ab13 100644 --- a/modules/sql/shared/src/test/scala/SqlCoalesceSuite.scala +++ b/modules/sql/shared/src/test/scala/SqlCoalesceSuite.scala @@ -17,7 +17,7 @@ import grackle.test.GraphQLResponseTests.assertWeaklyEqual trait SqlCoalesceSuite extends CatsEffectSuite { type Fragment - def mapping: IO[(QueryExecutor[IO, Json], SqlStatsMonitor[IO, Fragment])] + def mapping: IO[(Mapping[IO], SqlStatsMonitor[IO, Fragment])] test("simple coalesced query") { val query = """ diff --git a/modules/sql/shared/src/test/scala/SqlComposedWorldMapping.scala b/modules/sql/shared/src/test/scala/SqlComposedWorldMapping.scala index 722fdea0..60d85547 100644 --- a/modules/sql/shared/src/test/scala/SqlComposedWorldMapping.scala +++ b/modules/sql/shared/src/test/scala/SqlComposedWorldMapping.scala @@ -10,7 +10,6 @@ import edu.gemini.grackle.sql.Like import edu.gemini.grackle.syntax._ import io.circe.Json -import Cursor.{Context, Env} import Query._ import Predicate._ import Value._ @@ -55,24 +54,31 @@ class CurrencyMapping[F[_] : Sync](dataRef: Ref[F, CurrencyData], countRef: Ref[ val QueryType = schema.ref("Query") val CurrencyType = schema.ref("Currency") + override val selectElaborator = SelectElaborator { + case (QueryType, "exchangeRate", List(Binding("code", StringValue(code)))) => + Elab.env("code", code) + case (QueryType, "currencies", List(Binding("countryCodes", StringListValue(countryCodes)))) => + Elab.env("countryCodes", countryCodes) + } + val typeMappings = List( ValueObjectMapping[Unit]( tpe = QueryType, fieldMappings = List( - RootEffect.computeCursor("exchangeRate") { - case (Select(_, List(Binding("code", StringValue(code))), _), path, env) => + RootEffect.computeCursor("exchangeRate")((path, env) => + env.getR[String]("code").traverse(code => countRef.update(_+1) *> - dataRef.get.map(data => Result(valueCursor(path, env, data.exchangeRate(code)))) - case _ => Result.failure("Bad query").pure[F].widen - }, - RootEffect.computeCursor("currencies") { - case (Select(_, List(Binding("countryCodes", StringListValue(countryCodes))), _), path, env) => + dataRef.get.map(data => valueCursor(path, env, data.exchangeRate(code))) + ) + ), + RootEffect.computeCursor("currencies")((path, env) => + env.getR[List[String]]("countryCodes").traverse(countryCodes => countRef.update(_+1) *> - dataRef.get.map(data => Result(valueCursor(path, env, data.currencies(countryCodes)))) - case _ => Result.failure("Bad query").pure[F].widen - } + dataRef.get.map(data => valueCursor(path, env, data.currencies(countryCodes))) + ) + ) ) ), ValueObjectMapping[Currency]( @@ -92,7 +98,7 @@ class CurrencyMapping[F[_] : Sync](dataRef: Ref[F, CurrencyData], countRef: Ref[ val expandedQueries = queries.map { case (Select("currencies", _, child), c@Code(code)) => - (Select("currencies", List(Binding("countryCodes", ListValue(List(StringValue(code))))), child), c) + (SimpleCurrencyQuery(List(code), child), c) case other => other } @@ -107,7 +113,7 @@ class CurrencyMapping[F[_] : Sync](dataRef: Ref[F, CurrencyData], countRef: Ref[ def mkKey(q: ((Query, Cursor), String, Int)): (Query, Env, Context) = (q._1._1, q._1._2.fullEnv, q._1._2.context) - val grouped: List[((Select, Cursor), List[Int])] = + val grouped: List[((Query, Cursor), List[Int])] = (groupable.collect { case ((SimpleCurrencyQuery(code, child), cursor), i) => ((child, cursor), code, i) @@ -142,12 +148,12 @@ class CurrencyMapping[F[_] : Sync](dataRef: Ref[F, CurrencyData], countRef: Ref[ } object SimpleCurrencyQuery { - def apply(codes: List[String], child: Query): Select = - Select("currencies", List(Binding("countryCodes", StringListValue(codes))), child) + def apply(codes: List[String], child: Query): Query = + Environment(Env("countryCodes" -> codes), Select("currencies", child)) - def unapply(sel: Select): Option[(String, Query)] = + def unapply(sel: Query): Option[(String, Query)] = sel match { - case Select("currencies", List(Binding("countryCodes", StringListValue(List(code)))), child) => Some((code, child)) + case Environment(env, Select("currencies", None, child)) => env.get[List[String]]("countryCodes").flatMap(_.headOption).map((_, child)) case _ => None } @@ -257,14 +263,10 @@ class SqlComposedMapping[F[_] : Sync] ) ) - override val selectElaborator = new SelectElaborator(Map( - QueryType -> { - case Select("country", List(Binding("code", StringValue(code))), child) => - Select("country", Nil, Unique(Filter(Eql(CountryType / "code", Const(code)), child))).success - case Select("countries", _, child) => - Select("countries", Nil, child).success - case Select("cities", List(Binding("namePattern", StringValue(namePattern))), child) => - Select("cities", Nil, Filter(Like(CityType / "name", namePattern, true), child)).success - } - )) + override val selectElaborator = SelectElaborator { + case (QueryType, "country", List(Binding("code", StringValue(code)))) => + Elab.transformChild(child => Unique(Filter(Eql(CountryType / "code", Const(code)), child))) + case (QueryType, "cities", List(Binding("namePattern", StringValue(namePattern)))) => + Elab.transformChild(child => Filter(Like(CityType / "name", namePattern, true), child)) + } } diff --git a/modules/sql/shared/src/test/scala/SqlComposedWorldSuite.scala b/modules/sql/shared/src/test/scala/SqlComposedWorldSuite.scala index 177fb468..a21227f0 100644 --- a/modules/sql/shared/src/test/scala/SqlComposedWorldSuite.scala +++ b/modules/sql/shared/src/test/scala/SqlComposedWorldSuite.scala @@ -4,7 +4,6 @@ package edu.gemini.grackle.sql.test import cats.effect.IO -import io.circe.Json import io.circe.literal._ import munit.CatsEffectSuite @@ -13,7 +12,7 @@ import edu.gemini.grackle._ import grackle.test.GraphQLResponseTests.{assertWeaklyEqual, assertWeaklyEqualIO} trait SqlComposedWorldSuite extends CatsEffectSuite { - def mapping: IO[(CurrencyMapping[IO], QueryExecutor[IO, Json])] + def mapping: IO[(CurrencyMapping[IO], Mapping[IO])] test("simple effectful query") { val query = """ diff --git a/modules/sql/shared/src/test/scala/SqlCompositeKeySuite.scala b/modules/sql/shared/src/test/scala/SqlCompositeKeySuite.scala index 7e39dd59..c891dd78 100644 --- a/modules/sql/shared/src/test/scala/SqlCompositeKeySuite.scala +++ b/modules/sql/shared/src/test/scala/SqlCompositeKeySuite.scala @@ -4,7 +4,6 @@ package edu.gemini.grackle.sql.test import cats.effect.IO -import io.circe.Json import io.circe.literal._ import munit.CatsEffectSuite @@ -13,7 +12,7 @@ import edu.gemini.grackle._ import grackle.test.GraphQLResponseTests.assertWeaklyEqualIO trait SqlCompositeKeySuite extends CatsEffectSuite { - def mapping: QueryExecutor[IO, Json] + def mapping: Mapping[IO] test("root query") { val query = """ diff --git a/modules/sql/shared/src/test/scala/SqlCursorJsonMapping.scala b/modules/sql/shared/src/test/scala/SqlCursorJsonMapping.scala index 43083ae6..4575a710 100644 --- a/modules/sql/shared/src/test/scala/SqlCursorJsonMapping.scala +++ b/modules/sql/shared/src/test/scala/SqlCursorJsonMapping.scala @@ -90,10 +90,8 @@ trait SqlCursorJsonMapping[F[_]] extends SqlTestMapping[F] { PrimitiveMapping(CategoryType) ) - override val selectElaborator = new SelectElaborator(Map( - QueryType -> { - case Select("brands", List(Binding("id", IntValue(id))), child) => - Select("brands", Nil, Unique(Filter(Eql(BrandType / "id", Const(id)), child))).success - } - )) + override val selectElaborator = SelectElaborator { + case (QueryType, "brands", List(Binding("id", IntValue(id)))) => + Elab.transformChild(child => Unique(Filter(Eql(BrandType / "id", Const(id)), child))) + } } diff --git a/modules/sql/shared/src/test/scala/SqlCursorJsonSuite.scala b/modules/sql/shared/src/test/scala/SqlCursorJsonSuite.scala index a62fe69f..88ecf76f 100644 --- a/modules/sql/shared/src/test/scala/SqlCursorJsonSuite.scala +++ b/modules/sql/shared/src/test/scala/SqlCursorJsonSuite.scala @@ -3,7 +3,6 @@ package edu.gemini.grackle.sql.test -import io.circe.Json import cats.effect.IO import io.circe.literal._ import munit.CatsEffectSuite @@ -12,8 +11,7 @@ import edu.gemini.grackle._ import grackle.test.GraphQLResponseTests.assertWeaklyEqualIO trait SqlCursorJsonSuite extends CatsEffectSuite { - - def mapping: QueryExecutor[IO, Json] + def mapping: Mapping[IO] test("cursor field returns json") { val query = @@ -44,7 +42,6 @@ trait SqlCursorJsonSuite extends CatsEffectSuite { val res = mapping.compileAndRun(query) - assertWeaklyEqualIO(res, expected) } } diff --git a/modules/sql/shared/src/test/scala/SqlEmbedding2Mapping.scala b/modules/sql/shared/src/test/scala/SqlEmbedding2Mapping.scala index 22f8d449..e494212c 100644 --- a/modules/sql/shared/src/test/scala/SqlEmbedding2Mapping.scala +++ b/modules/sql/shared/src/test/scala/SqlEmbedding2Mapping.scala @@ -3,12 +3,12 @@ package edu.gemini.grackle.sql.test -import edu.gemini.grackle.Predicate._ -import edu.gemini.grackle.Query._ -import edu.gemini.grackle.QueryCompiler.SelectElaborator -import edu.gemini.grackle.Value._ -import edu.gemini.grackle.Result -import edu.gemini.grackle.syntax._ +import edu.gemini.grackle._ +import Predicate._ +import Query._ +import QueryCompiler._ +import Value._ +import syntax._ trait SqlEmbedding2Mapping[F[_]] extends SqlTestMapping[F] { @@ -72,18 +72,11 @@ trait SqlEmbedding2Mapping[F[_]] extends SqlTestMapping[F] { ) ) - override val selectElaborator = new SelectElaborator(Map( - QueryType -> { - case Select("program", List(Binding("programId", StringValue(id))), child) => - Result(Select("program", Nil, Unique(Filter(Eql(ProgramType / "id", Const(id)), child)))) - }, - ObservationSelectResultType -> { - case Select("matches", Nil, q) => - Result( - Select("matches", Nil, - OrderBy(OrderSelections(List(OrderSelection[String](ObservationType / "id", true, true))), q) - ) - ) - } - )) + override val selectElaborator = SelectElaborator { + case (QueryType, "program", List(Binding("programId", StringValue(id)))) => + Elab.transformChild(child => Unique(Filter(Eql(ProgramType / "id", Const(id)), child))) + + case (ObservationSelectResultType, "matches", Nil) => + Elab.transformChild(child => OrderBy(OrderSelections(List(OrderSelection[String](ObservationType / "id", true, true))), child)) + } } diff --git a/modules/sql/shared/src/test/scala/SqlEmbedding2Suite.scala b/modules/sql/shared/src/test/scala/SqlEmbedding2Suite.scala index d5f8a624..77d6dfa0 100644 --- a/modules/sql/shared/src/test/scala/SqlEmbedding2Suite.scala +++ b/modules/sql/shared/src/test/scala/SqlEmbedding2Suite.scala @@ -4,7 +4,6 @@ package edu.gemini.grackle.sql.test import cats.effect.IO -import io.circe.Json import io.circe.literal._ import munit.CatsEffectSuite @@ -13,7 +12,7 @@ import edu.gemini.grackle._ import grackle.test.GraphQLResponseTests.assertWeaklyEqualIO trait SqlEmbedding2Suite extends CatsEffectSuite { - def mapping: QueryExecutor[IO, Json] + def mapping: Mapping[IO] test("paging") { val query = """ diff --git a/modules/sql/shared/src/test/scala/SqlEmbedding3Suite.scala b/modules/sql/shared/src/test/scala/SqlEmbedding3Suite.scala index c5f91284..cb4ae7a6 100644 --- a/modules/sql/shared/src/test/scala/SqlEmbedding3Suite.scala +++ b/modules/sql/shared/src/test/scala/SqlEmbedding3Suite.scala @@ -4,7 +4,6 @@ package edu.gemini.grackle.sql.test import cats.effect.IO -import io.circe.Json import io.circe.literal._ import munit.CatsEffectSuite @@ -13,7 +12,7 @@ import edu.gemini.grackle._ import grackle.test.GraphQLResponseTests.assertWeaklyEqualIO trait SqlEmbedding3Suite extends CatsEffectSuite { - def mapping: QueryExecutor[IO, Json] + def mapping: Mapping[IO] test("paging") { val query = """ diff --git a/modules/sql/shared/src/test/scala/SqlEmbeddingSuite.scala b/modules/sql/shared/src/test/scala/SqlEmbeddingSuite.scala index 490459ed..1c9c8dce 100644 --- a/modules/sql/shared/src/test/scala/SqlEmbeddingSuite.scala +++ b/modules/sql/shared/src/test/scala/SqlEmbeddingSuite.scala @@ -4,7 +4,6 @@ package edu.gemini.grackle.sql.test import cats.effect.IO -import io.circe.Json import io.circe.literal._ import munit.CatsEffectSuite @@ -13,7 +12,7 @@ import edu.gemini.grackle._ import grackle.test.GraphQLResponseTests.assertWeaklyEqualIO trait SqlEmbeddingSuite extends CatsEffectSuite { - def mapping: QueryExecutor[IO, Json] + def mapping: Mapping[IO] test("simple embedded query (1)") { val query = """ diff --git a/modules/sql/shared/src/test/scala/SqlFilterJoinAliasMapping.scala b/modules/sql/shared/src/test/scala/SqlFilterJoinAliasMapping.scala index 2aeae853..c65821f8 100644 --- a/modules/sql/shared/src/test/scala/SqlFilterJoinAliasMapping.scala +++ b/modules/sql/shared/src/test/scala/SqlFilterJoinAliasMapping.scala @@ -8,8 +8,8 @@ import cats.implicits._ import edu.gemini.grackle._ import syntax._ import Predicate.{Const, Eql} -import Query.{Binding, Filter, Select, Unique} -import QueryCompiler.SelectElaborator +import Query.{Binding, Filter, Unique} +import QueryCompiler.{Elab, SelectElaborator} import Value.{AbsentValue, NullValue, ObjectValue, StringValue} trait SqlFilterJoinAliasMapping[F[_]] extends SqlTestMapping[F] { @@ -107,23 +107,11 @@ trait SqlFilterJoinAliasMapping[F[_]] extends SqlTestMapping[F] { } } - override val selectElaborator: SelectElaborator = new SelectElaborator(Map( - QueryType -> { - case Select("episode", List(Binding("id", StringValue(id))), child) => - Select( - "episode", - Nil, - Unique(Filter(Eql(EpisodeType / "id", Const(id)), child)) - ).success - }, - EpisodeType -> { - case Select("images", List(Binding("filter", filter)), child) => - for { - fc <- mkFilter(child, filter) - } yield Select("images", Nil, fc) + override val selectElaborator = SelectElaborator { + case (QueryType, "episode", List(Binding("id", StringValue(id)))) => + Elab.transformChild(child => Unique(Filter(Eql(EpisodeType / "id", Const(id)), child))) - case other => - other.success - } - )) + case (EpisodeType, "images", List(Binding("filter", filter))) => + Elab.transformChild(child => mkFilter(child, filter)) + } } diff --git a/modules/sql/shared/src/test/scala/SqlFilterOrderOffsetLimit2Mapping.scala b/modules/sql/shared/src/test/scala/SqlFilterOrderOffsetLimit2Mapping.scala index 3a035fff..471b0f14 100644 --- a/modules/sql/shared/src/test/scala/SqlFilterOrderOffsetLimit2Mapping.scala +++ b/modules/sql/shared/src/test/scala/SqlFilterOrderOffsetLimit2Mapping.scala @@ -4,8 +4,8 @@ package edu.gemini.grackle.sql.test import edu.gemini.grackle._, syntax._ -import Query.{Binding, Limit, Select} -import QueryCompiler.SelectElaborator +import Query.{Binding, Limit} +import QueryCompiler.{Elab, SelectElaborator} import Value.{AbsentValue, IntValue, NullValue} trait SqlFilterOrderOffsetLimit2Mapping[F[_]] extends SqlTestMapping[F] { @@ -115,50 +115,14 @@ trait SqlFilterOrderOffsetLimit2Mapping[F[_]] extends SqlTestMapping[F] { case other => Result.failure(s"Expected limit > 0, found $other") } - override val selectElaborator: SelectElaborator = new SelectElaborator(Map( - QueryType -> { - case Select("root", List(Binding("limit", limit)), child) => - for { - lc <- mkLimit(child, limit) - } yield Select("root", Nil, lc) + override val selectElaborator = SelectElaborator { + case (QueryType, "root"|"containers", List(Binding("limit", limit))) => + Elab.transformChild(child => mkLimit(child, limit)) - case Select("containers", List(Binding("limit", limit)), child) => - for { - lc <- mkLimit(child, limit) - } yield Select("containers", Nil, lc) + case (RootType, "containers"|"listA"|"listB", List(Binding("limit", limit))) => + Elab.transformChild(child => mkLimit(child, limit)) - case other => other.success - }, - RootType -> { - case Select("containers", List(Binding("limit", limit)), child) => - for { - lc <- mkLimit(child, limit) - } yield Select("containers", Nil, lc) - - case Select("listA", List(Binding("limit", limit)), child) => - for { - lc <- mkLimit(child, limit) - } yield Select("listA", Nil, lc) - - case Select("listB", List(Binding("limit", limit)), child) => - for { - lc <- mkLimit(child, limit) - } yield Select("listB", Nil, lc) - - case other => other.success - }, - ContainerType -> { - case Select("listA", List(Binding("limit", limit)), child) => - for { - lc <- mkLimit(child, limit) - } yield Select("listA", Nil, lc) - - case Select("listB", List(Binding("limit", limit)), child) => - for { - lc <- mkLimit(child, limit) - } yield Select("listB", Nil, lc) - - case other => other.success - } - )) + case (ContainerType, "listA"|"listB", List(Binding("limit", limit))) => + Elab.transformChild(child => mkLimit(child, limit)) + } } diff --git a/modules/sql/shared/src/test/scala/SqlFilterOrderOffsetLimit2Suite.scala b/modules/sql/shared/src/test/scala/SqlFilterOrderOffsetLimit2Suite.scala index 0f85adfa..fa91bc39 100644 --- a/modules/sql/shared/src/test/scala/SqlFilterOrderOffsetLimit2Suite.scala +++ b/modules/sql/shared/src/test/scala/SqlFilterOrderOffsetLimit2Suite.scala @@ -4,7 +4,6 @@ package edu.gemini.grackle.sql.test import cats.effect.IO -import io.circe.Json import io.circe.literal._ import munit.CatsEffectSuite @@ -13,7 +12,7 @@ import edu.gemini.grackle._ import grackle.test.GraphQLResponseTests.assertWeaklyEqualIO trait SqlFilterOrderOffsetLimit2Suite extends CatsEffectSuite { - def mapping: QueryExecutor[IO, Json] + def mapping: Mapping[IO] test("base query") { val query = """ diff --git a/modules/sql/shared/src/test/scala/SqlFilterOrderOffsetLimitMapping.scala b/modules/sql/shared/src/test/scala/SqlFilterOrderOffsetLimitMapping.scala index fc7606c6..0d535266 100644 --- a/modules/sql/shared/src/test/scala/SqlFilterOrderOffsetLimitMapping.scala +++ b/modules/sql/shared/src/test/scala/SqlFilterOrderOffsetLimitMapping.scala @@ -7,9 +7,9 @@ import cats.implicits._ import edu.gemini.grackle._, syntax._ import Predicate.{Const, Eql} -import Query.{Binding, Filter, Limit, Offset, OrderBy, OrderSelection, OrderSelections, Select} -import QueryCompiler.SelectElaborator -import Value.{AbsentValue, IntValue, NullValue, ObjectValue, StringValue, TypedEnumValue} +import Query.{Binding, Filter, Limit, Offset, OrderBy, OrderSelection, OrderSelections} +import QueryCompiler.{Elab, SelectElaborator} +import Value.{AbsentValue, EnumValue, IntValue, NullValue, ObjectValue, StringValue} trait SqlFilterOrderOffsetLimitMapping[F[_]] extends SqlTestMapping[F] { @@ -77,8 +77,8 @@ trait SqlFilterOrderOffsetLimitMapping[F[_]] extends SqlTestMapping[F] { } object OrderValue { - def unapply(tev: TypedEnumValue): Option[ListOrder] = - ListOrder.fromGraphQLString(tev.value.name) + def unapply(ev: EnumValue): Option[ListOrder] = + ListOrder.fromGraphQLString(ev.name) } val typeMappings = @@ -164,36 +164,34 @@ trait SqlFilterOrderOffsetLimitMapping[F[_]] extends SqlTestMapping[F] { case _ => Result.failure(s"Expected sort value, found $order") } - override val selectElaborator: SelectElaborator = new SelectElaborator(Map( - QueryType -> { - case Select("root", List(Binding("filter", filter), Binding("offset", offset), Binding("limit", limit)), child) => + override val selectElaborator = SelectElaborator { + case (QueryType, "root", List(Binding("filter", filter), Binding("offset", offset), Binding("limit", limit))) => + Elab.transformChild(child => for { fc <- mkFilter(child, RootType, filter) oc <- mkOffset(fc, offset) lc <- mkLimit(oc, limit) - } yield Select("root", Nil, lc) + } yield lc + ) - case other => other.success - }, - RootType -> { - case Select("listA", List(Binding("filter", filter), Binding("order", order), Binding("offset", offset), Binding("limit", limit)), child) => + case (RootType, "listA", List(Binding("filter", filter), Binding("order", order), Binding("offset", offset), Binding("limit", limit))) => + Elab.transformChild(child => for { fc <- mkFilter(child, ElemAType, filter) sc <- mkOrderBy(fc, order)(o => List(OrderSelection[Option[String]](ElemAType / "elemA", ascending = o.ascending))) oc <- mkOffset(sc, offset) lc <- mkLimit(oc, limit) - } yield Select("listA", Nil, lc) + } yield lc + ) - case Select("listB", List(Binding("filter", filter), Binding("order", order), Binding("offset", offset), Binding("limit", limit)), child) => + case (RootType, "listB", List(Binding("filter", filter), Binding("order", order), Binding("offset", offset), Binding("limit", limit))) => + Elab.transformChild(child => for { fc <- mkFilter(child, ElemBType, filter) sc <- mkOrderBy(fc, order)(o => List(OrderSelection[Option[Int]](ElemBType / "elemB", ascending = o.ascending))) oc <- mkOffset(sc, offset) lc <- mkLimit(oc, limit) - } yield Select("listB", Nil, lc) - - case other => - other.success - } - )) + } yield lc + ) + } } diff --git a/modules/sql/shared/src/test/scala/SqlFilterOrderOffsetLimitSuite.scala b/modules/sql/shared/src/test/scala/SqlFilterOrderOffsetLimitSuite.scala index 92386215..802a3953 100644 --- a/modules/sql/shared/src/test/scala/SqlFilterOrderOffsetLimitSuite.scala +++ b/modules/sql/shared/src/test/scala/SqlFilterOrderOffsetLimitSuite.scala @@ -4,7 +4,6 @@ package edu.gemini.grackle.sql.test import cats.effect.IO -import io.circe.Json import io.circe.literal._ import munit.CatsEffectSuite @@ -13,7 +12,7 @@ import edu.gemini.grackle._ import grackle.test.GraphQLResponseTests.assertWeaklyEqualIO trait SqlFilterOrderOffsetLimitSuite extends CatsEffectSuite { - def mapping: QueryExecutor[IO, Json] + def mapping: Mapping[IO] test("base query") { val query = """ diff --git a/modules/sql/shared/src/test/scala/SqlGraphMapping.scala b/modules/sql/shared/src/test/scala/SqlGraphMapping.scala index d530919b..295dbd30 100644 --- a/modules/sql/shared/src/test/scala/SqlGraphMapping.scala +++ b/modules/sql/shared/src/test/scala/SqlGraphMapping.scala @@ -70,12 +70,8 @@ trait SqlGraphMapping[F[_]] extends SqlTestMapping[F] { ) ) - override val selectElaborator: SelectElaborator = new SelectElaborator(Map( - QueryType -> { - case Select("node", List(Binding("id", IntValue(id))), child) => - Select("node", Nil, Unique(Filter(Eql(NodeType / "id", Const(id)), child))).success - - case other => other.success - } - )) + override val selectElaborator = SelectElaborator { + case (QueryType, "node", List(Binding("id", IntValue(id)))) => + Elab.transformChild(child => Unique(Filter(Eql(NodeType / "id", Const(id)), child))) + } } diff --git a/modules/sql/shared/src/test/scala/SqlGraphSuite.scala b/modules/sql/shared/src/test/scala/SqlGraphSuite.scala index fce4fb0d..b0837e40 100644 --- a/modules/sql/shared/src/test/scala/SqlGraphSuite.scala +++ b/modules/sql/shared/src/test/scala/SqlGraphSuite.scala @@ -4,7 +4,6 @@ package edu.gemini.grackle.sql.test import cats.effect.IO -import io.circe.Json import io.circe.literal._ import munit.CatsEffectSuite @@ -13,7 +12,7 @@ import edu.gemini.grackle._ import grackle.test.GraphQLResponseTests.assertWeaklyEqualIO trait SqlGraphSuite extends CatsEffectSuite { - def mapping: QueryExecutor[IO, Json] + def mapping: Mapping[IO] test("root query") { val query = """ diff --git a/modules/sql/shared/src/test/scala/SqlInterfacesMapping.scala b/modules/sql/shared/src/test/scala/SqlInterfacesMapping.scala index 36c9ad5a..36bae8df 100644 --- a/modules/sql/shared/src/test/scala/SqlInterfacesMapping.scala +++ b/modules/sql/shared/src/test/scala/SqlInterfacesMapping.scala @@ -199,12 +199,10 @@ trait SqlInterfacesMapping[F[_]] extends SqlTestMapping[F] { self => } } - override val selectElaborator = new SelectElaborator(Map( - QueryType -> { - case Select("films", Nil, child) => - Select("films", Nil, Filter(Eql[EntityType](FilmType / "entityType", Const(EntityType.Film)), child)).success - } - )) + override val selectElaborator = SelectElaborator { + case (QueryType, "films", Nil) => + Elab.transformChild(child => Filter(Eql[EntityType](FilmType / "entityType", Const(EntityType.Film)), child)) + } sealed trait EntityType extends Product with Serializable object EntityType { diff --git a/modules/sql/shared/src/test/scala/SqlInterfacesSuite.scala b/modules/sql/shared/src/test/scala/SqlInterfacesSuite.scala index 0d80ab6a..a00c92f9 100644 --- a/modules/sql/shared/src/test/scala/SqlInterfacesSuite.scala +++ b/modules/sql/shared/src/test/scala/SqlInterfacesSuite.scala @@ -4,7 +4,6 @@ package edu.gemini.grackle.sql.test import cats.effect.IO -import io.circe.Json import io.circe.literal._ import munit.CatsEffectSuite @@ -13,7 +12,7 @@ import edu.gemini.grackle._ import grackle.test.GraphQLResponseTests.assertWeaklyEqualIO trait SqlInterfacesSuite extends CatsEffectSuite { - def mapping: QueryExecutor[IO, Json] + def mapping: Mapping[IO] test("simple interface query") { val query = """ diff --git a/modules/sql/shared/src/test/scala/SqlInterfacesSuite2.scala b/modules/sql/shared/src/test/scala/SqlInterfacesSuite2.scala index a496e332..8461d450 100644 --- a/modules/sql/shared/src/test/scala/SqlInterfacesSuite2.scala +++ b/modules/sql/shared/src/test/scala/SqlInterfacesSuite2.scala @@ -4,7 +4,6 @@ package edu.gemini.grackle.sql.test import cats.effect.IO -import io.circe.Json import io.circe.literal._ import munit.CatsEffectSuite @@ -13,7 +12,7 @@ import edu.gemini.grackle._ import grackle.test.GraphQLResponseTests.assertWeaklyEqualIO trait SqlInterfacesSuite2 extends CatsEffectSuite { - def mapping: QueryExecutor[IO, Json] + def mapping: Mapping[IO] test("when discriminator fails the fragments should be ignored") { val query = """ diff --git a/modules/sql/shared/src/test/scala/SqlJsonbMapping.scala b/modules/sql/shared/src/test/scala/SqlJsonbMapping.scala index 81c17a8b..51946199 100644 --- a/modules/sql/shared/src/test/scala/SqlJsonbMapping.scala +++ b/modules/sql/shared/src/test/scala/SqlJsonbMapping.scala @@ -84,10 +84,8 @@ trait SqlJsonbMapping[F[_]] extends SqlTestMapping[F] { ), ) - override val selectElaborator = new SelectElaborator(Map( - QueryType -> { - case Select("record", List(Binding("id", IntValue(id))), child) => - Select("record", Nil, Unique(Filter(Eql(RowType / "id", Const(id)), child))).success - } - )) + override val selectElaborator = SelectElaborator { + case (QueryType, "record", List(Binding("id", IntValue(id)))) => + Elab.transformChild(child => Unique(Filter(Eql(RowType / "id", Const(id)), child))) + } } diff --git a/modules/sql/shared/src/test/scala/SqlJsonbSuite.scala b/modules/sql/shared/src/test/scala/SqlJsonbSuite.scala index e719b464..dbd7dc61 100644 --- a/modules/sql/shared/src/test/scala/SqlJsonbSuite.scala +++ b/modules/sql/shared/src/test/scala/SqlJsonbSuite.scala @@ -3,7 +3,6 @@ package edu.gemini.grackle.sql.test -import io.circe.Json import cats.effect.IO import io.circe.literal._ import munit.CatsEffectSuite @@ -14,7 +13,7 @@ import grackle.test.GraphQLResponseTests.assertWeaklyEqualIO trait SqlJsonbSuite extends CatsEffectSuite { - def mapping: QueryExecutor[IO, Json] + def mapping: Mapping[IO] test("simple jsonb query") { val query = """ diff --git a/modules/sql/shared/src/test/scala/SqlLikeMapping.scala b/modules/sql/shared/src/test/scala/SqlLikeMapping.scala index 19cc39f8..22bd7646 100644 --- a/modules/sql/shared/src/test/scala/SqlLikeMapping.scala +++ b/modules/sql/shared/src/test/scala/SqlLikeMapping.scala @@ -43,6 +43,10 @@ trait SqlLikeMapping[F[_]] extends SqlTestMapping[F] { fieldMappings = List( SqlObject("likes"), + SqlObject("likeNotNullableNotNullable"), + SqlObject("likeNotNullableNullable"), + SqlObject("likeNullableNotNullable"), + SqlObject("likeNullableNullable") ) ), ObjectMapping( @@ -81,18 +85,17 @@ trait SqlLikeMapping[F[_]] extends SqlTestMapping[F] { case Some(p) => Like(t, p, false) } - override val selectElaborator: SelectElaborator = new SelectElaborator(Map( - QueryType -> { - case Select(f@"likeNotNullableNotNullable", List(Binding("pattern", NonNullablePattern(pattern))), child) => - Rename(f, Select("likes", Nil, Filter(Like(LikeType / "notNullable", pattern, false), child))).success - case Select(f@"likeNotNullableNullable", List(Binding("pattern", NullablePattern(pattern))), child) => - Rename(f, Select("likes", Nil, Filter(mkPredicate(LikeType / "notNullable", pattern), child))).success - case Select(f@"likeNullableNotNullable", List(Binding("pattern", NonNullablePattern(pattern))), child) => - Rename(f, Select("likes", Nil, Filter(Like(LikeType / "nullable", pattern, false), child))).success - case Select(f@"likeNullableNullable", List(Binding("pattern", NullablePattern(pattern))), child) => - Rename(f, Select("likes", Nil, Filter(mkPredicate(LikeType / "nullable", pattern), child))).success + override val selectElaborator = SelectElaborator { + case (QueryType, "likeNotNullableNotNullable", List(Binding("pattern", NonNullablePattern(pattern)))) => + Elab.transformChild(child => Filter(Like(LikeType / "notNullable", pattern, false), child)) - case other => other.success - } - )) + case (QueryType, "likeNotNullableNullable", List(Binding("pattern", NullablePattern(pattern)))) => + Elab.transformChild(child => Filter(mkPredicate(LikeType / "notNullable", pattern), child)) + + case (QueryType, "likeNullableNotNullable", List(Binding("pattern", NonNullablePattern(pattern)))) => + Elab.transformChild(child => Filter(Like(LikeType / "nullable", pattern, false), child)) + + case (QueryType, "likeNullableNullable", List(Binding("pattern", NullablePattern(pattern)))) => + Elab.transformChild(child => Filter(mkPredicate(LikeType / "nullable", pattern), child)) + } } diff --git a/modules/sql/shared/src/test/scala/SqlLikeSuite.scala b/modules/sql/shared/src/test/scala/SqlLikeSuite.scala index 57cd52a8..c1112e22 100644 --- a/modules/sql/shared/src/test/scala/SqlLikeSuite.scala +++ b/modules/sql/shared/src/test/scala/SqlLikeSuite.scala @@ -4,7 +4,6 @@ package edu.gemini.grackle.sql.test import cats.effect.IO -import io.circe.Json import io.circe.literal._ import munit.CatsEffectSuite @@ -12,7 +11,7 @@ import edu.gemini.grackle._ import grackle.test.GraphQLResponseTests.assertWeaklyEqualIO trait SqlLikeSuite extends CatsEffectSuite { - def mapping: QueryExecutor[IO, Json] + def mapping: Mapping[IO] test("No filter") { val query = """ diff --git a/modules/sql/shared/src/test/scala/SqlMixedMapping.scala b/modules/sql/shared/src/test/scala/SqlMixedMapping.scala index b8d9b86c..5c923796 100644 --- a/modules/sql/shared/src/test/scala/SqlMixedMapping.scala +++ b/modules/sql/shared/src/test/scala/SqlMixedMapping.scala @@ -89,10 +89,8 @@ trait SqlMixedMapping[F[_]] extends SqlTestMapping[F] with ValueMappingLike[F] { Try(UUID.fromString(s.value)).toOption } - override val selectElaborator = new SelectElaborator(Map( - QueryType -> { - case Select("movie", List(Binding("id", UUIDValue(id))), child) => - Select("movie", Nil, Unique(Filter(Eql(MovieType / "id", Const(id)), child))).success - } - )) + override val selectElaborator = SelectElaborator { + case (QueryType, "movie", List(Binding("id", UUIDValue(id)))) => + Elab.transformChild(child => Unique(Filter(Eql(MovieType / "id", Const(id)), child))) + } } diff --git a/modules/sql/shared/src/test/scala/SqlMixedSuite.scala b/modules/sql/shared/src/test/scala/SqlMixedSuite.scala index b6c90e0a..b05da8e5 100644 --- a/modules/sql/shared/src/test/scala/SqlMixedSuite.scala +++ b/modules/sql/shared/src/test/scala/SqlMixedSuite.scala @@ -4,7 +4,6 @@ package edu.gemini.grackle.sql.test import cats.effect.IO -import io.circe.Json import io.circe.literal._ import munit.CatsEffectSuite @@ -13,7 +12,7 @@ import edu.gemini.grackle._ import grackle.test.GraphQLResponseTests.assertWeaklyEqualIO trait SqlMixedSuite extends CatsEffectSuite { - def mapping: QueryExecutor[IO, Json] + def mapping: Mapping[IO] test("DB query") { val query = """ diff --git a/modules/sql/shared/src/test/scala/SqlMovieMapping.scala b/modules/sql/shared/src/test/scala/SqlMovieMapping.scala index 316483a1..cd47bcab 100644 --- a/modules/sql/shared/src/test/scala/SqlMovieMapping.scala +++ b/modules/sql/shared/src/test/scala/SqlMovieMapping.scala @@ -168,8 +168,8 @@ trait SqlMovieMapping[F[_]] extends SqlTestMapping[F] { self => } object GenreValue { - def unapply(e: TypedEnumValue): Option[Genre] = - Genre.fromString(e.value.name) + def unapply(e: EnumValue): Option[Genre] = + Genre.fromString(e.name) } object GenreListValue { @@ -180,52 +180,50 @@ trait SqlMovieMapping[F[_]] extends SqlTestMapping[F] { self => } } - override val selectElaborator = new SelectElaborator(Map( - QueryType -> { - case Select("movieById", List(Binding("id", UUIDValue(id))), child) => - Select("movieById", Nil, Unique(Filter(Eql(MovieType / "id", Const(id)), child))).success - case Select("moviesByGenre", List(Binding("genre", GenreValue(genre))), child) => - Select("moviesByGenre", Nil, Filter(Eql(MovieType / "genre", Const(genre)), child)).success - case Select("moviesByGenres", List(Binding("genres", GenreListValue(genres))), child) => - Select("moviesByGenres", Nil, Filter(In(MovieType / "genre", genres), child)).success - case Select("moviesReleasedBetween", List(Binding("from", DateValue(from)), Binding("to", DateValue(to))), child) => - Select("moviesReleasedBetween", Nil, - Filter( - And( - Not(Lt(MovieType / "releaseDate", Const(from))), - Lt(MovieType / "releaseDate", Const(to)) - ), - child - ) - ).success - case Select("moviesLongerThan", List(Binding("duration", IntervalValue(duration))), child) => - Select("moviesLongerThan", Nil, - Filter( - Not(Lt(MovieType / "duration", Const(duration))), - child - ) - ).success - case Select("moviesShownLaterThan", List(Binding("time", TimeValue(time))), child) => - Select("moviesShownLaterThan", Nil, - Filter( - Not(Lt(MovieType / "showTime", Const(time))), - child - ) - ).success - case Select("moviesShownBetween", List(Binding("from", DateTimeValue(from)), Binding("to", DateTimeValue(to))), child) => - Select("moviesShownBetween", Nil, - Filter( - And( - Not(Lt(MovieType / "nextShowing", Const(from))), - Lt(MovieType / "nextShowing", Const(to)) - ), - child - ) - ).success - case Select("longMovies", Nil, child) => - Select("longMovies", Nil, Filter(Eql(MovieType / "isLong", Const(true)), child)).success - } - )) + override val selectElaborator = SelectElaborator { + case (QueryType, "movieById", List(Binding("id", UUIDValue(id)))) => + Elab.transformChild(child => Unique(Filter(Eql(MovieType / "id", Const(id)), child))) + case (QueryType, "moviesByGenre", List(Binding("genre", GenreValue(genre)))) => + Elab.transformChild(child => Filter(Eql(MovieType / "genre", Const(genre)), child)) + case (QueryType, "moviesByGenres", List(Binding("genres", GenreListValue(genres)))) => + Elab.transformChild(child => Filter(In(MovieType / "genre", genres), child)) + case (QueryType, "moviesReleasedBetween", List(Binding("from", DateValue(from)), Binding("to", DateValue(to)))) => + Elab.transformChild(child => + Filter( + And( + Not(Lt(MovieType / "releaseDate", Const(from))), + Lt(MovieType / "releaseDate", Const(to)) + ), + child + ) + ) + case (QueryType, "moviesLongerThan", List(Binding("duration", IntervalValue(duration)))) => + Elab.transformChild(child => + Filter( + Not(Lt(MovieType / "duration", Const(duration))), + child + ) + ) + case (QueryType, "moviesShownLaterThan", List(Binding("time", TimeValue(time)))) => + Elab.transformChild(child => + Filter( + Not(Lt(MovieType / "showTime", Const(time))), + child + ) + ) + case (QueryType, "moviesShownBetween", List(Binding("from", DateTimeValue(from)), Binding("to", DateTimeValue(to)))) => + Elab.transformChild(child => + Filter( + And( + Not(Lt(MovieType / "nextShowing", Const(from))), + Lt(MovieType / "nextShowing", Const(to)) + ), + child + ) + ) + case (QueryType, "longMovies", Nil) => + Elab.transformChild(child => Filter(Eql(MovieType / "isLong", Const(true)), child)) + } sealed trait Genre extends Product with Serializable object Genre { diff --git a/modules/sql/shared/src/test/scala/SqlMovieSuite.scala b/modules/sql/shared/src/test/scala/SqlMovieSuite.scala index 1c1edd40..1f33706a 100644 --- a/modules/sql/shared/src/test/scala/SqlMovieSuite.scala +++ b/modules/sql/shared/src/test/scala/SqlMovieSuite.scala @@ -4,7 +4,6 @@ package edu.gemini.grackle.sql.test import cats.effect.IO -import io.circe.Json import io.circe.literal._ import munit.CatsEffectSuite @@ -14,7 +13,7 @@ import grackle.test.GraphQLResponseTests.assertWeaklyEqualIO trait SqlMovieSuite extends CatsEffectSuite { - def mapping: QueryExecutor[IO, Json] + def mapping: Mapping[IO] test("query with UUID argument and custom scalar results") { val query = """ @@ -455,7 +454,7 @@ trait SqlMovieSuite extends CatsEffectSuite { { "errors" : [ { - "message" : "Unknown field 'isLong' in select" + "message" : "No field 'isLong' for type Movie" } ] } diff --git a/modules/sql/shared/src/test/scala/SqlMutationMapping.scala b/modules/sql/shared/src/test/scala/SqlMutationMapping.scala index 35933519..ac016c5c 100644 --- a/modules/sql/shared/src/test/scala/SqlMutationMapping.scala +++ b/modules/sql/shared/src/test/scala/SqlMutationMapping.scala @@ -57,6 +57,9 @@ trait SqlMutationMapping[F[_]] extends SqlTestMapping[F] { val CountryType = schema.ref("Country") val CityType = schema.ref("City") + case class UpdatePopulation(id: Int, population: Int) + case class CreateCity(name: String, countryCode: String, population: Int) + val typeMappings = List( ObjectMapping( @@ -68,25 +71,17 @@ trait SqlMutationMapping[F[_]] extends SqlTestMapping[F] { ObjectMapping( tpe = MutationType, fieldMappings = List( - RootEffect.computeQuery("updatePopulation")((query, _, env) => - (env.get[Int]("id"), env.get[Int]("population")).tupled match { - case Some((id, pop)) => - updatePopulation(id, pop).as(Result(query)) - case None => - Result.internalError(s"Implementation error, expected id and population in $env.").pure[F].widen + RootEffect.computeUnit("updatePopulation")(env => + env.getR[UpdatePopulation]("updatePopulation").traverse { + case UpdatePopulation(id, pop) => updatePopulation(id, pop) } ), - RootEffect.computeQuery("createCity")((query, _, env) => - (env.get[String]("name"), env.get[String]("countryCode"), env.get[Int]("population")).tupled match { - case Some((name, cc, pop)) => - query match { - case en@Environment(_, s@Select(_, _, child)) => - createCity(name, cc, pop).map { id => - Result(en.copy(child = s.copy(child = (Unique(Filter(Eql(CityType / "id", Const(id)), child)))))) - } - case _ => Result.internalError(s"Implementation error: expected Environment node.").pure[F].widen + RootEffect.computeChild("createCity")((child, _, env) => + env.getR[CreateCity]("createCity").flatTraverse { + case CreateCity(name, cc, pop) => + createCity(name, cc, pop).map { id => + Unique(Filter(Eql(CityType / "id", Const(id)), child)).success } - case None => Result.internalError(s"Implementation error: expected name, countryCode and population in $env.").pure[F].widen } ) ) @@ -111,29 +106,17 @@ trait SqlMutationMapping[F[_]] extends SqlTestMapping[F] { ), ) - override val selectElaborator = new SelectElaborator(Map( - QueryType -> { - case Select("city", List(Binding("id", IntValue(id))), child) => - Select("city", Nil, Unique(Filter(Eql(CityType / "id", Const(id)), child))).success - }, - MutationType -> { - - case Select("updatePopulation", List(Binding("id", IntValue(id)), Binding("population", IntValue(pop))), child) => - Environment( - Cursor.Env("id" -> id, "population" -> pop), - Select("updatePopulation", Nil, - // We could also do this in the SqlRoot's mutation, and in fact would need to do so if - // the mutation generated a new id. But for now it seems easiest to do it here. - Unique(Filter(Eql(CityType / "id", Const(id)), child)) - ) - ).success + override val selectElaborator = SelectElaborator { + case (QueryType, "city", List(Binding("id", IntValue(id)))) => + Elab.transformChild(child => Unique(Filter(Eql(CityType / "id", Const(id)), child))) - case Select("createCity", List(Binding("name", StringValue(name)), Binding("countryCode", StringValue(code)), Binding("population", IntValue(pop))), child) => - Environment( - Cursor.Env[Any]("name" -> name, "countryCode" -> code, "population" -> pop), - Select("createCity", Nil, child) - ).success + case (MutationType, "updatePopulation", List(Binding("id", IntValue(id)), Binding("population", IntValue(pop)))) => + for { + _ <- Elab.env("updatePopulation", UpdatePopulation(id, pop)) + _ <- Elab.transformChild(child => Unique(Filter(Eql(CityType / "id", Const(id)), child))) + } yield () - } - )) + case (MutationType, "createCity", List(Binding("name", StringValue(name)), Binding("countryCode", StringValue(code)), Binding("population", IntValue(pop)))) => + Elab.env("createCity", CreateCity(name, code, pop)) + } } diff --git a/modules/sql/shared/src/test/scala/SqlMutationSuite.scala b/modules/sql/shared/src/test/scala/SqlMutationSuite.scala index fa641ecf..3ba28291 100644 --- a/modules/sql/shared/src/test/scala/SqlMutationSuite.scala +++ b/modules/sql/shared/src/test/scala/SqlMutationSuite.scala @@ -4,7 +4,6 @@ package edu.gemini.grackle.sql.test import cats.effect.IO -import io.circe.Json import io.circe.literal._ import munit.CatsEffectSuite @@ -14,7 +13,7 @@ import grackle.test.GraphQLResponseTests.assertWeaklyEqualIO trait SqlMutationSuite extends CatsEffectSuite { - def mapping: QueryExecutor[IO, Json] + def mapping: Mapping[IO] test("simple read") { val query = """ diff --git a/modules/sql/shared/src/test/scala/SqlNestedEffectsMapping.scala b/modules/sql/shared/src/test/scala/SqlNestedEffectsMapping.scala index 15c8d7f1..0e45a292 100644 --- a/modules/sql/shared/src/test/scala/SqlNestedEffectsMapping.scala +++ b/modules/sql/shared/src/test/scala/SqlNestedEffectsMapping.scala @@ -12,8 +12,8 @@ import io.circe.generic.semiauto.deriveEncoder import edu.gemini.grackle._ import sql.Like import syntax._ -import Cursor.Env -import Query._, Predicate._, Value._ +import Query._ +import Predicate._, Value._ import QueryCompiler._ class CurrencyService[F[_] : Sync](dataRef: Ref[F, CurrencyData], countRef: Ref[F, Int]) { @@ -204,7 +204,7 @@ trait SqlNestedEffectsMapping[F[_]] extends SqlTestMapping[F] { val distinctCodes = queries.flatMap(_._2.fieldAs[String]("code2").toList).distinct val children = queries.flatMap { - case (PossiblyRenamedSelect(Select(name, _, child), alias), parentCursor) => + case (Select(name, alias, child), parentCursor) => parentCursor.context.forField(name, alias).toList.map(ctx => (ctx, child, parentCursor)) case _ => Nil } @@ -241,15 +241,15 @@ trait SqlNestedEffectsMapping[F[_]] extends SqlTestMapping[F] { val toCode = Map("BR" -> "BRA", "GB" -> "GBR", "NL" -> "NLD") def runEffects(queries: List[(Query, Cursor)]): F[Result[List[(Query, Cursor)]]] = { runGrouped(queries) { - case (PossiblyRenamedSelect(Select("country", _, child), alias), cursors, indices) => + case (Select("country", alias, child), cursors, indices) => val codes = cursors.flatMap(_.fieldAs[Json]("countryCode").toOption.flatMap(_.asString).toList).map(toCode) - val combinedQuery = PossiblyRenamedSelect(Select("country", Nil, Filter(In(CountryType / "code", codes), child)), alias) + val combinedQuery = Select("country", alias, Filter(In(CountryType / "code", codes), child)) (for { cursor <- ResultT(sqlCursor(combinedQuery, Env.empty)) } yield { codes.map { code => - (PossiblyRenamedSelect(Select("country", Nil, Unique(Filter(Eql(CountryType / "code", Const(code)), child))), alias), cursor) + (Select("country", alias, Unique(Filter(Eql(CountryType / "code", Const(code)), child))), cursor) }.zip(indices) }).value.widen @@ -269,16 +269,14 @@ trait SqlNestedEffectsMapping[F[_]] extends SqlTestMapping[F] { } } - override val selectElaborator = new SelectElaborator(Map( - QueryType -> { - case Select("cities", List(Binding("namePattern", StringValue(namePattern))), child) => - if (namePattern == "%") - Select("cities", Nil, child).success - else - Select("cities", Nil, Filter(Like(CityType / "name", namePattern, true), child)).success + override val selectElaborator = SelectElaborator { + case (QueryType, "cities", List(Binding("namePattern", StringValue(namePattern)))) => + if (namePattern == "%") + Elab.unit + else + Elab.transformChild(child => Filter(Like(CityType / "name", namePattern, true), child)) - case Select("country", List(Binding("code", StringValue(code))), child) => - Select("country", Nil, Unique(Filter(Eql(CountryType / "code", Const(code)), child))).success - } - )) + case (QueryType, "country", List(Binding("code", StringValue(code)))) => + Elab.transformChild(child => Unique(Filter(Eql(CountryType / "code", Const(code)), child))) + } } diff --git a/modules/sql/shared/src/test/scala/SqlNestedEffectsSuite.scala b/modules/sql/shared/src/test/scala/SqlNestedEffectsSuite.scala index f2f795bd..9c2d0d06 100644 --- a/modules/sql/shared/src/test/scala/SqlNestedEffectsSuite.scala +++ b/modules/sql/shared/src/test/scala/SqlNestedEffectsSuite.scala @@ -4,7 +4,6 @@ package edu.gemini.grackle.sql.test import cats.effect.IO -import io.circe.Json import io.circe.literal._ import munit.CatsEffectSuite @@ -13,7 +12,7 @@ import edu.gemini.grackle._ import grackle.test.GraphQLResponseTests.{assertWeaklyEqual, assertWeaklyEqualIO} trait SqlNestedEffectsSuite extends CatsEffectSuite { - def mapping: IO[(CurrencyService[IO], QueryExecutor[IO, Json])] + def mapping: IO[(CurrencyService[IO], Mapping[IO])] test("simple effectful service call") { val expected = json""" diff --git a/modules/sql/shared/src/test/scala/SqlPaging1Mapping.scala b/modules/sql/shared/src/test/scala/SqlPaging1Mapping.scala index c23fa60a..a0c543cd 100644 --- a/modules/sql/shared/src/test/scala/SqlPaging1Mapping.scala +++ b/modules/sql/shared/src/test/scala/SqlPaging1Mapping.scala @@ -3,13 +3,11 @@ package edu.gemini.grackle.sql.test -import cats.Order import cats.implicits._ import edu.gemini.grackle._, syntax._ -import Cursor.Env -import Query.{Binding, Count, Empty, Environment, Limit, Offset, OrderBy, OrderSelection, OrderSelections, Select} -import QueryCompiler.SelectElaborator +import Query.{Binding, Count, FilterOrderByOffsetLimit, OrderSelection, Select} +import QueryCompiler.{Elab, SelectElaborator} import Value.IntValue // Mapping illustrating paging in "counted" style: paged results can @@ -49,7 +47,7 @@ trait SqlPaging1Mapping[F[_]] extends SqlTestMapping[F] { type Country { code: String! name: String! - cities(offset: Int, limit: Int): PagedCity! + cities(offset: Int!, limit: Int!): PagedCity! } type PagedCity { offset: Int! @@ -81,8 +79,8 @@ trait SqlPaging1Mapping[F[_]] extends SqlTestMapping[F] { tpe = PagedCountryType, fieldMappings = List( - CursorField("offset", genValue("countryOffset"), Nil), - CursorField("limit", genValue("countryLimit"), Nil), + CursorField("offset", CountryPaging.genOffset, Nil), + CursorField("limit", CountryPaging.genLimit, Nil), SqlField("total", root.numCountries), SqlObject("items") ) @@ -102,8 +100,8 @@ trait SqlPaging1Mapping[F[_]] extends SqlTestMapping[F] { List( SqlField("code", country.code, key = true, hidden = true), SqlObject("items", Join(country.code, city.countrycode)), - CursorField("offset", genValue("cityOffset"), Nil), - CursorField("limit", genValue("cityLimit"), Nil), + CursorField("offset", CityPaging.genOffset, Nil), + CursorField("limit", CityPaging.genLimit, Nil), SqlField("total", country.numCities), ) ), @@ -121,47 +119,50 @@ trait SqlPaging1Mapping[F[_]] extends SqlTestMapping[F] { def genValue(key: String)(c: Cursor): Result[Int] = c.env[Int](key).toResultOrError(s"Missing key '$key'") - def transformChild[T: Order](query: Query, orderTerm: Term[T], off: Int, lim: Int): Result[Query] = - Query.mapFields(query) { - case Select("items", Nil, child) => - def order(query: Query): Query = - OrderBy(OrderSelections(List(OrderSelection(orderTerm))), query) + abstract class PagingConfig(key: String, countAttr: String, orderTerm: Term[String]) { + def setup(offset: Int, limit: Int): Elab[Unit] = + Elab.env(key -> new PagingInfo(offset, limit)) - def offset(query: Query): Query = - if (off < 1) query - else Offset(off, query) + def elabItems = Elab.envE[PagingInfo](key).flatMap(_.elabItems) + def elabTotal = Elab.envE[PagingInfo](key).flatMap(_.elabTotal) - def limit(query: Query): Query = - if (lim < 1) query - else Limit(lim, query) + def genOffset(c: Cursor): Result[Int] = info(c, _.offset) + def genLimit(c: Cursor): Result[Int] = info(c, _.limit) - Select("items", Nil, limit(offset(order(child)))).success + def info(c: Cursor, f: PagingInfo => Int): Result[Int] = + c.env[PagingInfo](key).map(f).toResultOrError(s"Missing key '$key'") - case other => other.success - } - - override val selectElaborator = new SelectElaborator(Map( - QueryType -> { - case Select("countries", List(Binding("offset", IntValue(off)), Binding("limit", IntValue(lim))), child) => { - transformChild[String](child, CountryType / "code", off, lim).map { child0 => - Select("countries", Nil, Environment(Env("countryOffset" -> off, "countryLimit" -> lim), child0)) - } - } - }, - PagedCountryType -> { - case Select("total", Nil, Empty) => - Count("total", Select("items", Nil, Select("code", Nil, Empty))).success - }, - CountryType -> { - case Select("cities", List(Binding("offset", IntValue(off)), Binding("limit", IntValue(lim))), child) => { - transformChild[String](child, CityType / "name", off, lim).map { child0 => - Select("cities", Nil, Environment(Env("cityOffset" -> off, "cityLimit" -> lim), child0)) + case class PagingInfo(offset: Int, limit: Int) { + def elabItems: Elab[Unit] = + Elab.transformChild { child => + FilterOrderByOffsetLimit(None, Some(List(OrderSelection(orderTerm))), Some(offset), Some(limit), child) } - } - }, - PagedCityType -> { - case Select("total", Nil, Empty) => - Count("total", Select("items", Nil, Select("id", Nil, Empty))).success + + def elabTotal: Elab[Unit] = + Elab.transformChild(_ => Count(Select("items", Select(countAttr)))) } - )) + } + + object CountryPaging extends PagingConfig("countryPaging", "code", CountryType / "code") + object CityPaging extends PagingConfig("cityPaging", "id", CityType / "name") + + override val selectElaborator = SelectElaborator { + case (QueryType, "countries", List(Binding("offset", IntValue(off)), Binding("limit", IntValue(lim)))) => + CountryPaging.setup(off, lim) + + case (PagedCountryType, "items", Nil) => + CountryPaging.elabItems + + case (PagedCountryType, "total", Nil) => + CountryPaging.elabTotal + + case (CountryType, "cities", List(Binding("offset", IntValue(off)), Binding("limit", IntValue(lim)))) => + CityPaging.setup(off, lim) + + case (PagedCityType, "items", Nil) => + CityPaging.elabItems + + case (PagedCityType, "total", Nil) => + CityPaging.elabTotal + } } diff --git a/modules/sql/shared/src/test/scala/SqlPaging1Suite.scala b/modules/sql/shared/src/test/scala/SqlPaging1Suite.scala index 39ea660e..99fdb053 100644 --- a/modules/sql/shared/src/test/scala/SqlPaging1Suite.scala +++ b/modules/sql/shared/src/test/scala/SqlPaging1Suite.scala @@ -4,7 +4,6 @@ package edu.gemini.grackle.sql.test import cats.effect.IO -import io.circe.Json import io.circe.literal._ import munit.CatsEffectSuite @@ -13,7 +12,7 @@ import edu.gemini.grackle._ import grackle.test.GraphQLResponseTests.assertWeaklyEqualIO trait SqlPaging1Suite extends CatsEffectSuite { - def mapping: QueryExecutor[IO, Json] + def mapping: Mapping[IO] test("paging (initial)") { val query = """ diff --git a/modules/sql/shared/src/test/scala/SqlPaging2Mapping.scala b/modules/sql/shared/src/test/scala/SqlPaging2Mapping.scala index e34b9c1c..766562ab 100644 --- a/modules/sql/shared/src/test/scala/SqlPaging2Mapping.scala +++ b/modules/sql/shared/src/test/scala/SqlPaging2Mapping.scala @@ -3,14 +3,12 @@ package edu.gemini.grackle.sql.test -import cats.Order import cats.implicits._ import edu.gemini.grackle._, syntax._ -import Cursor.Env -import Query.{Binding, Count, Empty, Environment, Group, Limit, Offset, OrderBy, OrderSelection, OrderSelections, Select} -import QueryCompiler.SelectElaborator -import Value.IntValue +import Query.{Binding, Count, FilterOrderByOffsetLimit, OrderSelection, Select} +import QueryCompiler.{Elab, SelectElaborator} +import Value._ // Mapping illustrating paging in "has more" style: paged results can // report whether there are more elements beyond the current sub list. @@ -77,7 +75,7 @@ trait SqlPaging2Mapping[F[_]] extends SqlTestMapping[F] { fieldMappings = List( SqlObject("items"), - CursorField("hasMore", genHasMore("countryLast", "numCountries"), List("numCountries")), + CursorField("hasMore", CountryPaging.genHasMore, List("numCountries")), SqlField("numCountries", root.numCountries, hidden = true) ) ), @@ -96,7 +94,7 @@ trait SqlPaging2Mapping[F[_]] extends SqlTestMapping[F] { List( SqlField("code", country.code, key = true, hidden = true), SqlObject("items", Join(country.code, city.countrycode)), - CursorField("hasMore", genHasMore("cityLast", "numCities"), List("numCities")), + CursorField("hasMore", CityPaging.genHasMore, List("numCities")), SqlField("numCities", country.numCities, hidden = true) ) ), @@ -111,59 +109,63 @@ trait SqlPaging2Mapping[F[_]] extends SqlTestMapping[F] { ) ) - def genHasMore(lastKey: String, numField: String)(c: Cursor): Result[Boolean] = - for { - last <- c.envR[Int](lastKey) - num <- c.fieldAs[Long](numField) - } yield num > last + abstract class PagingConfig(key: String, countField: String, countAttr: String, orderTerm: Term[String]) { + def setup(offset: Option[Int], limit: Option[Int]): Elab[Unit] = + Elab.env(key -> new PagingInfo(offset, limit)) - def transformChild[T: Order](query: Query, orderTerm: Term[T], off: Int, lim: Int): Result[Query] = - Query.mapFields(query) { - case Select("items", Nil, child) => - def order(query: Query): Query = - OrderBy(OrderSelections(List(OrderSelection(orderTerm))), query) + def elabItems = Elab.envE[PagingInfo](key).flatMap(_.elabItems) + def elabHasMore = Elab.envE[PagingInfo](key).flatMap(_.elabHasMore) - def offset(query: Query): Query = - if (off < 1) query - else Offset(off, query) + def genHasMore(c : Cursor): Result[Boolean] = + for { + info <- c.envR[PagingInfo](key) + c0 <- info.genHasMore(c) + } yield c0 - def limit(query: Query): Query = - if (lim < 1) query - else Limit(lim, query) + case class PagingInfo(offset: Option[Int], limit: Option[Int]) { + def elabItems: Elab[Unit] = + Elab.transformChild { child => + FilterOrderByOffsetLimit(None, Some(List(OrderSelection(orderTerm))), offset, limit, child) + } - Select("items", Nil, limit(offset(order(child)))).success + def elabHasMore: Elab[Unit] = + Elab.addAttribute(countField, Count(Select("items", Select(countAttr)))) - case other => other.success + def genHasMore(c: Cursor): Result[Boolean] = + for { + num <- c.fieldAs[Long](countField) + } yield num > offset.getOrElse(0)+limit.getOrElse(num.toInt) } + } - override val selectElaborator = new SelectElaborator(Map( - QueryType -> { - case Select("countries", List(Binding("offset", IntValue(off)), Binding("limit", IntValue(lim))), child) => { - transformChild[String](child, CountryType / "code", off, lim).map { child0 => - Select("countries", Nil, Environment(Env("countryLast" -> (off+lim)), child0)) - } - } - }, - PagedCountryType -> { - case s@Select("hasMore", Nil, Empty) => - Group(List( - s, - Count("numCountries", Select("items", Nil, Select("code", Nil, Empty))) - )).success - }, - CountryType -> { - case Select("cities", List(Binding("offset", IntValue(off)), Binding("limit", IntValue(lim))), child) => { - transformChild[String](child, CityType / "name", off, lim).map { child0 => - Select("cities", Nil, Environment(Env("cityLast" -> (off+lim)), child0)) - } - } - }, - PagedCityType -> { - case s@Select("hasMore", Nil, Empty) => - Group(List( - s, - Count("numCities", Select("items", Nil, Select("id", Nil, Empty))) - )).success + object CountryPaging extends PagingConfig("countryPaging", "numCountries", "code", CountryType / "code") + object CityPaging extends PagingConfig("cityPaging", "numCities", "name", CityType / "name") + + object OptIntValue { + def unapply(v: Value): Option[Option[Int]] = v match { + case IntValue(n) => Some(Some(n)) + case AbsentValue | NullValue => Some(None) + case _ => None } - )) + } + + override val selectElaborator = SelectElaborator { + case (QueryType, "countries", List(Binding("offset", OptIntValue(off)), Binding("limit", OptIntValue(lim)))) => + CountryPaging.setup(off, lim) + + case (PagedCountryType, "hasMore", Nil) => + CountryPaging.elabHasMore + + case (PagedCountryType, "items", Nil) => + CountryPaging.elabItems + + case (CountryType, "cities", List(Binding("offset", OptIntValue(off)), Binding("limit", OptIntValue(lim)))) => + CityPaging.setup(off, lim) + + case (PagedCityType, "hasMore", Nil) => + CityPaging.elabHasMore + + case (PagedCityType, "items", Nil) => + CityPaging.elabItems + } } diff --git a/modules/sql/shared/src/test/scala/SqlPaging2Suite.scala b/modules/sql/shared/src/test/scala/SqlPaging2Suite.scala index a572c5ef..d7a63513 100644 --- a/modules/sql/shared/src/test/scala/SqlPaging2Suite.scala +++ b/modules/sql/shared/src/test/scala/SqlPaging2Suite.scala @@ -4,7 +4,6 @@ package edu.gemini.grackle.sql.test import cats.effect.IO -import io.circe.Json import io.circe.literal._ import munit.CatsEffectSuite @@ -13,7 +12,7 @@ import edu.gemini.grackle._ import grackle.test.GraphQLResponseTests.assertWeaklyEqualIO trait SqlPaging2Suite extends CatsEffectSuite { - def mapping: QueryExecutor[IO, Json] + def mapping: Mapping[IO] test("paging (initial)") { val query = """ diff --git a/modules/sql/shared/src/test/scala/SqlPaging3Mapping.scala b/modules/sql/shared/src/test/scala/SqlPaging3Mapping.scala index 35cd5fb1..8becebc4 100644 --- a/modules/sql/shared/src/test/scala/SqlPaging3Mapping.scala +++ b/modules/sql/shared/src/test/scala/SqlPaging3Mapping.scala @@ -3,14 +3,13 @@ package edu.gemini.grackle.sql.test -import cats.Order import cats.implicits._ import edu.gemini.grackle._, syntax._ -import Cursor.{Env, ListTransformCursor} -import Query.{Binding, Count, Empty, Environment, Group, Limit, Offset, OrderBy, OrderSelection, OrderSelections, Select, TransformCursor} -import QueryCompiler.SelectElaborator -import Value.IntValue +import Cursor.ListTransformCursor +import Query.{Binding, Count, FilterOrderByOffsetLimit, OrderSelection, Select, TransformCursor} +import QueryCompiler.{Elab, SelectElaborator} +import Value._ // Mapping illustrating paging in "has more" style: paged results can report // whether there are more elements beyond the current sub list. @@ -84,7 +83,7 @@ trait SqlPaging3Mapping[F[_]] extends SqlTestMapping[F] { fieldMappings = List( SqlObject("items"), - CursorField("hasMore", genHasMore("country", "numCountries")), + CursorField("hasMore", CountryPaging.genHasMore), SqlField("numCountries", root.numCountries, hidden = true) ) ), @@ -103,7 +102,7 @@ trait SqlPaging3Mapping[F[_]] extends SqlTestMapping[F] { List( SqlField("code", country.code, key = true, hidden = true), SqlObject("items", Join(country.code, city.countrycode)), - CursorField("hasMore", genHasMore("city", "numCities")), + CursorField("hasMore", CityPaging.genHasMore), SqlField("numCities", country.numCities, hidden = true) ) ), @@ -118,100 +117,87 @@ trait SqlPaging3Mapping[F[_]] extends SqlTestMapping[F] { ) ) - def genHasMore(keyPrefix: String, countField: String)(c: Cursor): Result[Boolean] = { - val limitKey = keyPrefix+"Limit" - val lastKey = keyPrefix+"Last" - val aliasKey = keyPrefix+"Alias" - if(c.envContains(limitKey)) { + abstract class PagingConfig(key: String, countField: String, countAttr: String, orderTerm: Term[String]) { + def setup(offset: Option[Int], limit: Option[Int]): Elab[Unit] = for { - limit <- c.envR[Int](limitKey) - alias <- c.envR[Option[String]](aliasKey) - items <- c.field("items", alias) - size <- items.listSize - } yield limit < 0 || size > limit - } else if(c.envContains(lastKey)) { + hasHasMore <- Elab.hasField("hasMore") + hasItems <- Elab.hasField("items") + resultName <- Elab.fieldAlias("items") + _ <- Elab.env(key -> new PagingInfo(offset, limit, hasItems, resultName, hasHasMore)) + } yield () + + def elabItems = Elab.envE[PagingInfo](key).flatMap(_.elabItems) + def elabHasMore = Elab.envE[PagingInfo](key).flatMap(_.elabHasMore) + + def genHasMore(c : Cursor): Result[Boolean] = for { - last <- c.envR[Int](lastKey) - num <- c.fieldAs[Long](countField) - } yield num > last - } else - Result.internalError("Result has unexpected shape") + info <- c.envR[PagingInfo](key) + c0 <- info.genHasMore(c) + } yield c0 + + case class PagingInfo(offset: Option[Int], limit: Option[Int], hasItems: Boolean, itemsAlias: Option[String], hasHasMore: Boolean) { + def elabItems: Elab[Unit] = + Elab.transformChild { child => + val lim0 = if (hasHasMore) limit.map(_+1) else limit + val items = FilterOrderByOffsetLimit(None, Some(List(OrderSelection(orderTerm))), offset, lim0, child) + if (hasHasMore) TransformCursor(genItems, items) else items + } + + def genItems(c: Cursor): Result[Cursor] = { + for { + size <- c.listSize + elems <- c.asList(Seq) + } yield { + if(limit.map(size <= _).getOrElse(true)) c + else ListTransformCursor(c, size-1, elems.init) + } + } + + def elabHasMore: Elab[Unit] = + Elab.addAttribute(countField, Count(Select("items", Select(countAttr)))).whenA(!hasItems) + + def genHasMore(c: Cursor): Result[Boolean] = + if(hasItems) { + for { + items <- c.field("items", itemsAlias) + size <- items.listSize + } yield limit.map(size > _).getOrElse(false) + } else { + for { + num <- c.fieldAs[Long](countField) + } yield num > offset.getOrElse(0)+limit.getOrElse(num.toInt) + } + } } - def genItems(keyPrefix: String)(c: Cursor): Result[Cursor] = { - val limitKey = keyPrefix+"Limit" - for { - limit <- c.envR[Int](limitKey) - size <- c.listSize - elems <- c.asList(Seq) - } yield { - if(size <= limit) c - else ListTransformCursor(c, size-1, elems.init) + object CountryPaging extends PagingConfig("countryPaging", "numCountries", "code", CountryType / "code") + object CityPaging extends PagingConfig("cityPaging", "numCities", "name", CityType / "name") + + object OptIntValue { + def unapply(v: Value): Option[Option[Int]] = v match { + case IntValue(n) => Some(Some(n)) + case AbsentValue | NullValue => Some(None) + case _ => None } } - def transformChild[T: Order](query: Query, keyPrefix: String, orderTerm: Term[T], off: Int, lim: Int, hasHasMore: Boolean): Result[Query] = - Query.mapFields(query) { - case Select("items", Nil, child) => - def order(query: Query): Query = - OrderBy(OrderSelections(List(OrderSelection(orderTerm))), query) + override val selectElaborator = SelectElaborator { + case (QueryType, "countries", List(Binding("offset", OptIntValue(off)), Binding("limit", OptIntValue(lim)))) => + CountryPaging.setup(off, lim) - def offset(query: Query): Query = - if (off < 1) query - else Offset(off, query) + case (PagedCountryType, "items", Nil) => + CountryPaging.elabItems - def limit(query: Query): Query = - if (lim < 1) query - else Limit(if (hasHasMore) lim+1 else lim, query) + case (PagedCountryType, "hasMore", Nil) => + CountryPaging.elabHasMore - if(hasHasMore) - Select("items", Nil, TransformCursor(genItems(keyPrefix), limit(offset(order(child))))).success - else - Select("items", Nil, limit(offset(order(child)))).success + case (CountryType, "cities", List(Binding("offset", OptIntValue(off)), Binding("limit", OptIntValue(lim)))) => + CityPaging.setup(off, lim) - case other => other.success - } + case (PagedCityType, "items", Nil) => + CityPaging.elabItems - override val selectElaborator = new SelectElaborator(Map( - QueryType -> { - case Select("countries", List(Binding("offset", IntValue(off)), Binding("limit", IntValue(lim))), child) => { - val hasItems = Query.hasField(child, "items") - val hasHasMore = Query.hasField(child, "hasMore") - if(hasItems) { - val itemAlias = Query.fieldAlias(child, "items") - transformChild[String](child, "country", CountryType / "code", off, lim, hasHasMore).map { child0 => - Select("countries", Nil, Environment(Env[Any]("countryLimit" -> lim, "countryAlias" -> itemAlias), child0)) - } - } else - Select("countries", Nil, - Environment(Env("countryLast" -> (off+lim)), - Group(List( - Count("numCountries", Select("items", Nil, Select("code", Nil, Empty))), - child - )) - ) - ).success - } - }, - CountryType -> { - case Select("cities", List(Binding("offset", IntValue(off)), Binding("limit", IntValue(lim))), child) => { - val hasItems = Query.hasField(child, "items") - val hasHasMore = Query.hasField(child, "hasMore") - if(hasItems) { - val itemAlias = Query.fieldAlias(child, "items") - transformChild[String](child, "city", CityType / "name", off, lim, hasHasMore).map { child0 => - Select("cities", Nil, Environment(Env[Any]("cityLimit" -> lim, "cityAlias" -> itemAlias), child0)) - } - } else - Select("cities", Nil, - Environment(Env("cityLast" -> (off+lim)), - Group(List( - Count("numCities", Select("items", Nil, Select("id", Nil, Empty))), - child - )) - ) - ).success - } - } - )) + case (PagedCityType, "hasMore", Nil) => + CityPaging.elabHasMore + } } diff --git a/modules/sql/shared/src/test/scala/SqlPaging3Suite.scala b/modules/sql/shared/src/test/scala/SqlPaging3Suite.scala index 208391f6..3b2e9bfc 100644 --- a/modules/sql/shared/src/test/scala/SqlPaging3Suite.scala +++ b/modules/sql/shared/src/test/scala/SqlPaging3Suite.scala @@ -4,7 +4,6 @@ package edu.gemini.grackle.sql.test import cats.effect.IO -import io.circe.Json import io.circe.literal._ import munit.CatsEffectSuite @@ -13,7 +12,7 @@ import edu.gemini.grackle._ import grackle.test.GraphQLResponseTests.assertWeaklyEqualIO trait SqlPaging3Suite extends CatsEffectSuite { - def mapping: QueryExecutor[IO, Json] + def mapping: Mapping[IO] test("paging (initial)") { val query = """ diff --git a/modules/sql/shared/src/test/scala/SqlProjectionMapping.scala b/modules/sql/shared/src/test/scala/SqlProjectionMapping.scala index c5bce5e0..1bd4cd99 100644 --- a/modules/sql/shared/src/test/scala/SqlProjectionMapping.scala +++ b/modules/sql/shared/src/test/scala/SqlProjectionMapping.scala @@ -7,8 +7,8 @@ import cats.implicits._ import edu.gemini.grackle._, syntax._ import Predicate.{Const, Eql} -import Query.{Binding, Filter, Select} -import QueryCompiler.SelectElaborator +import Query.{Binding, Filter} +import QueryCompiler._ import Value.{BooleanValue, ObjectValue} trait SqlProjectionMapping[F[_]] extends SqlTestMapping[F] { @@ -126,50 +126,20 @@ trait SqlProjectionMapping[F[_]] extends SqlTestMapping[F] { } } - override val selectElaborator: SelectElaborator = new SelectElaborator(Map( - QueryType -> { - case Select("level0", List(Binding("filter", filter)), child) => - val f = filter match { - case Level0FilterValue(f) => Filter(f, child) - case _ => child - } - Select("level0", Nil, f).success - - case Select("level1", List(Binding("filter", filter)), child) => - val f = filter match { - case Level1FilterValue(f) => Filter(f, child) - case _ => child - } - Select("level1", Nil, f).success - - case Select("level2", List(Binding("filter", filter)), child) => - val f = filter match { - case Level2FilterValue(f) => Filter(f, child) - case _ => child - } - Select("level2", Nil, f).success - - case other => other.success - }, - Level0Type -> { - case Select("level1", List(Binding("filter", filter)), child) => - val f = filter match { - case Level1FilterValue(f) => Filter(f, child) - case _ => child - } - Select("level1", Nil, f).success - - case other => other.success - }, - Level1Type -> { - case Select("level2", List(Binding("filter", filter)), child) => - val f = filter match { - case Level2FilterValue(f) => Filter(f, child) - case _ => child - } - Select("level2", Nil, f).success - - case other => other.success - } - )) + override val selectElaborator = SelectElaborator { + case (QueryType, "level0", List(Binding("filter", Level0FilterValue(f)))) => + Elab.transformChild(child => Filter(f, child)) + + case (QueryType, "level1", List(Binding("filter", Level1FilterValue(f)))) => + Elab.transformChild(child => Filter(f, child)) + + case (QueryType, "level2", List(Binding("filter", Level2FilterValue(f)))) => + Elab.transformChild(child => Filter(f, child)) + + case (Level0Type, "level1", List(Binding("filter", Level1FilterValue(f)))) => + Elab.transformChild(child => Filter(f, child)) + + case (Level1Type, "level2", List(Binding("filter", Level2FilterValue(f)))) => + Elab.transformChild(child => Filter(f, child)) + } } diff --git a/modules/sql/shared/src/test/scala/SqlProjectionSuite.scala b/modules/sql/shared/src/test/scala/SqlProjectionSuite.scala index 0432ebe9..bfb76038 100644 --- a/modules/sql/shared/src/test/scala/SqlProjectionSuite.scala +++ b/modules/sql/shared/src/test/scala/SqlProjectionSuite.scala @@ -4,16 +4,15 @@ package edu.gemini.grackle.sql.test import cats.effect.IO -import io.circe.Json import io.circe.literal._ import munit.CatsEffectSuite -import edu.gemini.grackle.QueryExecutor +import edu.gemini.grackle._ import grackle.test.GraphQLResponseTests.assertWeaklyEqualIO trait SqlProjectionSuite extends CatsEffectSuite { - def mapping: QueryExecutor[IO, Json] + def mapping: Mapping[IO] test("base query") { val query = """ diff --git a/modules/sql/shared/src/test/scala/SqlRecursiveInterfacesSuite.scala b/modules/sql/shared/src/test/scala/SqlRecursiveInterfacesSuite.scala index 42ddceb9..15133bd7 100644 --- a/modules/sql/shared/src/test/scala/SqlRecursiveInterfacesSuite.scala +++ b/modules/sql/shared/src/test/scala/SqlRecursiveInterfacesSuite.scala @@ -4,7 +4,6 @@ package edu.gemini.grackle.sql.test import cats.effect.IO -import io.circe.Json import io.circe.literal._ import munit.CatsEffectSuite @@ -13,7 +12,7 @@ import edu.gemini.grackle._ import grackle.test.GraphQLResponseTests.assertWeaklyEqualIO trait SqlRecursiveInterfacesSuite extends CatsEffectSuite { - def mapping: QueryExecutor[IO, Json] + def mapping: Mapping[IO] test("specialized query on both sides") { val query = """ diff --git a/modules/sql/shared/src/test/scala/SqlSiblingListsMapping.scala b/modules/sql/shared/src/test/scala/SqlSiblingListsMapping.scala index 5ab038e9..207eef8c 100644 --- a/modules/sql/shared/src/test/scala/SqlSiblingListsMapping.scala +++ b/modules/sql/shared/src/test/scala/SqlSiblingListsMapping.scala @@ -4,11 +4,12 @@ package edu.gemini.grackle.sql.test import cats.implicits._ -import edu.gemini.grackle.Predicate.{Const, Eql} -import edu.gemini.grackle.Query.{Binding, Filter, Select, Unique} -import edu.gemini.grackle.syntax._ -import edu.gemini.grackle.QueryCompiler.SelectElaborator -import edu.gemini.grackle.Value.StringValue +import edu.gemini.grackle._ +import Predicate.{Const, Eql} +import Query.{Binding, Filter, Unique} +import QueryCompiler._ +import Value.StringValue +import syntax._ trait SqlSiblingListsData[F[_]] extends SqlTestMapping[F] { @@ -104,14 +105,8 @@ trait SqlSiblingListsData[F[_]] extends SqlTestMapping[F] { ) ) - override val selectElaborator: SelectElaborator = new SelectElaborator( - Map( - QueryType -> { - case Select("a", List(Binding("id", StringValue(id))), child) => - Select("a", Nil, Unique(Filter(Eql(AType / "id", Const(id)), child))).success - case other => other.success - } - ) - ) - + override val selectElaborator = SelectElaborator { + case (QueryType, "a", List(Binding("id", StringValue(id)))) => + Elab.transformChild(child => Unique(Filter(Eql(AType / "id", Const(id)), child))) + } } diff --git a/modules/sql/shared/src/test/scala/SqlSiblingListsSuite.scala b/modules/sql/shared/src/test/scala/SqlSiblingListsSuite.scala index 8895b1bc..7b7d7fa8 100644 --- a/modules/sql/shared/src/test/scala/SqlSiblingListsSuite.scala +++ b/modules/sql/shared/src/test/scala/SqlSiblingListsSuite.scala @@ -4,7 +4,6 @@ package edu.gemini.grackle.sql.test import cats.effect.IO -import io.circe.Json import io.circe.literal._ import munit.CatsEffectSuite @@ -13,7 +12,7 @@ import edu.gemini.grackle._ import grackle.test.GraphQLResponseTests.assertWeaklyEqualIO trait SqlSiblingListsSuite extends CatsEffectSuite { - def mapping: QueryExecutor[IO, Json] + def mapping: Mapping[IO] test("base query") { val query = """ diff --git a/modules/sql/shared/src/test/scala/SqlTreeMapping.scala b/modules/sql/shared/src/test/scala/SqlTreeMapping.scala index 687d2849..31b43138 100644 --- a/modules/sql/shared/src/test/scala/SqlTreeMapping.scala +++ b/modules/sql/shared/src/test/scala/SqlTreeMapping.scala @@ -53,12 +53,8 @@ trait SqlTreeMapping[F[_]] extends SqlTestMapping[F] { ) ) - override val selectElaborator: SelectElaborator = new SelectElaborator(Map( - QueryType -> { - case Select("bintree", List(Binding("id", IntValue(id))), child) => - Select("bintree", Nil, Unique(Filter(Eql(BinTreeType / "id", Const(id)), child))).success - - case other => other.success - } - )) + override val selectElaborator = SelectElaborator { + case (QueryType, "bintree", List(Binding("id", IntValue(id)))) => + Elab.transformChild(child => Unique(Filter(Eql(BinTreeType / "id", Const(id)), child))) + } } diff --git a/modules/sql/shared/src/test/scala/SqlTreeSuite.scala b/modules/sql/shared/src/test/scala/SqlTreeSuite.scala index 4ae798e1..23a21767 100644 --- a/modules/sql/shared/src/test/scala/SqlTreeSuite.scala +++ b/modules/sql/shared/src/test/scala/SqlTreeSuite.scala @@ -4,7 +4,6 @@ package edu.gemini.grackle.sql.test import cats.effect.IO -import io.circe.Json import io.circe.literal._ import munit.CatsEffectSuite @@ -13,7 +12,7 @@ import edu.gemini.grackle._ import grackle.test.GraphQLResponseTests.assertWeaklyEqualIO trait SqlTreeSuite extends CatsEffectSuite { - def mapping: QueryExecutor[IO, Json] + def mapping: Mapping[IO] test("root query") { val query = """ diff --git a/modules/sql/shared/src/test/scala/SqlUnionSuite.scala b/modules/sql/shared/src/test/scala/SqlUnionSuite.scala index 4813a0de..9cc7a710 100644 --- a/modules/sql/shared/src/test/scala/SqlUnionSuite.scala +++ b/modules/sql/shared/src/test/scala/SqlUnionSuite.scala @@ -4,7 +4,6 @@ package edu.gemini.grackle.sql.test import cats.effect.IO -import io.circe.Json import io.circe.literal._ import munit.CatsEffectSuite @@ -13,7 +12,7 @@ import edu.gemini.grackle._ import grackle.test.GraphQLResponseTests.assertWeaklyEqualIO trait SqlUnionSuite extends CatsEffectSuite { - def mapping: QueryExecutor[IO, Json] + def mapping: Mapping[IO] test("simple union query") { val query = """ diff --git a/modules/sql/shared/src/test/scala/SqlWorldCompilerSuite.scala b/modules/sql/shared/src/test/scala/SqlWorldCompilerSuite.scala index d6062eb3..b9ddfec8 100644 --- a/modules/sql/shared/src/test/scala/SqlWorldCompilerSuite.scala +++ b/modules/sql/shared/src/test/scala/SqlWorldCompilerSuite.scala @@ -9,7 +9,7 @@ import io.circe.literal._ import munit.CatsEffectSuite import edu.gemini.grackle._ -import Predicate._ +import Predicate._, Query._ import sql.{Like, SqlStatsMonitor} import grackle.test.GraphQLResponseTests.assertWeaklyEqual @@ -62,7 +62,7 @@ trait SqlWorldCompilerSuite extends CatsEffectSuite { assertEquals(stats, List( SqlStatsMonitor.SqlStats( - Query.Select("country", Nil, Query.Unique(Query.Filter(Eql(schema.ref("Country") / "code",Const("GBR")),Query.Select("name",List(),Query.Empty)))), + Select("country", Unique(Filter(Eql(schema.ref("Country") / "code",Const("GBR")), Select("name")))), simpleRestrictedQuerySql, List("GBR"), 1, @@ -117,7 +117,7 @@ trait SqlWorldCompilerSuite extends CatsEffectSuite { assertEquals(stats, List( SqlStatsMonitor.SqlStats( - Query.Select("cities", Nil, Query.Filter(Like(schema.ref("City") / "name","Linh%",true),Query.Select("name",List(),Query.Empty))), + Select("cities", Filter(Like(schema.ref("City") / "name","Linh%",true), Select("name"))), simpleFilteredQuerySql, List("Linh%"), 3, diff --git a/modules/sql/shared/src/test/scala/SqlWorldMapping.scala b/modules/sql/shared/src/test/scala/SqlWorldMapping.scala index 35008c03..d0793558 100644 --- a/modules/sql/shared/src/test/scala/SqlWorldMapping.scala +++ b/modules/sql/shared/src/test/scala/SqlWorldMapping.scala @@ -175,77 +175,72 @@ trait SqlWorldMapping[F[_]] extends SqlTestMapping[F] { } } - override val selectElaborator = new SelectElaborator(Map( + override val selectElaborator = SelectElaborator { + case (QueryType, "country", List(Binding("code", StringValue(code)))) => + Elab.transformChild(child => Unique(Filter(Eql(CountryType / "code", Const(code)), child))) + + case (QueryType, "city", List(Binding("id", IntValue(id)))) => + Elab.transformChild(child => Unique(Filter(Eql(CityType / "id", Const(id)), child))) + + case (QueryType, "countries", List(Binding("limit", IntValue(num)), Binding("offset", IntValue(off)), Binding("minPopulation", IntValue(min)), Binding("byPopulation", BooleanValue(byPop)))) => + def limit(query: Query): Query = + if (num < 1) query + else Limit(num, query) + + def offset(query: Query): Query = + if (off < 1) query + else Offset(off, query) + + def order(query: Query): Query = { + if (byPop) + OrderBy(OrderSelections(List(OrderSelection[Int](CountryType / "population"))), query) + else if (num > 0 || off > 0) + OrderBy(OrderSelections(List(OrderSelection[String](CountryType / "code"))), query) + else query + } - QueryType -> { + def filter(query: Query): Query = + if (min == 0) query + else Filter(GtEql(CountryType / "population", Const(min)), query) - case Select("country", List(Binding("code", StringValue(code))), child) => - Select("country", Nil, Unique(Filter(Eql(CountryType / "code", Const(code)), child))).success + Elab.transformChild(child => limit(offset(order(filter(child))))) - case Select("city", List(Binding("id", IntValue(id))), child) => - Select("city", Nil, Unique(Filter(Eql(CityType / "id", Const(id)), child))).success + case (QueryType, "cities", List(Binding("namePattern", StringValue(namePattern)))) => + if (namePattern == "%") + Elab.unit + else + Elab.transformChild(child => Filter(Like(CityType / "name", namePattern, true), child)) - case Select("countries", List(Binding("limit", IntValue(num)), Binding("offset", IntValue(off)), Binding("minPopulation", IntValue(min)), Binding("byPopulation", BooleanValue(byPop))), child) => - def limit(query: Query): Query = - if (num < 1) query - else Limit(num, query) + case (QueryType, "language", List(Binding("language", StringValue(language)))) => + Elab.transformChild(child => Unique(Filter(Eql(LanguageType / "language", Const(language)), child))) - def offset(query: Query): Query = - if (off < 1) query - else Offset(off, query) + case (QueryType, "languages", List(Binding("languages", StringListValue(languages)))) => + Elab.transformChild(child => Filter(In(CityType / "language", languages), child)) - def order(query: Query): Query = { - if (byPop) - OrderBy(OrderSelections(List(OrderSelection[Int](CountryType / "population"))), query) - else if (num > 0 || off > 0) - OrderBy(OrderSelections(List(OrderSelection[String](CountryType / "code"))), query) - else query - } + case (QueryType, "search", List(Binding("minPopulation", IntValue(min)), Binding("indepSince", IntValue(year)))) => + Elab.transformChild(child => + Filter( + And( + Not(Lt(CountryType / "population", Const(min))), + Not(Lt(CountryType / "indepyear", Const(Option(year)))) + ), + child + ) + ) + + case (QueryType, "search2", List(Binding("indep", BooleanValue(indep)), Binding("limit", IntValue(num)))) => + Elab.transformChild(child => Limit(num, Filter(IsNull[Int](CountryType / "indepyear", isNull = !indep), child))) - def filter(query: Query): Query = - if (min == 0) query - else Filter(GtEql(CountryType / "population", Const(min)), query) - - Select("countries", Nil, limit(offset(order(filter(child))))).success - - case Select("cities", List(Binding("namePattern", StringValue(namePattern))), child) => - if (namePattern == "%") - Select("cities", Nil, child).success - else - Select("cities", Nil, Filter(Like(CityType / "name", namePattern, true), child)).success - - case Select("language", List(Binding("language", StringValue(language))), child) => - Select("language", Nil, Unique(Filter(Eql(LanguageType / "language", Const(language)), child))).success - - case Select("languages", List(Binding("languages", StringListValue(languages))), child) => - Select("languages", Nil, Filter(In(CityType / "language", languages), child)).success - - case Select("search", List(Binding("minPopulation", IntValue(min)), Binding("indepSince", IntValue(year))), child) => - Select("search", Nil, - Filter( - And( - Not(Lt(CountryType / "population", Const(min))), - Not(Lt(CountryType / "indepyear", Const(Option(year)))) - ), - child - ) - ).success - - case Select("search2", List(Binding("indep", BooleanValue(indep)), Binding("limit", IntValue(num))), child) => - Select("search2", Nil, Limit(num, Filter(IsNull[Int](CountryType / "indepyear", isNull = !indep), child))).success - - case Select("numCountries", Nil, Empty) => - Count("numCountries", Select("countries", Nil, Select("code2", Nil, Empty))).success - }, - CountryType -> { - case Select("numCities", List(Binding("namePattern", AbsentValue)), Empty) => - Count("numCities", Select("cities", Nil, Select("name", Nil, Empty))).success - - case Select("numCities", List(Binding("namePattern", StringValue(namePattern))), Empty) => - Count("numCities", Select("cities", Nil, Filter(Like(CityType / "name", namePattern, true), Select("name", Nil, Empty)))).success - - case Select("city", List(Binding("id", IntValue(id))), child) => - Select("city", Nil, Unique(Filter(Eql(CityType / "id", Const(id)), child))).success - } - )) + case (QueryType, "numCountries", Nil) => + Elab.transformChild(_ => Count(Select("countries", Select("code2")))) + + case (CountryType, "numCities", List(Binding("namePattern", AbsentValue))) => + Elab.transformChild(_ => Count(Select("cities", Select("name")))) + + case (CountryType, "numCities", List(Binding("namePattern", StringValue(namePattern)))) => + Elab.transformChild(_ => Count(Select("cities", Filter(Like(CityType / "name", namePattern, true), Select("name"))))) + + case (CountryType, "city", List(Binding("id", IntValue(id)))) => + Elab.transformChild(child => Unique(Filter(Eql(CityType / "id", Const(id)), child))) + } } diff --git a/modules/sql/shared/src/test/scala/SqlWorldSuite.scala b/modules/sql/shared/src/test/scala/SqlWorldSuite.scala index f6afb96d..009fd7e9 100644 --- a/modules/sql/shared/src/test/scala/SqlWorldSuite.scala +++ b/modules/sql/shared/src/test/scala/SqlWorldSuite.scala @@ -5,7 +5,6 @@ package edu.gemini.grackle.sql.test import cats.effect.IO import cats.implicits._ -import io.circe.Json import io.circe.literal._ import munit.CatsEffectSuite @@ -15,7 +14,7 @@ import grackle.test.GraphQLResponseTests.{assertNoErrorsIO, assertWeaklyEqualIO} trait SqlWorldSuite extends CatsEffectSuite { - def mapping: QueryExecutor[IO, Json] + def mapping: Mapping[IO] test("simple query") { val query = """ @@ -1585,7 +1584,7 @@ trait SqlWorldSuite extends CatsEffectSuite { } """ - val expected = if (isJS) + val expected = if (isJS) json""" { "data" : { diff --git a/profile/src/main/scala/Bench.scala b/profile/src/main/scala/Bench.scala index e50279c7..a750397c 100644 --- a/profile/src/main/scala/Bench.scala +++ b/profile/src/main/scala/Bench.scala @@ -167,70 +167,65 @@ trait WorldMapping[F[_]] extends WorldPostgresSchema[F] { ) ) - override val selectElaborator = new SelectElaborator(Map( - - QueryType -> { - - case Select("country", List(Binding("code", StringValue(code))), child) => - Select("country", Nil, Unique(Filter(Eql(CountryType / "code", Const(code)), child))).success - - case Select("city", List(Binding("id", IntValue(id))), child) => - Select("city", Nil, Unique(Filter(Eql(CityType / "id", Const(id)), child))).success + override val selectElaborator = SelectElaborator { + case (QueryType, "country", List(Binding("code", StringValue(code)))) => + Elab.transformChild(child => Unique(Filter(Eql(CountryType / "code", Const(code)), child))) + + case (QueryType, "city", List(Binding("id", IntValue(id)))) => + Elab.transformChild(child => Unique(Filter(Eql(CityType / "id", Const(id)), child))) + + case (QueryType, "countries", List(Binding("limit", IntValue(num)), Binding("offset", IntValue(off)), Binding("minPopulation", IntValue(min)), Binding("byPopulation", BooleanValue(byPop)))) => + def limit(query: Query): Query = + if (num < 1) query + else Limit(num, query) + + def offset(query: Query): Query = + if (off < 1) query + else Offset(off, query) + + def order(query: Query): Query = { + if (byPop) + OrderBy(OrderSelections(List(OrderSelection[Int](CountryType / "population"))), query) + else if (num > 0 || off > 0) + OrderBy(OrderSelections(List(OrderSelection[String](CountryType / "code"))), query) + else query + } - case Select("countries", List(Binding("limit", IntValue(num)), Binding("offset", IntValue(off)), Binding("minPopulation", IntValue(min)), Binding("byPopulation", BooleanValue(byPop))), child) => - def limit(query: Query): Query = - if (num < 1) query - else Limit(num, query) + def filter(query: Query): Query = + if (min == 0) query + else Filter(GtEql(CountryType / "population", Const(min)), query) + + Elab.transformChild(child => limit(offset(order(filter(child))))) + + case (QueryType, "cities", List(Binding("namePattern", StringValue(namePattern)))) => + if (namePattern == "%") + Elab.unit + else + Elab.transformChild(child => Filter(Like(CityType / "name", namePattern, true), child)) + + case (QueryType, "language", List(Binding("language", StringValue(language)))) => + Elab.transformChild(child => Unique(Filter(Eql(LanguageType / "language", Const(language)), child))) + + case (QueryType, "search", List(Binding("minPopulation", IntValue(min)), Binding("indepSince", IntValue(year)))) => + Elab.transformChild(child => + Filter( + And( + Not(Lt(CountryType / "population", Const(min))), + Not(Lt(CountryType / "indepyear", Const(Option(year)))) + ), + child + ) + ) - def offset(query: Query): Query = - if (off < 1) query - else Offset(off, query) + case (QueryType, "search2", List(Binding("indep", BooleanValue(indep)), Binding("limit", IntValue(num)))) => + Elab.transformChild(child => Limit(num, Filter(IsNull[Int](CountryType / "indepyear", isNull = !indep), child))) - def order(query: Query): Query = { - if (byPop) - OrderBy(OrderSelections(List(OrderSelection[Int](CountryType / "population"))), query) - else if (num > 0 || off > 0) - OrderBy(OrderSelections(List(OrderSelection[String](CountryType / "code"))), query) - else query - } + case (CountryType, "numCities", List(Binding("namePattern", AbsentValue))) => + Elab.transformChild(_ => Count(Select("cities", Select("name")))) - def filter(query: Query): Query = - if (min == 0) query - else Filter(GtEql(CountryType / "population", Const(min)), query) - - Select("countries", Nil, limit(offset(order(filter(child))))).success - - case Select("cities", List(Binding("namePattern", StringValue(namePattern))), child) => - if (namePattern == "%") - Select("cities", Nil, child).success - else - Select("cities", Nil, Filter(Like(CityType / "name", namePattern, true), child)).success - - case Select("language", List(Binding("language", StringValue(language))), child) => - Select("language", Nil, Unique(Filter(Eql(LanguageType / "language", Const(language)), child))).success - - case Select("search", List(Binding("minPopulation", IntValue(min)), Binding("indepSince", IntValue(year))), child) => - Select("search", Nil, - Filter( - And( - Not(Lt(CountryType / "population", Const(min))), - Not(Lt(CountryType / "indepyear", Const(Option(year)))) - ), - child - ) - ).success - - case Select("search2", List(Binding("indep", BooleanValue(indep)), Binding("limit", IntValue(num))), child) => - Select("search2", Nil, Limit(num, Filter(IsNull[Int](CountryType / "indepyear", isNull = !indep), child))).success - }, - CountryType -> { - case Select("numCities", List(Binding("namePattern", AbsentValue)), Empty) => - Count("numCities", Select("cities", Nil, Select("name", Nil, Empty))).success - - case Select("numCities", List(Binding("namePattern", StringValue(namePattern))), Empty) => - Count("numCities", Select("cities", Nil, Filter(Like(CityType / "name", namePattern, true), Select("name", Nil, Empty)))).success - } - )) + case (CountryType, "numCities", List(Binding("namePattern", StringValue(namePattern)))) => + Elab.transformChild(_ => Count(Select("cities", Filter(Like(CityType / "name", namePattern, true), Select("name"))))) + } } object WorldMapping extends DoobieMappingCompanion { From c5aefa6f4c510887c04d75efe3f1f15d015c1709 Mon Sep 17 00:00:00 2001 From: Miles Sabin Date: Thu, 5 Oct 2023 15:44:30 +0100 Subject: [PATCH 2/2] Fix handling of grouped/ungrouped effects --- modules/core/src/main/scala/compiler.scala | 2 +- modules/core/src/main/scala/query.scala | 7 ++- .../src/main/scala/queryinterpreter.scala | 34 ++++------- .../shared/src/main/scala/SqlMapping.scala | 30 +++++----- .../test/scala/SqlNestedEffectsMapping.scala | 58 +++++++++++++------ .../test/scala/SqlNestedEffectsSuite.scala | 34 ++++++++++- 6 files changed, 105 insertions(+), 60 deletions(-) diff --git a/modules/core/src/main/scala/compiler.scala b/modules/core/src/main/scala/compiler.scala index d6ef3d3b..b234f56d 100644 --- a/modules/core/src/main/scala/compiler.scala +++ b/modules/core/src/main/scala/compiler.scala @@ -902,7 +902,7 @@ object QueryCompiler { } yield emapping.get((ref, fieldName)) match { case Some(handler) => - Effect(handler, s.copy(child = ec)) + Select(fieldName, resultName, Effect(handler, s.copy(child = ec))) case None => s.copy(child = ec) } diff --git a/modules/core/src/main/scala/query.scala b/modules/core/src/main/scala/query.scala index 7c4762e2..2819bc7c 100644 --- a/modules/core/src/main/scala/query.scala +++ b/modules/core/src/main/scala/query.scala @@ -93,7 +93,7 @@ object Query { } trait EffectHandler[F[_]] { - def runEffects(queries: List[(Query, Cursor)]): F[Result[List[(Query, Cursor)]]] + def runEffects(queries: List[(Query, Cursor)]): F[Result[List[Cursor]]] } /** Evaluates an introspection query relative to `schema` */ @@ -261,6 +261,11 @@ object Query { loop(q) } + def childContext(c: Context, query: Query): Result[Context] = + rootName(query).toResultOrError(s"Query has the wrong shape").flatMap { + case (fieldName, resultName) => c.forField(fieldName, resultName) + } + /** * Renames the root of `target` to match `source` if possible. */ diff --git a/modules/core/src/main/scala/queryinterpreter.scala b/modules/core/src/main/scala/queryinterpreter.scala index 1f15599a..5831523c 100644 --- a/modules/core/src/main/scala/queryinterpreter.scala +++ b/modules/core/src/main/scala/queryinterpreter.scala @@ -228,6 +228,11 @@ class QueryInterpreter[F[_]](mapping: Mapping[F]) { else size(c0) } yield List((sel.resultName, ProtoJson.fromJson(Json.fromInt(count)))) + case (sel@Select(_, _, Effect(handler, cont)), _) => + for { + value <- ProtoJson.effect(mapping, handler.asInstanceOf[EffectHandler[F]], cont, cursor).success + } yield List((sel.resultName, value)) + case (sel@Select(fieldName, resultName, child), _) => val fieldTpe = tpe.field(fieldName).getOrElse(ScalarType.AttributeType) for { @@ -241,12 +246,6 @@ class QueryInterpreter[F[_]](mapping: Mapping[F]) { value <- runValue(c, tpe, cursor) } yield List((componentName, ProtoJson.select(value, componentName))) - case (e@Effect(_, cont), _) => - for { - effectName <- resultName(cont).toResultOrError("Effect continuation has unexpected shape") - value <- runValue(e, tpe, cursor) - } yield List((effectName, value)) - case (Group(siblings), _) => siblings.flatTraverse(query => runFields(query, tpe, cursor)) @@ -410,9 +409,6 @@ class QueryInterpreter[F[_]](mapping: Mapping[F]) { } yield ProtoJson.component(mapping, renamedCont, cursor) } - case (Effect(handler, cont), _) => - ProtoJson.effect(mapping, handler.asInstanceOf[EffectHandler[F]], cont, cursor).success - case (Unique(child), _) => cursor.preunique.flatMap(c => runList(child, tpe.nonNull, c, true, tpe.isNullable) @@ -629,19 +625,8 @@ object QueryInterpreter { case p: Json => p case d: DeferredJson => subst(d) case ProtoObject(fields) => - val newFields: Seq[(String, Json)] = - fields.flatMap { case (label, pvalue) => - val value = loop(pvalue) - if (isDeferred(pvalue) && value.isObject) { - value.asObject.get.toList match { - case List((_, value)) => List((label, value)) - case other => other - } - } - else List((label, value)) - } - Json.fromFields(newFields) - + val fields0 = fields.map { case (label, pvalue) => (label, loop(pvalue)) } + Json.fromFields(fields0) case ProtoArray(elems) => val elems0 = elems.map(loop) Json.fromValues(elems0) @@ -667,8 +652,9 @@ object QueryInterpreter { ResultT(mapping.combineAndRun(queries)) case Some(handler) => for { - conts <- ResultT(handler.runEffects(queries)) - res <- ResultT(combineResults(conts.map { case (query, cursor) => mapping.interpreter.runValue(query, cursor.tpe, cursor) }).pure[F]) + cs <- ResultT(handler.runEffects(queries)) + conts <- ResultT(queries.traverse { case (q, _) => Query.extractChild(q).toResultOrError("Continuation query has the wrong shape") }.pure[F]) + res <- ResultT(combineResults((conts, cs).parMapN { case (query, cursor) => mapping.interpreter.runValue(query, cursor.tpe, cursor) }).pure[F]) } yield res } next <- ResultT(completeAll[F](pnext)) diff --git a/modules/sql/shared/src/main/scala/SqlMapping.scala b/modules/sql/shared/src/main/scala/SqlMapping.scala index 51002ea5..fd2381de 100644 --- a/modules/sql/shared/src/main/scala/SqlMapping.scala +++ b/modules/sql/shared/src/main/scala/SqlMapping.scala @@ -2916,6 +2916,21 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self case _: Count => Result.internalError("Count node must be a child of a Select node") + case Select(fieldName, _, Effect(_, _)) => + columnsForLeaf(context, fieldName).flatMap { + case Nil => EmptySqlQuery.success + case cols => + val constraintCols = if(exposeJoins) parentConstraints.lastOption.getOrElse(Nil).map(_._2) else Nil + val extraCols = keyColumnsForType(context) ++ constraintCols + for { + parentTable <- parentTableForType(context) + extraJoins <- parentConstraintsToSqlJoins(parentTable, parentConstraints) + } yield + SqlSelect(context, Nil, parentTable, (cols ++ extraCols).distinct, extraJoins, Nil, Nil, None, None, Nil, true, false) + } + + case Effect(_, _) => Result.internalError("Effect node must be a child of a Select node") + // Non-leaf non-Json element: compile subobject queries case s@Select(fieldName, resultName, child) => context.forField(fieldName, resultName).flatMap { fieldContext => @@ -2943,19 +2958,6 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self } } - case Effect(_, Select(fieldName, _, _)) => - columnsForLeaf(context, fieldName).flatMap { - case Nil => EmptySqlQuery.success - case cols => - val constraintCols = if(exposeJoins) parentConstraints.lastOption.getOrElse(Nil).map(_._2) else Nil - val extraCols = keyColumnsForType(context) ++ constraintCols - for { - parentTable <- parentTableForType(context) - extraJoins <- parentConstraintsToSqlJoins(parentTable, parentConstraints) - } yield - SqlSelect(context, Nil, parentTable, (cols ++ extraCols).distinct, extraJoins, Nil, Nil, None, None, Nil, true, false) - } - case TypeCase(default, narrows) => def isSimple(query: Query): Boolean = { def loop(query: Query): Boolean = @@ -3116,7 +3118,7 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self case TransformCursor(_, child) => loop(child, context, parentConstraints, exposeJoins) - case Empty | Query.Component(_, _, _) | Query.Effect(_, _) | (_: UntypedSelect) | (_: UntypedFragmentSpread) | (_: UntypedInlineFragment) | (_: Select) => + case Empty | Query.Component(_, _, _) | (_: UntypedSelect) | (_: UntypedFragmentSpread) | (_: UntypedInlineFragment) | (_: Select) => EmptySqlQuery.success } } diff --git a/modules/sql/shared/src/test/scala/SqlNestedEffectsMapping.scala b/modules/sql/shared/src/test/scala/SqlNestedEffectsMapping.scala index 0e45a292..545102a7 100644 --- a/modules/sql/shared/src/test/scala/SqlNestedEffectsMapping.scala +++ b/modules/sql/shared/src/test/scala/SqlNestedEffectsMapping.scala @@ -199,14 +199,13 @@ trait SqlNestedEffectsMapping[F[_]] extends SqlTestMapping[F] { ) object CurrencyQueryHandler extends EffectHandler[F] { - def runEffects(queries: List[(Query, Cursor)]): F[Result[List[(Query, Cursor)]]] = { + def runEffects(queries: List[(Query, Cursor)]): F[Result[List[Cursor]]] = { val countryCodes = queries.map(_._2.fieldAs[String]("code2").toOption) val distinctCodes = queries.flatMap(_._2.fieldAs[String]("code2").toList).distinct - val children = queries.flatMap { - case (Select(name, alias, child), parentCursor) => - parentCursor.context.forField(name, alias).toList.map(ctx => (ctx, child, parentCursor)) - case _ => Nil + val children0 = queries.traverse { + case (query, parentCursor) => + Query.childContext(parentCursor.context, query).map(ctx => (ctx, parentCursor)) } def unpackResults(res: Json): List[Json] = @@ -225,39 +224,60 @@ trait SqlNestedEffectsMapping[F[_]] extends SqlTestMapping[F] { case _ => Json.Null }).getOrElse(Nil) - for { - res <- currencyService.get(distinctCodes) + (for { + children <- ResultT(children0.pure[F]) + res <- ResultT(currencyService.get(distinctCodes).map(_.success)) } yield { unpackResults(res).zip(children).map { - case (res, (ctx, child, parentCursor)) => - val cursor = CirceCursor(ctx, res, Some(parentCursor), parentCursor.env) - (child, cursor) + case (res, (childContext, parentCursor)) => + val cursor = CirceCursor(childContext, res, Some(parentCursor), parentCursor.env) + cursor } - }.success + }).value.widen } } object CountryQueryHandler extends EffectHandler[F] { val toCode = Map("BR" -> "BRA", "GB" -> "GBR", "NL" -> "NLD") - def runEffects(queries: List[(Query, Cursor)]): F[Result[List[(Query, Cursor)]]] = { + def runEffects(queries: List[(Query, Cursor)]): F[Result[List[Cursor]]] = { + + def mkListCursor(cursor: Cursor, fieldName: String, resultName: Option[String]): Result[Cursor] = + for { + c <- cursor.field(fieldName, resultName) + lc <- c.preunique + } yield lc + + def extractCode(cursor: Cursor): Result[String] = + cursor.fieldAs[String]("code") + + def partitionCursor(codes: List[String], cursor: Cursor): Result[List[Cursor]] = { + for { + cursors <- cursor.asList + tagged <- cursors.traverse(c => (extractCode(c).map { code => (code, c) })) + } yield { + val m = tagged.toMap + codes.map(code => m(code)) + } + } + runGrouped(queries) { - case (Select("country", alias, child), cursors, indices) => + case (Select(_, _, child), cursors, indices) => val codes = cursors.flatMap(_.fieldAs[Json]("countryCode").toOption.flatMap(_.asString).toList).map(toCode) - val combinedQuery = Select("country", alias, Filter(In(CountryType / "code", codes), child)) + val combinedQuery = Select("country", None, Filter(In(CountryType / "code", codes), child)) (for { - cursor <- ResultT(sqlCursor(combinedQuery, Env.empty)) + cursor <- ResultT(sqlCursor(combinedQuery, Env.empty)) + cursor0 <- ResultT(mkListCursor(cursor, "country", None).pure[F]) + pcs <- ResultT(partitionCursor(codes, cursor0).pure[F]) } yield { - codes.map { code => - (Select("country", alias, Unique(Filter(Eql(CountryType / "code", Const(code)), child))), cursor) - }.zip(indices) + pcs.zip(indices) }).value.widen case _ => Result.internalError("Continuation query has the wrong shape").pure[F].widen } } - def runGrouped(ts: List[(Query, Cursor)])(op: (Query, List[Cursor], List[Int]) => F[Result[List[((Query, Cursor), Int)]]]): F[Result[List[(Query, Cursor)]]] = { + def runGrouped(ts: List[(Query, Cursor)])(op: (Query, List[Cursor], List[Int]) => F[Result[List[(Cursor, Int)]]]): F[Result[List[Cursor]]] = { val groupedAndIndexed = ts.zipWithIndex.groupMap(_._1._1)(ti => (ti._1._2, ti._2)).toList val groupedResults = groupedAndIndexed.map { case (q, cis) => diff --git a/modules/sql/shared/src/test/scala/SqlNestedEffectsSuite.scala b/modules/sql/shared/src/test/scala/SqlNestedEffectsSuite.scala index 9c2d0d06..fb4e119d 100644 --- a/modules/sql/shared/src/test/scala/SqlNestedEffectsSuite.scala +++ b/modules/sql/shared/src/test/scala/SqlNestedEffectsSuite.scala @@ -35,7 +35,39 @@ trait SqlNestedEffectsSuite extends CatsEffectSuite { assertWeaklyEqualIO(res, expected) } - test("simple composed query") { + test("simple composed query (1)") { + val query = """ + query { + country(code: "GBR") { + currencies { + code + exchangeRate + } + } + } + """ + + val expected = json""" + { + "data" : { + "country" : { + "currencies": [ + { + "code": "GBP", + "exchangeRate": 1.25 + } + ] + } + } + } + """ + + val res = mapping.flatMap(_._2.compileAndRun(query)) + + assertWeaklyEqualIO(res, expected) + } + + test("simple composed query (2)") { val query = """ query { country(code: "GBR") {