diff --git a/modules/core/src/main/scala/mapping.scala b/modules/core/src/main/scala/mapping.scala index 5cfc21bf..6e610235 100644 --- a/modules/core/src/main/scala/mapping.scala +++ b/modules/core/src/main/scala/mapping.scala @@ -381,6 +381,7 @@ abstract class Mapping[F[_]] { ((ancestralFieldMapping(context, fieldName), tms) match { case (Some(fm), List(om: ObjectMapping)) if !allImplsHaveFieldMapping => + addProblem(DeclaredFieldMappingIsHidden(om, fm)).whenA(fm.hidden) *> addSeenFieldMapping(om, fm) case (None, List(om: ObjectMapping)) if !(hasEnclosingSubtreeFieldMapping || allImplsHaveFieldMapping) => @@ -1371,6 +1372,25 @@ abstract class Mapping[F[_]] { |""".stripMargin } + /** Referenced field does not exist. */ + case class DeclaredFieldMappingIsHidden(objectMapping: ObjectMapping, fieldMapping: FieldMapping) + extends ValidationFailure(Severity.Error) { + override def toString: String = + s"$productPrefix(${showNamedType(objectMapping.tpe)}.${fieldMapping.fieldName})" + override def formattedMessage: String = + s"""|Declared field mapping is hidden. + | + |- The field ${graphql(s"${showNamedType(objectMapping.tpe)}.${fieldMapping.fieldName}")} is defined by a Schema at (1). + |- The ${scala(objectMapping.showMappingType)} at (2) contains a ${scala(fieldMapping.showMappingType)} mapping for field ${graphql(fieldMapping.fieldName)} at (3). + |- This field mapping is marked as hidden. + |- ${UNDERLINED}The mappings for declared fields must not be hidden.$RESET + | + |(1) ${schema.pos} + |(2) ${objectMapping.pos} + |(3) ${fieldMapping.pos} + |""".stripMargin + } + /** GraphQL type isn't applicable for mapping type. */ case class ObjectTypeExpected(objectMapping: ObjectMapping) extends ValidationFailure(Severity.Error) { diff --git a/modules/core/src/test/scala/mapping/MappingValidatorSuite.scala b/modules/core/src/test/scala/mapping/MappingValidatorSuite.scala index 556dcb3b..cfa2029e 100644 --- a/modules/core/src/test/scala/mapping/MappingValidatorSuite.scala +++ b/modules/core/src/test/scala/mapping/MappingValidatorSuite.scala @@ -462,6 +462,45 @@ final class ValidatorSuite extends CatsEffectSuite { } + test("declared fields must not be hidden") { + + object M extends TestMapping { + val schema = + schema""" + type Query { + foo: Foo + } + + type Foo { + bar: String + } + """ + + override val typeMappings = + List( + ObjectMapping( + schema.ref("Query"), + List( + CursorField[String]("foo", _ => ???, Nil) + ) + ), + ObjectMapping( + schema.ref("Foo"), + List( + CursorField[String]("bar", _ => ???, Nil, hidden = true), + ), + ) + ) + } + + val es = M.validate() + es match { + case List(M.DeclaredFieldMappingIsHidden(_, _)) => () + case _ => fail(es.foldMap(_.toErrorMessage)) + } + } + + test("unsafeValidate") { object M extends TestMapping { val schema = diff --git a/modules/generic/src/main/scala/genericmapping.scala b/modules/generic/src/main/scala/genericmapping.scala index 365f3212..5e2336c9 100644 --- a/modules/generic/src/main/scala/genericmapping.scala +++ b/modules/generic/src/main/scala/genericmapping.scala @@ -45,7 +45,7 @@ trait GenericMappingLike[F[_]] extends ScalaVersionSpecificGenericMappingLike[F] def subtree: Boolean = true } - def GenericField[T](fieldName: String, t: T, hidden: Boolean = true)(implicit cb: => CursorBuilder[T], pos: SourcePos): GenericField[T] = + def GenericField[T](fieldName: String, t: T, hidden: Boolean = false)(implicit cb: => CursorBuilder[T], pos: SourcePos): GenericField[T] = new GenericField(fieldName, t, () => cb, hidden) object semiauto { diff --git a/modules/sql/shared/src/test/scala/SqlSiblingListsMapping.scala b/modules/sql/shared/src/test/scala/SqlSiblingListsMapping.scala index 50d1f533..6312b32f 100644 --- a/modules/sql/shared/src/test/scala/SqlSiblingListsMapping.scala +++ b/modules/sql/shared/src/test/scala/SqlSiblingListsMapping.scala @@ -86,7 +86,7 @@ trait SqlSiblingListsData[F[_]] extends SqlTestMapping[F] { tpe = AType, fieldMappings = List( - SqlField("id", aTable.id, key = true, hidden = true), + SqlField("id", aTable.id, key = true), SqlObject("bs", Join(aTable.id, bTable.aId)) ) ), @@ -94,7 +94,7 @@ trait SqlSiblingListsData[F[_]] extends SqlTestMapping[F] { tpe = BType, fieldMappings = List( - SqlField("id", bTable.id, key = true, hidden = true), + SqlField("id", bTable.id, key = true), SqlObject("cs", Join(bTable.id, cTable.bId)), SqlObject("ds", Join(bTable.id, dTable.bId)) ) @@ -103,7 +103,7 @@ trait SqlSiblingListsData[F[_]] extends SqlTestMapping[F] { tpe = CType, fieldMappings = List( - SqlField("id", cTable.id, key = true, hidden = true), + SqlField("id", cTable.id, key = true), SqlField("nameC", cTable.nameC) ) ), @@ -111,7 +111,7 @@ trait SqlSiblingListsData[F[_]] extends SqlTestMapping[F] { tpe = DType, fieldMappings = List( - SqlField("id", dTable.id, key = true, hidden = true), + SqlField("id", dTable.id, key = true), SqlField("nameD", dTable.nameD) ) )