From 50983f60b33fa12d5978f07cb1d6da32b971114d Mon Sep 17 00:00:00 2001 From: Miles Sabin Date: Wed, 5 Jun 2024 11:43:42 +0100 Subject: [PATCH] Rework interfaces and implementations + Object type fields mapped at the interface level and interface type fields which are implemented or overridden at the object type level are now explicitly represented internally. This allows both more efficient lookup of inherited field mappings and correct lookup of overriden field mappings. + Field mapping lookup is now more effectively indexed TypeMappings. This might give a noticeable performance improvement for ValueMapping. Now that this indexing in centralised in TypeMappings, the per-ObjectMapping field indices have been removed. If this proves problematic for applications it could be reinstated. + Schema validation now enforces the uniqueness of interfaces in implements clauses. + Schema validation now enforces that object and interface types must directly implement all transitively implemented interfaces. The allInterfaces method on InterfaceType has been deprecated because with the preceeding validation change it would equivalent to interfaces. + The Mapping specific logic of mkCursorForField has been extracted to mkCursorForMappedField allowing simpler mapping-specific implementations. + Previously introspection did not report interfaces implemented by interfaces. + Added Schema#implementations which returns the implementing Object types of an interface. + The unsafe TypeMappings constructor has been deprecated and renamed to unchecked. + TypeMappings#unsafe has been renamed to unchecked and hidden + The implementations of hasField, nullableHasField, hasPath and hasListPath in Cursor had incorrect semantics and appear to be unused, so rather than fix them, they have been removed. + Various tests have been updated to conform to the newly implemented validation rules and changes to field mapping lookup. --- build.sbt | 2 +- .../circe/src/main/scala/circemapping.scala | 17 +- .../core/src/main/scala/composedmapping.scala | 15 +- modules/core/src/main/scala/cursor.scala | 56 --- .../core/src/main/scala/introspection.scala | 7 +- modules/core/src/main/scala/mapping.scala | 391 +++++++++++++----- .../src/main/scala/queryinterpreter.scala | 4 +- modules/core/src/main/scala/schema.scala | 111 +++-- .../core/src/main/scala/valuemapping.scala | 14 +- .../scala/extensions/ExtensionsSuite.scala | 7 +- .../introspection/IntrospectionSuite.scala | 6 +- .../scala/mapping/MappingValidatorSuite.scala | 16 +- .../src/test/scala/schema/SchemaSuite.scala | 101 ++++- .../test/scala/starwars/StarWarsData.scala | 10 + .../test/scala/starwars/StarWarsSuite.scala | 226 ++++++++++ .../src/main/scala-2/genericmapping2.scala | 5 - .../src/main/scala-3/genericmapping3.scala | 5 - .../src/main/scala/genericmapping.scala | 11 +- .../shared/src/main/scala/SqlMapping.scala | 54 +-- .../SqlMappingValidatorInvalidMapping.scala | 2 +- .../SqlMappingValidatorInvalidSuite.scala | 12 +- .../SqlMappingValidatorValidMapping.scala | 2 +- 22 files changed, 753 insertions(+), 321 deletions(-) diff --git a/build.sbt b/build.sbt index db5c9fb7..e69ab4d7 100644 --- a/build.sbt +++ b/build.sbt @@ -31,7 +31,7 @@ ThisBuild / scalaVersion := Scala2 ThisBuild / crossScalaVersions := Seq(Scala2, Scala3) ThisBuild / tlJdkRelease := Some(11) -ThisBuild / tlBaseVersion := "0.19" +ThisBuild / tlBaseVersion := "0.20" 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 98f430be..af41e522 100644 --- a/modules/circe/src/main/scala/circemapping.scala +++ b/modules/circe/src/main/scala/circemapping.scala @@ -56,18 +56,15 @@ trait CirceMappingLike[F[_]] extends Mapping[F] { else DeferredCursor(path, (context, parent) => CirceCursor(context, value, Some(parent), env).success) - override def mkCursorForField(parent: Cursor, fieldName: String, resultName: Option[String]): Result[Cursor] = { - val context = parent.context - val fieldContext = context.forFieldOrAttribute(fieldName, resultName) - (typeMappings.fieldMapping(context, fieldName), parent.focus) match { - case (Some(CirceField(_, json, _)), _) => + override def mkCursorForMappedField(parent: Cursor, fieldContext: Context, fm: FieldMapping): Result[Cursor] = + (fm, parent.focus) match { + case (CirceField(_, json, _), _) => CirceCursor(fieldContext, json, Some(parent), parent.env).success - case (Some(CursorFieldJson(_, f, _, _)), _) => + case (CursorFieldJson(_, f, _, _), _) => f(parent).map(res => CirceCursor(fieldContext, focus = res, parent = Some(parent), env = parent.env)) case _ => - super.mkCursorForField(parent, fieldName, resultName) + super.mkCursorForMappedField(parent, fieldContext, fm) } - } sealed trait CirceFieldMapping extends FieldMapping { def subtree: Boolean = true @@ -172,10 +169,6 @@ trait CirceMappingLike[F[_]] extends Mapping[F] { else Result.internalError(s"Focus ${focus} of static type $tpe cannot be narrowed to $subtpe") - def hasField(fieldName: String): Boolean = - tpe.hasField(fieldName) && focus.asObject.exists(_.contains(fieldName)) || - typeMappings.fieldMapping(context, fieldName).isDefined - def field(fieldName: String, resultName: Option[String]): Result[Cursor] = { val localField = for { diff --git a/modules/core/src/main/scala/composedmapping.scala b/modules/core/src/main/scala/composedmapping.scala index 4c53bcb1..2226de43 100644 --- a/modules/core/src/main/scala/composedmapping.scala +++ b/modules/core/src/main/scala/composedmapping.scala @@ -21,16 +21,8 @@ import Cursor.AbstractCursor import syntax._ abstract class ComposedMapping[F[_]](implicit val M: MonadThrow[F]) extends Mapping[F] { - override def mkCursorForField(parent: Cursor, fieldName: String, resultName: Option[String]): Result[Cursor] = { - val context = parent.context - val fieldContext = context.forFieldOrAttribute(fieldName, resultName) - typeMappings.fieldMapping(context, fieldName) match { - case Some(_) => - ComposedCursor(fieldContext, parent.env).success - case _ => - super.mkCursorForField(parent, fieldName, resultName) - } - } + override def mkCursorForMappedField(parent: Cursor, fieldContext: Context, fm: FieldMapping): Result[Cursor] = + ComposedCursor(fieldContext, parent.env).success case class ComposedCursor(context: Context, env: Env) extends AbstractCursor { val focus = null @@ -38,9 +30,6 @@ abstract class ComposedMapping[F[_]](implicit val M: MonadThrow[F]) extends Mapp def withEnv(env0: Env): Cursor = copy(env = env.add(env0)) - override def hasField(fieldName: String): Boolean = - typeMappings.fieldMapping(context, fieldName).isDefined - override def field(fieldName: String, resultName: Option[String]): Result[Cursor] = mkCursorForField(this, fieldName, resultName) } diff --git a/modules/core/src/main/scala/cursor.scala b/modules/core/src/main/scala/cursor.scala index 689dc691..ca0b48aa 100644 --- a/modules/core/src/main/scala/cursor.scala +++ b/modules/core/src/main/scala/cursor.scala @@ -137,9 +137,6 @@ trait Cursor { */ def narrow(subtpe: TypeRef): Result[Cursor] - /** Does the value at this `Cursor` have a field named `fieldName`? */ - def hasField(fieldName: String): Boolean - /** * Yield a `Cursor` corresponding to the value of the field `fieldName` of the * value at this `Cursor`, or an error on the left hand side if there is no @@ -161,18 +158,6 @@ trait Cursor { case _ => false }) - /** - * Does the possibly nullable value at this `Cursor` have a field named - * `fieldName`? - */ - def nullableHasField(fieldName: String): Boolean = - if (isNullable) - asNullable match { - case Result.Success(Some(c)) => c.nullableHasField(fieldName) - case _ => false - } - else hasField(fieldName) - /** * Yield a `Cursor` corresponding to the value of the possibly nullable field * `fieldName` of the value at this `Cursor`, or an error on the left hand @@ -186,19 +171,6 @@ trait Cursor { } else field(fieldName, None) - /** Does the value at this `Cursor` have a field identified by the path `fns`? */ - def hasPath(fns: List[String]): Boolean = fns match { - case Nil => true - case fieldName :: rest => - nullableHasField(fieldName) && { - nullableField(fieldName) match { - case Result.Success(c) => - !c.isList && c.hasPath(rest) - case _ => false - } - } - } - /** * Yield a `Cursor` corresponding to the value of the field identified by path * `fns` starting from the value at this `Cursor`, or an error on the left @@ -213,28 +185,6 @@ trait Cursor { } } - /** - * Does the value at this `Cursor` generate a list along the path `fns`? - * - * `true` if `fns` is a valid path from the value at this `Cursor` and passes - * through at least one field with a list type. - */ - def hasListPath(fns: List[String]): Boolean = { - def loop(c: Cursor, fns: List[String], seenList: Boolean): Boolean = fns match { - case Nil => seenList - case fieldName :: rest => - c.nullableHasField(fieldName) && { - c.nullableField(fieldName) match { - case Result.Success(c) => - loop(c, rest, c.isList) - case _ => false - } - } - } - - loop(this, fns, false) - } - /** * Yield a list of `Cursor`s corresponding to the values generated by * following the path `fns` from the value at this `Cursor`, or an error on @@ -306,8 +256,6 @@ object Cursor { def narrow(subtpe: TypeRef): Result[Cursor] = Result.internalError(s"Focus ${focus} of static type $tpe cannot be narrowed to $subtpe") - def hasField(fieldName: String): Boolean = false - def field(fieldName: String, resultName: Option[String]): Result[Cursor] = Result.internalError(s"No field '$fieldName' for type ${tpe.underlying}") } @@ -346,8 +294,6 @@ object Cursor { def narrow(subtpe: TypeRef): Result[Cursor] = underlying.narrow(subtpe) - def hasField(fieldName: String): Boolean = underlying.hasField(fieldName) - def field(fieldName: String, resultName: Option[String]): Result[Cursor] = underlying.field(fieldName, resultName) } @@ -387,8 +333,6 @@ object Cursor { def focus: Any = Result.internalError(s"Empty cursor has no focus") def withEnv(env0: Env): DeferredCursor = copy(env = env.add(env0)) - 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.underlying}") else diff --git a/modules/core/src/main/scala/introspection.scala b/modules/core/src/main/scala/introspection.scala index b7de8283..0f7e6b61 100644 --- a/modules/core/src/main/scala/introspection.scala +++ b/modules/core/src/main/scala/introspection.scala @@ -224,15 +224,12 @@ object Introspection { case _ => None }), ValueField("interfaces", flipNullityDealias andThen { - case ot: ObjectType => Some(ot.interfaces.map(_.nullable)) + case tf: TypeWithFields => Some(tf.interfaces.map(_.nullable)) case _ => None }), ValueField("possibleTypes", flipNullityDealias andThen { case u: UnionType => Some(u.members.map(_.nullable)) - case i: InterfaceType => - Some(allTypes.collect { - case o: ObjectType if o.interfaces.exists(_ =:= i) => NullableType(o) - }) + case i: InterfaceType => Some(targetSchema.implementations(i).map(_.nullable)) case _ => None }), ValueField("enumValues", flipNullityDealias andThen { diff --git a/modules/core/src/main/scala/mapping.scala b/modules/core/src/main/scala/mapping.scala index d52c46f4..884718b6 100644 --- a/modules/core/src/main/scala/mapping.scala +++ b/modules/core/src/main/scala/mapping.scala @@ -104,84 +104,75 @@ abstract class Mapping[F[_]] { def focus: Any = () - override def hasField(fieldName: String): Boolean = - typeMappings.fieldMapping(context, fieldName).isDefined - override def field(fieldName: String, resultName: Option[String]): Result[Cursor] = mkCursorForField(this, fieldName, resultName) } /** * Yields a `Cursor` suitable for traversing the query result corresponding to - * the `fieldName` child of `parent`. + * the `fieldName` child of `parent`, and with the given field `Context` and + * `FieldMapping`. * * This method is typically overridden in and delegated to by `Mapping` subtypes. */ - def mkCursorForField(parent: Cursor, fieldName: String, resultName: Option[String]): Result[Cursor] = { - val context = parent.context - val fieldContext = context.forFieldOrAttribute(fieldName, resultName) - + protected def mkCursorForMappedField(parent: Cursor, fieldContext: Context, fm: FieldMapping): Result[Cursor] = { def mkLeafCursor(focus: Any): Result[Cursor] = LeafCursor(fieldContext, focus, Some(parent), parent.env).success - typeMappings.fieldMapping(context, fieldName) match { - case Some(_ : EffectMapping) => + fm match { + case _ : EffectMapping => mkLeafCursor(parent.focus) - case Some(CursorField(_, f, _, _, _)) => + case CursorField(_, f, _, _, _) => f(parent).flatMap(res => mkLeafCursor(res)) case _ => - Result.failure(s"No field '$fieldName' for type ${parent.tpe}") + Result.internalError(s"Unhandled mapping of type ${fm.getClass.getName} for field '${fieldContext.path.head}' for type ${parent.tpe}") } } - case class TypeMappings private (mappings: Seq[TypeMapping], unsafe: Boolean) { + /** + * Yields a `Cursor` suitable for traversing the query result corresponding to + * 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, _)) + } + + final class TypeMappings private ( + val mappings: Seq[TypeMapping], + typeIndex: MMap[String, TypeMapping], + predicatedTypeIndex: MMap[String, Seq[TypeMapping]], + typeFieldIndex: MMap[String, MMap[String, FieldMapping]], + predicatedTypeFieldIndex: MMap[String, Seq[(ObjectMapping, MMap[String, FieldMapping])]], + unchecked: Boolean + ) { + import TypeMappings.{InheritedFieldMapping, PolymorphicFieldMapping} + /** Yields the `TypeMapping` associated with the provided context, if any. */ def typeMapping(context: Context): Option[TypeMapping] = { val nt = context.tpe.underlyingNamed val nme = nt.name - singleIndex.get(nme).orElse { + typeIndex.get(nme).orElse { val nc = context.asType(nt) - multipleIndex.get(nme).getOrElse(Nil).mapFilter { tm => + predicatedTypeIndex.getOrElse(nme, Nil).mapFilter { tm => tm.predicate(nc).map(prio => (prio, tm)) }.maxByOption(_._1).map(_._2) } } - private val (singleIndex, multipleIndex): (MMap[String, TypeMapping], MMap[String, Seq[TypeMapping]]) = { - val defaultLeafMappings: Seq[TypeMapping] = { - val intTypeEncoder: Encoder[Any] = - new Encoder[Any] { - def apply(i: Any): Json = (i: @unchecked) match { - case i: Int => Json.fromInt(i) - case l: Long => Json.fromLong(l) - } - } - - val floatTypeEncoder: Encoder[Any] = - new Encoder[Any] { - def apply(f: Any): Json = (f: @unchecked) match { - case f: Float => Json.fromFloatOrString(f) - case d: Double => Json.fromDoubleOrString(d) - case d: BigDecimal => Json.fromBigDecimal(d) - } - } - - Seq( - LeafMapping[String](ScalarType.StringType), - LeafMapping.DefaultLeafMapping[Any](MappingPredicate.TypeMatch(ScalarType.IntType), intTypeEncoder, typeName[Int]), - LeafMapping.DefaultLeafMapping[Any](MappingPredicate.TypeMatch(ScalarType.FloatType), floatTypeEncoder, typeName[Float]), - LeafMapping[Boolean](ScalarType.BooleanType), - LeafMapping[String](ScalarType.IDType) - ) + private def fieldIndex(context: Context): Option[MMap[String, FieldMapping]] = { + val nt = context.tpe.underlyingNamed + val nme = nt.name + typeFieldIndex.get(nme).orElse { + val nc = context.asType(nt) + predicatedTypeFieldIndex.getOrElse(nme, Nil).mapFilter { tm => + tm._1.predicate(nc).map(prio => (prio, tm)) + }.maxByOption(_._1).map(_._2._2) } - - val grouped = (mappings ++ defaultLeafMappings).groupBy(_.tpe.underlyingNamed.name) - val (single, multiple) = - grouped.partitionMap { - case (nme, tms) if tms.sizeCompare(1) == 0 => Left(nme -> tms.head) - case (nme, tms) => Right(nme -> tms) - } - (MMap.from(single), MMap.from(multiple)) } /** Yields the `ObjectMapping` associated with the provided context, if any. */ @@ -192,13 +183,30 @@ abstract class Mapping[F[_]] { /** 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 { - case Some(ot: ObjectType) => - ot.interfaces.collectFirstSome(nt => fieldMapping(context.asType(nt), fieldName)) - case _ => None - } + fieldIndex(context).flatMap(_.get(fieldName)).flatMap { + case ifm: InheritedFieldMapping => + ifm.select(context) + case pfm: PolymorphicFieldMapping => + pfm.select(context) + case fm => + Some(fm) + } + + /** + * 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] = { + val context = parent.context + fieldIndex(context).flatMap(_.get(fieldName)).flatMap { + case ifm: InheritedFieldMapping => + ifm.select(parent.context) + case pfm: PolymorphicFieldMapping => + pfm.select(parent) + case fm => + Some(fm) } + } /** Yields the `FieldMapping` directly or ancestrally associated with `fieldName` in `context`, if any. */ def ancestralFieldMapping(context: Context, fieldName: String): Option[FieldMapping] = @@ -210,23 +218,35 @@ abstract class Mapping[F[_]] { } yield fm } - /** Yields the `ObjectMapping` and `FieldMapping` associated with `fieldName` in `context`, if any. */ - def objectAndFieldMapping(context: Context, fieldName: String): Option[(ObjectMapping, FieldMapping)] = - objectMapping(context).flatMap(om => om.fieldMapping(fieldName).map(fm => (om, fm))).orElse { - context.tpe.underlyingObject match { - case Some(ot: ObjectType) => - ot.interfaces.collectFirstSome(nt => objectAndFieldMapping(context.asType(nt), fieldName)) - case _ => None - } - } - - /** Validates these type mappings against an unfolding of the schema */ - def validate: List[ValidationFailure] = { + /** + * Validatate this Mapping, yielding a list of `ValidationFailure`s of severity equal to or greater than the + * specified `Severity`. + */ + def validate(severity: Severity = Severity.Warning): List[ValidationFailure] = { val queryType = schema.schemaType.field("query").flatMap(_.nonNull.asNamed) val topLevelContexts = (queryType.toList ::: schema.mutationType.toList ::: schema.subscriptionType.toList).map(Context(_)) - validateRoots(topLevelContexts) + validateRoots(topLevelContexts).filter(_.severity >= severity) } + /** + * Validate this mapping, raising a `ValidationException` in `F` if there are any failures of + * severity equal to or greater than the specified `Severity`. + */ + def validateInto[G[_]](severity: Severity = Severity.Warning)( + implicit ev: ApplicativeError[G, Throwable] + ): G[Unit] = + NonEmptyList.fromList(validate(severity)).foldMapA(nec => ev.raiseError(ValidationException(nec))) + + /** + * Validate this Mapping, throwing a `ValidationException` if there are any failures of severity equal + * to or greater than the specified `Severity`. + */ + def unsafeValidate(severity: Severity = Severity.Warning): Unit = + validateInto[Either[Throwable, *]](severity).fold(throw _, _ => ()) + + private[Mapping] def unsafeValidateIfChecked(): Unit = + if(!unchecked) unsafeValidate() + /** Validates these type mappings against an unfolding of the schema */ @nowarn3 private def validateRoots(rootCtxts: List[Context]): List[ValidationFailure] = { @@ -239,10 +259,10 @@ abstract class Mapping[F[_]] { def allTypeMappings(context: Context): Seq[TypeMapping] = { val nt = context.tpe.underlyingNamed val nme = nt.name - singleIndex.get(nme) match { + typeIndex.get(nme) match { case Some(tm) => List(tm) case None => - multipleIndex.get(nme) match { + predicatedTypeIndex.get(nme) match { case None => Nil case Some(tms) => val nc = context.asType(nt) @@ -332,9 +352,7 @@ abstract class Mapping[F[_]] { val implCtxts = twf match { case it: InterfaceType => - schema.types.collect { - case ot: ObjectType if ot <:< it => context.asType(ot) - } + schema.implementations(it).map(context.asType) case _ => Nil } @@ -349,11 +367,19 @@ abstract class Mapping[F[_]] { def fieldCheck(fieldName: String): MV[Option[Context]] = { val fctx = context.forFieldOrAttribute(fieldName, None).forUnderlyingNamed + lazy val allImplsHaveFieldMapping = + implCtxts.nonEmpty && + implCtxts.forall { implCtxt => + objectMapping(implCtxt).exists { om => + om.fieldMapping(fieldName).isDefined + } + } + ((ancestralFieldMapping(context, fieldName), tms) match { - case (Some(fm), List(om: ObjectMapping)) => + case (Some(fm), List(om: ObjectMapping)) if !allImplsHaveFieldMapping => addSeenFieldMapping(om, fm) - case (None, List(om: ObjectMapping)) if !hasEnclosingSubtreeFieldMapping => + case (None, List(om: ObjectMapping)) if !(hasEnclosingSubtreeFieldMapping || allImplsHaveFieldMapping) => val field = context.tpe.fieldInfo(fieldName).get addProblem(MissingFieldMapping(om, field)) @@ -367,7 +393,7 @@ abstract class Mapping[F[_]] { seen <- seenType(twf) _ <- addSeenType(twf) _ <- objectCheck(seen) - ifCtxts <- twf.allInterfaces.traverseFilter(interfaceContext) + ifCtxts <- twf.interfaces.traverseFilter(interfaceContext) fCtxts <- fieldNames.traverseFilter(fieldCheck) pfCtxts <- MV.pure(if (!seen) allPrefixedMatchContexts(context) else Nil) } yield implCtxts ++ ifCtxts ++ fCtxts ++ pfCtxts @@ -431,7 +457,7 @@ abstract class Mapping[F[_]] { case om: ObjectMapping if !om.tpe.dealias.isUnion => for { sfms <- seenFieldMappings(om) - usfms = om.fieldMappings.filterNot { case _: Delegate => true ; case fm => fm.hidden || sfms(fm) } + usfms = om.fieldMappings.filterNot { case (_: Delegate) => true ; case fm => fm.hidden || sfms(fm) } _ <- usfms.traverse_(fm => addProblem(UnusedFieldMapping(om, fm))) } yield () case _ => @@ -453,21 +479,190 @@ abstract class Mapping[F[_]] { } object TypeMappings { + private val builtinLeafMappings: Seq[TypeMapping] = { + val intTypeEncoder: Encoder[Any] = + new Encoder[Any] { + def apply(i: Any): Json = (i: @unchecked) match { + case i: Int => Json.fromInt(i) + case l: Long => Json.fromLong(l) + } + } + + val floatTypeEncoder: Encoder[Any] = + new Encoder[Any] { + def apply(f: Any): Json = (f: @unchecked) match { + case f: Float => Json.fromFloatOrString(f) + case d: Double => Json.fromDoubleOrString(d) + case d: BigDecimal => Json.fromBigDecimal(d) + } + } + + Seq( + LeafMapping[String](ScalarType.StringType), + LeafMapping.DefaultLeafMapping[Any](MappingPredicate.TypeMatch(ScalarType.IntType), intTypeEncoder, typeName[Int]), + LeafMapping.DefaultLeafMapping[Any](MappingPredicate.TypeMatch(ScalarType.FloatType), floatTypeEncoder, typeName[Float]), + LeafMapping[Boolean](ScalarType.BooleanType), + LeafMapping[String](ScalarType.IDType) + ) + } + + /** A synthetic field mapping representing a field mapped in an implemented interface */ + private case class InheritedFieldMapping(candidates: Seq[(MappingPredicate, FieldMapping)])(implicit val pos: SourcePos) extends FieldMapping { + def fieldName: String = candidates.head._2.fieldName + def hidden: Boolean = false + def subtree: Boolean = false + + def select(context: Context): Option[FieldMapping] = { + val applicable = + candidates.mapFilter { case (pred, fm) => + pred(context.asType(pred.tpe)).map(prio => (prio, fm)) + } + applicable.maxByOption(_._1).map(_._2) + } + } + + /** A synthetic field mapping representing a field mapped by one or more implementing types */ + private case class PolymorphicFieldMapping(candidates: Seq[(MappingPredicate, FieldMapping)])(implicit val pos: SourcePos) extends FieldMapping { + def fieldName: String = candidates.head._2.fieldName + def hidden: Boolean = false + def subtree: Boolean = false + + def select(cursor: Cursor): Option[FieldMapping] = { + val context = cursor.context + 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 + } + applicable.maxByOption(_._1).map(_._2) + } + + def select(context: Context): Option[FieldMapping] = { + val applicable = + candidates.mapFilter { case (pred, fm) => + pred(context).map(prio => (prio, fm)) + } + applicable.maxByOption(_._1).map(_._2) + } + } + + private def mkMappings(mappings: Seq[TypeMapping], unchecked: Boolean): TypeMappings = { + val groupedMappings = (mappings ++ builtinLeafMappings).groupBy(_.tpe.underlyingNamed.name) + val (typeIndex0, predicatedTypeIndex0) = + groupedMappings.partitionMap { + case (nme, tms) if tms.sizeCompare(1) == 0 => Left(nme -> tms.head) + case (nme, tms) => Right(nme -> tms) + } + + val typeIndex = MMap.from(typeIndex0) + val predicatedTypeIndex = MMap.from(predicatedTypeIndex0) + + val objectFieldIndices = mappings.collect { + case om: ObjectMapping if om.tpe.dealias.isObject => + om.tpe.underlying match { + case o: ObjectType if o.interfaces.nonEmpty && !o.fields.forall(f => om.fieldMapping(f.name).isDefined) => + val ims = o.interfaces.flatMap { i => typeIndex.get(i.name).collect { case im: ObjectMapping => im } } + val attrs = om.fieldMappings.filterNot(fm => o.hasField(fm.fieldName)) + val newFields = + o.fields.flatMap { f => + om.fieldMapping(f.name).map(Seq(_)).getOrElse { + val candidates = + ims.flatMap { im => + im.fieldMapping(f.name).map { fm => (im.predicate, fm) } + } + + if (candidates.isEmpty) Seq.empty + else Seq(InheritedFieldMapping(candidates)) + } + } + val index = MMap.from((newFields ++ attrs).map(fm => fm.fieldName -> fm)) + (om, index) + + case _ => + val index = MMap.from(om.fieldMappings.map(fm => fm.fieldName -> fm)) + (om, index) + } + }.groupBy(_._1.tpe.underlyingNamed.name) + + val nonObjectFieldIndices = + mappings.collect { + case im: ObjectMapping if !im.tpe.dealias.isObject => + im.tpe.underlying match { + case i: InterfaceType => + val impls = schema.implementations(i) + val ms = impls.flatMap(impl => objectFieldIndices.getOrElse(impl.name, Seq.empty)) + val attrs = im.fieldMappings.filterNot(fm => i.hasField(fm.fieldName)) + val newFields = + i.fields.flatMap { f => + val candidates: Seq[(MappingPredicate, FieldMapping)] = + ms.flatMap { case (om, ofi) => + ofi.get(f.name).toSeq.flatMap { + case InheritedFieldMapping(ifms) => ifms.filterNot { case (p, _) => p.tpe =:= i } + case fm => Seq((om.predicate, fm)) + } + } + + val dfm = im.fieldMapping(f.name).toSeq + if (candidates.isEmpty) dfm + else { + val dfm0 = dfm.map(ifm => (im.predicate, ifm)) + val (tps, ifs) = + (dfm0 ++ candidates).partitionMap { + case pfm@(p, _) => + if (p.tpe.dealias.isInterface) Right(pfm) + else Left(pfm) + } + val cands = tps ++ ifs + Seq(PolymorphicFieldMapping(cands)) + } + } + + val index = MMap.from((newFields ++ attrs).map(fm => fm.fieldName -> fm)) + (im, index) + + case _ => + val index = MMap.from(im.fieldMappings.map(fm => fm.fieldName -> fm)) + (im, index) + } + }.groupBy(_._1.tpe.underlyingNamed.name) + + val (typeFieldIndex0, predicatedTypeFieldIndex0) = + (objectFieldIndices ++ nonObjectFieldIndices).partitionMap { + case (nme, tms) if tms.sizeCompare(1) == 0 => Left(nme -> tms.head._2) + case (nme, tms) => Right(nme -> tms) + } + + val typeFieldIndex = MMap.from(typeFieldIndex0) + val predicatedTypeFieldIndex = MMap.from(predicatedTypeFieldIndex0) + + new TypeMappings(mappings, typeIndex, predicatedTypeIndex, typeFieldIndex, predicatedTypeFieldIndex, unchecked) + } + def apply(mappings: Seq[TypeMapping]): TypeMappings = - new TypeMappings(mappings, false) + mkMappings(mappings, false) def apply(mappings: TypeMapping*)(implicit dummy: DummyImplicit): TypeMappings = apply(mappings) + def unchecked(mappings: Seq[TypeMapping]): TypeMappings = + mkMappings(mappings, true) + + def unchecked(mappings: TypeMapping*)(implicit dummy: DummyImplicit): TypeMappings = + unchecked(mappings) + + @deprecated("Use `unchecked` instead", "0.20.0") def unsafe(mappings: Seq[TypeMapping]): TypeMappings = - new TypeMappings(mappings, true) + unchecked(mappings) + @deprecated("Use `unchecked` instead", "0.20.0") def unsafe(mappings: TypeMapping*)(implicit dummy: DummyImplicit): TypeMappings = - unsafe(mappings) + unchecked(mappings) implicit def fromList(mappings: List[TypeMappingCompat]): TypeMappings = TypeMappings(mappings.flatMap(_.unwrap)) - val empty: TypeMappings = unsafe(Nil) + val empty: TypeMappings = unchecked(Nil) private type MappingValidator[T] = StateT[Id, MappingValidator.State, T] private object MappingValidator { @@ -530,25 +725,23 @@ abstract class Mapping[F[_]] { * Validatate this Mapping, yielding a chain of `Failure`s of severity equal to or greater than the * specified `Severity`. */ - def validate(severity: Severity = Severity.Warning): List[ValidationFailure] = { - typeMappings.validate.filter(_.severity >= severity) - } + def validate(severity: Severity = Severity.Warning): List[ValidationFailure] = + typeMappings.validate(severity) /** - * Run this validator, raising a `ValidationException` in `F` if there are any failures of + * Validate this mapping, raising a `ValidationException` in `F` if there are any failures of * severity equal to or greater than the specified `Severity`. */ def validateInto[G[_]](severity: Severity = Severity.Warning)( implicit ev: ApplicativeError[G, Throwable] - ): G[Unit] = - NonEmptyList.fromList(validate(severity)).foldMapA(nec => ev.raiseError(ValidationException(nec))) + ): G[Unit] = typeMappings.validateInto(severity)(ev) /** - * Run this validator, raising a `ValidationException` if there are any failures of severity equal + * Validate this Mapping, throwing a `ValidationException` if there are any failures of severity equal * to or greater than the specified `Severity`. */ def unsafeValidate(severity: Severity = Severity.Warning): Unit = - validateInto[Either[Throwable, *]](severity).fold(throw _, _ => ()) + typeMappings.unsafeValidate(severity) /** Yields the `RootEffect`, if any, associated with `fieldName`. */ def rootEffect(context: Context, fieldName: String): Option[RootEffect] = @@ -646,8 +839,7 @@ abstract class Mapping[F[_]] { case Some(_) => extendContext(ctx.forFieldOrAttribute(path.head, None), path.tail) case None => - val implementors = schema.types.collect { case ot: ObjectType if ot <:< it => ot } - implementors.collectFirstSome(tpe => extendContext(ctx.asType(tpe), path)) + schema.implementations(it).collectFirstSome(tpe => extendContext(ctx.asType(tpe), path)) } case ut: UnionType => ut.members.collectFirstSome(tpe => extendContext(ctx.asType(tpe), path)) @@ -744,10 +936,9 @@ abstract class Mapping[F[_]] { } abstract class ObjectMapping extends TypeMapping { - private lazy val fieldMappingIndex = fieldMappings.map(fm => (fm.fieldName, fm)).toMap - def fieldMappings: Seq[FieldMapping] - def fieldMapping(fieldName: String): Option[FieldMapping] = fieldMappingIndex.get(fieldName) + def fieldMapping(fieldName: String): Option[FieldMapping] = + fieldMappings.find(_.fieldName == fieldName) } object ObjectMapping { @@ -983,7 +1174,7 @@ abstract class Mapping[F[_]] { case Delegate(fieldName, mapping, join) => ComponentElaborator.ComponentMapping(schema.uncheckedRef(om.tpe), fieldName, mapping, join) } - case _ => Nil + case _ => Seq.empty } ComponentElaborator(componentMappings) @@ -997,7 +1188,7 @@ abstract class Mapping[F[_]] { case EffectField(fieldName, handler, _, _) => EffectElaborator.EffectMapping(schema.uncheckedRef(om.tpe), fieldName, handler) } - case _ => Nil + case _ => Seq.empty } EffectElaborator(effectMappings) @@ -1009,13 +1200,8 @@ abstract class Mapping[F[_]] { lazy val graphQLParser: GraphQLParser = GraphQLParser(parserConfig) lazy val queryParser: QueryParser = QueryParser(graphQLParser) - private def deferredValidate(): Unit = { - if(!typeMappings.unsafe) - unsafeValidate() - } - lazy val compiler: QueryCompiler = { - deferredValidate() + typeMappings.unsafeValidateIfChecked() new QueryCompiler(queryParser, schema, compilerPhases) } @@ -1083,7 +1269,6 @@ abstract class Mapping[F[_]] { def narrow(subtpe: TypeRef): Result[Cursor] = Result.failure(s"Cannot narrow $tpe to $subtpe") - def hasField(fieldName: String): Boolean = false def field(fieldName: String, resultName: Option[String]): Result[Cursor] = Result.failure(s"Cannot select field '$fieldName' from leaf type $tpe") } diff --git a/modules/core/src/main/scala/queryinterpreter.scala b/modules/core/src/main/scala/queryinterpreter.scala index fedf2a28..118dce32 100644 --- a/modules/core/src/main/scala/queryinterpreter.scala +++ b/modules/core/src/main/scala/queryinterpreter.scala @@ -204,8 +204,8 @@ class QueryInterpreter[F[_]](mapping: Mapping[F]) { (tpe.dealias match { case o: ObjectType => Some(o.name) case i: InterfaceType => - (schema.types.collectFirst { - case o: ObjectType if o <:< i && cursor.narrowsTo(schema.uncheckedRef(o)) => o.name + (schema.implementations(i).collectFirst { + case o if cursor.narrowsTo(schema.uncheckedRef(o)) => o.name }) case u: UnionType => (u.members.map(_.dealias).collectFirst { diff --git a/modules/core/src/main/scala/schema.scala b/modules/core/src/main/scala/schema.scala index 4339a51a..c8b73e93 100644 --- a/modules/core/src/main/scala/schema.scala +++ b/modules/core/src/main/scala/schema.scala @@ -15,6 +15,8 @@ package grackle +import scala.collection.mutable.{ Map => MMap } + import cats.implicits._ import io.circe.Json import org.tpolecat.sourcepos.SourcePos @@ -180,6 +182,21 @@ trait Schema { } } + private lazy val implIndex: MMap[String, List[ObjectType]] = { + val allIfs = types.collect { case i: InterfaceType => i } + val allObjs = types.collect { case o: ObjectType if o.interfaces.nonEmpty => o } + val grouped = + allIfs.map { i => + val impls = allObjs.filter(_ <:< i) + (i.name, impls) + } + MMap.from(grouped) + } + + /** Yields the `ObjectType` implementations of the given `InterfaceType`. */ + def implementations(it: InterfaceType): List[ObjectType] = + implIndex.getOrElse(it.name, Nil) + override def toString = SchemaRenderer.renderSchema(this) private def extendType(extns: List[TypeExtension])(baseType: NamedType): NamedType = { @@ -744,18 +761,8 @@ sealed trait TypeWithFields extends NamedType { override def fieldInfo(name: String): Option[Field] = fields.find(_.name == name) - def allInterfaces: List[NamedType] = { - @annotation.tailrec - def loop(pending: List[NamedType], acc: List[NamedType]): List[NamedType] = - pending match { - case Nil => acc.reverse - case (twf: TypeWithFields) :: tl => - loop(twf.interfaces.filterNot(i => acc.exists(_.name == i.name)).map(_.dealias) ::: tl, twf :: acc) - case _ :: tl => loop(tl, acc) - } - - loop(interfaces.map(_.dealias), Nil) - } + @deprecated("Use interfaces instead", "0.20.0") + def allInterfaces: List[NamedType] = interfaces } /** @@ -1838,38 +1845,68 @@ object SchemaValidator { } def validateImplementations(schema: Schema): List[Problem] = { - def validateImplementor(impl: TypeWithFields): List[Problem] = { import impl.{name, fields, interfaces} - interfaces.flatMap(_.dealias match { - case iface: InterfaceType => - iface.fields.flatMap { ifField => - fields.find(_.name == ifField.name).map { implField => - val ifTpe = ifField.tpe - val implTpe = implField.tpe + val duplicateImplementsProblems = { + val dupes = interfaces.groupBy(identity).collect { case (i, occurrences) if occurrences.sizeCompare(1) > 0 => i.name } + if (dupes.isEmpty) Nil + else { + val plural = if (dupes.sizeCompare(1) > 0) "s:" else "" + List(Problem(s"Implements clause of type '$name' has duplicate occurrences of interface$plural ${dupes.map(d => s"'$d'").mkString(", ")}")) + } + } - val rp = - if (implTpe <:< ifTpe) Nil - else List(Problem(s"Field '${implField.name}' of type '$name' has type '${renderType(implTpe)}', however implemented interface '${iface.name}' requires it to be a subtype of '${renderType(ifTpe)}'")) + def transitiveImplementsProblems = { + @annotation.tailrec + def loop(pending: List[NamedType], acc: Set[String]): Set[String] = + pending match { + case Nil => acc + case (twf: TypeWithFields) :: tl => + val unseen = twf.interfaces.filterNot(i => acc.contains(i.name)).map(_.dealias) + loop(unseen ::: tl, acc ++ unseen.map(_.name)) + case _ :: tl => loop(tl, acc) + } - val argsMatch = - implField.args.corresponds(ifField.args) { case (arg0, arg1) => - arg0.name == arg1.name && arg0.tpe == arg1.tpe - } + val missingTransitives = loop(interfaces.map(_.dealias), Set.empty) -- interfaces.map(_.name) - name + if (missingTransitives.isEmpty) Nil + else { + val plural = if (missingTransitives.sizeCompare(1) > 0) "s:" else "" + List(Problem(s"Type '$name' does not directly implement transitively implemented interface$plural ${missingTransitives.map(i => s"'$i'").mkString(", ")}")) + } + } - val ap = - if (argsMatch) Nil - else List(Problem(s"Field '${implField.name}' of type '$name' has has an argument list that does not conform to that specified by implemented interface '${iface.name}'")) + val implementationProblems = + interfaces.flatMap(_.dealias match { + case iface: InterfaceType => + iface.fields.flatMap { ifField => + fields.find(_.name == ifField.name).map { implField => + val ifTpe = ifField.tpe + val implTpe = implField.tpe + + val rp = + if (implTpe <:< ifTpe) Nil + else List(Problem(s"Field '${implField.name}' of type '$name' has type '${renderType(implTpe)}', however implemented interface '${iface.name}' requires it to be a subtype of '${renderType(ifTpe)}'")) + + val argsMatch = + implField.args.corresponds(ifField.args) { case (arg0, arg1) => + arg0.name == arg1.name && arg0.tpe == arg1.tpe + } + + val ap = + if (argsMatch) Nil + else List(Problem(s"Field '${implField.name}' of type '$name' has has an argument list that does not conform to that specified by implemented interface '${iface.name}'")) + + rp ++ ap + }.getOrElse(List(Problem(s"Field '${ifField.name}' from interface '${iface.name}' is not defined by implementing type '$name'"))) + } + case undefined: TypeRef => + List(Problem(s"Undefined type '${undefined.name}' declared as implemented by type '$name'")) + case other => + List(Problem(s"Non-interface type '${other.name}' declared as implemented by type '$name'")) + }) - rp ++ ap - }.getOrElse(List(Problem(s"Field '${ifField.name}' from interface '${iface.name}' is not defined by implementing type '$name'"))) - } - case undefined: TypeRef => - List(Problem(s"Undefined type '${undefined.name}' declared as implemented by type '$name'")) - case other => - List(Problem(s"Non-interface type '${other.name}' declared as implemented by type '$name'")) - }) + duplicateImplementsProblems ++ transitiveImplementsProblems ++ implementationProblems } val impls = schema.types.collect { case impl: TypeWithFields => impl } diff --git a/modules/core/src/main/scala/valuemapping.scala b/modules/core/src/main/scala/valuemapping.scala index 31319476..113b6cbf 100644 --- a/modules/core/src/main/scala/valuemapping.scala +++ b/modules/core/src/main/scala/valuemapping.scala @@ -43,11 +43,9 @@ trait ValueMappingLike[F[_]] extends Mapping[F] { DeferredCursor(path, (context, parent) => mkCursor(context, t, Some(parent), env).success) } - override def mkCursorForField(parent: Cursor, fieldName: String, resultName: Option[String]): Result[Cursor] = { - val context = parent.context - val fieldContext = context.forFieldOrAttribute(fieldName, resultName) - fieldMapping(context, fieldName) match { - case Some(ValueField(_, f, _)) => + override def mkCursorForMappedField(parent: Cursor, fieldContext: Context, fm: FieldMapping): Result[Cursor] = + fm match { + case ValueField(_, f, _) => val parentFocus: Any = parent match { case vc: ValueCursor => vc.focus case _ => () @@ -56,9 +54,8 @@ trait ValueMappingLike[F[_]] extends Mapping[F] { mkCursor(fieldContext, childFocus, Some(parent), Env.empty).success case _ => - super.mkCursorForField(parent, fieldName, resultName) + super.mkCursorForMappedField(parent, fieldContext, fm) } - } sealed trait ValueFieldMapping[T] extends FieldMapping { def unwrap: FieldMapping @@ -193,9 +190,6 @@ trait ValueMappingLike[F[_]] extends Mapping[F] { else Result.internalError(s"Focus ${focus} of static type $tpe cannot be narrowed to $subtpe") - def hasField(fieldName: String): Boolean = - fieldMapping(context, fieldName).isDefined - def field(fieldName: String, resultName: Option[String]): Result[Cursor] = mkCursorForField(this, fieldName, resultName) } diff --git a/modules/core/src/test/scala/extensions/ExtensionsSuite.scala b/modules/core/src/test/scala/extensions/ExtensionsSuite.scala index dbe189fa..d43230df 100644 --- a/modules/core/src/test/scala/extensions/ExtensionsSuite.scala +++ b/modules/core/src/test/scala/extensions/ExtensionsSuite.scala @@ -173,7 +173,7 @@ final class ExtensionsSuite extends CatsEffectSuite { extinct: Boolean } - extend type Human { + extend type Human implements Organism { id: ID! extinct: Boolean } @@ -192,7 +192,7 @@ final class ExtensionsSuite extends CatsEffectSuite { schema.definition("Human") match { case Some(obj: ObjectType) => assertEquals(obj.fields.map(_.name), List("name", "species", "id", "extinct")) - assertEquals(obj.interfaces.map(_.name), List("Animal")) + assertEquals(obj.interfaces.map(_.name), List("Animal", "Organism")) case _ => fail("Human type not found") } } @@ -219,6 +219,8 @@ final class ExtensionsSuite extends CatsEffectSuite { extend interface Animal implements Organism @Intrf + extend type Human implements Organism + directive @Intrf on INTERFACE """ @@ -594,6 +596,7 @@ final class ExtensionsSuite extends CatsEffectSuite { val expected = NonEmptyChain( "Duplicate definition of field 'species' for type 'Animal'", + "Type 'Human' does not directly implement transitively implemented interfaces: 'Organism', 'Mineral'", "Field 'extinct' from interface 'Organism' is not defined by implementing type 'Animal'", "Undefined type 'Mineral' declared as implemented by type 'Animal'", "Directive 'Sca' is not allowed on INTERFACE" diff --git a/modules/core/src/test/scala/introspection/IntrospectionSuite.scala b/modules/core/src/test/scala/introspection/IntrospectionSuite.scala index a8840aa3..c2bf94a2 100644 --- a/modules/core/src/test/scala/introspection/IntrospectionSuite.scala +++ b/modules/core/src/test/scala/introspection/IntrospectionSuite.scala @@ -473,7 +473,8 @@ final class IntrospectionSuite extends CatsEffectSuite { { "name" : "interface", "type" : { - "interfaces" : null + "interfaces" : [ + ] } }, { @@ -1209,7 +1210,8 @@ final class IntrospectionSuite extends CatsEffectSuite { } ], "inputFields" : null, - "interfaces" : null, + "interfaces" : [ + ], "enumValues" : null, "possibleTypes" : [ { diff --git a/modules/core/src/test/scala/mapping/MappingValidatorSuite.scala b/modules/core/src/test/scala/mapping/MappingValidatorSuite.scala index 996edb0d..556dcb3b 100644 --- a/modules/core/src/test/scala/mapping/MappingValidatorSuite.scala +++ b/modules/core/src/test/scala/mapping/MappingValidatorSuite.scala @@ -39,7 +39,7 @@ final class ValidatorSuite extends CatsEffectSuite { """ override val typeMappings = - TypeMappings.unsafe( + TypeMappings.unchecked( List( ObjectMapping( schema.ref("Query"), @@ -74,7 +74,7 @@ final class ValidatorSuite extends CatsEffectSuite { """ override val typeMappings = - TypeMappings.unsafe( + TypeMappings.unchecked( ObjectMapping(schema.ref("Query"))( CursorField[String]("foo", _ => ???, Nil) ), @@ -110,7 +110,7 @@ final class ValidatorSuite extends CatsEffectSuite { """ override val typeMappings = - TypeMappings.unsafe( + TypeMappings.unchecked( List( ObjectMapping( schema.ref("Query"), @@ -147,7 +147,7 @@ final class ValidatorSuite extends CatsEffectSuite { """ override val typeMappings = - TypeMappings.unsafe( + TypeMappings.unchecked( List( ObjectMapping( schema.ref("Query"), @@ -186,7 +186,7 @@ final class ValidatorSuite extends CatsEffectSuite { """ override val typeMappings = - TypeMappings.unsafe( + TypeMappings.unchecked( List( ObjectMapping( schema.ref("Query"), @@ -357,7 +357,7 @@ final class ValidatorSuite extends CatsEffectSuite { """ override val typeMappings = - TypeMappings.unsafe( + TypeMappings.unchecked( List( ObjectMapping( schema.ref("Query"), @@ -396,7 +396,7 @@ final class ValidatorSuite extends CatsEffectSuite { """ override val typeMappings = - TypeMappings.unsafe( + TypeMappings.unchecked( List( ObjectMapping( schema.ref("Query"), @@ -474,7 +474,7 @@ final class ValidatorSuite extends CatsEffectSuite { """ override val typeMappings = - TypeMappings.unsafe( + TypeMappings.unchecked( List( ObjectMapping( schema.ref("Query"), diff --git a/modules/core/src/test/scala/schema/SchemaSuite.scala b/modules/core/src/test/scala/schema/SchemaSuite.scala index 0083db2c..48e1c7e5 100644 --- a/modules/core/src/test/scala/schema/SchemaSuite.scala +++ b/modules/core/src/test/scala/schema/SchemaSuite.scala @@ -156,6 +156,35 @@ final class SchemaSuite extends CatsEffectSuite { } } + test("schema validation: object implementing duplicate interfaces") { + val schema = Schema( + """ + interface Character { + id: ID! + name: String! + email: String! + } + + type Human implements Character & Character { + id: ID! + name: String! + email: String! + } + """ + ) + + schema match { + case Result.Failure(ps) => + assertEquals( + ps.map(_.message), + NonEmptyChain( + "Implements clause of type 'Human' has duplicate occurrences of interface 'Character'" + ) + ) + case unexpected => fail(s"This was unexpected: $unexpected") + } + } + test("schema validation: object failing to implement interface fields") { val schema = Schema( """ @@ -173,7 +202,12 @@ final class SchemaSuite extends CatsEffectSuite { schema match { case Result.Failure(ps) => - assertEquals(ps.map(_.message), NonEmptyChain("Field 'id' from interface 'Character' is not defined by implementing type 'Human'", "Field 'email' from interface 'Character' is not defined by implementing type 'Human'")) + assertEquals(ps.map(_.message), + NonEmptyChain( + "Field 'id' from interface 'Character' is not defined by implementing type 'Human'", + "Field 'email' from interface 'Character' is not defined by implementing type 'Human'" + ) + ) case unexpected => fail(s"This was unexpected: $unexpected") } } @@ -195,7 +229,12 @@ final class SchemaSuite extends CatsEffectSuite { schema match { case Result.Failure(ps) => - assertEquals(ps.map(_.message), NonEmptyChain("Field 'id' from interface 'Character' is not defined by implementing type 'Named'", "Field 'email' from interface 'Character' is not defined by implementing type 'Named'")) + assertEquals(ps.map(_.message), + NonEmptyChain( + "Field 'id' from interface 'Character' is not defined by implementing type 'Named'", + "Field 'email' from interface 'Character' is not defined by implementing type 'Named'" + ) + ) case unexpected => fail(s"This was unexpected: $unexpected") } } @@ -215,7 +254,11 @@ final class SchemaSuite extends CatsEffectSuite { schema match { case Result.Failure(ps) => - assertEquals(ps.map(_.message), NonEmptyChain("Field 'name' of type 'Human' has type 'Int!', however implemented interface 'Character' requires it to be a subtype of 'String!'")) + assertEquals(ps.map(_.message), + NonEmptyChain( + "Field 'name' of type 'Human' has type 'Int!', however implemented interface 'Character' requires it to be a subtype of 'String!'" + ) + ) case unexpected => fail(s"This was unexpected: ${unexpected.getClass.getSimpleName}") } } @@ -235,7 +278,11 @@ final class SchemaSuite extends CatsEffectSuite { schema match { case Result.Failure(ps) => - assertEquals(ps.map(_.message), NonEmptyChain("Field 'name' of type 'Human' has has an argument list that does not conform to that specified by implemented interface 'Character'")) + assertEquals(ps.map(_.message), + NonEmptyChain( + "Field 'name' of type 'Human' has has an argument list that does not conform to that specified by implemented interface 'Character'" + ) + ) case unexpected => fail(s"This was unexpected: ${unexpected.getClass.getSimpleName}") } } @@ -260,7 +307,12 @@ final class SchemaSuite extends CatsEffectSuite { schema match { case Result.Failure(ps) => - assertEquals(ps.map(_.message), NonEmptyChain("Field 'id' from interface 'Character' is not defined by implementing type 'Human'", "Field 'id' from interface 'Character' is not defined by implementing type 'Dog'")) + assertEquals(ps.map(_.message), + NonEmptyChain( + "Field 'id' from interface 'Character' is not defined by implementing type 'Human'", + "Field 'id' from interface 'Character' is not defined by implementing type 'Dog'" + ) + ) case unexpected => fail(s"This was unexpected: $unexpected") } } @@ -285,7 +337,12 @@ final class SchemaSuite extends CatsEffectSuite { schema match { case Result.Failure(ps) => - assertEquals(ps.map(_.message), NonEmptyChain("Field 'id' from interface 'Character' is not defined by implementing type 'Human'", "Field 'email' from interface 'Contactable' is not defined by implementing type 'Human'")) + assertEquals(ps.map(_.message), + NonEmptyChain( + "Field 'id' from interface 'Character' is not defined by implementing type 'Human'", + "Field 'email' from interface 'Contactable' is not defined by implementing type 'Human'" + ) + ) case unexpected => fail(s"This was unexpected: $unexpected") } } @@ -315,6 +372,32 @@ final class SchemaSuite extends CatsEffectSuite { } } + test("schema validation: object failing to implement transitive interface") { + val schema = Schema( + """ + interface Node { + id: ID! + } + + interface Resource implements Node { + id: ID! + url: String + } + + type Human implements Resource { + id: ID! + url: String + } + """ + ) + + schema match { + case Result.Failure(ps) => + assertEquals(ps.map(_.message), NonEmptyChain("Type 'Human' does not directly implement transitively implemented interface 'Node'")) + case unexpected => fail(s"This was unexpected: $unexpected") + } + } + test("schema validation: implements non-interface") { val schema = Schema( """ @@ -373,19 +456,19 @@ final class SchemaSuite extends CatsEffectSuite { node: Node } - interface Node implements Tagged { + interface Node implements Tagged & Named { id: ID! name: String tag: String } - interface Named implements Node { + interface Named implements Node & Tagged { id: ID! name: String tag: String } - interface Tagged implements Named { + interface Tagged implements Named & Node { id: ID! name: String tag: String diff --git a/modules/core/src/test/scala/starwars/StarWarsData.scala b/modules/core/src/test/scala/starwars/StarWarsData.scala index 88318299..8454be6d 100644 --- a/modules/core/src/test/scala/starwars/StarWarsData.scala +++ b/modules/core/src/test/scala/starwars/StarWarsData.scala @@ -145,6 +145,8 @@ object StarWarsMapping extends ValueMapping[IO] { } interface Character { id: String! + taggedId1: String! + taggedId2: String! name: String numberOfFriends: Int friends: [Character!] @@ -152,6 +154,8 @@ object StarWarsMapping extends ValueMapping[IO] { } type Human implements Character { id: String! + taggedId1: String! + taggedId2: String! name: String numberOfFriends: Int friends: [Character!] @@ -160,6 +164,8 @@ object StarWarsMapping extends ValueMapping[IO] { } type Droid implements Character { id: String! + taggedId1: String! + taggedId2: String! name: String numberOfFriends: Int friends: [Character!] @@ -192,6 +198,7 @@ object StarWarsMapping extends ValueMapping[IO] { fieldMappings = List( ValueField("id", _.id), + ValueField("taggedId1", c => s"character-${c.id}"), ValueField("name", _.name), ValueField("appearsIn", _.appearsIn), ValueField("numberOfFriends", _ => 0), @@ -202,6 +209,8 @@ object StarWarsMapping extends ValueMapping[IO] { tpe = HumanType, fieldMappings = List( + ValueField("taggedId1", h => s"human-${h.id}"), + ValueField("taggedId2", h => s"human2-${h.id}"), ValueField("homePlanet", _.homePlanet) ) ), @@ -209,6 +218,7 @@ object StarWarsMapping extends ValueMapping[IO] { tpe = DroidType, fieldMappings = List( + ValueField("taggedId2", h => s"droid2-${h.id}"), ValueField("primaryFunction", _.primaryFunction) ) ), diff --git a/modules/core/src/test/scala/starwars/StarWarsSuite.scala b/modules/core/src/test/scala/starwars/StarWarsSuite.scala index 5e9a05eb..106b1665 100644 --- a/modules/core/src/test/scala/starwars/StarWarsSuite.scala +++ b/modules/core/src/test/scala/starwars/StarWarsSuite.scala @@ -478,4 +478,230 @@ final class StarWarsSuite extends CatsEffectSuite { assertIO(res, expected) } + + test("interface variant (1)") { + val query = """ + query { + characters(offset: 3, limit: 4) { + ... on Human { + taggedId1 + } + ... on Droid { + taggedId1 + } + } + } + """ + + val expected = json""" + { + "data" : { + "characters" : [ + { + "taggedId1" : "human-1003" + }, + { + "taggedId1" : "human-1004" + }, + { + "taggedId1" : "character-2000" + }, + { + "taggedId1" : "character-2001" + } + ] + } + } + """ + + val res = StarWarsMapping.compileAndRun(query) + + assertIO(res, expected) + } + + test("interface variant (2)") { + val query = """ + query { + characters(offset: 3, limit: 4) { + taggedId1 + } + } + """ + + val expected = json""" + { + "data" : { + "characters" : [ + { + "taggedId1" : "human-1003" + }, + { + "taggedId1" : "human-1004" + }, + { + "taggedId1" : "character-2000" + }, + { + "taggedId1" : "character-2001" + } + ] + } + } + """ + + val res = StarWarsMapping.compileAndRun(query) + + assertIO(res, expected) + } + + test("interface variant (3)") { + val query = """ + query { + characters(offset: 3, limit: 4) { + taggedId1 + ... on Human { + taggedId1 + } + } + } + """ + + val expected = json""" + { + "data" : { + "characters" : [ + { + "taggedId1" : "human-1003" + }, + { + "taggedId1" : "human-1004" + }, + { + "taggedId1" : "character-2000" + }, + { + "taggedId1" : "character-2001" + } + ] + } + } + """ + + val res = StarWarsMapping.compileAndRun(query) + + assertIO(res, expected) + } + + test("interface variant (4)") { + val query = """ + query { + characters(offset: 3, limit: 4) { + ... on Human { + taggedId2 + } + ... on Droid { + taggedId2 + } + } + } + """ + + val expected = json""" + { + "data" : { + "characters" : [ + { + "taggedId2" : "human2-1003" + }, + { + "taggedId2" : "human2-1004" + }, + { + "taggedId2" : "droid2-2000" + }, + { + "taggedId2" : "droid2-2001" + } + ] + } + } + """ + + val res = StarWarsMapping.compileAndRun(query) + + assertIO(res, expected) + } + + test("interface variant (5)") { + val query = """ + query { + characters(offset: 3, limit: 4) { + taggedId2 + } + } + """ + + val expected = json""" + { + "data" : { + "characters" : [ + { + "taggedId2" : "human2-1003" + }, + { + "taggedId2" : "human2-1004" + }, + { + "taggedId2" : "droid2-2000" + }, + { + "taggedId2" : "droid2-2001" + } + ] + } + } + """ + + val res = StarWarsMapping.compileAndRun(query) + + assertIO(res, expected) + } + + test("interface variant (6)") { + val query = """ + query { + characters(offset: 3, limit: 4) { + taggedId2 + ... on Human { + taggedId2 + } + } + } + """ + + val expected = json""" + { + "data" : { + "characters" : [ + { + "taggedId2" : "human2-1003" + }, + { + "taggedId2" : "human2-1004" + }, + { + "taggedId2" : "droid2-2000" + }, + { + "taggedId2" : "droid2-2001" + } + ] + } + } + """ + + val res = StarWarsMapping.compileAndRun(query) + + assertIO(res, expected) + } } diff --git a/modules/generic/src/main/scala-2/genericmapping2.scala b/modules/generic/src/main/scala-2/genericmapping2.scala index 4b074991..4feea20b 100644 --- a/modules/generic/src/main/scala-2/genericmapping2.scala +++ b/modules/generic/src/main/scala-2/genericmapping2.scala @@ -83,9 +83,6 @@ trait ScalaVersionSpecificGenericMappingLike[F[_]] extends Mapping[F] { self: Ge extends AbstractCursor { def withEnv(env0: Env): Cursor = copy(env = env.add(env0)) - override def hasField(fieldName: String): Boolean = - fieldMap.contains(fieldName) || typeMappings.fieldMapping(context, fieldName).isDefined - override def field(fieldName: String, resultName: Option[String]): Result[Cursor] = { val localField = fieldMap.get(fieldName).toResult(s"No field '$fieldName' for type $tpe").flatMap { f => @@ -137,8 +134,6 @@ trait ScalaVersionSpecificGenericMappingLike[F[_]] extends Mapping[F] { self: Ge def focus: Any = cursor.focus val context: Context = cursor.context.asType(tpe0) - override def hasField(fieldName: String): Boolean = cursor.hasField(fieldName) - override def field(fieldName: String, resultName: Option[String]): Result[Cursor] = cursor.field(fieldName, resultName) orElse mkCursorForField(this, fieldName, resultName) diff --git a/modules/generic/src/main/scala-3/genericmapping3.scala b/modules/generic/src/main/scala-3/genericmapping3.scala index d2be67bf..328332bb 100644 --- a/modules/generic/src/main/scala-3/genericmapping3.scala +++ b/modules/generic/src/main/scala-3/genericmapping3.scala @@ -73,9 +73,6 @@ trait ScalaVersionSpecificGenericMappingLike[F[_]] extends Mapping[F] { self: Ge extends AbstractCursor { def withEnv(env0: Env): Cursor = copy(env = env.add(env0)) - override def hasField(fieldName: String): Boolean = - fieldMap.contains(fieldName) || typeMappings.fieldMapping(context, fieldName).isDefined - override def field(fieldName: String, resultName: Option[String]): Result[Cursor] = { val localField = fieldMap.get(fieldName).toResult(s"No field '$fieldName' for type $tpe").flatMap { f => @@ -116,8 +113,6 @@ trait ScalaVersionSpecificGenericMappingLike[F[_]] extends Mapping[F] { self: Ge def focus: Any = cursor.focus val context: Context = cursor.context.asType(tpe0) - override def hasField(fieldName: String): Boolean = cursor.hasField(fieldName) - override def field(fieldName: String, resultName: Option[String]): Result[Cursor] = cursor.field(fieldName, resultName) orElse mkCursorForField(this, fieldName, resultName) diff --git a/modules/generic/src/main/scala/genericmapping.scala b/modules/generic/src/main/scala/genericmapping.scala index ee927e85..365f3212 100644 --- a/modules/generic/src/main/scala/genericmapping.scala +++ b/modules/generic/src/main/scala/genericmapping.scala @@ -31,16 +31,13 @@ trait GenericMappingLike[F[_]] extends ScalaVersionSpecificGenericMappingLike[F] else DeferredCursor(path, (context, parent) => cb.build(context, t, Some(parent), env)).success - override def mkCursorForField(parent: Cursor, fieldName: String, resultName: Option[String]): Result[Cursor] = { - val context = parent.context - val fieldContext = context.forFieldOrAttribute(fieldName, resultName) - typeMappings.fieldMapping(context, fieldName) match { - case Some(GenericField(_, t, cb, _)) => + override def mkCursorForMappedField(parent: Cursor, fieldContext: Context, fm: FieldMapping): Result[Cursor] = + fm match { + case GenericField(_, t, cb, _) => cb().build(fieldContext, t, Some(parent), parent.env) case _ => - super.mkCursorForField(parent, fieldName, resultName) + super.mkCursorForMappedField(parent, fieldContext, fm) } - } case class GenericField[T](val fieldName: String, t: T, cb: () => CursorBuilder[T], hidden: Boolean)( implicit val pos: SourcePos diff --git a/modules/sql/shared/src/main/scala/SqlMapping.scala b/modules/sql/shared/src/main/scala/SqlMapping.scala index 5358d44f..3e4739b1 100644 --- a/modules/sql/shared/src/main/scala/SqlMapping.scala +++ b/modules/sql/shared/src/main/scala/SqlMapping.scala @@ -709,14 +709,12 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self def isLocallyMapped(context: Context, query: Query): Boolean = rootFieldMapping(context, query) match { - //case Some(_: SqlFieldMapping) => true // Scala 3 thinks this is unreachable - case Some(fm) if fm.isInstanceOf[SqlFieldMapping] => true + case Some(_: SqlFieldMapping) => true case Some(re: EffectMapping) => val fieldContext = context.forFieldOrAttribute(re.fieldName, None) typeMappings.objectMapping(fieldContext).exists { om => om.fieldMappings.exists { - //case _: SqlFieldMapping => true // Scala 3 thinks this is unreachable - case fm if fm.isInstanceOf[SqlFieldMapping] => true + case _: SqlFieldMapping => true case _ => false } } @@ -894,16 +892,14 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self }.toList val interfaceKeys = context.tpe.underlyingObject match { - case Some(ot: ObjectType) => - ot.interfaces.flatMap(nt => keyColumnsForType(context.asType(nt))) + case Some(twf: TypeWithFields) => + twf.interfaces.flatMap(nt => keyColumnsForType(context.asType(nt))) case _ => Nil } (objectKeys ++ interfaceKeys).distinct }.getOrElse(Nil) - assert(cols.nonEmpty || parentTableForType(context).map(_.isRoot).getOrElse(false), s"No key columns for type ${context.tpe}") - cols } @@ -3521,14 +3517,11 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self } else Result.internalError(s"Cannot narrow $tpe to $subtpe") } - def hasField(fieldName: String): Boolean = - typeMappings.fieldMapping(context, fieldName).isDefined - def field(fieldName: String, resultName: Option[String]): Result[Cursor] = { val fieldContext = context.forFieldOrAttribute(fieldName, resultName) val fieldTpe = fieldContext.tpe val localField = - typeMappings.fieldMapping(context, fieldName) match { + typeMappings.fieldMapping(this, fieldName) match { case Some(_: SqlJson) => asTable.flatMap { table => def mkCirceCursor(f: Json): Result[Cursor] = @@ -3575,7 +3568,6 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self def withEnv(env0: Env): MultiRootCursor = copy(roots = roots.map(_.withEnv(env0))) def context: Context = roots.head.context - override def hasField(fieldName: String): Boolean = roots.exists(_.hasField(fieldName)) override def field(fieldName: String, resultName: Option[String]): Result[Cursor] = { roots.find(_.mapped.containsRoot(fieldName, resultName)).map(_.field(fieldName, resultName)). getOrElse(Result.internalError(s"No field '$fieldName' for type ${context.tpe}")) @@ -3593,24 +3585,9 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self // Union mappings have no SqlObjects or SqlJson fields // Union field mappings must be hidden - def hasKey(om: ObjectMapping): List[ValidationFailure] = { - def hasKey(om: ObjectMapping, context: Context): Boolean = - om.fieldMappings.exists { - case sf: SqlField => sf.key - case _ => false - } || (context.tpe.underlyingObject match { - case Some(ot: ObjectType) => - ot.interfaces.exists { nt => - val ctx = context.asType(nt) - val nom = mappings.objectMapping(ctx) - nom.map(hasKey(_, ctx)).getOrElse(false) - } - case _ => false - }) - - if (hasKey(om, context)) Nil + def checkKey(om: ObjectMapping): List[ValidationFailure] = + if (keyColumnsForType(context).nonEmpty || parentTableForType(context).map(_.isRoot).getOrElse(false)) Nil else List(NoKeyInObjectTypeMapping(om)) - } def checkAssoc(om: ObjectMapping): List[ValidationFailure] = om.fieldMappings.iterator.collect { @@ -3618,14 +3595,9 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self AssocFieldNotKey(om, sf) }.toList - def hasDiscriminator(om: ObjectMapping): List[ValidationFailure] = { - val hasDiscriminator = om.fieldMappings.exists { - case sf: SqlField => sf.discriminator - case _ => false - } - if (hasDiscriminator) Nil + def hasDiscriminator(om: ObjectMapping): List[ValidationFailure] = + if (discriminatorColumnsForType(context).nonEmpty) Nil else List(NoDiscriminatorInObjectTypeMapping(om)) - } def checkSplit(om: ObjectMapping): List[ValidationFailure] = { val tables = allTables(List(om)) @@ -3636,7 +3608,7 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self def checkSuperInterfaces(om: ObjectMapping): List[ValidationFailure] = { val allMappings = om.tpe.dealias match { - case twf: TypeWithFields => om :: twf.allInterfaces.flatMap(nt => mappings.objectMapping(context.asType(nt))) + case twf: TypeWithFields => om :: twf.interfaces.flatMap(nt => mappings.objectMapping(context.asType(nt))) case _ => Nil } val tables = allTables(allMappings) @@ -3678,20 +3650,20 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self tm match { case im: SqlInterfaceMapping => - hasKey(im) ++ + checkKey(im) ++ checkAssoc(im) ++ hasDiscriminator(im) ++ checkSplit(im) ++ checkSuperInterfaces(im) case um: SqlUnionMapping => - hasKey(um) ++ + checkKey(um) ++ checkAssoc(um) ++ hasDiscriminator(um) ++ checkSplit(um) ++ checkUnionMembers(um) ++ checkUnionFields(um) case om: ObjectMapping if isSql(om) => - (if(schema.isRootType(om.tpe)) Nil else hasKey(om)) ++ + (if(schema.isRootType(om.tpe)) Nil else checkKey(om)) ++ checkAssoc(om) ++ checkSplit(om) ++ checkSuperInterfaces(om) diff --git a/modules/sql/shared/src/test/scala/SqlMappingValidatorInvalidMapping.scala b/modules/sql/shared/src/test/scala/SqlMappingValidatorInvalidMapping.scala index 68bb11f8..dd770860 100644 --- a/modules/sql/shared/src/test/scala/SqlMappingValidatorInvalidMapping.scala +++ b/modules/sql/shared/src/test/scala/SqlMappingValidatorInvalidMapping.scala @@ -133,7 +133,7 @@ trait SqlMappingValidatorInvalidMapping[F[_]] extends SqlTestMapping[F] { val SubObj3Type = schema.ref("SubObj3") override val typeMappings = - TypeMappings.unsafe( + TypeMappings.unchecked( List( ObjectMapping( tpe = QueryType, diff --git a/modules/sql/shared/src/test/scala/SqlMappingValidatorInvalidSuite.scala b/modules/sql/shared/src/test/scala/SqlMappingValidatorInvalidSuite.scala index 7bfd2450..630b4831 100644 --- a/modules/sql/shared/src/test/scala/SqlMappingValidatorInvalidSuite.scala +++ b/modules/sql/shared/src/test/scala/SqlMappingValidatorInvalidSuite.scala @@ -168,6 +168,15 @@ trait SqlMappingValidatorInvalidSuite extends CatsEffectSuite { } } + object UFM { + def unapply(vf: ValidationFailure): Option[(String, String)] = + vf match { + case M.UnusedFieldMapping(om, fm) => + Some((om.tpe.name, fm.fieldName)) + case _ => None + } + } + object TypeNames { val BooleanTypeName = typeName[Boolean] val IntTypeName = typeName[Int] @@ -212,7 +221,8 @@ trait SqlMappingValidatorInvalidSuite extends CatsEffectSuite { NHUFM("Union", "id"), SUM("Union", List("Obj1", "Obj2"), List("obj1", "obj2")), ND("Union"), - NK("Union") + NK("Union"), + UFM("Intrf", "id") ) => } } diff --git a/modules/sql/shared/src/test/scala/SqlMappingValidatorValidMapping.scala b/modules/sql/shared/src/test/scala/SqlMappingValidatorValidMapping.scala index 8f5bc53b..0b145729 100644 --- a/modules/sql/shared/src/test/scala/SqlMappingValidatorValidMapping.scala +++ b/modules/sql/shared/src/test/scala/SqlMappingValidatorValidMapping.scala @@ -207,7 +207,7 @@ trait SqlMappingValidatorValidMapping[F[_]] extends SqlTestMapping[F] { val SubObj2Type = schema.ref("SubObj2") override val typeMappings = - TypeMappings.unsafe( + TypeMappings.unchecked( List( ObjectMapping( tpe = QueryType,