From 6f5a7dd853759ea840203db33406248cc17ec10a Mon Sep 17 00:00:00 2001 From: Miles Sabin Date: Sat, 14 Sep 2024 16:22:44 +0100 Subject: [PATCH] Fixes for polymorphic field mappings + The resolution of polymorphic field mappings now takes the runtime context into account properly. + All narrowing operations on cursors yield a value in Result allowing mappings to report internal errors for failed discriminators. + In SQL mappings, failed discriminators now fail the entire query with an internal error. + The TypeCase extractor has been removed because it can't take the context into account. --- build.sbt | 2 +- .../circe/src/main/scala/circemapping.scala | 16 +- modules/core/src/main/scala/cursor.scala | 6 +- modules/core/src/main/scala/mapping.scala | 54 +- modules/core/src/main/scala/query.scala | 29 - .../src/main/scala/queryinterpreter.scala | 45 +- .../core/src/main/scala/valuemapping.scala | 16 +- .../src/main/scala-2/genericmapping2.scala | 10 +- .../src/main/scala-3/genericmapping3.scala | 10 +- .../shared/src/main/scala/SqlMapping.scala | 130 +++-- .../src/test/resources/db/interfaces.sql | 18 +- .../src/test/scala/SqlInterfacesMapping.scala | 24 +- .../test/scala/SqlInterfacesMapping2.scala | 12 +- .../src/test/scala/SqlInterfacesSuite.scala | 549 ++++++++++++++++++ .../src/test/scala/SqlInterfacesSuite2.scala | 48 +- .../SqlMappingValidatorInvalidMapping.scala | 5 +- .../SqlMappingValidatorValidMapping.scala | 5 +- .../scala/SqlRecursiveInterfacesMapping.scala | 8 +- .../src/test/scala/SqlUnionsMapping.scala | 8 +- 19 files changed, 781 insertions(+), 214 deletions(-) diff --git a/build.sbt b/build.sbt index 0e1f2a3b..9d05a9a5 100644 --- a/build.sbt +++ b/build.sbt @@ -30,7 +30,7 @@ ThisBuild / scalaVersion := Scala2 ThisBuild / crossScalaVersions := Seq(Scala2, Scala3) ThisBuild / tlJdkRelease := Some(11) -ThisBuild / tlBaseVersion := "0.21" +ThisBuild / tlBaseVersion := "0.22" ThisBuild / startYear := Some(2019) ThisBuild / licenses := Seq(License.Apache2) ThisBuild / developers := List( diff --git a/modules/circe/src/main/scala/circemapping.scala b/modules/circe/src/main/scala/circemapping.scala index af41e522..607d524e 100644 --- a/modules/circe/src/main/scala/circemapping.scala +++ b/modules/circe/src/main/scala/circemapping.scala @@ -152,8 +152,8 @@ trait CirceMappingLike[F[_]] extends Mapping[F] { case _ => Result.internalError(s"Expected Nullable type, found $focus for $tpe") } - def narrowsTo(subtpe: TypeRef): Boolean = - subtpe <:< tpe && + def narrowsTo(subtpe: TypeRef): Result[Boolean] = + (subtpe <:< tpe && ((subtpe.dealias, focus.asObject) match { case (nt: TypeWithFields, Some(obj)) => nt.fields.forall { f => @@ -161,13 +161,15 @@ trait CirceMappingLike[F[_]] extends Mapping[F] { } && obj.keys.forall(nt.hasField) case _ => false - }) + })).success def narrow(subtpe: TypeRef): Result[Cursor] = - if (narrowsTo(subtpe)) - mkChild(context.asType(subtpe)).success - else - Result.internalError(s"Focus ${focus} of static type $tpe cannot be narrowed to $subtpe") + narrowsTo(subtpe).flatMap { n => + if (n) + mkChild(context.asType(subtpe)).success + else + Result.internalError(s"Focus ${focus} of static type $tpe cannot be narrowed to $subtpe") + } def field(fieldName: String, resultName: Option[String]): Result[Cursor] = { val localField = diff --git a/modules/core/src/main/scala/cursor.scala b/modules/core/src/main/scala/cursor.scala index ca0b48aa..d4303535 100644 --- a/modules/core/src/main/scala/cursor.scala +++ b/modules/core/src/main/scala/cursor.scala @@ -128,7 +128,7 @@ trait Cursor { def isDefined: Result[Boolean] /** Is the value at this `Cursor` narrowable to `subtpe`? */ - def narrowsTo(subtpe: TypeRef): Boolean + def narrowsTo(subtpe: TypeRef): Result[Boolean] /** * Yield a `Cursor` corresponding to the value at this `Cursor` narrowed to @@ -251,7 +251,7 @@ object Cursor { def isDefined: Result[Boolean] = Result.internalError(s"Expected Nullable type, found $focus for $tpe") - def narrowsTo(subtpe: TypeRef): Boolean = false + def narrowsTo(subtpe: TypeRef): Result[Boolean] = false.success def narrow(subtpe: TypeRef): Result[Cursor] = Result.internalError(s"Focus ${focus} of static type $tpe cannot be narrowed to $subtpe") @@ -290,7 +290,7 @@ object Cursor { def isDefined: Result[Boolean] = underlying.isDefined - def narrowsTo(subtpe: TypeRef): Boolean = underlying.narrowsTo(subtpe) + def narrowsTo(subtpe: TypeRef): Result[Boolean] = underlying.narrowsTo(subtpe) def narrow(subtpe: TypeRef): Result[Cursor] = underlying.narrow(subtpe) diff --git a/modules/core/src/main/scala/mapping.scala b/modules/core/src/main/scala/mapping.scala index c1b552a4..123f3cd2 100644 --- a/modules/core/src/main/scala/mapping.scala +++ b/modules/core/src/main/scala/mapping.scala @@ -134,12 +134,13 @@ abstract class Mapping[F[_]] { * the `fieldName` child of `parent`. */ protected final def mkCursorForField(parent: Cursor, fieldName: String, resultName: Option[String]): Result[Cursor] = { - val context = parent.context - val fieldContext = context.forFieldOrAttribute(fieldName, resultName) - typeMappings.fieldMapping(parent, fieldName). - toResultOrError(s"No mapping for field '$fieldName' for type ${parent.tpe}"). - flatMap(mkCursorForMappedField(parent, fieldContext, _)) + flatMap(_.toResultOrError(s"No mapping for field '$fieldName' for type ${parent.tpe}")). + flatMap { + case (np, fm) => + val fieldContext = np.context.forFieldOrAttribute(fieldName, resultName) + mkCursorForMappedField(np, fieldContext, fm) + } } final class TypeMappings private ( @@ -200,18 +201,25 @@ abstract class Mapping[F[_]] { * Yields the `FieldMapping` associated with `fieldName` in the runtime context * determined by the given `Cursor`, if any. */ - def fieldMapping(parent: Cursor, fieldName: String): Option[FieldMapping] = { + def fieldMapping(parent: Cursor, fieldName: String): Result[Option[(Cursor, FieldMapping)]] = { val context = parent.context - fieldIndex(context).flatMap(_.get(fieldName)).flatMap { - case ifm: InheritedFieldMapping => - ifm.select(parent.context) - case pfm: PolymorphicFieldMapping => + fieldIndex(context).flatMap(_.get(fieldName)) match { + case Some(ifm: InheritedFieldMapping) => + ifm.select(parent.context).map((parent, _)).success + case Some(pfm: PolymorphicFieldMapping) => pfm.select(parent) - case fm => - Some(fm) + case Some(fm) => + Option((parent, fm)).success + case None => None.success } } + def fieldIsPolymorphic(context: Context, fieldName: String): Boolean = + rawFieldMapping(context, fieldName).exists { + case _: PolymorphicFieldMapping => true + case _ => false + } + /** Yields the `FieldMapping` directly or ancestrally associated with `fieldName` in `context`, if any. */ def ancestralFieldMapping(context: Context, fieldName: String): Option[FieldMapping] = fieldMapping(context, fieldName).orElse { @@ -532,16 +540,20 @@ abstract class Mapping[F[_]] { def hidden: Boolean = false def subtree: Boolean = false - def select(cursor: Cursor): Option[FieldMapping] = { - val context = cursor.context + def select(cursor: Cursor): Result[Option[(Cursor, FieldMapping)]] = { val applicable = - candidates.mapFilter { - case (pred, fm) if cursor.narrowsTo(schema.uncheckedRef(pred.tpe)) => - pred(context.asType(pred.tpe)).map(prio => (prio, fm)) - case _ => - None + candidates.traverseFilter { + case (pred, fm) => + cursor.narrowsTo(schema.uncheckedRef(pred.tpe)).flatMap { narrows => + if (narrows) + for { + nc <- cursor.narrow(schema.uncheckedRef(pred.tpe)) + } yield pred(nc.context).map(prio => (prio, (nc, fm))) + else + None.success + } } - applicable.maxByOption(_._1).map(_._2) + applicable.map(_.maxByOption(_._1).map(_._2)) } def select(context: Context): Option[FieldMapping] = { @@ -1266,7 +1278,7 @@ abstract class Mapping[F[_]] { case _ => Result.internalError(s"Not nullable at ${context.path}") } - def narrowsTo(subtpe: TypeRef): Boolean = false + def narrowsTo(subtpe: TypeRef): Result[Boolean] = false.success def narrow(subtpe: TypeRef): Result[Cursor] = Result.failure(s"Cannot narrow $tpe to $subtpe") diff --git a/modules/core/src/main/scala/query.scala b/modules/core/src/main/scala/query.scala index 901eb808..f4cc6560 100644 --- a/modules/core/src/main/scala/query.scala +++ b/modules/core/src/main/scala/query.scala @@ -456,35 +456,6 @@ object Query { } } - /** Extractor for grouped Narrow patterns in the query algebra */ - object TypeCase { - def unapply(q: Query): Option[(Query, List[Narrow])] = { - def branch(q: Query): Option[TypeRef] = - q match { - case Narrow(subtpe, _) => Some(subtpe) - case _ => None - } - - val grouped = ungroup(q).groupBy(branch).toList - val (default0, narrows0) = grouped.partition(_._1.isEmpty) - if (narrows0.isEmpty) None - else { - val default = default0.flatMap(_._2) match { - case Nil => Empty - case children => Group(children) - } - val narrows = narrows0.collect { - case (Some(subtpe), narrows) => - narrows.collect { case Narrow(_, child) => child } match { - case List(child) => Narrow(subtpe, child) - case children => Narrow(subtpe, Group(children)) - } - } - Some((default, narrows)) - } - } - } - /** Construct a query which yields all the supplied paths */ def mkPathQuery(paths: List[List[String]]): List[Query] = paths match { diff --git a/modules/core/src/main/scala/queryinterpreter.scala b/modules/core/src/main/scala/queryinterpreter.scala index 118dce32..198852e0 100644 --- a/modules/core/src/main/scala/queryinterpreter.scala +++ b/modules/core/src/main/scala/queryinterpreter.scala @@ -201,22 +201,23 @@ class QueryInterpreter[F[_]](mapping: Mapping[F]) { siblings.flatTraverse(query => runFields(query, tpe, cursor)) case Introspect(schema, s@Select("__typename", _, Empty)) if tpe.isNamed => - (tpe.dealias match { - case o: ObjectType => Some(o.name) + val fail = Result.failure(s"'__typename' cannot be applied to non-selectable type '$tpe'") + def mkTypeNameFields(name: String) = + List((s.resultName, ProtoJson.fromJson(Json.fromString(name)))).success + def mkTypeNameFieldsOrFail(name: Option[String]) = + name.map(mkTypeNameFields).getOrElse(fail) + + tpe.dealias match { + case o: ObjectType => mkTypeNameFields(o.name) case i: InterfaceType => - (schema.implementations(i).collectFirst { - case o if cursor.narrowsTo(schema.uncheckedRef(o)) => o.name - }) + schema.implementations(i).collectFirstSomeM { o => + cursor.narrowsTo(schema.uncheckedRef(o)).ifF(Some(o.name), None) + }.flatMap(mkTypeNameFieldsOrFail) case u: UnionType => - (u.members.map(_.dealias).collectFirst { - case nt: NamedType if cursor.narrowsTo(schema.uncheckedRef(nt)) => nt.name - }) - case _ => None - }) match { - case Some(name) => - List((s.resultName, ProtoJson.fromJson(Json.fromString(name)))).success - case None => - Result.failure(s"'__typename' cannot be applied to non-selectable type '$tpe'") + u.members.map(_.dealias).collectFirstSomeM { nt => + cursor.narrowsTo(schema.uncheckedRef(nt)).ifF(Some(nt.name), None) + }.flatMap(mkTypeNameFieldsOrFail) + case _ => fail } case sel: Select if tpe.isNullable => @@ -250,13 +251,15 @@ class QueryInterpreter[F[_]](mapping: Mapping[F]) { value <- runValue(child, fieldTpe, c) } yield List((sel.resultName, value)) - case Narrow(tp1, child) if cursor.narrowsTo(tp1) => - for { - c <- cursor.narrow(tp1) - fields <- runFields(child, tp1, c) - } yield fields - - case _: Narrow => Nil.success + case Narrow(tp1, child) => + cursor.narrowsTo(tp1).flatMap { n => + if (!n) Nil.success + else + for { + c <- cursor.narrow(tp1) + fields <- runFields(child, tp1, c) + } yield fields + } case c@Component(_, _, cont) => for { diff --git a/modules/core/src/main/scala/valuemapping.scala b/modules/core/src/main/scala/valuemapping.scala index 113b6cbf..ace00837 100644 --- a/modules/core/src/main/scala/valuemapping.scala +++ b/modules/core/src/main/scala/valuemapping.scala @@ -175,20 +175,22 @@ trait ValueMappingLike[F[_]] extends Mapping[F] { case _ => Result.internalError(s"Expected Nullable type, found $focus for $tpe") } - def narrowsTo(subtpe: TypeRef): Boolean = - subtpe <:< tpe && + def narrowsTo(subtpe: TypeRef): Result[Boolean] = + (subtpe <:< tpe && objectMapping(context.asType(subtpe)).exists { case ValueObjectMapping(_, _, classTag) => classTag.runtimeClass.isInstance(focus) case _ => false - } + }).success def narrow(subtpe: TypeRef): Result[Cursor] = - if (narrowsTo(subtpe)) - mkChild(context.asType(subtpe)).success - else - Result.internalError(s"Focus ${focus} of static type $tpe cannot be narrowed to $subtpe") + narrowsTo(subtpe).flatMap { n => + if(n) + mkChild(context.asType(subtpe)).success + else + Result.internalError(s"Focus ${focus} of static type $tpe cannot be narrowed to $subtpe") + } def field(fieldName: String, resultName: Option[String]): Result[Cursor] = mkCursorForField(this, fieldName, resultName) diff --git a/modules/generic/src/main/scala-2/genericmapping2.scala b/modules/generic/src/main/scala-2/genericmapping2.scala index 4feea20b..c61e4d37 100644 --- a/modules/generic/src/main/scala-2/genericmapping2.scala +++ b/modules/generic/src/main/scala-2/genericmapping2.scala @@ -137,12 +137,14 @@ trait ScalaVersionSpecificGenericMappingLike[F[_]] extends Mapping[F] { self: Ge override def field(fieldName: String, resultName: Option[String]): Result[Cursor] = cursor.field(fieldName, resultName) orElse mkCursorForField(this, fieldName, resultName) - override def narrowsTo(subtpe: TypeRef): Boolean = - subtpe <:< tpe && rtpe <:< subtpe + override def narrowsTo(subtpe: TypeRef): Result[Boolean] = + (subtpe <:< tpe && rtpe <:< subtpe).success override def narrow(subtpe: TypeRef): Result[Cursor] = - if (narrowsTo(subtpe)) copy(tpe0 = subtpe).success - else Result.internalError(s"Focus ${focus} of static type $tpe cannot be narrowed to $subtpe") + narrowsTo(subtpe).flatMap { n => + if (n) copy(tpe0 = subtpe).success + else Result.internalError(s"Focus ${focus} of static type $tpe cannot be narrowed to $subtpe") + } } } } diff --git a/modules/generic/src/main/scala-3/genericmapping3.scala b/modules/generic/src/main/scala-3/genericmapping3.scala index 328332bb..d57c6d33 100644 --- a/modules/generic/src/main/scala-3/genericmapping3.scala +++ b/modules/generic/src/main/scala-3/genericmapping3.scala @@ -116,12 +116,14 @@ trait ScalaVersionSpecificGenericMappingLike[F[_]] extends Mapping[F] { self: Ge override def field(fieldName: String, resultName: Option[String]): Result[Cursor] = cursor.field(fieldName, resultName) orElse mkCursorForField(this, fieldName, resultName) - override def narrowsTo(subtpe: TypeRef): Boolean = - subtpe <:< tpe && rtpe <:< subtpe + override def narrowsTo(subtpe: TypeRef): Result[Boolean] = + (subtpe <:< tpe && rtpe <:< subtpe).success override def narrow(subtpe: TypeRef): Result[Cursor] = - if (narrowsTo(subtpe)) copy(tpe0 = subtpe).success - else Result.internalError(s"Focus ${focus} of static type $tpe cannot be narrowed to $subtpe") + narrowsTo(subtpe).flatMap { n => + if (n) copy(tpe0 = subtpe).success + else Result.internalError(s"Focus ${focus} of static type $tpe cannot be narrowed to $subtpe") + } } } } diff --git a/modules/sql/shared/src/main/scala/SqlMapping.scala b/modules/sql/shared/src/main/scala/SqlMapping.scala index ccb2b909..5eff5339 100644 --- a/modules/sql/shared/src/main/scala/SqlMapping.scala +++ b/modules/sql/shared/src/main/scala/SqlMapping.scala @@ -761,7 +761,7 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self /** Discriminator for the branches of an interface/union */ trait SqlDiscriminator { /** yield a predicate suitable for filtering row corresponding to the supplied type */ - def narrowPredicate(tpe: Type): Option[Predicate] + def narrowPredicate(tpe: Type): Result[Predicate] /** compute the concrete type of the value at the cursor */ def discriminate(cursor: Cursor): Result[Type] @@ -2826,6 +2826,53 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self /** Compile the given GraphQL query to SQL in the given `Context` */ def apply(q: Query, context: Context): Result[MappedQuery] = { def loop(q: Query, context: Context, parentConstraints: List[List[(SqlColumn, SqlColumn)]], exposeJoins: Boolean): Result[SqlQuery] = { + + object TypeCase { + def unapply(q: Query): Option[(Query, List[Narrow])] = { + def isPolySelect(q: Query): Boolean = + q match { + case Select(fieldName, _, _) => + typeMappings.fieldIsPolymorphic(context, fieldName) + case _ => false + } + + def branch(q: Query): Option[TypeRef] = + q match { + case Narrow(subtpe, _) => Some(subtpe) + case _ => None + } + + val ungrouped = ungroup(q).flatMap { + case sel@Select(fieldName, _, _) if isPolySelect(sel) => + typeMappings.rawFieldMapping(context, fieldName) match { + case Some(TypeMappings.PolymorphicFieldMapping(cands)) => + cands.map { case (pred, _) => Narrow(schema.uncheckedRef(pred.tpe), sel) } + case _ => Seq(sel) + } + + case other => Seq(other) + } + + val grouped = ungrouped.groupBy(branch).toList + val (default0, narrows0) = grouped.partition(_._1.isEmpty) + if (narrows0.isEmpty) None + else { + val default = default0.flatMap(_._2) match { + case Nil => Empty + case children => Group(children) + } + val narrows = narrows0.collect { + case (Some(subtpe), narrows) => + narrows.collect { case Narrow(_, child) => child } match { + case List(child) => Narrow(subtpe, child) + case children => Narrow(subtpe, Group(children)) + } + } + Some((default, narrows)) + } + } + } + def group(queries: List[Query]): Result[SqlQuery] = { queries.foldLeft(List.empty[SqlQuery].success) { case (nodes, q) => @@ -3019,11 +3066,6 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self assert(supertpe.underlying.isInterface || supertpe.underlying.isUnion || (subtpes.sizeCompare(1) == 0 && subtpes.head =:= supertpe)) subtpes.foreach(subtpe => assert(subtpe <:< supertpe)) - val discriminator = discriminatorForType(context) - val narrowPredicates = subtpes.map { subtpe => - (subtpe, discriminator.flatMap(_.discriminator.narrowPredicate(subtpe))) - } - val exhaustive = schema.exhaustive(supertpe, subtpes) val exclusive = default == Empty val allSimple = narrows.forall(narrow => isSimple(narrow.child)) @@ -3050,24 +3092,24 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self val dquery = if(exhaustive) EmptySqlQuery(context).success - else { - val allPreds = narrowPredicates.collect { - case (_, Some(pred)) => pred - } - if (exclusive) { - for { - parentTable <- parentTableForType(context) - allPreds0 <- allPreds.traverse(pred => SqlQuery.contextualiseWhereTerms(context, parentTable, pred).map(Not(_))) - } yield { + else + discriminatorForType(context).map { disc => + subtpes.traverse(disc.discriminator.narrowPredicate) + }.getOrElse(Nil.success).flatMap { allPreds => + if (exclusive) { + for { + parentTable <- parentTableForType(context) + allPreds0 <- allPreds.traverse(pred => SqlQuery.contextualiseWhereTerms(context, parentTable, pred).map(Not(_))) + } yield { + val defaultPredicate = And.combineAll(allPreds0) + SqlSelect(context, Nil, parentTable, extraCols, Nil, defaultPredicate :: Nil, Nil, None, None, Nil, true, false) + } + } else { + val allPreds0 = allPreds.map(Not(_)) val defaultPredicate = And.combineAll(allPreds0) - SqlSelect(context, Nil, parentTable, extraCols, Nil, defaultPredicate :: Nil, Nil, None, None, Nil, true, false) + loop(Filter(defaultPredicate, default), context, parentConstraints, exposeJoins).flatMap(_.withContext(context, extraCols, Nil)) } - } else { - val allPreds0 = allPreds.map(Not(_)) - val defaultPredicate = And.combineAll(allPreds0) - loop(Filter(defaultPredicate, default), context, parentConstraints, exposeJoins).flatMap(_.withContext(context, extraCols, Nil)) } - } for { dquery0 <- dquery @@ -3496,36 +3538,39 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self case _ => Result.internalError(s"Not nullable at ${context.path}") } - def narrowsTo(subtpe: TypeRef): Boolean = { + def narrowsTo(subtpe: TypeRef): Result[Boolean] = { def check(ctpe: Type): Boolean = if (ctpe =:= tpe) asTable.map(table => mapped.narrowsTo(context.asType(subtpe), table)).toOption.getOrElse(false) else ctpe <:< subtpe - (subtpe <:< tpe) && - (discriminatorForType(context) match { - case Some(disc) => disc.discriminator.discriminate(this).map(check).getOrElse(false) - case _ => check(tpe) - }) + if (!(subtpe <:< tpe)) false.success + else + discriminatorForType(context) match { + case Some(disc) => disc.discriminator.discriminate(this).map(check) + case _ => check(tpe).success + } } def narrow(subtpe: TypeRef): Result[Cursor] = { - if (narrowsTo(subtpe)) { - val narrowedContext = context.asType(subtpe) - asTable.map { table => - mkChild(context = narrowedContext, focus = mapped.narrow(narrowedContext, table)) - } - } else Result.internalError(s"Cannot narrow $tpe to $subtpe") + narrowsTo(subtpe).flatMap { n => + if (n) { + val narrowedContext = context.asType(subtpe) + asTable.map { table => + mkChild(context = narrowedContext, focus = mapped.narrow(narrowedContext, table)) + } + } else Result.internalError(s"Cannot narrow $tpe to $subtpe") + } } def field(fieldName: String, resultName: Option[String]): Result[Cursor] = { - val fieldContext = context.forFieldOrAttribute(fieldName, resultName) - val fieldTpe = fieldContext.tpe val localField = - typeMappings.fieldMapping(this, fieldName) match { - case Some(_: SqlJson) => + typeMappings.fieldMapping(this, fieldName).flatMap { + case Some((np, _: SqlJson)) => + val fieldContext = np.context.forFieldOrAttribute(fieldName, resultName) + val fieldTpe = fieldContext.tpe asTable.flatMap { table => def mkCirceCursor(f: Json): Result[Cursor] = - CirceCursor(fieldContext, focus = f, parent = Some(this), Env.empty).success + CirceCursor(fieldContext, focus = f, parent = Some(np), Env.empty).success mapped.selectAtomicField(context, fieldName, table).flatMap(_ match { case Some(j: Json) if fieldTpe.isNullable => mkCirceCursor(j) case None => mkCirceCursor(Json.Null) @@ -3535,7 +3580,9 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self }) } - case Some(_: SqlField) => + case Some((np, _: SqlField)) => + val fieldContext = np.context.forFieldOrAttribute(fieldName, resultName) + val fieldTpe = fieldContext.tpe asTable.flatMap(table => mapped.selectAtomicField(context, fieldName, table).map { leaf => val leafFocus = leaf match { @@ -3543,11 +3590,12 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self case other => other } assert(leafFocus != FailedJoin) - LeafCursor(fieldContext, leafFocus, Some(this), Env.empty) + LeafCursor(fieldContext, leafFocus, Some(np), Env.empty) } ) - case Some(_: SqlObject) | Some(_: EffectMapping) => + case Some((np, (_: SqlObject | _: EffectMapping))) => + val fieldContext = np.context.forFieldOrAttribute(fieldName, resultName) asTable.map { table => val focussed = mapped.narrow(fieldContext, table) mkChild(context = fieldContext, focus = focussed) diff --git a/modules/sql/shared/src/test/resources/db/interfaces.sql b/modules/sql/shared/src/test/resources/db/interfaces.sql index c8dc71cb..03c8cd0b 100644 --- a/modules/sql/shared/src/test/resources/db/interfaces.sql +++ b/modules/sql/shared/src/test/resources/db/interfaces.sql @@ -7,7 +7,9 @@ CREATE TABLE entities ( film_rating TEXT, film_label INTEGER, series_number_of_episodes INTEGER, - series_label TEXT + series_label TEXT, + image_url TEXT, + hidden_image_url TEXT ); CREATE TABLE episodes ( @@ -18,13 +20,13 @@ CREATE TABLE episodes ( synopsis_long TEXT ); -COPY entities (id, entity_type, title, synopsis_short, synopsis_long, film_rating, film_label, series_number_of_episodes, series_label) FROM STDIN WITH DELIMITER '|' NULL AS ''; -1|1|Film 1|Short film 1|Long film 1|PG|1|| -2|1|Film 2|Short film 2|Long film 2|U|2|| -3|1|Film 3|Short film 3|Long film 3|15|3|| -4|2|Series 1|Short series 1|Long series 1|||5|One -5|2|Series 2|Short series 2|Long series 2|||6|Two -6|2|Series 3|Short series 3|Long series 3|||7|Three +COPY entities (id, entity_type, title, synopsis_short, synopsis_long, film_rating, film_label, series_number_of_episodes, series_label, image_url, hidden_image_url) FROM STDIN WITH DELIMITER '|' NULL AS ''; +1|1|Film 1|Short film 1|Long film 1|PG|1|||http://www.example.com/film1.jpg| +2|1|Film 2|Short film 2|Long film 2|U|2|||http://www.example.com/film2.jpg| +3|1|Film 3|Short film 3|Long film 3|15|3|||http://www.example.com/film3.jpg| +4|2|Series 1|Short series 1|Long series 1|||5|One||hidden_series1.jpg +5|2|Series 2|Short series 2|Long series 2|||6|Two||hidden_series2.jpg +6|2|Series 3|Short series 3|Long series 3|||7|Three||hidden_series3.jpg \. COPY episodes (id, series_id, title, synopsis_short, synopsis_long) FROM STDIN WITH DELIMITER '|' NULL AS ''; diff --git a/modules/sql/shared/src/test/scala/SqlInterfacesMapping.scala b/modules/sql/shared/src/test/scala/SqlInterfacesMapping.scala index 0a866850..83c2781b 100644 --- a/modules/sql/shared/src/test/scala/SqlInterfacesMapping.scala +++ b/modules/sql/shared/src/test/scala/SqlInterfacesMapping.scala @@ -38,6 +38,8 @@ trait SqlInterfacesMapping[F[_]] extends SqlTestMapping[F] { self => val seriesLabel = col("series_label", nullable(text)) val synopsisShort = col("synopsis_short", nullable(text)) val synopsisLong = col("synopsis_long", nullable(text)) + val imageUrl = col("image_url", nullable(text)) + val hiddenImageUrl = col("hidden_image_url", nullable(text)) } object episodes extends TableDef("episodes") { @@ -59,12 +61,14 @@ trait SqlInterfacesMapping[F[_]] extends SqlTestMapping[F] { self => entityType: EntityType! title: String synopses: Synopses + imageUrl: String } type Film implements Entity { id: ID! entityType: EntityType! title: String synopses: Synopses + imageUrl: String rating: String label: Int } @@ -73,6 +77,7 @@ trait SqlInterfacesMapping[F[_]] extends SqlTestMapping[F] { self => entityType: EntityType! title: String synopses: Synopses + imageUrl: String numberOfEpisodes: Int episodes: [Episode!]! label: String @@ -126,16 +131,20 @@ trait SqlInterfacesMapping[F[_]] extends SqlTestMapping[F] { self => fieldMappings = List( SqlField("rating", entities.filmRating), - SqlField("label", entities.filmLabel) + SqlField("label", entities.filmLabel), + SqlField("imageUrl", entities.imageUrl) ) ), ObjectMapping( tpe = SeriesType, fieldMappings = List( + SqlField("title", entities.title), SqlField("numberOfEpisodes", entities.seriesNumberOfEpisodes), SqlObject("episodes", Join(entities.id, episodes.seriesId)), - SqlField("label", entities.seriesLabel) + SqlField("label", entities.seriesLabel), + SqlField("hiddenImageUrl", entities.hiddenImageUrl, hidden = true), + CursorField("imageUrl", mkSeriesImageUrl, List("hiddenImageUrl")) ) ), ObjectMapping( @@ -197,18 +206,21 @@ trait SqlInterfacesMapping[F[_]] extends SqlTestMapping[F] { self => } } - def narrowPredicate(subtpe: Type): Option[Predicate] = { - def mkPredicate(tpe: EntityType): Option[Predicate] = - Some(Eql(EType / "entityType", Const(tpe))) + def narrowPredicate(subtpe: Type): Result[Predicate] = { + def mkPredicate(tpe: EntityType): Result[Predicate] = + Eql(EType / "entityType", Const(tpe)).success subtpe match { case FilmType => mkPredicate(EntityType.Film) case SeriesType => mkPredicate(EntityType.Series) - case _ => None + case _ => Result.internalError(s"Invalid discriminator: $subtpe") } } } + def mkSeriesImageUrl(c: Cursor): Result[Option[String]] = + c.fieldAs[Option[String]]("hiddenImageUrl").map(_.map(hiu => s"http://example.com/series/$hiu")) + override val selectElaborator = SelectElaborator { case (QueryType, "films", Nil) => Elab.transformChild(child => Filter(Eql[EntityType](FilmType / "entityType", Const(EntityType.Film)), child)) diff --git a/modules/sql/shared/src/test/scala/SqlInterfacesMapping2.scala b/modules/sql/shared/src/test/scala/SqlInterfacesMapping2.scala index d51f7337..891eb7bc 100644 --- a/modules/sql/shared/src/test/scala/SqlInterfacesMapping2.scala +++ b/modules/sql/shared/src/test/scala/SqlInterfacesMapping2.scala @@ -16,6 +16,7 @@ package grackle.sql.test import grackle._ +import grackle.syntax._ import Predicate._ trait SqlInterfacesMapping2[F[_]] extends SqlInterfacesMapping[F] { self => @@ -24,19 +25,18 @@ trait SqlInterfacesMapping2[F[_]] extends SqlInterfacesMapping[F] { self => // discriminator always fails def discriminate(c: Cursor): Result[Type] = - Result.failure("no") + Result.internalError("Boom!!!") // same as in SqlInterfacesMapping - def narrowPredicate(subtpe: Type): Option[Predicate] = { - def mkPredicate(tpe: EntityType): Option[Predicate] = - Some(Eql(EType / "entityType", Const(tpe))) + def narrowPredicate(subtpe: Type): Result[Predicate] = { + def mkPredicate(tpe: EntityType): Result[Predicate] = + Eql(EType / "entityType", Const(tpe)).success subtpe match { case FilmType => mkPredicate(EntityType.Film) case SeriesType => mkPredicate(EntityType.Series) - case _ => None + case _ => Result.internalError(s"Invalid discriminator: $subtpe") } } } - } diff --git a/modules/sql/shared/src/test/scala/SqlInterfacesSuite.scala b/modules/sql/shared/src/test/scala/SqlInterfacesSuite.scala index ed8dde6b..27ef3f35 100644 --- a/modules/sql/shared/src/test/scala/SqlInterfacesSuite.scala +++ b/modules/sql/shared/src/test/scala/SqlInterfacesSuite.scala @@ -805,4 +805,553 @@ trait SqlInterfacesSuite extends CatsEffectSuite { assertWeaklyEqualIO(res, expected) } + + test("interface query with polymorphic cursor field (1)") { + val query = """ + query { + entities { + id + ... on Film { + imageUrl + } + ... on Series { + imageUrl + } + } + } + """ + + val expected = json""" + { + "data" : { + "entities" : [ + { + "id" : "4", + "imageUrl" : "http://example.com/series/hidden_series1.jpg" + }, + { + "id" : "5", + "imageUrl" : "http://example.com/series/hidden_series2.jpg" + }, + { + "id" : "2", + "imageUrl" : "http://www.example.com/film2.jpg" + }, + { + "id" : "3", + "imageUrl" : "http://www.example.com/film3.jpg" + }, + { + "id" : "6", + "imageUrl" : "http://example.com/series/hidden_series3.jpg" + }, + { + "id" : "1", + "imageUrl" : "http://www.example.com/film1.jpg" + } + ] + } + } + """ + + val res = mapping.compileAndRun(query) + + assertWeaklyEqualIO(res, expected) + } + + test("interface query with polymorphic cursor field (2)") { + val query = """ + query { + entities { + id + imageUrl + } + } + """ + + val expected = json""" + { + "data" : { + "entities" : [ + { + "id" : "4", + "imageUrl" : "http://example.com/series/hidden_series1.jpg" + }, + { + "id" : "5", + "imageUrl" : "http://example.com/series/hidden_series2.jpg" + }, + { + "id" : "2", + "imageUrl" : "http://www.example.com/film2.jpg" + }, + { + "id" : "3", + "imageUrl" : "http://www.example.com/film3.jpg" + }, + { + "id" : "6", + "imageUrl" : "http://example.com/series/hidden_series3.jpg" + }, + { + "id" : "1", + "imageUrl" : "http://www.example.com/film1.jpg" + } + ] + } + } + """ + + val res = mapping.compileAndRun(query) + + assertWeaklyEqualIO(res, expected) + } + + test("interface query with polymorphic cursor field (3)") { + val query = """ + query { + entities { + id + imageUrl + ... on Film { + rating + imageUrl + } + } + } + """ + + val expected = json""" + { + "data" : { + "entities" : [ + { + "id" : "4", + "imageUrl" : "http://example.com/series/hidden_series1.jpg" + }, + { + "id" : "5", + "imageUrl" : "http://example.com/series/hidden_series2.jpg" + }, + { + "id" : "2", + "imageUrl" : "http://www.example.com/film2.jpg", + "rating" : "U" + }, + { + "id" : "3", + "imageUrl" : "http://www.example.com/film3.jpg", + "rating" : "15" + }, + { + "id" : "6", + "imageUrl" : "http://example.com/series/hidden_series3.jpg" + }, + { + "id" : "1", + "imageUrl" : "http://www.example.com/film1.jpg", + "rating" : "PG" + } + ] + } + } + """ + + val res = mapping.compileAndRun(query) + + assertWeaklyEqualIO(res, expected) + } + + test("interface query with polymorphic cursor field (4)") { + val query = """ + query { + entities { + id + imageUrl + ... FilmFields + } + } + + fragment FilmFields on Film { + rating + imageUrl + } + """ + + val expected = json""" + { + "data" : { + "entities" : [ + { + "id" : "4", + "imageUrl" : "http://example.com/series/hidden_series1.jpg" + }, + { + "id" : "5", + "imageUrl" : "http://example.com/series/hidden_series2.jpg" + }, + { + "id" : "2", + "imageUrl" : "http://www.example.com/film2.jpg", + "rating" : "U" + }, + { + "id" : "3", + "imageUrl" : "http://www.example.com/film3.jpg", + "rating" : "15" + }, + { + "id" : "6", + "imageUrl" : "http://example.com/series/hidden_series3.jpg" + }, + { + "id" : "1", + "imageUrl" : "http://www.example.com/film1.jpg", + "rating" : "PG" + } + ] + } + } + """ + + val res = mapping.compileAndRun(query) + + assertWeaklyEqualIO(res, expected) + } + + test("interface query with polymorphic cursor field (5)") { + val query = """ + query { + entities { + id + ... on Film { + rating + imageUrl + } + imageUrl + } + } + """ + + val expected = json""" + { + "data" : { + "entities" : [ + { + "id" : "4", + "imageUrl" : "http://example.com/series/hidden_series1.jpg" + }, + { + "id" : "5", + "imageUrl" : "http://example.com/series/hidden_series2.jpg" + }, + { + "id" : "2", + "rating" : "U", + "imageUrl" : "http://www.example.com/film2.jpg" + }, + { + "id" : "3", + "rating" : "15", + "imageUrl" : "http://www.example.com/film3.jpg" + }, + { + "id" : "6", + "imageUrl" : "http://example.com/series/hidden_series3.jpg" + }, + { + "id" : "1", + "rating" : "PG", + "imageUrl" : "http://www.example.com/film1.jpg" + } + ] + } + } + """ + + val res = mapping.compileAndRun(query) + + assertWeaklyEqualIO(res, expected) + } + + test("interface query with polymorphic cursor field (6)") { + val query = """ + query { + entities { + id + ... FilmFields + imageUrl + } + } + + fragment FilmFields on Film { + rating + imageUrl + } + """ + + val expected = json""" + { + "data" : { + "entities" : [ + { + "id" : "4", + "imageUrl" : "http://example.com/series/hidden_series1.jpg" + }, + { + "id" : "5", + "imageUrl" : "http://example.com/series/hidden_series2.jpg" + }, + { + "id" : "2", + "rating" : "U", + "imageUrl" : "http://www.example.com/film2.jpg" + }, + { + "id" : "3", + "rating" : "15", + "imageUrl" : "http://www.example.com/film3.jpg" + }, + { + "id" : "6", + "imageUrl" : "http://example.com/series/hidden_series3.jpg" + }, + { + "id" : "1", + "rating" : "PG", + "imageUrl" : "http://www.example.com/film1.jpg" + } + ] + } + } + """ + + val res = mapping.compileAndRun(query) + + assertWeaklyEqualIO(res, expected) + } + + test("interface query with polymorphic cursor field (7)") { + val query = """ + query { + entities { + id + imageUrl + ... on Series { + label + imageUrl + } + } + } + """ + + val expected = json""" + { + "data" : { + "entities" : [ + { + "id" : "4", + "imageUrl" : "http://example.com/series/hidden_series1.jpg", + "label" : "One" + }, + { + "id" : "5", + "imageUrl" : "http://example.com/series/hidden_series2.jpg", + "label" : "Two" + }, + { + "id" : "2", + "imageUrl" : "http://www.example.com/film2.jpg" + }, + { + "id" : "3", + "imageUrl" : "http://www.example.com/film3.jpg" + }, + { + "id" : "6", + "imageUrl" : "http://example.com/series/hidden_series3.jpg", + "label" : "Three" + }, + { + "id" : "1", + "imageUrl" : "http://www.example.com/film1.jpg" + } + ] + } + } + """ + + val res = mapping.compileAndRun(query) + + assertWeaklyEqualIO(res, expected) + } + + test("interface query with polymorphic cursor field (8)") { + val query = """ + query { + entities { + id + imageUrl + ... SeriesFields + } + } + + fragment SeriesFields on Series { + label + imageUrl + } + """ + + val expected = json""" + { + "data" : { + "entities" : [ + { + "id" : "4", + "imageUrl" : "http://example.com/series/hidden_series1.jpg", + "label" : "One" + }, + { + "id" : "5", + "imageUrl" : "http://example.com/series/hidden_series2.jpg", + "label" : "Two" + }, + { + "id" : "2", + "imageUrl" : "http://www.example.com/film2.jpg" + }, + { + "id" : "3", + "imageUrl" : "http://www.example.com/film3.jpg" + }, + { + "id" : "6", + "imageUrl" : "http://example.com/series/hidden_series3.jpg", + "label" : "Three" + }, + { + "id" : "1", + "imageUrl" : "http://www.example.com/film1.jpg" + } + ] + } + } + """ + + val res = mapping.compileAndRun(query) + + assertWeaklyEqualIO(res, expected) + } + + test("interface query with polymorphic cursor field (9)") { + val query = """ + query { + entities { + id + ... on Series { + label + imageUrl + } + imageUrl + } + } + """ + + val expected = json""" + { + "data" : { + "entities" : [ + { + "id" : "4", + "label" : "One", + "imageUrl" : "http://example.com/series/hidden_series1.jpg" + }, + { + "id" : "5", + "label" : "Two", + "imageUrl" : "http://example.com/series/hidden_series2.jpg" + }, + { + "id" : "2", + "imageUrl" : "http://www.example.com/film2.jpg" + }, + { + "id" : "3", + "imageUrl" : "http://www.example.com/film3.jpg" + }, + { + "id" : "6", + "label" : "Three", + "imageUrl" : "http://example.com/series/hidden_series3.jpg" + }, + { + "id" : "1", + "imageUrl" : "http://www.example.com/film1.jpg" + } + ] + } + } + """ + + val res = mapping.compileAndRun(query) + + assertWeaklyEqualIO(res, expected) + } + + test("interface query with polymorphic cursor field (10)") { + val query = """ + query { + entities { + id + ... SeriesFields + imageUrl + } + } + + fragment SeriesFields on Series { + label + imageUrl + } + """ + + val expected = json""" + { + "data" : { + "entities" : [ + { + "id" : "4", + "label" : "One", + "imageUrl" : "http://example.com/series/hidden_series1.jpg" + }, + { + "id" : "5", + "label" : "Two", + "imageUrl" : "http://example.com/series/hidden_series2.jpg" + }, + { + "id" : "2", + "imageUrl" : "http://www.example.com/film2.jpg" + }, + { + "id" : "3", + "imageUrl" : "http://www.example.com/film3.jpg" + }, + { + "id" : "6", + "label" : "Three", + "imageUrl" : "http://example.com/series/hidden_series3.jpg" + }, + { + "id" : "1", + "imageUrl" : "http://www.example.com/film1.jpg" + } + ] + } + } + """ + + val res = mapping.compileAndRun(query) + + assertWeaklyEqualIO(res, expected) + } } diff --git a/modules/sql/shared/src/test/scala/SqlInterfacesSuite2.scala b/modules/sql/shared/src/test/scala/SqlInterfacesSuite2.scala index 0cc5352e..320244f8 100644 --- a/modules/sql/shared/src/test/scala/SqlInterfacesSuite2.scala +++ b/modules/sql/shared/src/test/scala/SqlInterfacesSuite2.scala @@ -16,17 +16,14 @@ package grackle.sql.test import cats.effect.IO -import io.circe.literal._ import munit.CatsEffectSuite import grackle._ -import grackle.test.GraphQLResponseTests.assertWeaklyEqualIO - trait SqlInterfacesSuite2 extends CatsEffectSuite { def mapping: Mapping[IO] - test("when discriminator fails the fragments should be ignored") { + test("when discriminator fails the entire query should fail") { val query = """ query { entities { @@ -43,47 +40,10 @@ trait SqlInterfacesSuite2 extends CatsEffectSuite { } """ - val expected = json""" - { - "data" : { - "entities" : [ - { - "id" : "1", - "entityType" : "FILM", - "title" : "Film 1" - }, - { - "id" : "2", - "entityType" : "FILM", - "title" : "Film 2" - }, - { - "id" : "3", - "entityType" : "FILM", - "title" : "Film 3" - }, - { - "id" : "4", - "entityType" : "SERIES", - "title" : "Series 1" - }, - { - "id" : "5", - "entityType" : "SERIES", - "title" : "Series 2" - }, - { - "id" : "6", - "entityType" : "SERIES", - "title" : "Series 3" - } - ] - } - } - """ + val expected = Left("Boom!!!") - val res = mapping.compileAndRun(query) + val res = mapping.compileAndRun(query).attempt - assertWeaklyEqualIO(res, expected) + assertIO(res.map(_.left.map(_.getMessage)), expected) } } diff --git a/modules/sql/shared/src/test/scala/SqlMappingValidatorInvalidMapping.scala b/modules/sql/shared/src/test/scala/SqlMappingValidatorInvalidMapping.scala index dd770860..1f95a8ad 100644 --- a/modules/sql/shared/src/test/scala/SqlMappingValidatorInvalidMapping.scala +++ b/modules/sql/shared/src/test/scala/SqlMappingValidatorInvalidMapping.scala @@ -234,8 +234,9 @@ trait SqlMappingValidatorInvalidMapping[F[_]] extends SqlTestMapping[F] { lazy val objectTypeDiscriminator = new SqlDiscriminator { def discriminate(c: Cursor): Result[Type] = - Result.failure("discriminator not implemented") + Result.internalError("discriminator not implemented") - def narrowPredicate(subtpe: Type): Option[Predicate] = None + def narrowPredicate(subtpe: Type): Result[Predicate] = + Result.internalError("discriminator not implemented") } } diff --git a/modules/sql/shared/src/test/scala/SqlMappingValidatorValidMapping.scala b/modules/sql/shared/src/test/scala/SqlMappingValidatorValidMapping.scala index 0b145729..49176164 100644 --- a/modules/sql/shared/src/test/scala/SqlMappingValidatorValidMapping.scala +++ b/modules/sql/shared/src/test/scala/SqlMappingValidatorValidMapping.scala @@ -326,9 +326,10 @@ trait SqlMappingValidatorValidMapping[F[_]] extends SqlTestMapping[F] { lazy val objectTypeDiscriminator = new SqlDiscriminator { def discriminate(c: Cursor): Result[Type] = - Result.failure("discriminator not implemented") + Result.internalError("discriminator not implemented") - def narrowPredicate(subtpe: Type): Option[Predicate] = None + def narrowPredicate(subtpe: Type): Result[Predicate] = + Result.internalError("discriminator not implemented") } sealed trait Genre extends Product with Serializable diff --git a/modules/sql/shared/src/test/scala/SqlRecursiveInterfacesMapping.scala b/modules/sql/shared/src/test/scala/SqlRecursiveInterfacesMapping.scala index 96fde13f..75b04c20 100644 --- a/modules/sql/shared/src/test/scala/SqlRecursiveInterfacesMapping.scala +++ b/modules/sql/shared/src/test/scala/SqlRecursiveInterfacesMapping.scala @@ -111,14 +111,14 @@ trait SqlRecursiveInterfacesMapping[F[_]] extends SqlTestMapping[F] { self => } } - def narrowPredicate(subtpe: Type): Option[Predicate] = { - def mkPredicate(tpe: ItemType): Option[Predicate] = - Some(Eql(IType / "itemType", Const(tpe))) + def narrowPredicate(subtpe: Type): Result[Predicate] = { + def mkPredicate(tpe: ItemType): Result[Predicate] = + Eql(IType / "itemType", Const(tpe)).success subtpe match { case ItemAType => mkPredicate(ItemType.ItemA) case ItemBType => mkPredicate(ItemType.ItemB) - case _ => None + case _ => Result.internalError(s"Invalid discriminator: $subtpe") } } } diff --git a/modules/sql/shared/src/test/scala/SqlUnionsMapping.scala b/modules/sql/shared/src/test/scala/SqlUnionsMapping.scala index f57bb766..9b351636 100644 --- a/modules/sql/shared/src/test/scala/SqlUnionsMapping.scala +++ b/modules/sql/shared/src/test/scala/SqlUnionsMapping.scala @@ -92,14 +92,14 @@ trait SqlUnionsMapping[F[_]] extends SqlTestMapping[F] { case "ItemB" => ItemBType } - def narrowPredicate(subtpe: Type): Option[Predicate] = { - def mkPredicate(tpe: String): Option[Predicate] = - Some(Eql(ItemType / "itemType", Const(tpe))) + def narrowPredicate(subtpe: Type): Result[Predicate] = { + def mkPredicate(tpe: String): Result[Predicate] = + Eql(ItemType / "itemType", Const(tpe)).success subtpe match { case ItemAType => mkPredicate("ItemA") case ItemBType => mkPredicate("ItemB") - case _ => None + case _ => Result.internalError(s"Invalid discriminator: $subtpe") } } }