diff --git a/modules/core/src/main/scala/compiler.scala b/modules/core/src/main/scala/compiler.scala index ef8aadc1..e95ee630 100644 --- a/modules/core/src/main/scala/compiler.scala +++ b/modules/core/src/main/scala/compiler.scala @@ -234,7 +234,7 @@ class QueryCompiler(schema: Schema, phases: List[Phase]) { rootTpe <- op.rootTpe(schema) res <- ( for { - query <- allPhases.foldLeftM(op.query) { (acc, phase) => phase.transform(acc) } + query <- allPhases.foldLeftM(op.query) { (acc, phase) => phase.transformFragments *> phase.transform(acc) } } yield Operation(query, rootTpe, op.directives) ).runA( ElabState( @@ -339,6 +339,12 @@ object QueryCompiler { /** 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")) + def transformFragments(f: Map[String, UntypedFragment] => Elab[Map[String, UntypedFragment]]): Elab[Unit] = + for { + fs <- fragments + fs0 <- f(fs) + _ <- StateT.modify(_.copy(fragments = fs0)): Elab[Unit] + } yield () /** `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 */ @@ -432,6 +438,8 @@ object QueryCompiler { /** A QueryCompiler phase. */ trait Phase { + def transformFragments: Elab[Unit] = Elab.unit + /** * Transform the supplied query algebra term `query`. */ @@ -553,6 +561,21 @@ object QueryCompiler { * A phase which elaborates GraphQL introspection queries into the query algrebra. */ class IntrospectionElaborator(level: IntrospectionLevel) extends Phase { + override def transformFragments: Elab[Unit] = + Elab.transformFragments { fs => + fs.toList.traverse { + case (nme, f@UntypedFragment(_, tpnme, _, child)) => + for { + s <- Elab.schema + c <- Elab.context + tpe <- Elab.liftR(Result.fromOption(s.definition(tpnme).orElse(Introspection.schema.definition(tpnme)), s"Unknown type '$tpnme' in fragment definition")) + _ <- Elab.push(c.asType(tpe), child) + ec <- transform(child) + _ <- Elab.pop + } yield (nme, f.copy(child = ec)) + }.map(_.toMap) + } + override def transform(query: Query): Elab[Query] = query match { case s@UntypedSelect(fieldName @ ("__typename" | "__schema" | "__type"), _, _, _, _) => diff --git a/modules/core/src/test/scala/introspection/IntrospectionSuite.scala b/modules/core/src/test/scala/introspection/IntrospectionSuite.scala index 392065ed..12efb4a6 100644 --- a/modules/core/src/test/scala/introspection/IntrospectionSuite.scala +++ b/modules/core/src/test/scala/introspection/IntrospectionSuite.scala @@ -105,6 +105,56 @@ final class IntrospectionSuite extends CatsEffectSuite { assertIO(res, expected) } + test("simple type query with variables") { + val query = """ + query getType($name: String!) { + __type(name: $name) { + name + fields { + name + type { + name + } + } + } + } + """ + + val variables = json""" + { + "name": "User" + } + """ + + val expected = json""" + { + "data" : { + "__type": { + "name": "User", + "fields": [ + { + "name": "id", + "type": { "name": "String" } + }, + { + "name": "name", + "type": { "name": "String" } + }, + { + "name": "birthday", + "type": { "name": "Date" } + } + ] + } + } + } + """ + + val res = TestMapping.compileAndRun(query, untypedVars = Some(variables)) + + assertIO(res, expected) + } + test("kind/name/description/ofType query") { val query = """ { @@ -879,6 +929,36 @@ final class IntrospectionSuite extends CatsEffectSuite { assertIO(res, Some(expected)) } + test("simple schema query with fragment") { + val query = """ + { + __schema { + ...SchemaFields + } + } + + fragment SchemaFields on __Schema { + queryType { name } + } + """ + + val expected = json""" + { + "data" : { + "__schema" : { + "queryType" : { + "name" : "Query" + } + } + } + } + """ + + val res = TestMapping.compileAndRun(query) + + assertIO(res, expected) + } + test("standard introspection query") { val query = """ |query IntrospectionQuery { @@ -1266,6 +1346,39 @@ final class IntrospectionSuite extends CatsEffectSuite { assertIO(res, expected) } + test("typename in a fragment") { + val query = """ + { + users { + ...UserFields + } + } + fragment UserFields on User { + __typename + renamed: __typename + name + } + """ + + val expected = json""" + { + "data" : { + "users" : [ + { + "__typename" : "User", + "renamed" : "User", + "name" : "Luke Skywalker" + } + ] + } + } + """ + + val res = SmallMapping.compileAndRun(query) + + assertIO(res, expected) + } + test("mixed query") { val query = """ { diff --git a/modules/generic/src/main/scala-2/genericmapping2.scala b/modules/generic/src/main/scala-2/genericmapping2.scala index a568bdb3..81132aa3 100644 --- a/modules/generic/src/main/scala-2/genericmapping2.scala +++ b/modules/generic/src/main/scala-2/genericmapping2.scala @@ -87,13 +87,6 @@ trait ScalaVersionSpecificGenericMappingLike[F[_]] extends Mapping[F] { self: Ge override def field(fieldName: String, resultName: Option[String]): Result[Cursor] = mkCursorForField(this, fieldName, resultName) orElse { - fieldMap.get(fieldName) match { - case None => - println(s"No field '$fieldName' for type $tpe") - println(fieldMap) - case _ => - } - fieldMap.get(fieldName).toResult(s"No field '$fieldName' for type $tpe").flatMap { f => f(context.forFieldOrAttribute(fieldName, resultName), focus, Some(this), Env.empty) }