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..bbfea097 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,193 @@ 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 interfaceMappingIndex = + MMap.from(mappings.collect { case om: ObjectMapping if om.tpe.dealias.isInterface => om.tpe.name -> om }) + + 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 => interfaceMappingIndex.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 +728,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 +842,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 +939,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 +1177,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 +1191,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 +1203,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 +1272,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,