Skip to content

Commit

Permalink
Merge pull request #521 from typelevel/topic/fragment-introspection
Browse files Browse the repository at this point in the history
Elaborate introspection in out of line fragments
  • Loading branch information
milessabin authored Nov 27, 2023
2 parents c3f68dd + 76d4526 commit b21d8ce
Show file tree
Hide file tree
Showing 3 changed files with 137 additions and 8 deletions.
25 changes: 24 additions & 1 deletion modules/core/src/main/scala/compiler.scala
Original file line number Diff line number Diff line change
Expand Up @@ -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(
Expand Down Expand Up @@ -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 */
Expand Down Expand Up @@ -432,6 +438,8 @@ object QueryCompiler {

/** A QueryCompiler phase. */
trait Phase {
def transformFragments: Elab[Unit] = Elab.unit

/**
* Transform the supplied query algebra term `query`.
*/
Expand Down Expand Up @@ -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"), _, _, _, _) =>
Expand Down
113 changes: 113 additions & 0 deletions modules/core/src/test/scala/introspection/IntrospectionSuite.scala
Original file line number Diff line number Diff line change
Expand Up @@ -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 = """
{
Expand Down Expand Up @@ -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 {
Expand Down Expand Up @@ -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 = """
{
Expand Down
7 changes: 0 additions & 7 deletions modules/generic/src/main/scala-2/genericmapping2.scala
Original file line number Diff line number Diff line change
Expand Up @@ -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)
}
Expand Down

0 comments on commit b21d8ce

Please sign in to comment.