diff --git a/demo/src/main/scala/demo/starwars/StarWarsMapping.scala b/demo/src/main/scala/demo/starwars/StarWarsMapping.scala index d770312d..e0d2f1d1 100644 --- a/demo/src/main/scala/demo/starwars/StarWarsMapping.scala +++ b/demo/src/main/scala/demo/starwars/StarWarsMapping.scala @@ -69,17 +69,13 @@ trait StarWarsMapping[F[_]] extends GenericMapping[F] { self: StarWarsData[F] => val DroidType = schema.ref("Droid") val typeMappings = - List( + TypeMappings( // #root - ObjectMapping( - tpe = QueryType, - fieldMappings = - List( - GenericField("hero", characters), - GenericField("character", characters), - GenericField("human", characters.collect { case h: Human => h }), - GenericField("droid", characters.collect { case d: Droid => d }) - ) + ObjectMapping(QueryType)( + GenericField("hero", characters), + GenericField("character", characters), + GenericField("human", characters.collect { case h: Human => h }), + GenericField("droid", characters.collect { case d: Droid => d }) ) // #root ) diff --git a/demo/src/main/scala/demo/world/WorldMapping.scala b/demo/src/main/scala/demo/world/WorldMapping.scala index fe3247be..b907f69d 100644 --- a/demo/src/main/scala/demo/world/WorldMapping.scala +++ b/demo/src/main/scala/demo/world/WorldMapping.scala @@ -119,65 +119,53 @@ trait WorldMapping[F[_]] extends DoobieMapping[F] { val LanguageType = schema.ref("Language") val typeMappings = - List( + TypeMappings( // #root - ObjectMapping( - tpe = QueryType, - fieldMappings = List( - SqlObject("cities"), - SqlObject("city"), - SqlObject("country"), - SqlObject("countries"), - SqlObject("language"), - SqlObject("search"), - SqlObject("search2") - ) + ObjectMapping(QueryType)( + SqlObject("cities"), + SqlObject("city"), + SqlObject("country"), + SqlObject("countries"), + SqlObject("language"), + SqlObject("search"), + SqlObject("search2") ), // #root // #type_mappings - ObjectMapping( - tpe = CountryType, - fieldMappings = List( - SqlField("code", country.code, key = true), - SqlField("name", country.name), - SqlField("continent", country.continent), - SqlField("region", country.region), - SqlField("surfacearea", country.surfacearea), - SqlField("indepyear", country.indepyear), - SqlField("population", country.population), - SqlField("lifeexpectancy", country.lifeexpectancy), - SqlField("gnp", country.gnp), - SqlField("gnpold", country.gnpold), - SqlField("localname", country.localname), - SqlField("governmentform", country.governmentform), - SqlField("headofstate", country.headofstate), - SqlField("capitalId", country.capitalId), - SqlField("code2", country.code2), - SqlField("numCities", country.numCities), - SqlObject("cities", Join(country.code, city.countrycode)), - SqlObject("languages", Join(country.code, countrylanguage.countrycode)) - ), + ObjectMapping(CountryType)( + SqlField("code", country.code, key = true), + SqlField("name", country.name), + SqlField("continent", country.continent), + SqlField("region", country.region), + SqlField("surfacearea", country.surfacearea), + SqlField("indepyear", country.indepyear), + SqlField("population", country.population), + SqlField("lifeexpectancy", country.lifeexpectancy), + SqlField("gnp", country.gnp), + SqlField("gnpold", country.gnpold), + SqlField("localname", country.localname), + SqlField("governmentform", country.governmentform), + SqlField("headofstate", country.headofstate), + SqlField("capitalId", country.capitalId), + SqlField("code2", country.code2), + SqlField("numCities", country.numCities), + SqlObject("cities", Join(country.code, city.countrycode)), + SqlObject("languages", Join(country.code, countrylanguage.countrycode)) ), - ObjectMapping( - tpe = CityType, - fieldMappings = List( - SqlField("id", city.id, key = true, hidden = true), - SqlField("countrycode", city.countrycode, hidden = true), - SqlField("name", city.name), - SqlField("district", city.district), - SqlField("population", city.population), - SqlObject("country", Join(city.countrycode, country.code)), - ) + ObjectMapping(CityType)( + SqlField("id", city.id, key = true, hidden = true), + SqlField("countrycode", city.countrycode, hidden = true), + SqlField("name", city.name), + SqlField("district", city.district), + SqlField("population", city.population), + SqlObject("country", Join(city.countrycode, country.code)), ), - ObjectMapping( - tpe = LanguageType, - fieldMappings = List( - SqlField("language", countrylanguage.language, key = true, associative = true), - SqlField("isOfficial", countrylanguage.isOfficial), - SqlField("percentage", countrylanguage.percentage), - SqlField("countrycode", countrylanguage.countrycode, hidden = true), - SqlObject("countries", Join(countrylanguage.countrycode, country.code)) - ) + ObjectMapping(LanguageType)( + SqlField("language", countrylanguage.language, key = true, associative = true), + SqlField("isOfficial", countrylanguage.isOfficial), + SqlField("percentage", countrylanguage.percentage), + SqlField("countrycode", countrylanguage.countrycode, hidden = true), + SqlObject("countries", Join(countrylanguage.countrycode, country.code)) ) // #type_mappings ) diff --git a/modules/circe/src/main/scala/circemapping.scala b/modules/circe/src/main/scala/circemapping.scala index 57b6e78c..98f430be 100644 --- a/modules/circe/src/main/scala/circemapping.scala +++ b/modules/circe/src/main/scala/circemapping.scala @@ -59,7 +59,7 @@ trait CirceMappingLike[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) - (fieldMapping(context, fieldName), parent.focus) match { + (typeMappings.fieldMapping(context, fieldName), parent.focus) match { case (Some(CirceField(_, json, _)), _) => CirceCursor(fieldContext, json, Some(parent), parent.env).success case (Some(CursorFieldJson(_, f, _, _)), _) => @@ -70,7 +70,7 @@ trait CirceMappingLike[F[_]] extends Mapping[F] { } sealed trait CirceFieldMapping extends FieldMapping { - def withParent(tpe: Type): FieldMapping = this + def subtree: Boolean = true } case class CirceField(fieldName: String, value: Json, hidden: Boolean = false)(implicit val pos: SourcePos) extends CirceFieldMapping @@ -174,7 +174,7 @@ trait CirceMappingLike[F[_]] extends Mapping[F] { def hasField(fieldName: String): Boolean = tpe.hasField(fieldName) && focus.asObject.exists(_.contains(fieldName)) || - fieldMapping(context, fieldName).isDefined + typeMappings.fieldMapping(context, fieldName).isDefined def field(fieldName: String, resultName: Option[String]): Result[Cursor] = { val localField = diff --git a/modules/circe/src/test/scala/CircePrioritySuite.scala b/modules/circe/src/test/scala/CircePrioritySuite.scala index 16fa1fe2..9e98bc8f 100644 --- a/modules/circe/src/test/scala/CircePrioritySuite.scala +++ b/modules/circe/src/test/scala/CircePrioritySuite.scala @@ -44,7 +44,7 @@ object CircePriorityMapping extends CirceMapping[IO] { val BarrelType = schema.ref("Barrel") val FooType = schema.ref("Foo") - val typeMappings: List[TypeMapping] = + val typeMappings = List( ObjectMapping( tpe = QueryType, diff --git a/modules/core/src/main/scala/compiler.scala b/modules/core/src/main/scala/compiler.scala index 6836f044..55c5281a 100644 --- a/modules/core/src/main/scala/compiler.scala +++ b/modules/core/src/main/scala/compiler.scala @@ -400,7 +400,7 @@ class QueryCompiler(parser: QueryParser, schema: Schema, phases: List[Phase]) { else { val hd = pendingFrags.head if (seen.contains(hd)) None - else checkCycle(fragRefs(hd)._2 ++ pendingFrags.tail, seen + hd) + else checkCycle(fragRefs.get(hd).map(_._2).getOrElse(Set.empty) ++ pendingFrags.tail, seen + hd) } } @@ -1231,7 +1231,7 @@ object QueryCompiler { case class ComponentMapping[F[_]](tpe: TypeRef, fieldName: String, mapping: Mapping[F], join: (Query, Cursor) => Result[Query] = TrivialJoin) - def apply[F[_]](mappings: List[ComponentMapping[F]]): ComponentElaborator[F] = + def apply[F[_]](mappings: Seq[ComponentMapping[F]]): ComponentElaborator[F] = new ComponentElaborator(mappings.map(m => ((m.tpe, m.fieldName), (m.mapping, m.join))).toMap) } @@ -1279,7 +1279,7 @@ object QueryCompiler { object EffectElaborator { case class EffectMapping[F[_]](tpe: TypeRef, fieldName: String, handler: EffectHandler[F]) - def apply[F[_]](mappings: List[EffectMapping[F]]): EffectElaborator[F] = + def apply[F[_]](mappings: Seq[EffectMapping[F]]): EffectElaborator[F] = new EffectElaborator(mappings.map(m => ((m.tpe, m.fieldName), m.handler)).toMap) } diff --git a/modules/core/src/main/scala/composedmapping.scala b/modules/core/src/main/scala/composedmapping.scala new file mode 100644 index 00000000..4c53bcb1 --- /dev/null +++ b/modules/core/src/main/scala/composedmapping.scala @@ -0,0 +1,47 @@ +// Copyright (c) 2016-2023 Association of Universities for Research in Astronomy, Inc. (AURA) +// Copyright (c) 2016-2023 Grackle Contributors +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +package grackle + +import cats.MonadThrow + +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) + } + } + + case class ComposedCursor(context: Context, env: Env) extends AbstractCursor { + val focus = null + val parent = None + + 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 3fe62d59..689dc691 100644 --- a/modules/core/src/main/scala/cursor.scala +++ b/modules/core/src/main/scala/cursor.scala @@ -453,6 +453,12 @@ case class Context( copy(path = fieldName :: path, resultPath = resultName.getOrElse(fieldName) :: resultPath, typePath = fieldTpe :: typePath) } + def forUnderlyingNamed: Context = + typePath match { + case Nil => copy(rootTpe.underlyingNamed.dealias) + case hd :: tl => copy(typePath = hd.underlyingNamed.dealias :: tl) + } + override def equals(other: Any): Boolean = other match { case Context(oRootTpe, oPath, oResultPath, _) => diff --git a/modules/core/src/main/scala/introspection.scala b/modules/core/src/main/scala/introspection.scala index 32fa82a9..b7de8283 100644 --- a/modules/core/src/main/scala/introspection.scala +++ b/modules/core/src/main/scala/introspection.scala @@ -188,124 +188,96 @@ object Introspection { val schema = Introspection.schema val typeMappings = - List( - ValueObjectMapping[Unit]( - tpe = QueryType, - fieldMappings = - List( - ValueField("__schema", _ => targetSchema), - ValueField("__type", _ => allTypes.map(_.nullable)) - ) + TypeMappings( + ValueObjectMapping(QueryType).on[Unit]( + ValueField("__schema", _ => targetSchema), + ValueField("__type", _ => allTypes.map(_.nullable)) ), - ValueObjectMapping[Schema]( - tpe = __SchemaType, - fieldMappings = - List( - ValueField("types", _ => allTypes.map(_.nullable)), - ValueField("queryType", _.queryType.dealias.nullable), - ValueField("mutationType", _.mutationType.map(_.dealias.nullable)), - ValueField("subscriptionType", _.subscriptionType.map(_.dealias.nullable)), - ValueField("directives", _.directives) - ) + ValueObjectMapping(__SchemaType).on[Schema]( + ValueField("types", _ => allTypes.map(_.nullable)), + ValueField("queryType", _.queryType.dealias.nullable), + ValueField("mutationType", _.mutationType.map(_.dealias.nullable)), + ValueField("subscriptionType", _.subscriptionType.map(_.dealias.nullable)), + ValueField("directives", _.directives) ), - ValueObjectMapping[Type]( - tpe = __TypeType, - fieldMappings = - List( - ValueField("kind", flipNullityDealias andThen { - case _: ScalarType => TypeKind.SCALAR - case _: ObjectType => TypeKind.OBJECT - case _: UnionType => TypeKind.UNION - case _: InterfaceType => TypeKind.INTERFACE - case _: EnumType => TypeKind.ENUM - case _: InputObjectType => TypeKind.INPUT_OBJECT - case _: ListType => TypeKind.LIST - case _: NonNullType => TypeKind.NON_NULL - }), - ValueField("name", flipNullityDealias andThen { - case nt: NamedType => Some(nt.name) - case _ => None - }), - ValueField("description", flipNullityDealias andThen { - case nt: NamedType => nt.description - case _ => None - }), - ValueField("fields", flipNullityDealias andThen { - case tf: TypeWithFields => Some(tf.fields) - case _ => None - }), - ValueField("interfaces", flipNullityDealias andThen { - case ot: ObjectType => Some(ot.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 _ => None - }), - ValueField("enumValues", flipNullityDealias andThen { - case e: EnumType => Some(e.enumValues) - case _ => None - }), - ValueField("inputFields", flipNullityDealias andThen { - case i: InputObjectType => Some(i.inputFields) - case _ => None - }), - ValueField("ofType", flipNullityDealias andThen { - case l: ListType => Some(l.ofType) - case NonNullType(t) => Some(NullableType(t)) - case _ => None + ValueObjectMapping(__TypeType).on[Type]( + ValueField("kind", flipNullityDealias andThen { + case _: ScalarType => TypeKind.SCALAR + case _: ObjectType => TypeKind.OBJECT + case _: UnionType => TypeKind.UNION + case _: InterfaceType => TypeKind.INTERFACE + case _: EnumType => TypeKind.ENUM + case _: InputObjectType => TypeKind.INPUT_OBJECT + case _: ListType => TypeKind.LIST + case _: NonNullType => TypeKind.NON_NULL + }), + ValueField("name", flipNullityDealias andThen { + case nt: NamedType => Some(nt.name) + case _ => None + }), + ValueField("description", flipNullityDealias andThen { + case nt: NamedType => nt.description + case _ => None + }), + ValueField("fields", flipNullityDealias andThen { + case tf: TypeWithFields => Some(tf.fields) + case _ => None + }), + ValueField("interfaces", flipNullityDealias andThen { + case ot: ObjectType => Some(ot.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 _ => None + }), + ValueField("enumValues", flipNullityDealias andThen { + case e: EnumType => Some(e.enumValues) + case _ => None + }), + ValueField("inputFields", flipNullityDealias andThen { + case i: InputObjectType => Some(i.inputFields) + case _ => None + }), + ValueField("ofType", flipNullityDealias andThen { + case l: ListType => Some(l.ofType) + case NonNullType(t) => Some(NullableType(t)) + case _ => None + }) ), - ValueObjectMapping[Field]( - tpe = __FieldType, - fieldMappings = - List( - ValueField("name", _.name), - ValueField("description", _.description), - ValueField("args", _.args), - ValueField("type", _.tpe.dealias), - ValueField("isDeprecated", _.isDeprecated), - ValueField("deprecationReason", _.deprecationReason) - ) + ValueObjectMapping(__FieldType).on[Field]( + ValueField("name", _.name), + ValueField("description", _.description), + ValueField("args", _.args), + ValueField("type", _.tpe.dealias), + ValueField("isDeprecated", _.isDeprecated), + ValueField("deprecationReason", _.deprecationReason) ), - ValueObjectMapping[InputValue]( - tpe = __InputValueType, - fieldMappings = - List( - ValueField("name", _.name), - ValueField("description", _.description), - ValueField("type", _.tpe.dealias), - ValueField("defaultValue", _.defaultValue.map(SchemaRenderer.renderValue)) - ) + ValueObjectMapping(__InputValueType).on[InputValue]( + ValueField("name", _.name), + ValueField("description", _.description), + ValueField("type", _.tpe.dealias), + ValueField("defaultValue", _.defaultValue.map(SchemaRenderer.renderValue)) ), - ValueObjectMapping[EnumValueDefinition]( - tpe = __EnumValueType, - fieldMappings = - List( - ValueField("name", _.name), - ValueField("description", _.description), - ValueField("isDeprecated", _.isDeprecated), - ValueField("deprecationReason", _.deprecationReason) - ) + ValueObjectMapping(__EnumValueType).on[EnumValueDefinition]( + ValueField("name", _.name), + ValueField("description", _.description), + ValueField("isDeprecated", _.isDeprecated), + ValueField("deprecationReason", _.deprecationReason) ), - ValueObjectMapping[DirectiveDef]( - tpe = __DirectiveType, - fieldMappings = - List( - ValueField("name", _.name), - ValueField("description", _.description), - ValueField("locations", _.locations), - ValueField("args", _.args), - ValueField("isRepeatable", _.isRepeatable) - ) + ValueObjectMapping(__DirectiveType).on[DirectiveDef]( + ValueField("name", _.name), + ValueField("description", _.description), + ValueField("locations", _.locations), + ValueField("args", _.args), + ValueField("isRepeatable", _.isRepeatable) ), LeafMapping[TypeKind.Value](__TypeKindType), LeafMapping[Ast.DirectiveLocation](__DirectiveLocationType) - ) + ) } } diff --git a/modules/core/src/main/scala/mapping.scala b/modules/core/src/main/scala/mapping.scala index aaa9cde4..a8d5bde5 100644 --- a/modules/core/src/main/scala/mapping.scala +++ b/modules/core/src/main/scala/mapping.scala @@ -16,16 +16,18 @@ package grackle import scala.collection.Factory +import scala.collection.mutable.{ Map => MMap } import scala.reflect.ClassTag -import cats.MonadThrow -import cats.data.Chain +import cats.{ApplicativeError, Id, MonadThrow} +import cats.data.{Chain, NonEmptyList, StateT} import cats.implicits._ import fs2.{ Stream, Compiler } import io.circe.{Encoder, Json} import io.circe.syntax._ import org.tpolecat.sourcepos.SourcePos import org.tpolecat.typename._ +import org.typelevel.scalaccompat.annotation._ import syntax._ import Cursor.{AbstractCursor, ProxyCursor} @@ -33,6 +35,7 @@ import Query.EffectHandler import QueryCompiler.{ComponentElaborator, EffectElaborator, IntrospectionLevel, SelectElaborator} import QueryInterpreter.ProtoJson import IntrospectionLevel._ +import ValidationFailure.Severity /** * Represents a mapping between a GraphQL schema and an underlying abstract data source. @@ -40,7 +43,7 @@ import IntrospectionLevel._ abstract class Mapping[F[_]] { implicit val M: MonadThrow[F] val schema: Schema - val typeMappings: List[TypeMapping] + val typeMappings: TypeMappings /** * Compile and run a single GraphQL query or mutation. @@ -102,7 +105,7 @@ abstract class Mapping[F[_]] { def focus: Any = () override def hasField(fieldName: String): Boolean = - fieldMapping(context, fieldName).isDefined + typeMappings.fieldMapping(context, fieldName).isDefined override def field(fieldName: String, resultName: Option[String]): Result[Cursor] = mkCursorForField(this, fieldName, resultName) @@ -121,7 +124,7 @@ abstract class Mapping[F[_]] { def mkLeafCursor(focus: Any): Result[Cursor] = LeafCursor(fieldContext, focus, Some(parent), parent.env).success - fieldMapping(context, fieldName) match { + typeMappings.fieldMapping(context, fieldName) match { case Some(_ : EffectMapping) => mkLeafCursor(parent.focus) case Some(CursorField(_, f, _, _, _)) => @@ -131,154 +134,592 @@ abstract class Mapping[F[_]] { } } - /** Yields the `TypeMapping` associated with the provided type, if any. */ - def typeMapping(tpe: NamedType): Option[TypeMapping] = - typeMappingIndex.get(tpe.name) + case class TypeMappings private (mappings: Seq[TypeMapping], unsafe: Boolean) { + /** 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 { + val nc = context.asType(nt) + multipleIndex.get(nme).getOrElse(Nil).mapFilter { tm => + tm.predicate(nc).map(prio => (prio, tm)) + }.maxByOption(_._1).map(_._2) + } + } - private lazy val typeMappingIndex = - typeMappings.flatMap(tm => tm.tpe.asNamed.map(tpe => (tpe.name, tm)).toList).toMap + 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 validator: MappingValidator = - MappingValidator(this) + 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) + } + } - /** Yields the `ObjectMapping` associated with the provided context, if any. */ - def objectMapping(context: Context): Option[ObjectMapping] = - context.tpe.underlyingObject.flatMap { obj => - obj.asNamed.flatMap(typeMapping) match { - case Some(om: ObjectMapping) => Some(om) - case Some(pm: PrefixedMapping) => - val revPath = context.path.reverse - pm.mappings.filter(m => revPath.endsWith(m._1)).maxByOption(_._1.length).map(_._2) - case _ => None + 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) + ) } + + 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 `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 + /** Yields the `ObjectMapping` associated with the provided context, if any. */ + def objectMapping(context: Context): Option[ObjectMapping] = + typeMapping(context).collect { + case om: ObjectMapping => om + } + + /** 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 + } } + + /** Yields the `FieldMapping` directly or ancestrally associated with `fieldName` in `context`, if any. */ + def ancestralFieldMapping(context: Context, fieldName: String): Option[FieldMapping] = + fieldMapping(context, fieldName).orElse { + for { + parent <- context.parent + fm <- ancestralFieldMapping(parent, context.path.head) + if fm.subtree + } 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] = { + val queryType = schema.schemaType.field("query").flatMap(_.nonNull.asNamed) + val topLevelContexts = (queryType.toList ::: schema.mutationType.toList ::: schema.subscriptionType.toList).map(Context(_)) + validateRoots(topLevelContexts) } + /** Validates these type mappings against an unfolding of the schema */ + @nowarn3 + private def validateRoots(rootCtxts: List[Context]): List[ValidationFailure] = { + import TypeMappings.{MappingValidator => MV} // Bogus unused import warning with Scala 3.3.3 + import MV.{ + initialState, addSeenType, addSeenTypeMapping, addSeenFieldMapping, addProblem, + addProblems, seenType, seenTypeMappings, seenFieldMappings + } + + def allPrefixedMatchContexts(ctx: Context): Seq[Context] = { + def hasPath(ctx: Context, path: List[String]): Option[Context] = + if(path.isEmpty) Some(ctx) + else + ctx.tpe.underlyingNamed.dealias match { + case ot: ObjectType => + ot.fieldInfo(path.head) match { + case Some(_) => + hasPath(ctx.forFieldOrAttribute(path.head, None), path.tail) + case None => + None + } + case it: InterfaceType => + it.fieldInfo(path.head) match { + case Some(_) => + hasPath(ctx.forFieldOrAttribute(path.head, None), path.tail) + case None => + val implementors = schema.types.collect { case ot: ObjectType if ot <:< it => ot } + implementors.collectFirstSome(tpe => hasPath(ctx.asType(tpe), path)) + } + case ut: UnionType => + ut.members.collectFirstSome(tpe => hasPath(ctx.asType(tpe), path)) + + case _ => + None + } + + mappings.flatMap { tm => + tm.predicate match { + case p: MappingPredicate.PathMatch if p.path.rootTpe =:= ctx.tpe => + hasPath(ctx, p.path.path) + case p: MappingPredicate.PrefixedTypeMatch => + hasPath(ctx, p.prefix) + case _ => Nil + } + } + } + + def step(context: Context): MV[List[Context]] = { + lazy val hasEnclosingSubtreeFieldMapping = + if (context.path.isEmpty) false + else ancestralFieldMapping(context.parent.get, context.path.head).map(_.subtree).getOrElse(false) + + def typeChecks(om: ObjectMapping): List[ValidationFailure] = + validateTypeMapping(this, context, om) ++ + om.fieldMappings.reverse.flatMap(validateFieldMapping(this, context, om, _)) + + (context.tpe.underlyingNamed.dealias, typeMapping(context)) match { + case (lt@((_: ScalarType) | (_: EnumType)), Some(lm: LeafMapping[_])) => + addSeenType(lt) *> + addSeenTypeMapping(lm) *> + MV.pure(Nil) + + case (lt@((_: ScalarType) | (_: EnumType)), None) if hasEnclosingSubtreeFieldMapping => + addSeenType(lt) *> + MV.pure(Nil) + + case (lt@((_: ScalarType) | (_: EnumType)), Some(om: ObjectMapping)) => + addSeenType(lt) *> + addSeenTypeMapping(om) *> + addProblem(ObjectTypeExpected(om)) *> + MV.pure(Nil) + + case (ut: UnionType, Some(om: ObjectMapping)) => + val vs = ut.members.map(context.asType) + addSeenType(ut) *> + addSeenTypeMapping(om) *> + addProblems(typeChecks(om)) *> + MV.pure(vs) + + case (ut: UnionType, Some(lm: LeafMapping[_])) => + val vs = ut.members.map(context.asType) + addSeenType(ut) *> + addSeenTypeMapping(lm) *> + addProblem(LeafTypeExpected(lm)) *> + MV.pure(vs) + + case (ut: UnionType, None) => + val vs = ut.members.map(context.asType) + addSeenType(ut) *> + MV.pure(vs) + + case (twf: TypeWithFields, tm) => + def objectCheck(seen: Boolean) = + (twf, tm) match { + case (_, Some(om: ObjectMapping)) => + addSeenTypeMapping(om) *> + addProblems(typeChecks(om)).whenA(!seen) + + case (_, Some(lm: LeafMapping[_])) => + addSeenTypeMapping(lm) *> + addProblem(LeafTypeExpected(lm)) + + case (_: InterfaceType, None) => + MV.unit + + case (_, None) if !hasEnclosingSubtreeFieldMapping => + addProblem(MissingTypeMapping(context.tpe)) + + case _ => + MV.unit + } + + val implCtxts = + twf match { + case it: InterfaceType => + schema.types.collect { + case ot: ObjectType if ot <:< it => context.asType(ot) + } + case _ => Nil + } + + def interfaceContext(it: NamedType): MV[Option[Context]] = { + val v = context.asType(it) + for { + seen <- seenType(v.tpe.underlyingNamed) + } yield if(seen) None else Some(v) + } + + val fieldNames = twf.fields.map(_.name) + + def fieldCheck(fieldName: String): MV[Option[Context]] = { + val fctx = context.forFieldOrAttribute(fieldName, None).forUnderlyingNamed + ((ancestralFieldMapping(context, fieldName), tm) match { + case (Some(fm), Some(om: ObjectMapping)) => + addSeenFieldMapping(om, fm) + + case (None, Some(om: ObjectMapping)) if !hasEnclosingSubtreeFieldMapping => + val field = context.tpe.fieldInfo(fieldName).get + addProblem(MissingFieldMapping(om, field)) + + case _ => // Other errors will have been reported earlier + MV.unit + }) *> + seenType(fctx.tpe).map(if(_) None else Some(fctx)) + } + + for { + seen <- seenType(twf) + _ <- addSeenType(twf) + _ <- objectCheck(seen) + ifCtxts <- twf.allInterfaces.traverseFilter(interfaceContext) + fCtxts <- fieldNames.traverseFilter(fieldCheck) + pfCtxts <- MV.pure(if (!seen) allPrefixedMatchContexts(context) else Nil) + } yield implCtxts ++ ifCtxts ++ fCtxts ++ pfCtxts + + case (_: TypeRef | _: InputObjectType, _) => + MV.pure(Nil) // Errors will have been reported earlier + + case (tpe, None) => + addProblem(MissingTypeMapping(tpe)) *> + MV.pure(Nil) + } + } + + def validateAll(pending: List[Context]): MV[Unit] = { + pending.tailRecM { + case Nil => + MV.pure(Right(())) + case head :: tail => + step(head).map(next => Left(next ::: tail)) + } + } + + def unseenTypeMappings(seen: Set[TypeMapping]): Seq[TypeMapping] = { + @annotation.tailrec + def loop(pending: Seq[TypeMapping], acc: Seq[TypeMapping]): Seq[TypeMapping] = + pending match { + case Seq(om: ObjectMapping, tail @ _*) => + if (seen(om) || om.fieldMappings.forall { case _: Delegate => true ; case _ => false }) + loop(tail, acc) + else + loop(tail, om +: acc) + + case Seq(tm, tail @ _*) if seen(tm) => + loop(tail, acc) + + case Seq(tm, tail @ _*) => + loop(tail, tm +: acc) + + case _ => acc.reverse + } + + loop(mappings, Nil) + } + + def refChecks(tm: TypeMapping): MV[Unit] = { + addProblem(ReferencedTypeDoesNotExist(tm)).whenA(!tm.tpe.exists) *> + (tm match { + 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.traverse_(fm => addProblem(UnusedFieldMapping(om, fm))) + } yield () + case _ => + MV.unit + }) + } + + val res = + for { + _ <- addProblem(MissingTypeMapping(schema.uncheckedRef("Query"))).whenA(schema.schemaType.field("query").isEmpty) + _ <- validateAll(rootCtxts) + seen <- seenTypeMappings + _ <- unseenTypeMappings(seen).traverse_(tm => addProblem(UnusedTypeMapping(tm))) + _ <- mappings.traverse_(refChecks) + } yield () + + res.runS(initialState).problems.reverse + } + } + + object TypeMappings { + def apply(mappings: Seq[TypeMapping]): TypeMappings = + new TypeMappings(mappings, false) + + def apply(mappings: TypeMapping*)(implicit dummy: DummyImplicit): TypeMappings = + apply(mappings) + + def unsafe(mappings: Seq[TypeMapping]): TypeMappings = + new TypeMappings(mappings, true) + + def unsafe(mappings: TypeMapping*)(implicit dummy: DummyImplicit): TypeMappings = + unsafe(mappings) + + implicit def fromList(mappings: List[TypeMappingCompat]): TypeMappings = TypeMappings(mappings.flatMap(_.unwrap)) + + val empty: TypeMappings = unsafe(Nil) + + private type MappingValidator[T] = StateT[Id, MappingValidator.State, T] + private object MappingValidator { + type MV[T] = MappingValidator[T] + def unit: MV[Unit] = StateT.pure(()) + def pure[T](t: T): MV[T] = StateT.pure(t) + def addProblem(p: ValidationFailure): MV[Unit] = StateT.modify(_.addProblem(p)) + def addProblems(ps: List[ValidationFailure]): MV[Unit] = StateT.modify(_.addProblems(ps)) + def seenType(tpe: Type): MV[Boolean] = StateT.inspect(_.seenType(tpe)) + def addSeenType(tpe: Type): MV[Unit] = StateT.modify(_.addSeenType(tpe)) + def addSeenTypeMapping(tm: TypeMapping): MV[Unit] = StateT.modify(_.addSeenTypeMapping(tm)) + def seenTypeMappings: MV[Set[TypeMapping]] = StateT.inspect(_.seenTypeMappings) + def addSeenFieldMapping(om: ObjectMapping, fm: FieldMapping): MV[Unit] = StateT.modify(_.addSeenFieldMapping(om, fm)) + def seenFieldMappings(om: ObjectMapping): MV[Set[FieldMapping]] = StateT.inspect(_.seenFieldMappings(om)) + + case class State( + seenTypes: Set[String], + seenTypeMappings: Set[TypeMapping], + seenFieldMappings0: Map[ObjectMapping, Set[FieldMapping]], + problems: List[ValidationFailure] + ) { + def addProblem(p: ValidationFailure): State = + copy(problems = p :: problems) + def addProblems(ps: List[ValidationFailure]): State = + copy(problems = ps ::: problems) + def seenType(tpe: Type): Boolean = + tpe match { + case nt: NamedType => seenTypes(nt.name) + case _ => false + } + def addSeenType(tpe: Type): State = + tpe match { + case nt: NamedType => copy(seenTypes = seenTypes + nt.name) + case _ => this + } + + def addSeenTypeMapping(tm: TypeMapping): State = + copy(seenTypeMappings = seenTypeMappings + tm) + + def addSeenFieldMapping(om: ObjectMapping, fm: FieldMapping): State = + copy(seenTypeMappings = seenTypeMappings + om, seenFieldMappings0 = seenFieldMappings0.updatedWith(om)(_.map(_ + fm).orElse(Set(fm).some))) + + def seenFieldMappings(om: ObjectMapping): Set[FieldMapping] = + seenFieldMappings0.getOrElse(om, Set.empty) + } + + def initialState: State = + State(Set.empty, Set.empty, Map.empty, Nil) + } + } + + + /** Check Mapping specific TypeMapping validity */ + protected def validateTypeMapping(mappings: TypeMappings, context: Context, tm: TypeMapping): List[ValidationFailure] = Nil + + /** Check Mapping specific FieldMapping validity */ + protected def validateFieldMapping(mappings: TypeMappings, context: Context, om: ObjectMapping, fm: FieldMapping): List[ValidationFailure] = Nil + + /** + * 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) + } + + /** + * Run this validator, 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))) + + /** + * Run this validator, raising 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 _, _ => ()) + /** Yields the `RootEffect`, if any, associated with `fieldName`. */ def rootEffect(context: Context, fieldName: String): Option[RootEffect] = - fieldMapping(context, fieldName).collect { + typeMappings.fieldMapping(context, fieldName).collect { case re: RootEffect => re } /** Yields the `RootStream`, if any, associated with `fieldName`. */ def rootStream(context: Context, fieldName: String): Option[RootStream] = - fieldMapping(context, fieldName).collect { + typeMappings.fieldMapping(context, fieldName).collect { case rs: RootStream => rs } - /** Yields the `LeafMapping` associated with the provided type, if any. */ - def leafMapping[T](tpe: Type): Option[LeafMapping[T]] = - typeMappings.collectFirst { - case lm@LeafMapping(tpe0, _) if tpe0 =:= tpe => lm.asInstanceOf[LeafMapping[T]] + /** Yields the `Encoder` associated with the provided leaf context, if any. */ + def encoderForLeaf(context: Context): Option[Encoder[Any]] = + typeMappings.typeMapping(context).collect { + case lm: LeafMapping[_] => lm.encoder.asInstanceOf[Encoder[Any]] } - /** - * True if the supplied type is a leaf with respect to the GraphQL schema - * or mapping, false otherwise. - */ - def isLeaf(tpe: Type): Boolean = tpe.underlying match { - case (_: ScalarType)|(_: EnumType) => true - case tpe => leafMapping(tpe).isDefined + sealed trait TypeMappingCompat { + private[grackle] def unwrap: Seq[TypeMapping] = + this match { + case TypeMappingCompat.PrefixedMappingCompat(mappings) => mappings + case tm: TypeMapping => Seq(tm) + } } - /** Yields the `Encoder` associated with the provided type, if any. */ - def encoderForLeaf(tpe: Type): Option[Encoder[Any]] = - encoderMemo.get(tpe.dealias) + object TypeMappingCompat { + case class PrefixedMappingCompat(mappings0: Seq[TypeMapping]) extends TypeMappingCompat + } - private lazy val encoderMemo: scala.collection.immutable.Map[Type, Encoder[Any]] = { - 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) - } - } + protected def unpackPrefixedMapping(prefix: List[String], om: ObjectMapping): ObjectMapping = + om match { + case om: ObjectMapping.DefaultObjectMapping => + om.copy(predicate = MappingPredicate.PrefixedTypeMatch(prefix, om.predicate.tpe)) + case other => other + } - 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) - } - } + /** Backwards compatible constructor for legacy `PrefixedMapping` */ + def PrefixedMapping(tpe: NamedType, mappings: List[(List[String], ObjectMapping)]): TypeMappingCompat.PrefixedMappingCompat = { + if (!mappings.forall(_._2.predicate.tpe =:= tpe)) + throw new IllegalArgumentException("All prefixed mappings must have the same type") + + TypeMappingCompat.PrefixedMappingCompat( + mappings.map { case (prefix, om) => unpackPrefixedMapping(prefix, om) } + ) + } - val definedEncoders: List[(Type, Encoder[Any])] = - typeMappings.collect { case lm: LeafMapping[_] => (lm.tpe.dealias -> lm.encoder.asInstanceOf[Encoder[Any]]) } + sealed trait TypeMapping extends TypeMappingCompat with Product with Serializable { + def predicate: MappingPredicate + def pos: SourcePos - val defaultEncoders: List[(Type, Encoder[Any])] = - List( - ScalarType.StringType -> Encoder[String].asInstanceOf[Encoder[Any]], - ScalarType.IntType -> intTypeEncoder, - ScalarType.FloatType -> floatTypeEncoder, - ScalarType.BooleanType -> Encoder[Boolean].asInstanceOf[Encoder[Any]], - ScalarType.IDType -> Encoder[String].asInstanceOf[Encoder[Any]] - ) + def tpe: NamedType = predicate.tpe - (definedEncoders ++ defaultEncoders).toMap + def showMappingType: String = productPrefix } - trait TypeMapping extends Product with Serializable { - def tpe: Type - def pos: SourcePos + /** A predicate determining the applicability of a `TypeMapping` in a given context */ + trait MappingPredicate { + /** The type to which this predicate applies */ + def tpe: NamedType + /** + * Does this predicate apply to the given context? + * + * Yields the priority of the corresponding `TypeMapping` if the predicate applies, + * or `None` otherwise. + */ + def apply(ctx: Context): Option[Int] } - case class PrimitiveMapping(tpe: Type)(implicit val pos: SourcePos) extends TypeMapping + object MappingPredicate { + /** A predicate that matches a specific type in any context */ + case class TypeMatch(tpe: NamedType) extends MappingPredicate { + def apply(ctx: Context): Option[Int] = + if (ctx.tpe =:= tpe) + Some(0) + else + None + } + + /** + * A predicate that matches a specific type with a given path prefix. + * + * This predicate corresponds to the semantics of the `PrefixedMapping` in earlier + * releases. + */ + case class PrefixedTypeMatch(prefix: List[String], tpe: NamedType) extends MappingPredicate { + def apply(ctx: Context): Option[Int] = + if ( + ctx.tpe =:= tpe && + ctx.path.startsWith(prefix.reverse) + ) + Some(prefix.length) + else + None + } - abstract class ObjectMapping extends TypeMapping { - def tpe: NamedType + /** + * A predicate that matches the given `Path` as the suffix of the context path. + * + * Note that a `Path` corresponds to an initial type, followed by a sequence of + * field selectors. The type found by following the field selectors from the initial + * type determines the final type to which the predicate applies. + * + * This predicate is thus a slightly more restrictive variant of `PrefixedTypeMatch` + * which will match in any context with the given path and final type, irrespective + * of the initial type. + * + * In practice `PathMatch` is more convenient to use in most + * circumstances and should be preferred to `PrefixedTypeMatch` unless the semantics + * of the latter are absolutely required. + */ + case class PathMatch(path: Path) extends MappingPredicate { + lazy val tpe: NamedType = path.tpe.get.underlyingNamed + def apply(ctx: Context): Option[Int] = + if ( + ctx.tpe =:= tpe && + ctx.path.startsWith(path.path.reverse) && + ctx.typePath.drop(path.path.length).headOption.exists(_ =:= path.rootTpe) + ) + Some(path.path.length+1) + else + None + } + } + abstract class ObjectMapping extends TypeMapping { private lazy val fieldMappingIndex = fieldMappings.map(fm => (fm.fieldName, fm)).toMap - def fieldMappings: List[FieldMapping] + def fieldMappings: Seq[FieldMapping] def fieldMapping(fieldName: String): Option[FieldMapping] = fieldMappingIndex.get(fieldName) } object ObjectMapping { - - case class DefaultObjectMapping(tpe: NamedType, fieldMappings: List[FieldMapping])( + case class DefaultObjectMapping(predicate: MappingPredicate, fieldMappings: Seq[FieldMapping])( implicit val pos: SourcePos - ) extends ObjectMapping + ) extends ObjectMapping { + override def showMappingType: String = "ObjectMapping" + } + + def apply(predicate: MappingPredicate)(fieldMappings: FieldMapping*)( + implicit pos: SourcePos + ): ObjectMapping = + DefaultObjectMapping(predicate, fieldMappings) + + def apply(tpe: NamedType)(fieldMappings: FieldMapping*)( + implicit pos: SourcePos + ): ObjectMapping = + DefaultObjectMapping(MappingPredicate.TypeMatch(tpe), fieldMappings) def apply(tpe: NamedType, fieldMappings: List[FieldMapping])( implicit pos: SourcePos ): ObjectMapping = - DefaultObjectMapping(tpe, fieldMappings.map(_.withParent(tpe))) + DefaultObjectMapping(MappingPredicate.TypeMatch(tpe), fieldMappings) } - case class PrefixedMapping(tpe: Type, mappings: List[(List[String], ObjectMapping)])( - implicit val pos: SourcePos - ) extends TypeMapping - trait FieldMapping extends Product with Serializable { def fieldName: String def hidden: Boolean - def withParent(tpe: Type): FieldMapping + def subtree: Boolean def pos: SourcePos - } - case class PrimitiveField(fieldName: String, hidden: Boolean = false)(implicit val pos: SourcePos) extends FieldMapping { - def withParent(tpe: Type): PrimitiveField = this + def showMappingType: String = productPrefix } /** * Abstract type of field mappings with effects. */ - trait EffectMapping extends FieldMapping + trait EffectMapping extends FieldMapping { + def subtree: Boolean = true + } case class EffectField(fieldName: String, handler: EffectHandler[F], required: List[String] = Nil, hidden: Boolean = false)(implicit val pos: SourcePos) - extends EffectMapping { - def withParent(tpe: Type): EffectField = this - } + extends EffectMapping /** * Root effects can perform an intial effect prior to computing the resulting @@ -298,7 +739,6 @@ abstract class Mapping[F[_]] { case class RootEffect private (fieldName: String, effect: (Query, Path, Env) => F[Result[(Query, Cursor)]])(implicit val pos: SourcePos) extends EffectMapping { def hidden = false - def withParent(tpe: Type): RootEffect = this def toRootStream: RootStream = RootStream(fieldName)((q, p, e) => Stream.eval(effect(q, p, e))) } @@ -366,7 +806,6 @@ abstract class Mapping[F[_]] { case class RootStream private (fieldName: String, effect: (Query, Path, Env) => Stream[F, Result[(Query, Cursor)]])(implicit val pos: SourcePos) extends EffectMapping { def hidden = false - def withParent(tpe: Type): RootStream = this } object RootStream { @@ -413,19 +852,24 @@ abstract class Mapping[F[_]] { } trait LeafMapping[T] extends TypeMapping { - def tpe: Type def encoder: Encoder[T] def scalaTypeName: String def pos: SourcePos } + object LeafMapping { - case class DefaultLeafMapping[T](tpe: Type, encoder: Encoder[T], scalaTypeName: String)( + case class DefaultLeafMapping[T](predicate: MappingPredicate, encoder: Encoder[T], scalaTypeName: String)( implicit val pos: SourcePos - ) extends LeafMapping[T] + ) extends LeafMapping[T] { + override def showMappingType: String = "LeafMapping" + } + + def apply[T: TypeName](predicate: MappingPredicate)(implicit encoder: Encoder[T], pos: SourcePos): LeafMapping[T] = + DefaultLeafMapping(predicate, encoder, typeName) - def apply[T: TypeName](tpe: Type)(implicit encoder: Encoder[T], pos: SourcePos): LeafMapping[T] = - DefaultLeafMapping(tpe, encoder, typeName) + def apply[T: TypeName](tpe: NamedType)(implicit encoder: Encoder[T], pos: SourcePos): LeafMapping[T] = + DefaultLeafMapping(MappingPredicate.TypeMatch(tpe), encoder, typeName) def unapply[T](lm: LeafMapping[T]): Option[(Type, Encoder[T])] = Some((lm.tpe, lm.encoder)) @@ -434,7 +878,7 @@ abstract class Mapping[F[_]] { case class CursorField[T](fieldName: String, f: Cursor => Result[T], encoder: Encoder[T], required: List[String], hidden: Boolean)( implicit val pos: SourcePos ) extends FieldMapping { - def withParent(tpe: Type): CursorField[T] = this + def subtree = false } object CursorField { def apply[T](fieldName: String, f: Cursor => Result[T], required: List[String] = Nil, hidden: Boolean = false)(implicit encoder: Encoder[T], di: DummyImplicit): CursorField[T] = @@ -447,14 +891,14 @@ abstract class Mapping[F[_]] { join: (Query, Cursor) => Result[Query] = ComponentElaborator.TrivialJoin )(implicit val pos: SourcePos) extends FieldMapping { def hidden = false - def withParent(tpe: Type): Delegate = this + def subtree = true } val selectElaborator: SelectElaborator = SelectElaborator.identity lazy val componentElaborator = { val componentMappings = - typeMappings.flatMap { + typeMappings.mappings.flatMap { case om: ObjectMapping => om.fieldMappings.collect { case Delegate(fieldName, mapping, join) => @@ -468,7 +912,7 @@ abstract class Mapping[F[_]] { lazy val effectElaborator = { val effectMappings = - typeMappings.flatMap { + typeMappings.mappings.flatMap { case om: ObjectMapping => om.fieldMappings.collect { case EffectField(fieldName, handler, _, _) => @@ -486,7 +930,15 @@ abstract class Mapping[F[_]] { lazy val graphQLParser: GraphQLParser = GraphQLParser(parserConfig) lazy val queryParser: QueryParser = QueryParser(graphQLParser) - lazy val compiler: QueryCompiler = new QueryCompiler(queryParser, schema, compilerPhases) + private def deferredValidate(): Unit = { + if(!typeMappings.unsafe) + unsafeValidate() + } + + lazy val compiler: QueryCompiler = { + deferredValidate() + new QueryCompiler(queryParser, schema, compilerPhases) + } val interpreter: QueryInterpreter[F] = new QueryInterpreter(this) @@ -500,7 +952,7 @@ abstract class Mapping[F[_]] { def isLeaf: Boolean = tpe.isLeaf def asLeaf: Result[Json] = - encoderForLeaf(tpe).map(enc => enc(focus).success).getOrElse(Result.internalError( + encoderForLeaf(context).map(enc => enc(focus).success).getOrElse(Result.internalError( s"Cannot encode value $focus at ${context.path.reverse.mkString("/")} (of GraphQL type ${context.tpe}). Did you forget a LeafMapping?".stripMargin.trim )) @@ -520,12 +972,12 @@ abstract class Mapping[F[_]] { } def asList[C](factory: Factory[Cursor, C]): Result[C] = (tpe, focus) match { - case (ListType(tpe), it: List[_]) => it.view.map(f => mkChild(context.asType(tpe), focus = f)).to(factory).success + case (ListType(tpe), it: Seq[_]) => it.view.map(f => mkChild(context.asType(tpe), focus = f)).to(factory).success case _ => Result.internalError(s"Expected List type, found $tpe") } def listSize: Result[Int] = (tpe, focus) match { - case (ListType(_), it: List[_]) => it.size.success + case (ListType(_), it: Seq[_]) => it.size.success case _ => Result.internalError(s"Expected List type, found $tpe") } @@ -595,30 +1047,119 @@ abstract class Mapping[F[_]] { case Result.InternalError(err) => M.raiseError(err) case _ => mkResponse(result.toOption, result.toProblems).pure[F] } -} -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) - fieldMapping(context, fieldName) match { - case Some(_) => - ComposedCursor(fieldContext, parent.env).success - case _ => - super.mkCursorForField(parent, fieldName, resultName) - } + /** Missing type mapping. */ + case class MissingTypeMapping(tpe: Type) + extends ValidationFailure(Severity.Error) { + override def toString: String = + s"$productPrefix(${showNamedType(tpe)})" + override def formattedMessage: String = + s"""|Missing type mapping. + | + |- The type ${graphql(showNamedType(tpe))} is defined by a Schema at (1). + |- ${UNDERLINED}No mapping was found for this type.$RESET + | + |(1) ${schema.pos} + |""".stripMargin } - case class ComposedCursor(context: Context, env: Env) extends AbstractCursor { - val focus = null - val parent = None + /** Object type `owner` declares `field` but no such mapping exists. */ + case class MissingFieldMapping(objectMapping: ObjectMapping, field: Field) + extends ValidationFailure(Severity.Error) { + override def toString: String = + s"$productPrefix(${showNamedType(objectMapping.tpe)}.${field.name}:${showType(field.tpe)})" + override def formattedMessage: String = + s"""|Missing field mapping. + | + |- The field ${graphql(s"${showNamedType(objectMapping.tpe)}.${field.name}:${showType(field.tpe)}")} is defined by a Schema at (1). + |- The ${scala(objectMapping.showMappingType)} for ${graphql(showNamedType(objectMapping.tpe))} at (2) ${UNDERLINED}does not define a mapping for this field$RESET. + | + |(1) ${schema.pos} + |(2) ${objectMapping.pos} + |""".stripMargin + } - def withEnv(env0: Env): Cursor = copy(env = env.add(env0)) + /** GraphQL type isn't applicable for mapping type. */ + case class ObjectTypeExpected(objectMapping: ObjectMapping) + extends ValidationFailure(Severity.Error) { + override def toString: String = + s"$productPrefix(${objectMapping.showMappingType}, ${showNamedType(objectMapping.tpe)})" + override def formattedMessage: String = + s"""|Inapplicable GraphQL type. + | + |- The ${typeKind(objectMapping.tpe)} ${graphql(showNamedType(objectMapping.tpe))} is defined by a Schema at (1). + |- It is mapped by the ${scala(objectMapping.showMappingType)} at (2), which expects an object type. + |- ${UNDERLINED}Use a different kind of mapping for this type.$RESET + | + |(1) ${schema.pos} + |(2) ${objectMapping.pos} + |""".stripMargin + } - override def hasField(fieldName: String): Boolean = - fieldMapping(context, fieldName).isDefined + /** GraphQL type isn't applicable for mapping type. */ + case class LeafTypeExpected(leafMapping: LeafMapping[_]) + extends ValidationFailure(Severity.Error) { + override def toString: String = + s"$productPrefix(${leafMapping.showMappingType}, ${showNamedType(leafMapping.tpe)})" + override def formattedMessage: String = + s"""|Inapplicable GraphQL type. + | + |- The ${typeKind(leafMapping.tpe)} ${graphql(showNamedType(leafMapping.tpe))} is defined by a Schema at (1). + |- It is mapped by the ${scala(leafMapping.showMappingType)} at (2), which expects a leaf type. + |- ${UNDERLINED}Use a different kind of mapping for this type.$RESET + | + |(1) ${schema.pos} + |(2) ${leafMapping.pos} + |""".stripMargin + } - override def field(fieldName: String, resultName: Option[String]): Result[Cursor] = - mkCursorForField(this, fieldName, resultName) + + /** Referenced type does not exist. */ + case class ReferencedTypeDoesNotExist(typeMapping: TypeMapping) + extends ValidationFailure(Severity.Error) { + override def toString: String = + s"$productPrefix(${typeMapping.showMappingType}, ${showNamedType(typeMapping.tpe)})" + override def formattedMessage: String = + s"""|Referenced type does not exist. + | + |- The ${scala(typeMapping.showMappingType)} at (1) references type ${graphql(showNamedType(typeMapping.tpe))}. + |- ${UNDERLINED}This type is undeclared$RESET in the Schema defined at (2). + | + |(1) ${typeMapping.pos} + |(2) ${schema.pos} + |""".stripMargin + } + + /** Type mapping is unused. */ + case class UnusedTypeMapping(typeMapping: TypeMapping) + extends ValidationFailure(Severity.Warning) { + override def toString: String = + s"$productPrefix(${typeMapping.showMappingType}, ${showNamedType(typeMapping.tpe)})" + override def formattedMessage: String = + s"""|Type mapping is unused. + | + |- The ${scala(typeMapping.showMappingType)} at (1) references type ${graphql(showNamedType(typeMapping.tpe))}. + |- ${UNDERLINED}This type mapping is unused$RESET by queries conforming to the Schema at (2). + | + |(1) ${typeMapping.pos} + |(2) ${schema.pos} + |""".stripMargin + } + + /** Referenced field does not exist. */ + case class UnusedFieldMapping(objectMapping: ObjectMapping, fieldMapping: FieldMapping) + extends ValidationFailure(Severity.Error) { + override def toString: String = + s"$productPrefix(${showNamedType(objectMapping.tpe)}.${fieldMapping.fieldName})" + override def formattedMessage: String = + s"""|Field mapping is unused. + | + |- The ${scala(objectMapping.showMappingType)} at (1) contains a ${scala(fieldMapping.showMappingType)} mapping for field ${graphql(fieldMapping.fieldName)} at (2). + |- ${UNDERLINED}This field mapping is unused$RESET by queries conforming to the Schema at (3). + | + |(1) ${objectMapping.pos} + |(2) ${fieldMapping.pos} + |(3) ${schema.pos} + |""".stripMargin } } diff --git a/modules/core/src/main/scala/mappingvalidator.scala b/modules/core/src/main/scala/mappingvalidator.scala deleted file mode 100644 index 8d1510b6..00000000 --- a/modules/core/src/main/scala/mappingvalidator.scala +++ /dev/null @@ -1,303 +0,0 @@ -// Copyright (c) 2016-2023 Association of Universities for Research in Astronomy, Inc. (AURA) -// Copyright (c) 2016-2023 Grackle Contributors -// -// Licensed under the Apache License, Version 2.0 (the "License"); -// you may not use this file except in compliance with the License. -// You may obtain a copy of the License at -// -// http://www.apache.org/licenses/LICENSE-2.0 -// -// Unless required by applicable law or agreed to in writing, software -// distributed under the License is distributed on an "AS IS" BASIS, -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -// See the License for the specific language governing permissions and -// limitations under the License. - -package grackle - -import cats._ -import cats.data.{ Chain, NonEmptyList } -import cats.implicits._ -import scala.io.AnsiColor -import scala.util.control.NoStackTrace - -trait MappingValidator { - - type F[_] - type M <: Mapping[F] - - val mapping: M - - import MappingValidator._ - import mapping._ - - /** Can't validate this kind of `FieldMapping`. */ - case class CannotValidateTypeMapping(typeMapping: TypeMapping) - extends Failure(Severity.Info, typeMapping.tpe, None) { - override def toString: String = - s"$productPrefix(${typeMapping.tpe}, ${typeMapping.productPrefix})" - override def formattedMessage: String = - s"""|The ${typeMapping.productPrefix} for ${typeMapping.tpe} cannot be validated. - |""".stripMargin - } - - /** Can't validate this kind of `FieldMapping`. */ - case class CannotValidateFieldMapping(owner: ObjectType, field: Field, fieldMapping: FieldMapping) - extends Failure(Severity.Info, owner, Some(fieldMapping.fieldName)) { - override def toString: String = - s"$productPrefix($owner.${field.name}:${field.tpe}, ${fieldMapping.productPrefix})" - override def formattedMessage: String = - s"""|Field mapping cannot be validated. - | - |- Field ${graphql(s"$owner.${field.name}: ${field.tpe}")} is defined by a Schema at (1). - |- Its mapping to Scala is defined by a ${scala(fieldMapping.productPrefix)} at (2). - |- ${UNDERLINED}This kind of mapping canont be validated.$RESET Ensure you have unit tests. - | - |(1) ${schema.pos} - |(2) ${fieldMapping.pos} - |""".stripMargin - } - - /** Object type `owner` declares `field` but no such mapping exists. */ - case class MissingFieldMapping(owner: ObjectMapping, field: Field) - extends Failure(Severity.Error, owner.tpe, Some(field.name)) { - override def toString: String = - s"$productPrefix(${owner.tpe}.${field.name}:${field.tpe})" - override def formattedMessage: String = - s"""|Missing field mapping. - | - |- Field ${graphql(s"${owner.tpe}.${field.name}: ${field.tpe}")} is defined by a Schema at (1). - |- The ${scala(owner.productPrefix)} for ${graphql(owner.tpe)} at (2) ${UNDERLINED}does not define a mapping for this field$RESET. - | - |(1) ${schema.pos} - |(2) ${owner.pos} - |""".stripMargin - } - - /** GraphQL type isn't applicable for mapping type. */ - case class InapplicableGraphQLType(typeMapping: TypeMapping, expected: String) - extends Failure(Severity.Error, typeMapping.tpe, None) { - override def toString: String = - s"$productPrefix(${typeMapping.productPrefix}, ${typeMapping.tpe.productPrefix})" - override def formattedMessage: String = - s"""|Inapplicable GraphQL type. - | - |- Type ${graphql(typeMapping.tpe)} (${scala(typeMapping.tpe.dealias.productPrefix)}) is defined by a Schema at (1). - |- It is mapped by a ${graphql(typeMapping.productPrefix)} at (2), which expects ${scala(expected)}. - |- ${UNDERLINED}Use a different kind of mapping for this type.$RESET - | - |(1) ${schema.pos} - |(2) ${typeMapping.pos} - |""".stripMargin - } - - /** Referenced type does not exist. */ - case class ReferencedTypeDoesNotExist(typeMapping: TypeMapping) - extends Failure(Severity.Error, typeMapping.tpe, None) { - override def toString: String = - s"$productPrefix(${typeMapping.productPrefix}, ${typeMapping.tpe})" - override def formattedMessage: String = - s"""|Referenced type does not exist. - | - |- A ${graphql(typeMapping.productPrefix)} at (1) references type ${graphql(typeMapping.tpe)}. - |- ${UNDERLINED}This type is undeclared$RESET in referenced Schema at (2). - | - |(1) ${typeMapping.pos} - |(2) ${schema.pos} - |""".stripMargin - } - - /** Referenced field does not exist. */ - case class ReferencedFieldDoesNotExist(objectMapping: ObjectMapping, fieldMapping: FieldMapping) - extends Failure(Severity.Error, objectMapping.tpe, Some(fieldMapping.fieldName)) { - override def toString: String = - s"$productPrefix(${objectMapping.tpe}.${fieldMapping.fieldName})" - override def formattedMessage: String = - s"""|Referenced field does not exist. - | - |- ${objectMapping.tpe} is defined in a Schema at (1). - |- A ${graphql(objectMapping.productPrefix)} at (2) references field ${graphql(fieldMapping.fieldName)}. - |- ${UNDERLINED}This field does not exist in the Schema.$RESET - | - |(1) ${schema.pos} - |(1) ${objectMapping.pos} - |""".stripMargin - } - - /** Missing type mapping. */ - case class MissingTypeMapping(tpe: Type) - extends Failure(Severity.Error, tpe, None) { - override def toString: String = - s"$productPrefix($tpe)" - override def formattedMessage: String = - s"""|Missing type mapping. - | - |- ${tpe} is defined in a Schema at (1). - |- ${UNDERLINED}No mapping was found for this type.$RESET - | - |(1) ${schema.pos} - |""".stripMargin - } - - /** - * Run this validator, yielding a chain of `Failure`s of severity equal to or greater than the - * specified `Severity`. - */ - def validateMapping(severity: Severity = Severity.Warning): List[Failure] = - List( - missingTypeMappings, - typeMappings.foldMap(validateTypeMapping).filter(_.severity >= severity) - ).foldMap(_.toList) - - /** - * Run this validator, raising a `ValidationException` in `G` if there are any failures of - * severity equal to or greater than the specified `Severity`. - */ - def validate[G[_]](severity: Severity = Severity.Warning)( - implicit ev: ApplicativeError[G, Throwable] - ): G[Unit] = - NonEmptyList.fromList(validateMapping(severity)).foldMapA(nec => ev.raiseError(ValidationException(nec))) - - /** - * Run this validator, raising a `ValidationException` if there are any failures of severity equal - * to or greater than the specified `Severity`. - */ - def unsafeValidate(severity: Severity = Severity.Warning): Unit = - validate[Either[Throwable, *]](severity).fold(throw _, _ => ()) - - protected def missingTypeMappings: Chain[Failure] = - schema.types.filter { - case _: InputObjectType => false - case tpe => typeMapping(tpe).isEmpty - }.foldMap { tpe => - Chain(MissingTypeMapping(tpe)) - } - - protected def validateTypeMapping(tm: TypeMapping): Chain[Failure] = { - if (!tm.tpe.dealias.exists) Chain(ReferencedTypeDoesNotExist(tm)) - else tm match { - case om: ObjectMapping => validateObjectMapping(om) - case lm: LeafMapping[_] => validateLeafMapping(lm) - case _: PrimitiveMapping => Chain.empty - case tm => Chain(CannotValidateTypeMapping(tm)) - } - } - - protected def validateLeafMapping(lm: LeafMapping[_]): Chain[Failure] = - lm.tpe.dealias match { - case (_: ScalarType)|(_: EnumType)|(_: ListType) => - Chain.empty // these are valid on construction. Nothing to do. - case _ => Chain(InapplicableGraphQLType(lm, "Leaf Type")) - } - - protected def validateFieldMapping(owner: ObjectType, field: Field, fieldMapping: FieldMapping): Chain[Failure] = - Chain(CannotValidateFieldMapping(owner, field, fieldMapping)) - - /** All interfaces for `t`, transtiviely, including `t` if it is an interface. */ - protected def interfaces(t: Type): List[InterfaceType] = - t.dealias match { - case it: InterfaceType => it :: it.interfaces.flatMap(interfaces) - case ot: ObjectType => ot.interfaces.flatMap(interfaces) - case _ => Nil - } - - /** - * Mappings for all fields defined transitively for interfaces of `tpe`, including `tpe` if it - * is an interface. - */ - protected def transitiveInterfaceFieldMappings(tpe: Type): List[FieldMapping] = - interfaces(tpe).flatMap { iface => - typeMapping(iface).toList.collect { - case om: ObjectMapping => om.fieldMappings - }.flatten - } - - protected def validateObjectFieldMappings(m: ObjectMapping, tpe: ObjectType): Chain[Failure] = { - val fms = m.fieldMappings ++ transitiveInterfaceFieldMappings(tpe) - val missing = tpe.fields.foldMap { f => - fms.find(_.fieldName == f.name) match { - case Some(fm) => validateFieldMapping(tpe, f, fm) - case None => Chain(MissingFieldMapping(m, f)) - } - } - val unknown = fms.foldMap { fm => - tpe.fields.find(_.name == fm.fieldName || fm.hidden) match { - case Some(_) => Chain.empty - case None => Chain(ReferencedFieldDoesNotExist(m, fm)) - } - } - missing ++ unknown - } - - protected def validateObjectMapping(m: ObjectMapping): Chain[Failure] = - m.tpe.dealias match { - case ot: ObjectType => validateObjectFieldMappings(m, ot) - case _: InterfaceType => Chain(CannotValidateTypeMapping(m)) - case _ => Chain(InapplicableGraphQLType(m, "ObjectType")) - } - -} - -object MappingValidator { - - def apply[G[_]](m: Mapping[G]): MappingValidator = - new MappingValidator { - type F[a] = G[a] - type M = Mapping[F] - val mapping = m - } - - sealed trait Severity extends Product - object Severity { - case object Error extends Severity - case object Warning extends Severity - case object Info extends Severity - - implicit val OrderSeverity: Order[Severity] = - Order.by { - case Error => 3 - case Warning => 2 - case Info => 1 - } - - } - - abstract class Failure( - val severity: Severity, - val graphQLTypeName: String, - val fieldName: Option[String], - ) extends AnsiColor { - - def this( - severity: Severity, - tpe: Type, - fieldName: Option[String], - ) = this(severity, tpe.toString, fieldName) - - protected def formattedMessage: String = s"$toString (no detail given)" - - private val prefix: String = - severity match { - case Severity.Error => "🛑 " - case Severity.Warning => "⚠️ " - case Severity.Info => "ℹ️ " - } - - protected def graphql(a: Any) = s"$BLUE$a$RESET" - protected def scala(a: Any) = s"$RED$a$RESET" - protected def sql(a: Any) = s"$GREEN$a$RESET" - - final def toErrorMessage: String = - s"""|$formattedMessage - |Color Key: ${scala("◼")} Scala | ${graphql("◼")} GraphQL | ${sql("◼")} SQL - |""".stripMargin.linesIterator.mkString(s"$prefix\n$prefix", s"\n$prefix", s"\n$prefix\n") - - } - - final case class ValidationException(failures: NonEmptyList[Failure]) extends RuntimeException with NoStackTrace { - override def getMessage(): String = - s"\n\n${failures.foldMap(_.toErrorMessage)}\n" - } - -} - diff --git a/modules/core/src/main/scala/schema.scala b/modules/core/src/main/scala/schema.scala index dfc4941d..4339a51a 100644 --- a/modules/core/src/main/scala/schema.scala +++ b/modules/core/src/main/scala/schema.scala @@ -580,6 +580,10 @@ sealed trait Type extends Product { def isObject: Boolean = false + def isScalar: Boolean = false + + def isEnum: Boolean = false + def /(pathElement: String): Path = Path.from(this) / pathElement @@ -641,6 +645,8 @@ case class ScalarType( ) extends Type with NamedType { import ScalarType._ + override def isScalar: Boolean = true + /** True if this is one of the five built-in Scalar types defined in the GraphQL Specification. */ def isBuiltIn: Boolean = this match { @@ -651,7 +657,6 @@ case class ScalarType( IDType => true case _ => false } - } object ScalarType { @@ -738,6 +743,19 @@ sealed trait TypeWithFields extends NamedType { def interfaces: List[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) + } } /** @@ -844,6 +862,8 @@ case class EnumType( enumValues: List[EnumValueDefinition], directives: List[Directive] ) extends Type with NamedType { + override def isEnum: Boolean = true + def hasValue(name: String): Boolean = enumValues.exists(_.name == name) def value(name: String): Option[EnumValue] = valueDefinition(name).map(_ => EnumValue(name)) @@ -1666,6 +1686,7 @@ object SchemaValidator { validateUniqueFields(schema) ++ validateUnionMembers(schema) ++ validateUniqueEnumValues(schema) ++ + validateInterfaces(schema) ++ validateImplementations(schema) ++ validateTypeExtensions(defns, typeExtnDefns) ++ Directive.validateDirectivesForSchema(schema) @@ -1780,6 +1801,42 @@ object SchemaValidator { } } + def validateInterfaces(schema: Schema): List[Problem] = { + val ifs = schema.types.collect { case i: InterfaceType => i } + if (ifs.isEmpty) Nil + else { + val ifRefs: Map[String, Set[String]] = + ifs.map { i => (i.name, i.interfaces.map(_.name).toSet) }.toMap + + @annotation.tailrec + def checkCycle(pendingIfs: Set[String], seen: Set[String]): Option[Set[String]] = { + if (pendingIfs.isEmpty) Some(seen) + else { + val hd = pendingIfs.head + if (seen.contains(hd)) None + else checkCycle(ifRefs.getOrElse(hd, Set.empty) ++ pendingIfs.tail, seen + hd) + } + } + + @annotation.tailrec + def loop(pendingIfs: Set[String]): Either[String, Set[String]] = { + if(pendingIfs.isEmpty) Right(Set.empty[String]) + else { + val hd = pendingIfs.head + checkCycle(Set(hd), Set.empty[String]) match { + case None => Left(hd) + case Some(seen) => loop(pendingIfs.tail.diff(seen)) + } + } + } + + loop(ifs.map(_.name).toSet) match { + case Left(from) => List(Problem(s"Interface cycle starting from '$from'")) + case _ => Nil + } + } + } + def validateImplementations(schema: Schema): List[Problem] = { def validateImplementor(impl: TypeWithFields): List[Problem] = { diff --git a/modules/core/src/main/scala/validationfailure.scala b/modules/core/src/main/scala/validationfailure.scala new file mode 100644 index 00000000..d540e47d --- /dev/null +++ b/modules/core/src/main/scala/validationfailure.scala @@ -0,0 +1,81 @@ +// Copyright (c) 2016-2023 Association of Universities for Research in Astronomy, Inc. (AURA) +// Copyright (c) 2016-2023 Grackle Contributors +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +package grackle + +import scala.io.AnsiColor +import scala.util.control.NoStackTrace + +import cats._ +import cats.data.NonEmptyList +import cats.implicits._ + +import ValidationFailure.Severity + +abstract class ValidationFailure(val severity: Severity) extends AnsiColor { + protected def formattedMessage: String + + private val prefix: String = + severity match { + case Severity.Error => "🛑 " + case Severity.Warning => "⚠️ " + case Severity.Info => "ℹ️ " + } + + protected def graphql(a: Any) = s"$BLUE$a$RESET" + protected def scala(a: Any) = s"$RED$a$RESET" + protected def key: String = + s"Color Key: ${scala("◼")} Scala | ${graphql("◼")} GraphQL" + + + final def toErrorMessage: String = + s"""|$formattedMessage + |$key + |""".stripMargin.linesIterator.mkString(s"$prefix\n$prefix", s"\n$prefix", s"\n$prefix\n") + + protected def typeKind(tpe: Type): String = + tpe.dealias match { + case _: ObjectType => "object type" + case _: InterfaceType => "interface type" + case _: UnionType => "union type" + case _: EnumType => "enum type" + case _: ScalarType => "scalar type" + case _ => "type" + } + + protected def showType(tpe: Type): String = SchemaRenderer.renderType(tpe) + protected def showNamedType(tpe: Type): String = tpe.underlyingNamed.name +} + +object ValidationFailure { + sealed trait Severity extends Product + object Severity { + case object Error extends Severity + case object Warning extends Severity + case object Info extends Severity + + implicit val OrderSeverity: Order[Severity] = + Order.by { + case Error => 3 + case Warning => 2 + case Info => 1 + } + } +} + +final case class ValidationException(failures: NonEmptyList[ValidationFailure]) extends RuntimeException with NoStackTrace { + override def getMessage(): String = + s"\n\n${failures.foldMap(_.toErrorMessage)}\n" +} diff --git a/modules/core/src/main/scala/valuemapping.scala b/modules/core/src/main/scala/valuemapping.scala index 7c7379b6..a6e9712c 100644 --- a/modules/core/src/main/scala/valuemapping.scala +++ b/modules/core/src/main/scala/valuemapping.scala @@ -28,8 +28,10 @@ import Cursor.{DeferredCursor} abstract class ValueMapping[F[_]](implicit val M: MonadThrow[F]) extends Mapping[F] with ValueMappingLike[F] trait ValueMappingLike[F[_]] extends Mapping[F] { + import typeMappings._ + def mkCursor(context: Context, focus: Any, parent: Option[Cursor], env: Env): Cursor = - if(isLeaf(context.tpe)) + if(context.tpe.isUnderlyingLeaf) LeafCursor(context, focus, parent, env) else ValueCursor(context, focus, parent, env) @@ -58,34 +60,65 @@ trait ValueMappingLike[F[_]] extends Mapping[F] { } } - sealed trait ValueFieldMapping[T] extends FieldMapping + sealed trait ValueFieldMapping[T] extends FieldMapping { + def unwrap: FieldMapping + } + object ValueFieldMapping { implicit def wrap[T](fm: FieldMapping): ValueFieldMapping[T] = Wrap(fm) case class Wrap[T](fm: FieldMapping)(implicit val pos: SourcePos) extends ValueFieldMapping[T] { def fieldName = fm.fieldName def hidden = fm.hidden - def withParent(tpe: Type): FieldMapping = fm.withParent(tpe) + def subtree = fm.subtree + def unwrap = fm } } case class ValueField[T](fieldName: String, f: T => Any, hidden: Boolean = false)(implicit val pos: SourcePos) extends ValueFieldMapping[T] { - def withParent(tpe: Type): ValueField[T] = this + def subtree: Boolean = false + def unwrap: FieldMapping = this } object ValueField { def fromValue[T](fieldName: String, t: T, hidden: Boolean = false)(implicit pos: SourcePos): ValueField[Unit] = new ValueField[Unit](fieldName, _ => t, hidden) } - case class ValueObjectMapping[T]( - tpe: NamedType, - fieldMappings: List[FieldMapping], - classTag: ClassTag[T] - )(implicit val pos: SourcePos) extends ObjectMapping - - def ValueObjectMapping[T]( - tpe: NamedType, - fieldMappings: List[ValueFieldMapping[T]] - )(implicit classTag: ClassTag[T], pos: SourcePos): ValueObjectMapping[T] = - new ValueObjectMapping(tpe, fieldMappings.map(_.withParent(tpe)), classTag) + object ValueObjectMapping { + case class DefaultValueObjectMapping( + predicate: MappingPredicate, + fieldMappings: Seq[FieldMapping], + classTag: ClassTag[_] + )(implicit val pos: SourcePos) extends ObjectMapping { + override def showMappingType: String = "ValueObjectMapping" + } + + class Builder(predicate: MappingPredicate, pos: SourcePos) { + def on[T](fieldMappings: ValueFieldMapping[T]*)(implicit classTag: ClassTag[T]): ObjectMapping = + DefaultValueObjectMapping(predicate, fieldMappings.map(_.unwrap), classTag)(pos) + } + + def apply(predicate: MappingPredicate)(implicit pos: SourcePos): Builder = + new Builder(predicate, pos) + + def apply(tpe: NamedType)(implicit pos: SourcePos): Builder = + new Builder(MappingPredicate.TypeMatch(tpe), pos) + + def apply[T]( + tpe: NamedType, + fieldMappings: List[ValueFieldMapping[T]] + )(implicit classTag: ClassTag[T], pos: SourcePos): ObjectMapping = + DefaultValueObjectMapping(MappingPredicate.TypeMatch(tpe), fieldMappings.map(_.unwrap), classTag) + + def unapply(om: DefaultValueObjectMapping): Option[(MappingPredicate, Seq[FieldMapping], ClassTag[_])] = { + Some((om.predicate, om.fieldMappings, om.classTag)) + } + } + + override protected def unpackPrefixedMapping(prefix: List[String], om: ObjectMapping): ObjectMapping = + om match { + case vom: ValueObjectMapping.DefaultValueObjectMapping => + vom.copy(predicate = MappingPredicate.PrefixedTypeMatch(prefix, om.predicate.tpe)) + case _ => super.unpackPrefixedMapping(prefix, om) + } case class ValueCursor( context: Context, diff --git a/modules/core/src/test/scala/compiler/CompilerSuite.scala b/modules/core/src/test/scala/compiler/CompilerSuite.scala index 973382fe..faaeb68a 100644 --- a/modules/core/src/test/scala/compiler/CompilerSuite.scala +++ b/modules/core/src/test/scala/compiler/CompilerSuite.scala @@ -16,6 +16,7 @@ package compiler import cats.data.NonEmptyChain +import cats.effect.IO import cats.implicits._ import munit.CatsEffectSuite @@ -351,7 +352,7 @@ final class CompilerSuite extends CatsEffectSuite { ) ) - val res = ComposedMapping.compiler.compile(query) + val res = TestComposedMapping.compiler.compile(query) assertEquals(res.map(_.query), Result.Success(expected)) } @@ -500,7 +501,7 @@ object ComponentA extends DummyComponent object ComponentB extends DummyComponent object ComponentC extends DummyComponent -object ComposedMapping extends TestMapping { +object TestComposedMapping extends ComposedMapping[IO] { val schema = schema""" type Query { diff --git a/modules/core/src/test/scala/compiler/TestMapping.scala b/modules/core/src/test/scala/compiler/TestMapping.scala index 7f66af69..203bfdea 100644 --- a/modules/core/src/test/scala/compiler/TestMapping.scala +++ b/modules/core/src/test/scala/compiler/TestMapping.scala @@ -21,5 +21,5 @@ import cats.effect.IO import grackle._ abstract class TestMapping(implicit val M: MonadThrow[IO]) extends Mapping[IO] { - val typeMappings: List[TypeMapping] = Nil + val typeMappings: TypeMappings = TypeMappings.empty } diff --git a/modules/core/src/test/scala/directives/DirectiveValidationSuite.scala b/modules/core/src/test/scala/directives/DirectiveValidationSuite.scala index 1175ba63..3b32bdb1 100644 --- a/modules/core/src/test/scala/directives/DirectiveValidationSuite.scala +++ b/modules/core/src/test/scala/directives/DirectiveValidationSuite.scala @@ -370,7 +370,7 @@ object ExecutableDirectiveMapping extends Mapping[IO] { val MutationType = schema.mutationType.get val SubscriptionType = schema.subscriptionType.get - val typeMappings: List[TypeMapping] = Nil + val typeMappings = TypeMappings.empty override val selectElaborator = PreserveArgsElaborator diff --git a/modules/core/src/test/scala/introspection/IntrospectionSuite.scala b/modules/core/src/test/scala/introspection/IntrospectionSuite.scala index 12efb4a6..397adf4b 100644 --- a/modules/core/src/test/scala/introspection/IntrospectionSuite.scala +++ b/modules/core/src/test/scala/introspection/IntrospectionSuite.scala @@ -1486,11 +1486,13 @@ final class IntrospectionSuite extends CatsEffectSuite { } } -object TestMapping extends compiler.TestMapping { +object TestMapping extends ValueMapping[IO] { val schema = schema""" type Query { users: [User!]! + kind: KindTest + deprecation: DeprecationTest } "User object type" @@ -1547,6 +1549,76 @@ object TestMapping extends compiler.TestMapping { flags: Flags } """ + + val QueryType = schema.queryType + val UserType = schema.ref("User") + val ProfileType = schema.ref("Profile") + val DateType = schema.ref("Date") + val FlagsType = schema.ref("Flags") + val KindTestType = schema.ref("KindTest") + val DeprecationTestType = schema.ref("DeprecationTest") + + val typeMappings = + List( + ValueObjectMapping[Unit]( + tpe = QueryType, + fieldMappings = + List( + ValueField("users", identity), + ValueField("kind", identity), + ValueField("deprecation", identity), + ) + ), + ValueObjectMapping[Unit]( + tpe = UserType, + fieldMappings = + List( + //ValueField("id", identity), + ValueField("name", identity), + ValueField("age", identity), + ValueField("birthday", identity) + ) + ), + ValueObjectMapping[Unit]( + tpe = ProfileType, + fieldMappings = + List( + ValueField("id", identity) + ) + ), + ValueObjectMapping[Unit]( + tpe = DateType, + fieldMappings = + List( + ValueField("day", identity), + ValueField("month", identity), + ValueField("year", identity) + ) + ), + ValueObjectMapping[Unit]( + tpe = KindTestType, + fieldMappings = + List( + ValueField("scalar", identity), + ValueField("object", identity), + ValueField("interface", identity), + ValueField("union", identity), + ValueField("enum", identity), + ValueField("list", identity), + ValueField("nonnull", identity), + ValueField("nonnulllistnonnull", identity) + ) + ), + ValueObjectMapping[Unit]( + tpe = DeprecationTestType, + fieldMappings = + List( + ValueField("user", identity), + ValueField("flags", identity) + ) + ), + LeafMapping[String](FlagsType) + ) } object SmallData { @@ -1587,6 +1659,8 @@ object SmallMapping extends ValueMapping[IO] { """ val QueryType = schema.queryType + val SubscriptionType = schema.subscriptionType.get + val MutationType = schema.mutationType.get val UserType = schema.ref("User") val ProfileType = schema.ref("Profile") @@ -1600,6 +1674,20 @@ object SmallMapping extends ValueMapping[IO] { ValueField("profiles", _ => users) ) ), + ObjectMapping( + tpe = SubscriptionType, + fieldMappings = + List( + ValueField.fromValue("dummy", 0) + ) + ), + ObjectMapping( + tpe = MutationType, + fieldMappings = + List( + ValueField.fromValue("dummy", 1) + ) + ), ValueObjectMapping[User]( tpe = UserType, fieldMappings = diff --git a/modules/core/src/test/scala/mapping/MappingValidatorSuite.scala b/modules/core/src/test/scala/mapping/MappingValidatorSuite.scala index 1e62c615..d4de19d6 100644 --- a/modules/core/src/test/scala/mapping/MappingValidatorSuite.scala +++ b/modules/core/src/test/scala/mapping/MappingValidatorSuite.scala @@ -18,23 +18,42 @@ package validator import cats.syntax.all._ import munit.CatsEffectSuite -import grackle.{ ListType, MappingValidator } -import grackle.MappingValidator.ValidationException +import grackle.ValidationException import grackle.syntax._ import compiler.TestMapping final class ValidatorSuite extends CatsEffectSuite { - test("missing type mapping") { object M extends TestMapping { - val schema = schema"type Foo { bar: String }" + val schema = + schema""" + type Query { + foo: Foo + } + + type Foo { + bar: String + } + """ + + override val typeMappings = + TypeMappings.unsafe( + List( + ObjectMapping( + schema.ref("Query"), + List( + CursorField[String]("foo", _ => ???, Nil) + ) + ) + ) + ) } - val es = M.validator.validateMapping() + val es = M.validate() es match { - case List(M.validator.MissingTypeMapping(_)) => () + case List(M.MissingTypeMapping(_)) => () case _ => fail(es.foldMap(_.toErrorMessage)) } @@ -43,13 +62,37 @@ final class ValidatorSuite extends CatsEffectSuite { test("missing field mapping") { object M extends TestMapping { - val schema = schema"type Foo { bar: String }" - override val typeMappings = List(ObjectMapping(schema.ref("Foo"), Nil)) + val schema = + schema""" + type Query { + foo: Foo + } + + type Foo { + bar: String + } + """ + + override val typeMappings = + TypeMappings.unsafe( + List( + ObjectMapping( + schema.ref("Query"), + List( + CursorField[String]("foo", _ => ???, Nil) + ) + ), + ObjectMapping( + schema.ref("Foo"), + Nil + ) + ) + ) } - val es = M.validator.validateMapping() + val es = M.validate() es match { - case List(M.validator.MissingFieldMapping(_, f)) if f.name == "bar" => () + case List(M.MissingFieldMapping(_, f)) if f.name == "bar" => () case _ => fail(es.foldMap(_.toErrorMessage)) } @@ -58,13 +101,35 @@ final class ValidatorSuite extends CatsEffectSuite { test("inapplicable type (object mapping for scalar)") { object M extends TestMapping { - val schema = schema"scalar Foo" - override val typeMappings = List(ObjectMapping(schema.ref("Foo"), Nil)) + val schema = + schema""" + type Query { + foo: Foo + } + + scalar Foo + """ + + override val typeMappings = + TypeMappings.unsafe( + List( + ObjectMapping( + schema.ref("Query"), + List( + CursorField[String]("foo", _ => ???, Nil) + ) + ), + ObjectMapping( + schema.ref("Foo"), + Nil + ) + ) + ) } - val es = M.validator.validateMapping() + val es = M.validate() es match { - case List(M.validator.InapplicableGraphQLType(_, _)) => () + case List(M.ObjectTypeExpected(_)) => () case _ => fail(es.foldMap(_.toErrorMessage)) } @@ -73,13 +138,34 @@ final class ValidatorSuite extends CatsEffectSuite { test("inapplicable type (leaf mapping for object)") { object M extends TestMapping { - val schema = schema"type Foo { bar: String }" - override val typeMappings = List(LeafMapping[String](schema.ref("Foo"))) + val schema = + schema""" + type Query { + foo: Foo + } + + type Foo { + bar: String + } + """ + + override val typeMappings = + TypeMappings.unsafe( + List( + ObjectMapping( + schema.ref("Query"), + List( + CursorField[String]("foo", _ => ???, Nil) + ) + ), + LeafMapping[String](schema.ref("Foo")) + ) + ) } - val es = M.validator.validateMapping() + val es = M.validate() es match { - case List(M.validator.InapplicableGraphQLType(_, _)) => () + case List(M.LeafTypeExpected(_)) => () case _ => fail(es.foldMap(_.toErrorMessage)) } @@ -88,11 +174,28 @@ final class ValidatorSuite extends CatsEffectSuite { test("enums are valid leaf mappings") { object M extends TestMapping { - val schema = schema"enum Foo { BAR }" - override val typeMappings = List(LeafMapping[String](schema.ref("Foo"))) + val schema = + schema""" + type Query { + foo: Foo + } + + enum Foo { BAR } + """ + + override val typeMappings = + List( + ObjectMapping( + schema.ref("Query"), + List( + CursorField[String]("foo", _ => ???, Nil) + ) + ), + LeafMapping[String](schema.ref("Foo")) + ) } - val es = M.validator.validateMapping() + val es = M.validate() es match { case Nil => () case _ => fail(es.foldMap(_.toErrorMessage)) @@ -103,20 +206,38 @@ final class ValidatorSuite extends CatsEffectSuite { test("lists are valid leaf mappings") { object M extends TestMapping { - val schema = schema"enum Foo { BAR } type Baz { quux: [Foo] }" - override val typeMappings = List( + val schema = + schema""" + type Query { + baz: Baz + } + + enum Foo { BAR } + + type Baz { + quux: [Foo] + } + """ + + override val typeMappings = + List( + ObjectMapping( + schema.ref("Query"), + List( + CursorField[String]("baz", _ => ???, Nil) + ) + ), ObjectMapping( schema.ref("Baz"), List( CursorField[String]("quux", _ => ???, Nil) ) ), - LeafMapping[String](schema.ref("Foo")), - LeafMapping[String](ListType(schema.ref("Foo"))) + LeafMapping[String](schema.ref("Foo")) ) } - val es = M.validator.validateMapping() + val es = M.validate() es match { case Nil => () case _ => fail(es.foldMap(_.toErrorMessage)) @@ -127,10 +248,27 @@ final class ValidatorSuite extends CatsEffectSuite { test("input object types don't require mappings") { object M extends TestMapping { - val schema = schema"input Foo { bar: String }" + val schema = + schema""" + type Query { + foo(in: Foo): Int + } + + input Foo { bar: String } + """ + + override val typeMappings = + List( + ObjectMapping( + schema.ref("Query"), + List( + CursorField[String]("foo", _ => ???, Nil) + ) + ) + ) } - val es = M.validator.validateMapping() + val es = M.validate() es match { case Nil => () case _ => fail(es.foldMap(_.toErrorMessage)) @@ -141,13 +279,29 @@ final class ValidatorSuite extends CatsEffectSuite { test("input only enums are valid primitive mappings") { object M extends TestMapping { - val schema = schema"input Foo { bar: Bar } enum Bar { BAZ }" - override val typeMappings = List( - PrimitiveMapping(schema.ref("Bar")) - ) + val schema = + schema""" + type Query { + foo(in: Foo): Int + } + + input Foo { bar: Bar } + + enum Bar { BAZ } + """ + + override val typeMappings = + List( + ObjectMapping( + schema.ref("Query"), + List( + CursorField[String]("foo", _ => ???, Nil) + ) + ) + ) } - val es = M.validator.validateMapping() + val es = M.validate() es match { case Nil => () case _ => fail(es.foldMap(_.toErrorMessage)) @@ -159,13 +313,33 @@ final class ValidatorSuite extends CatsEffectSuite { test("nonexistent type (type mapping)") { object M extends TestMapping { - val schema = schema"" - override val typeMappings = List(ObjectMapping(schema.uncheckedRef("Foo"), Nil)) + val schema = + schema""" + type Query { + foo: Int + } + """ + + override val typeMappings = + TypeMappings.unsafe( + List( + ObjectMapping( + schema.ref("Query"), + List( + CursorField[String]("foo", _ => ???, Nil) + ) + ), + ObjectMapping( + schema.uncheckedRef("Foo"), + Nil + ) + ) + ) } - val es = M.validator.validateMapping() + val es = M.validate() es match { - case List(M.validator.ReferencedTypeDoesNotExist(_)) => () + case List(M.ReferencedTypeDoesNotExist(_)) => () case _ => fail(es.foldMap(_.toErrorMessage)) } @@ -174,42 +348,77 @@ final class ValidatorSuite extends CatsEffectSuite { test("unknown field") { object M extends TestMapping { - val schema = schema"type Foo { bar: String }" - override val typeMappings = List( - ObjectMapping( - schema.ref("Foo"), + val schema = + schema""" + type Query { + foo: Foo + } + + type Foo { + bar: String + } + """ + + override val typeMappings = + TypeMappings.unsafe( List( - CursorField[String]("bar", _ => ???, Nil), - CursorField[String]("quz", _ => ???, Nil), - ), + ObjectMapping( + schema.ref("Query"), + List( + CursorField[String]("foo", _ => ???, Nil) + ) + ), + ObjectMapping( + schema.ref("Foo"), + List( + CursorField[String]("bar", _ => ???, Nil), + CursorField[String]("quz", _ => ???, Nil), + ) + ) + ) ) - ) } - val es = M.validator.validateMapping() + val es = M.validate() es match { - case List(M.validator.ReferencedFieldDoesNotExist(_, _)) => () + case List(M.UnusedFieldMapping(_, _)) => () case _ => fail(es.foldMap(_.toErrorMessage)) } - } test("non-field attributes are valid") { object M extends TestMapping { - val schema = schema"type Foo { bar: String }" - override val typeMappings = List( - ObjectMapping( - schema.ref("Foo"), - List( - CursorField[String]("bar", _ => ???, Nil), - CursorField[String]("baz", _ => ???, Nil, hidden = true), + 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), + CursorField[String]("baz", _ => ???, Nil, hidden = true), + ), + ) ) - ) } - val es = M.validator.validateMapping() + val es = M.validate() es match { case Nil => () case _ => fail(es.foldMap(_.toErrorMessage)) @@ -219,12 +428,31 @@ final class ValidatorSuite extends CatsEffectSuite { test("unsafeValidate") { object M extends TestMapping { - val schema = schema"scalar Bar" - override val typeMappings = List(ObjectMapping(schema.uncheckedRef("Foo"), Nil)) + val schema = + schema""" + type Query { + foo: Foo + } + + scalar Foo + """ + + override val typeMappings = + TypeMappings.unsafe( + List( + ObjectMapping( + schema.ref("Query"), + List( + CursorField[String]("foo", _ => ???, Nil) + ) + ), + ObjectMapping(schema.uncheckedRef("Bar"), Nil) + ) + ) } + intercept[ValidationException] { - MappingValidator(M).unsafeValidate() + M.unsafeValidate() } } - } diff --git a/modules/core/src/test/scala/schema/SchemaSuite.scala b/modules/core/src/test/scala/schema/SchemaSuite.scala index 88287cd9..0083db2c 100644 --- a/modules/core/src/test/scala/schema/SchemaSuite.scala +++ b/modules/core/src/test/scala/schema/SchemaSuite.scala @@ -340,6 +340,66 @@ final class SchemaSuite extends CatsEffectSuite { } } + test("schema validation: direct interface cycles") { + val schema = Schema( + """ + type Query { + node: Node + } + + interface Node implements Named & Node { + id: ID! + name: String + } + + interface Named implements Node & Named { + id: ID! + name: String + } + """ + ) + + schema match { + case Result.Failure(ps) => + assertEquals(ps.map(_.message), NonEmptyChain("Interface cycle starting from 'Node'")) + case unexpected => fail(s"This was unexpected: $unexpected") + } + } + + test("schema validation: indirect interface cycles") { + val schema = Schema( + """ + type Query { + node: Node + } + + interface Node implements Tagged { + id: ID! + name: String + tag: String + } + + interface Named implements Node { + id: ID! + name: String + tag: String + } + + interface Tagged implements Named { + id: ID! + name: String + tag: String + } + """ + ) + + schema match { + case Result.Failure(ps) => + assertEquals(ps.map(_.message), NonEmptyChain("Interface cycle starting from 'Node'")) + case unexpected => fail(s"This was unexpected: $unexpected") + } + } + test("explicit Schema type (complete)") { val schema = diff --git a/modules/core/src/test/scala/starwars/StarWarsData.scala b/modules/core/src/test/scala/starwars/StarWarsData.scala index 129bae71..88318299 100644 --- a/modules/core/src/test/scala/starwars/StarWarsData.scala +++ b/modules/core/src/test/scala/starwars/StarWarsData.scala @@ -194,7 +194,7 @@ object StarWarsMapping extends ValueMapping[IO] { ValueField("id", _.id), ValueField("name", _.name), ValueField("appearsIn", _.appearsIn), - PrimitiveField("numberOfFriends"), + ValueField("numberOfFriends", _ => 0), ValueField("friends", resolveFriends _) ) ), diff --git a/modules/core/src/test/scala/starwars/StarWarsSuite.scala b/modules/core/src/test/scala/starwars/StarWarsSuite.scala index e4a4bd98..5e9a05eb 100644 --- a/modules/core/src/test/scala/starwars/StarWarsSuite.scala +++ b/modules/core/src/test/scala/starwars/StarWarsSuite.scala @@ -15,20 +15,10 @@ package starwars -import cats.implicits._ import io.circe.literal._ import munit.CatsEffectSuite final class StarWarsSuite extends CatsEffectSuite { - - test("validate mapping") { - val es = StarWarsMapping.validator.validateMapping() - es match { - case Nil => () - case _ => fail(es.foldMap(_.toErrorMessage)) - } - } - test("simple query") { val query = """ query { diff --git a/modules/core/src/test/scala/subscription/SubscriptionSuite.scala b/modules/core/src/test/scala/subscription/SubscriptionSuite.scala index 03b8918d..d4c95042 100644 --- a/modules/core/src/test/scala/subscription/SubscriptionSuite.scala +++ b/modules/core/src/test/scala/subscription/SubscriptionSuite.scala @@ -50,7 +50,7 @@ final class SubscriptionSuite extends CatsEffectSuite { val MutationType = schema.ref("Mutation") val SubscriptionType = schema.ref("Subscription") - val typeMappings: List[TypeMapping] = + val typeMappings = List( ObjectMapping(QueryType, List( RootEffect.computeCursor("get")((path, env) => ref.get.map(n => Result(valueCursor(path, env, n)))) diff --git a/modules/doobie-pg/src/main/scala/DoobieMapping.scala b/modules/doobie-pg/src/main/scala/DoobieMapping.scala index 4800c739..dd2d16a9 100644 --- a/modules/doobie-pg/src/main/scala/DoobieMapping.scala +++ b/modules/doobie-pg/src/main/scala/DoobieMapping.scala @@ -48,6 +48,7 @@ trait DoobieMappingLike[F[_]] extends Mapping[F] with SqlMappingLike[F] { type Fragment = DoobieFragment def toEncoder(c: Codec): Encoder = (c._1.put, c._2) + def isNullable(c: Codec): Boolean = c._2 def intCodec = (Meta[Int], false) def intEncoder = (Put[Int], false) diff --git a/modules/doobie-pg/src/test/scala/DoobieDatabaseSuite.scala b/modules/doobie-pg/src/test/scala/DoobieDatabaseSuite.scala index eaff09ab..25f88d4b 100644 --- a/modules/doobie-pg/src/test/scala/DoobieDatabaseSuite.scala +++ b/modules/doobie-pg/src/test/scala/DoobieDatabaseSuite.scala @@ -46,32 +46,34 @@ trait DoobieDatabaseSuite extends SqlDatabaseSuite { abstract class DoobieTestMapping[F[_]: Sync](transactor: Transactor[F], monitor: DoobieMonitor[F] = DoobieMonitor.noopMonitor[IO]) extends DoobieMapping[F](transactor, monitor) with SqlTestMapping[F] { - def bool: Codec = (Meta[Boolean], false) - def text: Codec = (Meta[String], false) - def varchar: Codec = (Meta[String], false) - def bpchar(len: Int): Codec = (Meta[String], false) - def int2: Codec = (Meta[Int], false) - def int4: Codec = (Meta[Int], false) - def int8: Codec = (Meta[Long], false) - def float4: Codec = (Meta[Float], false) - def float8: Codec = (Meta[Double], false) - def numeric(precision: Int, scale: Int): Codec = (Meta[BigDecimal], false) + type TestCodec[T] = (Meta[T], Boolean) - def uuid: Codec = (Meta[UUID], false) - def localDate: Codec = (Meta[LocalDate], false) - def localTime: Codec = (Meta[LocalTime], false) - def offsetDateTime: Codec = (Meta[OffsetDateTime], false) - def duration: Codec = (Meta[Long].timap(Duration.ofMillis)(_.toMillis), false) + def bool: TestCodec[Boolean] = (Meta[Boolean], false) + def text: TestCodec[String] = (Meta[String], false) + def varchar: TestCodec[String] = (Meta[String], false) + def bpchar(len: Int): TestCodec[String] = (Meta[String], false) + def int2: TestCodec[Int] = (Meta[Int], false) + def int4: TestCodec[Int] = (Meta[Int], false) + def int8: TestCodec[Long] = (Meta[Long], false) + def float4: TestCodec[Float] = (Meta[Float], false) + def float8: TestCodec[Double] = (Meta[Double], false) + def numeric(precision: Int, scale: Int): TestCodec[BigDecimal] = (Meta[BigDecimal], false) - def jsonb: Codec = (new Meta(Get[Json], Put[Json]), false) + def uuid: TestCodec[UUID] = (Meta[UUID], false) + def localDate: TestCodec[LocalDate] = (Meta[LocalDate], false) + def localTime: TestCodec[LocalTime] = (Meta[LocalTime], false) + def offsetDateTime: TestCodec[OffsetDateTime] = (Meta[OffsetDateTime], false) + def duration: TestCodec[Duration] = (Meta[Long].timap(Duration.ofMillis)(_.toMillis), false) - def nullable(c: Codec): Codec = (c._1, true) + def jsonb: TestCodec[Json] = (new Meta(Get[Json], Put[Json]), false) - def list(c: Codec): Codec = { - val cm = c._1.asInstanceOf[Meta[Any]] - val decode = cm.get.get.k.asInstanceOf[String => Any] - val encode = cm.put.put.k.asInstanceOf[Any => String] - val cl: Meta[List[Any]] = Meta.Advanced.array[String]("VARCHAR", "_VARCHAR").imap(_.toList.map(decode))(_.map(encode).toArray) + def nullable[T](c: TestCodec[T]): TestCodec[T] = (c._1, true) + + def list[T](c: TestCodec[T]): TestCodec[List[T]] = { + val cm = c._1 + val decode = cm.get.get.k.asInstanceOf[String => T] + val encode = cm.put.put.k.asInstanceOf[T => String] + val cl = Meta.Advanced.array[String]("VARCHAR", "_VARCHAR").imap(_.toList.map(decode))(_.map(encode).toArray) (cl, false) } } diff --git a/modules/doobie-pg/src/test/scala/DoobieSuites.scala b/modules/doobie-pg/src/test/scala/DoobieSuites.scala index 8d85489f..dd6c06c2 100644 --- a/modules/doobie-pg/src/test/scala/DoobieSuites.scala +++ b/modules/doobie-pg/src/test/scala/DoobieSuites.scala @@ -18,6 +18,8 @@ package grackle.doobie.test import cats.effect.IO import doobie.implicits._ import doobie.Meta +import munit.catseffect.IOFixture + import grackle.doobie.postgres.DoobieMonitor import grackle.sql.SqlStatsMonitor @@ -81,7 +83,7 @@ final class GraphSuite extends DoobieDatabaseSuite with SqlGraphSuite { final class InterfacesSuite extends DoobieDatabaseSuite with SqlInterfacesSuite { lazy val mapping = new DoobieTestMapping(xa) with SqlInterfacesMapping[IO] { - def entityType: Codec = + def entityType: TestCodec[EntityType] = (Meta[Int].timap(EntityType.fromInt)(EntityType.toInt), false) } } @@ -89,7 +91,7 @@ final class InterfacesSuite extends DoobieDatabaseSuite with SqlInterfacesSuite final class InterfacesSuite2 extends DoobieDatabaseSuite with SqlInterfacesSuite2 { lazy val mapping = new DoobieTestMapping(xa) with SqlInterfacesMapping2[IO] { - def entityType: Codec = + def entityType: TestCodec[EntityType] = (Meta[Int].timap(EntityType.fromInt)(EntityType.toInt), false) } } @@ -102,6 +104,21 @@ final class LikeSuite extends DoobieDatabaseSuite with SqlLikeSuite { lazy val mapping = new DoobieTestMapping(xa) with SqlLikeMapping[IO] } +final class MappingValidatorValidSuite extends DoobieDatabaseSuite with SqlMappingValidatorValidSuite { + // no DB instance needed for this suite + lazy val mapping = new DoobieTestMapping(null) with SqlMappingValidatorValidMapping[IO] { + def genre: TestCodec[Genre] = (Meta[Int].imap(Genre.fromInt)(Genre.toInt), false) + def feature: TestCodec[Feature] = (Meta[String].imap(Feature.fromString)(_.toString), false) + } + override def munitFixtures: Seq[IOFixture[_]] = Nil +} + +final class MappingValidatorInvalidSuite extends DoobieDatabaseSuite with SqlMappingValidatorInvalidSuite { + // no DB instance needed for this suite + lazy val mapping = new DoobieTestMapping(null) with SqlMappingValidatorInvalidMapping[IO] + override def munitFixtures: Seq[IOFixture[_]] = Nil +} + final class MixedSuite extends DoobieDatabaseSuite with SqlMixedSuite { lazy val mapping = new DoobieTestMapping(xa) with SqlMixedMapping[IO] } @@ -109,8 +126,9 @@ final class MixedSuite extends DoobieDatabaseSuite with SqlMixedSuite { final class MovieSuite extends DoobieDatabaseSuite with SqlMovieSuite { lazy val mapping = new DoobieTestMapping(xa) with SqlMovieMapping[IO] { - def genre: Codec = (Meta[Int].imap(Genre.fromInt)(Genre.toInt), false) - def feature: Codec = (Meta[String].imap(Feature.fromString)(_.toString), false) + def genre: TestCodec[Genre] = (Meta[Int].imap(Genre.fromInt)(Genre.toInt), false) + def feature: TestCodec[Feature] = (Meta[String].imap(Feature.fromString)(_.toString), false) + def tagList: TestCodec[List[String]] = (Meta[Int].imap(Tags.fromInt)(Tags.toInt), false) } } @@ -167,7 +185,7 @@ final class ProjectionSuite extends DoobieDatabaseSuite with SqlProjectionSuite final class RecursiveInterfacesSuite extends DoobieDatabaseSuite with SqlRecursiveInterfacesSuite { lazy val mapping = new DoobieTestMapping(xa) with SqlRecursiveInterfacesMapping[IO] { - def itemType: Codec = + def itemType: TestCodec[ItemType] = (Meta[Int].timap(ItemType.fromInt)(ItemType.toInt), false) } } diff --git a/modules/generic/src/main/scala-2/genericmapping2.scala b/modules/generic/src/main/scala-2/genericmapping2.scala index 0e2c50c8..4b074991 100644 --- a/modules/generic/src/main/scala-2/genericmapping2.scala +++ b/modules/generic/src/main/scala-2/genericmapping2.scala @@ -84,7 +84,7 @@ trait ScalaVersionSpecificGenericMappingLike[F[_]] extends Mapping[F] { self: Ge def withEnv(env0: Env): Cursor = copy(env = env.add(env0)) override def hasField(fieldName: String): Boolean = - fieldMap.contains(fieldName) || fieldMapping(context, fieldName).isDefined + fieldMap.contains(fieldName) || typeMappings.fieldMapping(context, fieldName).isDefined override def field(fieldName: String, resultName: Option[String]): Result[Cursor] = { val localField = diff --git a/modules/generic/src/main/scala-3/genericmapping3.scala b/modules/generic/src/main/scala-3/genericmapping3.scala index 935431f8..d2be67bf 100644 --- a/modules/generic/src/main/scala-3/genericmapping3.scala +++ b/modules/generic/src/main/scala-3/genericmapping3.scala @@ -74,7 +74,7 @@ trait ScalaVersionSpecificGenericMappingLike[F[_]] extends Mapping[F] { self: Ge def withEnv(env0: Env): Cursor = copy(env = env.add(env0)) override def hasField(fieldName: String): Boolean = - fieldMap.contains(fieldName) || fieldMapping(context, fieldName).isDefined + fieldMap.contains(fieldName) || typeMappings.fieldMapping(context, fieldName).isDefined override def field(fieldName: String, resultName: Option[String]): Result[Cursor] = { val localField = diff --git a/modules/generic/src/main/scala/genericmapping.scala b/modules/generic/src/main/scala/genericmapping.scala index aa48910b..ee927e85 100644 --- a/modules/generic/src/main/scala/genericmapping.scala +++ b/modules/generic/src/main/scala/genericmapping.scala @@ -34,23 +34,22 @@ trait GenericMappingLike[F[_]] extends ScalaVersionSpecificGenericMappingLike[F] 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(GenericField(_, _, t, cb, _)) => + typeMappings.fieldMapping(context, fieldName) match { + case Some(GenericField(_, t, cb, _)) => cb().build(fieldContext, t, Some(parent), parent.env) case _ => super.mkCursorForField(parent, fieldName, resultName) } } - case class GenericField[T](val tpe: Option[Type], val fieldName: String, t: T, cb: () => CursorBuilder[T], hidden: Boolean)( + case class GenericField[T](val fieldName: String, t: T, cb: () => CursorBuilder[T], hidden: Boolean)( implicit val pos: SourcePos ) extends FieldMapping { - def withParent(tpe: Type): GenericField[T] = - new GenericField(Some(tpe), fieldName, t, cb, hidden) + def subtree: Boolean = true } def GenericField[T](fieldName: String, t: T, hidden: Boolean = true)(implicit cb: => CursorBuilder[T], pos: SourcePos): GenericField[T] = - new GenericField(None, fieldName, t, () => cb, hidden) + new GenericField(fieldName, t, () => cb, hidden) object semiauto { final def deriveObjectCursorBuilder[T](tpe: Type) diff --git a/modules/skunk/js-jvm/src/test/scala/SkunkDatabaseSuite.scala b/modules/skunk/js-jvm/src/test/scala/SkunkDatabaseSuite.scala index 291a238c..3a515a9f 100644 --- a/modules/skunk/js-jvm/src/test/scala/SkunkDatabaseSuite.scala +++ b/modules/skunk/js-jvm/src/test/scala/SkunkDatabaseSuite.scala @@ -15,11 +15,14 @@ package grackle.skunk.test -import java.time.Duration +import java.time.{Duration, LocalDate, LocalTime, OffsetDateTime} +import java.util.UUID import cats.effect.{IO, Resource, Sync} +import io.circe.Json +import munit.catseffect.IOFixture import natchez.Trace.Implicits.noop -import skunk.Session +import skunk.{ Codec => SCodec, Session } import skunk.codec.{ all => codec } import skunk.circe.codec.{ all => ccodec } @@ -45,39 +48,42 @@ trait SkunkDatabaseSuite extends SqlDatabaseSuite { } val poolFixture = ResourceSuiteLocalFixture("skunk", poolResource) - override def munitFixtures = Seq(poolFixture) + override def munitFixtures: Seq[IOFixture[_]] = Seq(poolFixture) def pool = Resource.eval(IO(poolFixture())) abstract class SkunkTestMapping[F[_]: Sync](pool: Resource[F,Session[F]], monitor: SkunkMonitor[F] = SkunkMonitor.noopMonitor[IO]) extends SkunkMapping[F](pool, monitor) with SqlTestMapping[F] { - def bool: Codec = (codec.bool, false) - def text: Codec = (codec.text, false) - def varchar: Codec = (codec.varchar, false) - def bpchar(len: Int): Codec = (codec.bpchar(len), false) - def int2: Codec = (codec.int2.imap(_.toInt)(_.toShort), false) - def int4: Codec = (codec.int4, false) - def int8: Codec = (codec.int8, false) - def float4: Codec = (codec.float4, false) - def float8: Codec = (codec.float8, false) - def numeric(precision: Int, scale: Int): Codec = (codec.numeric(precision, scale), false) - - def uuid: Codec = (codec.uuid, false) - def localDate: Codec = (codec.date, false) - def localTime: Codec = (codec.time, false) - def offsetDateTime: Codec = (codec.timestamptz, false) - def duration: Codec = (codec.int8.imap(Duration.ofMillis)(_.toMillis), false) - - def jsonb: Codec = (ccodec.jsonb, false) - - def nullable(c: Codec): Codec = (c._1.opt, true) - - def list(c: Codec): Codec = { - val cc = c._1.asInstanceOf[_root_.skunk.Codec[Any]] + + type TestCodec[T] = (SCodec[T], Boolean) + + def bool: TestCodec[Boolean] = (codec.bool, false) + def text: TestCodec[String] = (codec.text, false) + def varchar: TestCodec[String] = (codec.varchar, false) + def bpchar(len: Int): TestCodec[String] = (codec.bpchar(len), false) + def int2: TestCodec[Int] = (codec.int2.imap(_.toInt)(_.toShort), false) + def int4: TestCodec[Int] = (codec.int4, false) + def int8: TestCodec[Long] = (codec.int8, false) + def float4: TestCodec[Float] = (codec.float4, false) + def float8: TestCodec[Double] = (codec.float8, false) + def numeric(precision: Int, scale: Int): TestCodec[BigDecimal] = (codec.numeric(precision, scale), false) + + def uuid: TestCodec[UUID] = (codec.uuid, false) + def localDate: TestCodec[LocalDate] = (codec.date, false) + def localTime: TestCodec[LocalTime] = (codec.time, false) + def offsetDateTime: TestCodec[OffsetDateTime] = (codec.timestamptz, false) + def duration: TestCodec[Duration] = (codec.int8.imap(Duration.ofMillis)(_.toMillis), false) + + def jsonb: TestCodec[Json] = (ccodec.jsonb, false) + + def nullable[T](c: TestCodec[T]): TestCodec[T] = (c._1.opt, true).asInstanceOf[TestCodec[T]] + + def list[T](c: TestCodec[T]): TestCodec[List[T]] = { + val cc = c._1.asInstanceOf[SCodec[Any]] val ty = _root_.skunk.data.Type(s"_${cc.types.head.name}", cc.types) val encode = (elem: Any) => cc.encode(elem).head.get val decode = (str: String) => cc.decode(0, List(Some(str))).left.map(_.message) - (_root_.skunk.Codec.array(encode, decode, ty), false) + (SCodec.array(encode, decode, ty), false).asInstanceOf[TestCodec[List[T]]] } } } diff --git a/modules/skunk/js-jvm/src/test/scala/SkunkSuites.scala b/modules/skunk/js-jvm/src/test/scala/SkunkSuites.scala index 734fc4f1..3662a6f5 100644 --- a/modules/skunk/js-jvm/src/test/scala/SkunkSuites.scala +++ b/modules/skunk/js-jvm/src/test/scala/SkunkSuites.scala @@ -16,6 +16,7 @@ package grackle.skunk.test import cats.effect.IO +import munit.catseffect.IOFixture import skunk.codec.{all => codec} import skunk.implicits._ @@ -83,7 +84,7 @@ final class GraphSuite extends SkunkDatabaseSuite with SqlGraphSuite { final class InterfacesSuite extends SkunkDatabaseSuite with SqlInterfacesSuite { lazy val mapping = new SkunkTestMapping(pool) with SqlInterfacesMapping[IO] { - def entityType: Codec = + def entityType: TestCodec[EntityType] = (codec.int4.imap(EntityType.fromInt)(EntityType.toInt), false) } } @@ -91,7 +92,7 @@ final class InterfacesSuite extends SkunkDatabaseSuite with SqlInterfacesSuite { final class InterfacesSuite2 extends SkunkDatabaseSuite with SqlInterfacesSuite2 { lazy val mapping = new SkunkTestMapping(pool) with SqlInterfacesMapping2[IO] { - def entityType: Codec = + def entityType: TestCodec[EntityType] = (codec.int4.imap(EntityType.fromInt)(EntityType.toInt), false) } } @@ -104,6 +105,22 @@ final class LikeSuite extends SkunkDatabaseSuite with SqlLikeSuite { lazy val mapping = new SkunkTestMapping(pool) with SqlLikeMapping[IO] } +final class MappingValidatorValidSuite extends SkunkDatabaseSuite with SqlMappingValidatorValidSuite { + // no DB instance needed for this suite + lazy val mapping = + new SkunkTestMapping(null) with SqlMappingValidatorValidMapping[IO] { + def genre: TestCodec[Genre] = (codec.int4.imap(Genre.fromInt)(Genre.toInt), false) + def feature: TestCodec[Feature] = (codec.varchar.imap(Feature.fromString)(_.toString), false) + } + override def munitFixtures: Seq[IOFixture[_]] = Nil +} + +final class MappingValidatorInvalidSuite extends SkunkDatabaseSuite with SqlMappingValidatorInvalidSuite { + // no DB instance needed for this suite + lazy val mapping = new SkunkTestMapping(null) with SqlMappingValidatorInvalidMapping[IO] + override def munitFixtures: Seq[IOFixture[_]] = Nil +} + final class MixedSuite extends SkunkDatabaseSuite with SqlMixedSuite { lazy val mapping = new SkunkTestMapping(pool) with SqlMixedMapping[IO] } @@ -111,8 +128,9 @@ final class MixedSuite extends SkunkDatabaseSuite with SqlMixedSuite { final class MovieSuite extends SkunkDatabaseSuite with SqlMovieSuite { lazy val mapping = new SkunkTestMapping(pool) with SqlMovieMapping[IO] { - def genre: Codec = (codec.int4.imap(Genre.fromInt)(Genre.toInt), false) - def feature: Codec = (codec.varchar.imap(Feature.fromString)(_.toString), false) + def genre: TestCodec[Genre] = (codec.int4.imap(Genre.fromInt)(Genre.toInt), false) + def feature: TestCodec[Feature] = (codec.varchar.imap(Feature.fromString)(_.toString), false) + def tagList: TestCodec[List[String]] = (codec.int4.imap(Tags.fromInt)(Tags.toInt), false) } } @@ -172,7 +190,7 @@ final class ProjectionSuite extends SkunkDatabaseSuite with SqlProjectionSuite { final class RecursiveInterfacesSuite extends SkunkDatabaseSuite with SqlRecursiveInterfacesSuite { lazy val mapping = new SkunkTestMapping(pool) with SqlRecursiveInterfacesMapping[IO] { - def itemType: Codec = + def itemType: TestCodec[ItemType] = (codec.int4.imap(ItemType.fromInt)(ItemType.toInt), false) } } diff --git a/modules/skunk/shared/src/main/scala/SkunkMapping.scala b/modules/skunk/shared/src/main/scala/SkunkMapping.scala index a0e25316..d9e47b47 100644 --- a/modules/skunk/shared/src/main/scala/SkunkMapping.scala +++ b/modules/skunk/shared/src/main/scala/SkunkMapping.scala @@ -49,6 +49,7 @@ trait SkunkMappingLike[F[_]] extends Mapping[F] with SqlMappingLike[F] { outer = type Fragment = _root_.skunk.AppliedFragment def toEncoder(c: Codec): Encoder = c._1 + def isNullable(c: Codec): Boolean = c._2 // Also we need to know how to encode the basic GraphQL types. def booleanEncoder = bool diff --git a/modules/sql/shared/src/main/scala/SqlMapping.scala b/modules/sql/shared/src/main/scala/SqlMapping.scala index b5b712ed..c931b126 100644 --- a/modules/sql/shared/src/main/scala/SqlMapping.scala +++ b/modules/sql/shared/src/main/scala/SqlMapping.scala @@ -25,20 +25,18 @@ import cats.data.{NonEmptyList, OptionT, StateT} import cats.implicits._ import io.circe.Json import org.tpolecat.sourcepos.SourcePos +import org.tpolecat.typename.typeName +import circe.CirceMappingLike import syntax._ import Predicate._ import Query._ -import circe.CirceMappingLike +import ValidationFailure.Severity abstract class SqlMapping[F[_]](implicit val M: MonadThrow[F]) extends Mapping[F] with SqlMappingLike[F] /** An abstract mapping that is backed by a SQL database. */ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self => - - override val validator: SqlMappingValidator = - SqlMappingValidator(this) - import SqlQuery.{EmptySqlQuery, SqlJoin, SqlSelect, SqlUnion} import TableExpr.{DerivedTableRef, SubqueryRef, TableRef, WithRef} @@ -706,7 +704,7 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self def rootFieldMapping(context: Context, query: Query): Option[FieldMapping] = for { rn <- Query.rootName(query) - fm <- fieldMapping(context, rn._1) + fm <- typeMappings.fieldMapping(context, rn._1) } yield fm def isLocallyMapped(context: Context, query: Query): Boolean = @@ -715,7 +713,7 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self case Some(fm) if fm.isInstanceOf[SqlFieldMapping] => true case Some(re: EffectMapping) => val fieldContext = context.forFieldOrAttribute(re.fieldName, None) - objectMapping(fieldContext).exists { om => + typeMappings.objectMapping(fieldContext).exists { om => om.fieldMappings.exists { //case _: SqlFieldMapping => true // Scala 3 thinks this is unreachable case fm if fm.isInstanceOf[SqlFieldMapping] => true @@ -725,9 +723,7 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self case _ => false } - sealed trait SqlFieldMapping extends FieldMapping { - final def withParent(tpe: Type): FieldMapping = this - } + sealed trait SqlFieldMapping extends FieldMapping case class SqlField( fieldName: String, @@ -736,12 +732,15 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self discriminator: Boolean = false, hidden: Boolean = false, associative: Boolean = false // a key which is also associative might occur multiple times in the table, ie. it is not a DB primary key - )(implicit val pos: SourcePos) extends SqlFieldMapping + )(implicit val pos: SourcePos) extends SqlFieldMapping { + def subtree: Boolean = false + } case class SqlObject(fieldName: String, joins: List[Join])( implicit val pos: SourcePos ) extends SqlFieldMapping { final def hidden = false + final def subtree: Boolean = false } object SqlObject { def apply(fieldName: String, joins: Join*): SqlObject = apply(fieldName, joins.toList) @@ -751,6 +750,7 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self implicit val pos: SourcePos ) extends SqlFieldMapping { def hidden: Boolean = false + def subtree: Boolean = true } /** @@ -773,53 +773,107 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self object SqlInterfaceMapping { - case class DefaultInterfaceMapping(tpe: NamedType, fieldMappings: List[FieldMapping], path: List[String], discriminator: SqlDiscriminator)( + case class DefaultInterfaceMapping(predicate: MappingPredicate, fieldMappings: Seq[FieldMapping], discriminator: SqlDiscriminator)( implicit val pos: SourcePos - ) extends SqlInterfaceMapping + ) extends SqlInterfaceMapping { + override def showMappingType: String = "SqlInterfaceMapping" + } + + def apply( + predicate: MappingPredicate, + discriminator: SqlDiscriminator + )( + fieldMappings: FieldMapping* + )( + implicit pos: SourcePos + ): ObjectMapping = + DefaultInterfaceMapping(predicate, fieldMappings, discriminator) + + def apply( + tpe: NamedType, + discriminator: SqlDiscriminator + )( + fieldMappings: FieldMapping* + )( + implicit pos: SourcePos + ): ObjectMapping = + DefaultInterfaceMapping(MappingPredicate.TypeMatch(tpe), fieldMappings, discriminator) + def apply( tpe: NamedType, fieldMappings: List[FieldMapping], - path: List[String] = Nil, discriminator: SqlDiscriminator )( implicit pos: SourcePos ): ObjectMapping = - DefaultInterfaceMapping(tpe, fieldMappings.map(_.withParent(tpe)), path, discriminator) + DefaultInterfaceMapping(MappingPredicate.TypeMatch(tpe), fieldMappings, discriminator) } sealed trait SqlUnionMapping extends ObjectMapping with SqlDiscriminatedType object SqlUnionMapping { - case class DefaultUnionMapping(tpe: NamedType, fieldMappings: List[FieldMapping], path: List[String], discriminator: SqlDiscriminator)( + case class DefaultUnionMapping(predicate: MappingPredicate, fieldMappings: Seq[FieldMapping], discriminator: SqlDiscriminator)( implicit val pos: SourcePos - ) extends SqlUnionMapping + ) extends SqlUnionMapping { + override def showMappingType: String = "SqlUnionMapping" + } + + def apply( + predicate: MappingPredicate, + discriminator: SqlDiscriminator + )( + fieldMappings: FieldMapping* + )( + implicit pos: SourcePos + ): ObjectMapping = + DefaultUnionMapping(predicate, fieldMappings, discriminator) + + def apply( + tpe: NamedType, + discriminator: SqlDiscriminator + )( + fieldMappings: FieldMapping* + )( + implicit pos: SourcePos + ): ObjectMapping = + DefaultUnionMapping(MappingPredicate.TypeMatch(tpe), fieldMappings, discriminator) + def apply( tpe: NamedType, fieldMappings: List[FieldMapping], - path: List[String] = Nil, discriminator: SqlDiscriminator, )( implicit pos: SourcePos ): ObjectMapping = - DefaultUnionMapping(tpe, fieldMappings.map(_.withParent(tpe)), path, discriminator) + DefaultUnionMapping(MappingPredicate.TypeMatch(tpe), fieldMappings, discriminator) } + override protected def unpackPrefixedMapping(prefix: List[String], om: ObjectMapping): ObjectMapping = + om match { + case im: SqlInterfaceMapping.DefaultInterfaceMapping => + im.copy(predicate = MappingPredicate.PrefixedTypeMatch(prefix, om.predicate.tpe)) + case um: SqlUnionMapping.DefaultUnionMapping => + um.copy(predicate = MappingPredicate.PrefixedTypeMatch(prefix, om.predicate.tpe)) + case _ => super.unpackPrefixedMapping(prefix, om) + } + + /** Returns the discriminator columns for the context type */ def discriminatorColumnsForType(context: Context): List[SqlColumn] = - objectMapping(context).map(_.fieldMappings.collect { + typeMappings.objectMapping(context).map(_.fieldMappings.iterator.collect { case cm: SqlField if cm.discriminator => SqlColumn.TableColumn(context, cm.columnRef, cm.fieldName :: context.resultPath) - }).getOrElse(Nil) + }.toList).getOrElse(Nil) /** Returns the key columns for the context type */ def keyColumnsForType(context: Context): List[SqlColumn] = { val cols = - objectMapping(context).map { obj => - val objectKeys = obj.fieldMappings.collect { + typeMappings.objectMapping(context).map { obj => + val objectKeys = obj.fieldMappings.iterator.collect { case cm: SqlField if cm.key => SqlColumn.TableColumn(context, cm.columnRef, cm.fieldName :: context.resultPath) - } + }.toList val interfaceKeys = context.tpe.underlyingObject match { case Some(ot: ObjectType) => @@ -837,7 +891,7 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self /** Returns the columns for leaf field `fieldName` in `context` */ def columnsForLeaf(context: Context, fieldName: String): Result[List[SqlColumn]] = - fieldMapping(context, fieldName) match { + typeMappings.fieldMapping(context, fieldName) match { case Some(SqlField(_, cr, _, _, _, _)) => List(SqlColumn.TableColumn(context, cr, fieldName :: context.resultPath)).success case Some(SqlJson(_, cr)) => List(SqlColumn.TableColumn(context, cr, fieldName :: context.resultPath)).success case Some(CursorFieldJson(_, _, required, _)) => @@ -867,7 +921,7 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self /** Returns the aliased column corresponding to the atomic field `fieldName` in `context` */ def columnForAtomicField(context: Context, fieldName: String): Result[SqlColumn] = { - fieldMapping(context, fieldName) match { + typeMappings.fieldMapping(context, fieldName) match { case Some(SqlField(_, cr, _, _, _, _)) => SqlColumn.TableColumn(context, cr, fieldName :: context.resultPath).success case Some(SqlJson(_, cr)) => SqlColumn.TableColumn(context, cr, fieldName :: context.resultPath).success case _ => Result.internalError(s"No column for atomic field '$fieldName' in context $context") @@ -892,7 +946,7 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self /** Returns the discriminator for the type at `context` */ def discriminatorForType(context: Context): Option[SqlDiscriminatedType] = - objectMapping(context) collect { + typeMappings.objectMapping(context) collect { //case d: SqlDiscriminatedType => d // Fails in 2.13.6 due to https://github.com/scala/bug/issues/12398 case i: SqlInterfaceMapping => i case u: SqlUnionMapping => u @@ -901,7 +955,7 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self /** Returns the table for the type at `context` */ def parentTableForType(context: Context): Result[TableRef] = { def noTable = s"No table for type ${context.tpe}" - objectMapping(context).toResultOrError(noTable).flatMap { om => + typeMappings.objectMapping(context).toResultOrError(noTable).flatMap { om => om.fieldMappings.collectFirst { case SqlField(_, cr, _, _, _, _) => TableRef(context, cr.table) }.toResultOrError(noTable).orElse { context.tpe.underlyingObject match { case Some(ot: ObjectType) => @@ -914,7 +968,7 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self /** Is `fieldName` in `context` Jsonb? */ def isJsonb(context: Context, fieldName: String): Boolean = - fieldMapping(context, fieldName) match { + typeMappings.fieldMapping(context, fieldName) match { case Some(_: SqlJson) => true case Some(_: CursorFieldJson) => true case _ => false @@ -922,7 +976,7 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self /** Is `fieldName` in `context` computed? */ def isComputedField(context: Context, fieldName: String): Boolean = - fieldMapping(context, fieldName) match { + typeMappings.fieldMapping(context, fieldName) match { case Some(_: CursorField[_]) => true case _ => false } @@ -942,7 +996,7 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self /** Is the context type mapped to an associative table? */ def isAssociative(context: Context): Boolean = - objectMapping(context).exists(_.fieldMappings.exists { + typeMappings.objectMapping(context).exists(_.fieldMappings.exists { case sf: SqlField => sf.associative case _ => false }) @@ -951,7 +1005,7 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self def nonLeafList(context: Context, fieldName: String): Boolean = context.tpe.underlyingField(fieldName).exists { fieldTpe => fieldTpe.nonNull.isList && ( - fieldMapping(context, fieldName).exists { + typeMappings.fieldMapping(context, fieldName).exists { case SqlObject(_, joins) => joins.nonEmpty case _ => false } @@ -1638,7 +1692,7 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self def isEmbeddedIn(inner: Context, outer: Context): Boolean = { def directlyEmbedded(child: Context, parent: Context): Boolean = - fieldMapping(parent, child.path.head) match { + typeMappings.fieldMapping(parent, child.path.head) match { case Some(_: CursorFieldJson) | Some(SqlObject(_, Nil)) => true case _ => false } @@ -1894,7 +1948,7 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self val fieldName = context.path.head val nested = - fieldMapping(parentContext, fieldName) match { + typeMappings.fieldMapping(parentContext, fieldName) match { case Some(_: CursorFieldJson) | Some(SqlObject(_, Nil)) => val embeddedCols = cols.map { col => if(table.owns(col)) SqlColumn.EmbeddedColumn(parentTable, col) @@ -2783,7 +2837,7 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self } def isEmbedded(context: Context, fieldName: String): Boolean = - fieldMapping(context, fieldName) match { + typeMappings.fieldMapping(context, fieldName) match { case Some(_: CursorFieldJson) | Some(SqlObject(_, Nil)) => true case _ => false } @@ -2806,7 +2860,7 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self def parentConstraintsFromJoins(parentContext: Context, fieldName: String, resultName: String): Result[List[List[(SqlColumn, SqlColumn)]]] = { val tableContext = unembed(parentContext) parentContext.forField(fieldName, resultName).map { childContext => - fieldMapping(parentContext, fieldName) match { + typeMappings.fieldMapping(parentContext, fieldName) match { case Some(SqlObject(_, Nil)) => Nil case Some(SqlObject(_, join :: Nil)) => @@ -3450,13 +3504,13 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self } def hasField(fieldName: String): Boolean = - fieldMapping(context, fieldName).isDefined + 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 = - fieldMapping(context, fieldName) match { + typeMappings.fieldMapping(context, fieldName) match { case Some(_: SqlJson) => asTable.flatMap { table => def mkCirceCursor(f: Json): Result[Cursor] = @@ -3509,4 +3563,541 @@ trait SqlMappingLike[F[_]] extends CirceMappingLike[F] with SqlModule[F] { self getOrElse(Result.internalError(s"No field '$fieldName' for type ${context.tpe}")) } } + + /** Check SqlMapping specific TypeMapping validity */ + override protected def validateTypeMapping(mappings: TypeMappings, context: Context, tm: TypeMapping): List[ValidationFailure] = { + // ObjectMappings must have a key column + // Associative fields must be keys + // Unions and interfaces must have a discriminator + // ObjectMappings must have columnRefs in the same table + // Implementors of interfaces must have columns in the same table + // Members of unions must have columns in the same table + // 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 + else List(NoKeyInObjectTypeMapping(om)) + } + + def checkAssoc(om: ObjectMapping): List[ValidationFailure] = + om.fieldMappings.iterator.collect { + case sf: SqlField if sf.associative && !sf.key => + 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 + else List(NoDiscriminatorInObjectTypeMapping(om)) + } + + def checkSplit(om: ObjectMapping): List[ValidationFailure] = { + val tables = allTables(List(om)) + val split = tables.sizeCompare(1) > 0 + if (!split) Nil + else List(SplitObjectTypeMapping(om, tables)) + } + + 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 _ => Nil + } + val tables = allTables(allMappings) + val split = tables.sizeCompare(1) > 0 + if (!split) Nil + else List(SplitInterfaceTypeMapping(om, allMappings, tables)) + } + + def checkUnionMembers(om: ObjectMapping): List[ValidationFailure] = { + om.tpe.dealias match { + case ut: UnionType => + val allMappings = ut.members.flatMap(nt => mappings.objectMapping(context.asType(nt))) + val tables = allTables(allMappings) + val split = tables.sizeCompare(1) > 0 + if (!split) Nil + else List(SplitUnionTypeMapping(om, allMappings, tables)) + + case _ => Nil + } + } + + def checkUnionFields(om: ObjectMapping): List[ValidationFailure] = + om.fieldMappings.iterator.collect { + case so: SqlObject => + IllegalSubobjectInUnionTypeMapping(om, so) + case sj: SqlJson => + IllegalJsonInUnionTypeMapping(om, sj) + case sf: SqlField if !sf.hidden => + NonHiddenUnionFieldMapping(om, sf) + }.toList + + def isSql(om: ObjectMapping): Boolean = + om.fieldMappings.exists { + case sf: SqlField => !TableName.isRoot(sf.columnRef.table) + case sj: SqlJson => !TableName.isRoot(sj.columnRef.table) + case SqlObject(_, joins) => joins.nonEmpty + case _ => false + } + + tm match { + case im: SqlInterfaceMapping => + hasKey(im) ++ + checkAssoc(im) ++ + hasDiscriminator(im) ++ + checkSplit(im) ++ + checkSuperInterfaces(im) + case um: SqlUnionMapping => + hasKey(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)) ++ + checkAssoc(om) ++ + checkSplit(om) ++ + checkSuperInterfaces(om) + case _ => + super.validateTypeMapping(mappings, context, tm) + } + } + + /** Check SqlMapping specific FieldMapping validity */ + override protected def validateFieldMapping(mappings: TypeMappings, context: Context, om: ObjectMapping, fm: FieldMapping): List[ValidationFailure] = { + // GraphQL and DB schema nullability must be compatible + // GraphQL and DB schema types must be compatible + // Embedded objects are in the same table as their parent + // Joins must have at least one join condition + // Parallel joins must relate the same tables + // Serial joins must chain correctly + + val IntTypeName = typeName[Int] + val LongTypeName = typeName[Long] + val FloatTypeName = typeName[Float] + val DoubleTypeName = typeName[Double] + val BigDecimalTypeName = typeName[BigDecimal] + val JsonTypeName = typeName[Json] + + val tpe = om.tpe.dealias + + (fm, tpe.fieldInfo(fm.fieldName)) match { + case (sf: SqlField, Some(field)) => + val fieldIsNullable = field.tpe.isNullable + val colIsNullable = isNullable(sf.columnRef.codec) + + val fieldContext = context.forFieldOrAttribute(sf.fieldName, None) + val leafMapping0 = mappings.typeMapping(fieldContext).collectFirst { case lm: LeafMapping[_] => lm } + (field.tpe.dealias.nonNull, leafMapping0) match { + case ((_: ScalarType)|(_: EnumType), Some(leafMapping)) => + if(colIsNullable != fieldIsNullable) + List(InconsistentlyNullableFieldMapping(om, sf, field, sf.columnRef, colIsNullable)) + else + (sf.columnRef.scalaTypeName, leafMapping.scalaTypeName) match { + case (t0, t1) if t0 == t1 => Nil + case (LongTypeName, IntTypeName) => Nil + case (DoubleTypeName, FloatTypeName) => Nil + case (BigDecimalTypeName, FloatTypeName) => Nil + case _ => + List(InconsistentFieldLeafMapping(om, sf, field, sf.columnRef, leafMapping)) + } + + case _ => + // Fallback to check only matching top level nullability + // Missing LeafMapping will be reported elsewhere + if(colIsNullable != fieldIsNullable) + List(InconsistentlyNullableFieldMapping(om, sf, field, sf.columnRef, colIsNullable)) + else Nil + + } + + case (sj: SqlJson, Some(field)) => + if(sj.columnRef.scalaTypeName != JsonTypeName) + List(InconsistentFieldTypeMapping(om, sj, field, sj.columnRef, JsonTypeName)) + else Nil + + case (fm@SqlObject(fieldName, Nil), _) if !schema.isRootType(tpe) => + val parentTables0 = allTables(List(om)) + if(parentTables0.forall(TableName.isRoot)) Nil + else { + val parentTables = parentTables0.filterNot(TableName.isRoot) + (for { + fieldContext <- context.forField(fieldName, None).toOption + com <- mappings.objectMapping(fieldContext) + } yield { + val childTables = allTables(List(com)) + if (parentTables.sameElements(childTables)) Nil + else List(SplitEmbeddedObjectTypeMapping(om, fm, com, parentTables, childTables)) + }).getOrElse(Nil) + } + + case (SqlObject(fieldName, joins), _) if !schema.isRootType(tpe) => + val com0 = + for { + fieldContext <- context.forField(fieldName, None).toOption + com <- mappings.objectMapping(fieldContext) + } yield com + + com0 match { + case None => Nil // Missing mapping will be reported elsewhere + case Some(com) => + val parentTables = allTables(List(om)).filterNot(TableName.isRoot) + val childTables = allTables(List(com)).filterNot(TableName.isRoot) + + (parentTables, childTables) match { + case (parentTable :: _, childTable :: _) => + val nonEmpty = + if(joins.forall(_.conditions.nonEmpty)) Nil + else List(NoJoinConditions(om, fm)) + + def consistentConditions(j: Join): Boolean = + j.conditions match { + case Nil => true + case hd :: tl => + val parent = hd._1.table + val child = hd._2.table + tl.forall { case (p, c) => p.table == parent && c.table == child } + } + + val parConsistent = joins.filterNot(consistentConditions).map { j => + InconsistentJoinConditions(om, fm, j.conditions.map(_._1.table).distinct, j.conditions.map(_._2.table).distinct) + } + + val serConsistent = { + val nonEmptyJoins = joins.filter(_.conditions.nonEmpty) + nonEmptyJoins match { + case Nil => Nil // Empty joins will be reported elsewhere + case hd :: tl => + val headIsParent = hd.conditions.head._1.table == parentTable + val lastIsChild = nonEmptyJoins.last.conditions.head._2.table == childTable + val consistentChain = + nonEmptyJoins.zip(tl).forall { + case (j0, j1) => j0.conditions.head._2.table == j1.conditions.head._1.table + } + + if(headIsParent && lastIsChild && consistentChain) Nil + else { + val path = nonEmptyJoins.map(j => (j.conditions.head._1.table, j.conditions.last._2.table)) + + List(MisalignedJoins(om, fm, parentTable, childTable, path)) + } + } + } + + nonEmpty ++ parConsistent ++ serConsistent + + case _ => Nil // No or multiple tables will be reported elsewhere + } + } + + case (other, _) => + super.validateFieldMapping(mappings, context, om, other) + } + } + + private def allTables(oms: List[ObjectMapping]): List[String] = + oms.flatMap(_.fieldMappings.flatMap { + case SqlField(_, columnRef, _, _, _, _) => List(columnRef.table) + case SqlJson(_, columnRef) => List(columnRef.table) + case SqlObject(_, Nil) => Nil + case SqlObject(_, joins) => joins.head.conditions.map(_._1.table) + case _ => Nil + }).distinct + + abstract class SqlValidationFailure(severity: Severity) extends ValidationFailure(severity) { + protected def sql(a: Any) = s"$GREEN$a$RESET" + protected override def key: String = + s"Color Key: ${scala("◼")} Scala | ${graphql("◼")} GraphQL | ${sql("◼")} SQL" + } + + /* Join has no join conditions */ + case class NoJoinConditions(objectMapping: ObjectMapping, fieldMapping: FieldMapping) + extends SqlValidationFailure(Severity.Error) { + override def toString: String = + s"$productPrefix(${objectMapping.showMappingType}, ${showNamedType(objectMapping.tpe)}, ${fieldMapping.fieldName})" + override def formattedMessage: String = + s"""|No join conditions in field mapping. + | + |- The ${scala(objectMapping.showMappingType)} for type ${graphql(showNamedType(objectMapping.tpe))} at (1) has a ${scala("SqlObject")} field mapping for the field ${graphql(fieldMapping.fieldName)} at (2) with a ${scala("Join")} with no join conditions. + |- ${UNDERLINED}Joins must include at least one join condition.$RESET + | + |(1) ${objectMapping.pos} + |(2) ${fieldMapping.pos} + |""".stripMargin + } + + /** Parallel joins relate different tables */ + case class InconsistentJoinConditions(objectMapping: ObjectMapping, fieldMapping: FieldMapping, parents: List[String], children: List[String]) + extends SqlValidationFailure(Severity.Error) { + override def toString: String = + s"$productPrefix(${objectMapping.showMappingType}, ${showNamedType(objectMapping.tpe)}, ${fieldMapping.fieldName}, (${parents.mkString(", ")}), (${children.mkString(", ")}))" + override def formattedMessage: String = + s"""|Inconsistent join conditions in field mapping. + | + |- The ${scala(objectMapping.showMappingType)} for type ${graphql(showNamedType(objectMapping.tpe))} at (1) has a ${scala("SqlObject")} field mapping for the field ${graphql(fieldMapping.fieldName)} at (2) with a Join with inconsistent join conditions: ${sql(s"(${parents.mkString(", ")}) -> (${children.mkString(", ")})")}. + |- ${UNDERLINED}All join conditions must relate the same tables.$RESET + | + |(1) ${objectMapping.pos} + |(2) ${fieldMapping.pos} + |""".stripMargin + } + + /** Serial joins are misaligned */ + case class MisalignedJoins(objectMapping: ObjectMapping, fieldMapping: FieldMapping, parent: String, child: String, path: List[(String, String)]) + extends SqlValidationFailure(Severity.Error) { + override def toString: String = + s"$productPrefix(${objectMapping.showMappingType}, ${showNamedType(objectMapping.tpe)}, ${fieldMapping.fieldName}, $parent, $child, ${path.mkString(", ")})" + override def formattedMessage: String = + s"""|Misaligned joins in field mapping. + | + |- The ${scala(objectMapping.showMappingType)} for type ${graphql(showNamedType(objectMapping.tpe))} at (1) has a ${scala("SqlObject")} field mapping for the field ${graphql(fieldMapping.fieldName)} at (2) with misaligned joins: ${sql(s"$parent, $child, ${path.mkString(", ")}")}. + |- ${UNDERLINED}Sequential joins must relate the parent table to the child table and chain correctly.$RESET + | + |(1) ${objectMapping.pos} + |(2) ${fieldMapping.pos} + |""".stripMargin + } + + /** Object type mapping has no key */ + case class NoKeyInObjectTypeMapping(objectMapping: ObjectMapping) + extends SqlValidationFailure(Severity.Error) { + override def toString: String = + s"$productPrefix(${objectMapping.showMappingType}, ${showNamedType(objectMapping.tpe)})" + override def formattedMessage: String = + s"""|No key field mapping in object type mapping. + | + |- The ${scala(objectMapping.showMappingType)} for type ${graphql(showNamedType(objectMapping.tpe))} at (1) has no direct or inherited key field mapping. + |- ${UNDERLINED}Object type mappings must include at least one direct or inherited key field mapping.$RESET + | + |(1) ${objectMapping.pos} + |""".stripMargin + } + + /** Object type mapping is split across multiple tables */ + case class SplitObjectTypeMapping(objectMapping: ObjectMapping, tables: List[String]) + extends SqlValidationFailure(Severity.Error) { + override def toString: String = + s"$productPrefix(${objectMapping.showMappingType}, ${showNamedType(objectMapping.tpe)}, (${tables.mkString(", ")}))" + override def formattedMessage: String = + s"""|Object type mapping is split across multiple tables. + | + |- The ${scala(objectMapping.showMappingType)} for type ${graphql(showNamedType(objectMapping.tpe))} defined at (1) is split across multiple tables: ${sql(s"${tables.mkString(", ")}")}. + |- ${UNDERLINED}Object types must map to a single database table.$RESET + | + |(1) ${objectMapping.pos} + |""".stripMargin + } + + /** Embedded object type mapping is split across non-parent tables */ + case class SplitEmbeddedObjectTypeMapping(parent: ObjectMapping, parentField: FieldMapping, child: ObjectMapping, parentTables: List[String], childTables: List[String]) + extends SqlValidationFailure(Severity.Error) { + override def toString: String = + s"$productPrefix(${parent.showMappingType}, ${showNamedType(parent.tpe)}.${parentField.fieldName}, ${child.showMappingType}, ${showNamedType(child.tpe)}, (${childTables.mkString(", ")}))" + override def formattedMessage: String = + s"""|Embedded object type maps to non-parent tables. + | + |- The ${scala(parent.showMappingType)} for type ${graphql(showNamedType(parent.tpe))} defined at (1) embeds the ${scala(child.showMappingType)} for type ${graphql(showNamedType(child.tpe))} defined at (2) via field mapping ${graphql(s"${showNamedType(parent.tpe)}.${parentField.fieldName}")} at (3). + |- The parent object is in table(s) ${sql(parentTables.mkString(", "))}. + |- The embedded object is in non-parent table(s) ${sql(childTables.mkString(", "))}. + |- ${UNDERLINED}Embedded objects must map to the same database tables as their parents.$RESET + | + |(1) ${parent.pos} + |(2) ${child.pos} + |(3) ${parentField.pos} + |""".stripMargin + } + + /** Interface/union implementation mappings split across multiple tables */ + case class SplitInterfaceTypeMapping(objectMapping: ObjectMapping, intrfs: List[ObjectMapping], tables: List[String]) + extends SqlValidationFailure(Severity.Error) { + override def toString: String = + s"$productPrefix(${objectMapping.showMappingType}, ${showNamedType(objectMapping.tpe)}, (${intrfs.map(_.tpe.name).mkString(", ")}), (${tables.mkString(", ")}))" + override def formattedMessage: String = + s"""|Interface implementors are split across multiple tables. + | + |- The ${scala(objectMapping.showMappingType)} for type ${graphql(showNamedType(objectMapping.tpe))} at (1) has implementors (${intrfs.map(_.tpe.name).mkString(", ")}) which are split across multiple tables: ${sql(s"${tables.mkString(", ")}")}. + |- ${UNDERLINED}All implmentors of an interface must map to a single database table.$RESET + | + |(1) ${objectMapping.pos} + |""".stripMargin + } + + /** Interface/union implementation mappings split across multiple tables */ + case class SplitUnionTypeMapping(objectMapping: ObjectMapping, members: List[ObjectMapping], tables: List[String]) + extends SqlValidationFailure(Severity.Error) { + override def toString: String = + s"$productPrefix(${objectMapping.showMappingType}, ${showNamedType(objectMapping.tpe)}, (${members.map(_.tpe.name).mkString(", ")}), (${tables.mkString(", ")}))" + override def formattedMessage: String = + s"""|Union member mappings are split across multiple tables. + | + |- The ${scala(objectMapping.showMappingType)} for type ${graphql(showNamedType(objectMapping.tpe))} at (1) has members (${members.map(_.tpe.name).mkString(", ")}) which are split across multiple tables: ${sql(s"${tables.mkString(", ")}")}. + |- ${UNDERLINED}All members of a union must map to a single database table.$RESET + | + |(1) ${objectMapping.pos} + |""".stripMargin + } + + /** Interface/union type mapping has no discriminator */ + case class NoDiscriminatorInObjectTypeMapping(objectMapping: ObjectMapping) + extends SqlValidationFailure(Severity.Error) { + override def toString: String = + s"$productPrefix(${objectMapping.showMappingType}, ${showNamedType(objectMapping.tpe)})" + override def formattedMessage: String = + s"""|No discriminator field mapping in interface/union type mapping. + | + |- The ${scala(objectMapping.showMappingType)} for type ${graphql(showNamedType(objectMapping.tpe))} at (1) has no discriminator field mapping. + |- ${UNDERLINED}interface/union type mappings must include at least one discriminator field mapping.$RESET + | + |(1) ${objectMapping.pos} + |""".stripMargin + } + + /** Subobject field mappings not allowed in union type mappings */ + case class IllegalSubobjectInUnionTypeMapping(objectMapping: ObjectMapping, fieldMapping: FieldMapping) + extends SqlValidationFailure(Severity.Error) { + override def toString: String = + s"$productPrefix(${objectMapping.showMappingType}, ${showNamedType(objectMapping.tpe)}.${fieldMapping.fieldName})" + override def formattedMessage: String = + s"""|Illegal subobject field mapping in union type mapping. + | + |- The ${scala(objectMapping.showMappingType)} for type ${graphql(showNamedType(objectMapping.tpe))} at (1) contains a subobject field mapping ${graphql(fieldMapping.fieldName)} at (2). + |- ${UNDERLINED}Subobject field mappings are not allowed in union type mappings.$RESET + | + |(1) ${objectMapping.pos} + |(2) ${fieldMapping.pos} + |""".stripMargin + } + + /** SqlJson field mappings not allowed in union type mappings */ + case class IllegalJsonInUnionTypeMapping(objectMapping: ObjectMapping, fieldMapping: FieldMapping) + extends SqlValidationFailure(Severity.Error) { + override def toString: String = + s"$productPrefix(${objectMapping.showMappingType}, ${showNamedType(objectMapping.tpe)}.${fieldMapping.fieldName})" + override def formattedMessage: String = + s"""|Illegal json field mapping in union type mapping. + | + |- The ${scala(objectMapping.showMappingType)} for type ${graphql(showNamedType(objectMapping.tpe))} at (1) contains a subobject field mapping ${graphql(fieldMapping.fieldName)} at (2). + |- ${UNDERLINED}SqlJson field mappings are not allowed in union type mappings.$RESET + | + |(1) ${objectMapping.pos} + |(2) ${fieldMapping.pos} + |""".stripMargin + } + + /** Associative field must be a key */ + case class AssocFieldNotKey(objectMapping: ObjectMapping, fieldMapping: FieldMapping) + extends SqlValidationFailure(Severity.Error) { + override def toString: String = + s"$productPrefix(${objectMapping.showMappingType}, ${showNamedType(objectMapping.tpe)}.${fieldMapping.fieldName})" + override def formattedMessage: String = + s"""|Non-key associatitve field mapping in object type mapping. + | + |- The ${scala(objectMapping.showMappingType)} for type ${graphql(showNamedType(objectMapping.tpe))} at (1) contains an associative field mapping ${graphql(fieldMapping.fieldName)} at (2) which is not a key. + |- ${UNDERLINED}All associative field mappings must be keys.$RESET + | + |(1) ${objectMapping.pos} + |(2) ${fieldMapping.pos} + |""".stripMargin + } + + /** Union field mappings must be hidden */ + case class NonHiddenUnionFieldMapping(objectMapping: ObjectMapping, fieldMapping: FieldMapping) + extends SqlValidationFailure(Severity.Error) { + override def toString: String = + s"$productPrefix(${objectMapping.showMappingType}, ${showNamedType(objectMapping.tpe)}.${fieldMapping.fieldName})" + override def formattedMessage: String = + s"""|Non-hidden field mapping in union type mapping. + | + |- The ${scala(objectMapping.showMappingType)} for type ${graphql(showNamedType(objectMapping.tpe))} at (1) contains a field mapping ${graphql(fieldMapping.fieldName)} at (2) which is not hidden. + |- ${UNDERLINED}All fields mappings in a union type mapping must be hidden.$RESET + | + |(1) ${objectMapping.pos} + |(2) ${fieldMapping.pos} + |""".stripMargin + } + + /** SqlField codec and LeafMapping are inconsistent. */ + case class InconsistentFieldLeafMapping(objectMapping: ObjectMapping, fieldMapping: FieldMapping, field: Field, columnRef: ColumnRef, leafMapping: LeafMapping[_]) + extends SqlValidationFailure(Severity.Error) { + override def toString() = + s"$productPrefix(${objectMapping.showMappingType}, ${showNamedType(objectMapping.tpe)}.${fieldMapping.fieldName}, ${columnRef.table}.${columnRef.column}:${columnRef.scalaTypeName}, ${showNamedType(leafMapping.tpe)}:${leafMapping.scalaTypeName})" + override def formattedMessage: String = { + s"""|Inconsistent field leaf mapping. + | + |- The field ${graphql(s"${showNamedType(objectMapping.tpe)}.${fieldMapping.fieldName}: ${showType(field.tpe)}")} is defined by a Schema at (1). + |- The ${scala(fieldMapping.showMappingType)} at (2) and ColumnRef for ${sql(s"${columnRef.table}.${columnRef.column}")} at (3) map ${graphql(showNamedType(leafMapping.tpe))} to Scala type ${scala(columnRef.scalaTypeName)}. + |- A ${scala(leafMapping.showMappingType)} at (4) maps ${graphql(showNamedType(leafMapping.tpe))} to Scala type ${scala(leafMapping.scalaTypeName)}. + |- ${UNDERLINED}The Scala types are inconsistent.$RESET + | + |(1) ${schema.pos} + |(2) ${fieldMapping.pos} + |(3) ${columnRef.pos} + |(4) ${leafMapping.pos} + |""".stripMargin + } + } + + /** SqlField codec and LeafMapping are inconsistent. */ + case class InconsistentFieldTypeMapping(objectMapping: ObjectMapping, fieldMapping: FieldMapping, field: Field, columnRef: ColumnRef, scalaTypeName: String) + extends SqlValidationFailure(Severity.Error) { + override def toString() = + s"$productPrefix(${objectMapping.showMappingType}, ${showNamedType(objectMapping.tpe)}.${fieldMapping.fieldName}:${showType(field.tpe)}, ${columnRef.table}.${columnRef.column}:${columnRef.scalaTypeName}, ${scalaTypeName})" + override def formattedMessage: String = { + s"""|Inconsistent field type mapping. + | + |- The field ${graphql(s"${showNamedType(objectMapping.tpe)}.${fieldMapping.fieldName}: ${showType(field.tpe)}")} is defined by a Schema at (1). + |- The ${scala(fieldMapping.showMappingType)} at (2) and ColumnRef for ${sql(s"${columnRef.table}.${columnRef.column}")} at (3) map to Scala type ${scala(columnRef.scalaTypeName)}. + |- The expected Scala type is ${scala(scalaTypeName)}. + |- ${UNDERLINED}The Scala types are inconsistent.$RESET + | + |(1) ${schema.pos} + |(2) ${fieldMapping.pos} + |(3) ${columnRef.pos} + |""".stripMargin + } + } + + + /** SqlField codec and LeafMapping are inconsistent. */ + case class InconsistentlyNullableFieldMapping(objectMapping: ObjectMapping, fieldMapping: FieldMapping, field: Field, columnRef: ColumnRef, colIsNullable: Boolean) + extends SqlValidationFailure(Severity.Error) { + override def toString() = + s"$productPrefix(${objectMapping.showMappingType}, ${showNamedType(objectMapping.tpe)}.${fieldMapping.fieldName}, ${columnRef.table}.${columnRef.column})" + override def formattedMessage: String = { + val fieldNullability = if(field.tpe.isNullable) "nullable" else "not nullable" + val colNullability = if(colIsNullable) "nullable" else "not nullable" + + s"""|Inconsistently nullable field mapping. + | + |- The ${scala(objectMapping.showMappingType)} for type ${graphql(showNamedType(objectMapping.tpe))} at (1) contains a field mapping ${graphql(fieldMapping.fieldName)} at (2). + |- In the schema at (3) the field is ${fieldNullability}. + |- The corresponding ColumnRef for ${sql(s"${columnRef.table}.${columnRef.column}")} at (4) is ${colNullability}. + |- ${UNDERLINED}The nullabilities are inconsistent.$RESET + | + |(1) ${objectMapping.pos} + |(2) ${fieldMapping.pos} + |(3) ${schema.pos} + |(3) ${columnRef.pos} + |""".stripMargin + } + } } diff --git a/modules/sql/shared/src/main/scala/SqlMappingValidator.scala b/modules/sql/shared/src/main/scala/SqlMappingValidator.scala deleted file mode 100644 index c2bdc236..00000000 --- a/modules/sql/shared/src/main/scala/SqlMappingValidator.scala +++ /dev/null @@ -1,111 +0,0 @@ -// Copyright (c) 2016-2023 Association of Universities for Research in Astronomy, Inc. (AURA) -// Copyright (c) 2016-2023 Grackle Contributors -// -// Licensed under the Apache License, Version 2.0 (the "License"); -// you may not use this file except in compliance with the License. -// You may obtain a copy of the License at -// -// http://www.apache.org/licenses/LICENSE-2.0 -// -// Unless required by applicable law or agreed to in writing, software -// distributed under the License is distributed on an "AS IS" BASIS, -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -// See the License for the specific language governing permissions and -// limitations under the License. - -package grackle -package sql - -import cats.data.Chain -import org.tpolecat.typename.typeName - -trait SqlMappingValidator extends MappingValidator { - - type F[_] - type M <: SqlMappingLike[F] - - val mapping: M - - import MappingValidator._ - import mapping._ - - /** SqlField codec and LeafMapping are inconsistent. */ - case class InconsistentTypeMapping(owner: ObjectType, field: Field, sf: SqlField, lm: LeafMapping[_]) extends Failure(Severity.Error, owner, Some(sf.fieldName)) { - override def toString() = - s"$productPrefix(${owner.name}.${sf.fieldName}, ${sf.columnRef.table}.${sf.columnRef.column}:${sf.columnRef.scalaTypeName}, ${lm.tpe}:${lm.scalaTypeName})" - override def formattedMessage: String = - s"""|Inconsistent type mapping. - | - |- Field ${graphql(s"$owner.${field.name}: ${field.tpe}")} is defined by a Schema at (1). - |- A ${scala(lm.productPrefix)} at (2) maps ${graphql(field.tpe)} to Scala type ${scala(lm.scalaTypeName)}. - |- The ${scala(sf.productPrefix)} at (3) and ColumnRef for ${sql(s"${sf.columnRef.table}.${sf.columnRef.column}")} at (4) map ${graphql(field.tpe)} to Scala type ${scala(sf.columnRef.scalaTypeName)}. - |- ${UNDERLINED}The Scala types are inconsistent.$RESET - | - |(1) ${schema.pos} - |(2) ${lm.pos} - |(3) ${sf.pos} - |(4) ${sf.columnRef.pos} - |""".stripMargin - } - - override protected def validateFieldMapping(owner: ObjectType, field: Field, fieldMapping: mapping.FieldMapping): Chain[MappingValidator.Failure] = - fieldMapping match { - case sf @ SqlField(_, columnRef, _, _, _, _) => - - field.tpe.dealias match { - - case ScalarType.BooleanType if columnRef.scalaTypeName == typeName[Boolean] => Chain.empty - case ScalarType.FloatType if columnRef.scalaTypeName == typeName[Double] => Chain.empty - case ScalarType.StringType if columnRef.scalaTypeName == typeName[String] => Chain.empty - case ScalarType.IDType if columnRef.scalaTypeName == typeName[String] => Chain.empty - case ScalarType.IntType if columnRef.scalaTypeName == typeName[Int] => Chain.empty - - case NullableType(ScalarType.BooleanType) if columnRef.scalaTypeName == typeName[Option[Boolean]] => Chain.empty - case NullableType(ScalarType.FloatType) if columnRef.scalaTypeName == typeName[Option[Double]] => Chain.empty - case NullableType(ScalarType.StringType) if columnRef.scalaTypeName == typeName[Option[String]] => Chain.empty - case NullableType(ScalarType.IDType) if columnRef.scalaTypeName == typeName[Option[String]] => Chain.empty - case NullableType(ScalarType.IntType) if columnRef.scalaTypeName == typeName[Option[Int]] => Chain.empty - - case tpe: ScalarType => - typeMapping(tpe) match { - case Some(lm: LeafMapping[_]) => - if (lm.scalaTypeName == columnRef.scalaTypeName) Chain.empty - else Chain(InconsistentTypeMapping(owner, field, sf, lm)) - case None => Chain.empty // missing type mapping; superclass will catch this - case _ => super.validateFieldMapping(owner, field, fieldMapping) - } - - case NullableType(ofType) => - ofType.dealias match { - case s: ScalarType => - typeMapping(s) match { - case Some(lm: LeafMapping[_]) => - if (lm.scalaTypeName == columnRef.scalaTypeName) Chain.empty - else Chain(InconsistentTypeMapping(owner, field, sf, lm)) - case None => Chain.empty // missing type mapping; superclass will catch this - case _ => super.validateFieldMapping(owner, field, fieldMapping) - } - case _ => super.validateFieldMapping(owner, field, fieldMapping) - } - - case _ => super.validateFieldMapping(owner, field, fieldMapping) - - } - - - - case other => super.validateFieldMapping(owner, field, other) - } - -} - -object SqlMappingValidator { - - def apply[G[_]](m: SqlMappingLike[G]): SqlMappingValidator = - new SqlMappingValidator { - type F[a] = G[a] - type M = SqlMappingLike[F] - val mapping = m - } - -} diff --git a/modules/sql/shared/src/main/scala/SqlModule.scala b/modules/sql/shared/src/main/scala/SqlModule.scala index 930c0cbe..dd90c8cc 100644 --- a/modules/sql/shared/src/main/scala/SqlModule.scala +++ b/modules/sql/shared/src/main/scala/SqlModule.scala @@ -32,6 +32,8 @@ trait SqlModule[F[_]] { /** Extract an encoder from a codec. */ def toEncoder(c: Codec): Encoder + def isNullable(c: Codec): Boolean + /** Typeclass for SQL fragments. */ trait SqlFragment[T] extends Monoid[T] { diff --git a/modules/sql/shared/src/test/resources/db/movies.sql b/modules/sql/shared/src/test/resources/db/movies.sql index ebbfbd0b..62d35298 100644 --- a/modules/sql/shared/src/test/resources/db/movies.sql +++ b/modules/sql/shared/src/test/resources/db/movies.sql @@ -7,18 +7,19 @@ CREATE TABLE movies ( nextshowing TIMESTAMP WITH TIME ZONE NOT NULL, duration BIGINT NOT NULL, categories VARCHAR[] NOT NULL, - features VARCHAR[] NOT NULL + features VARCHAR[] NOT NULL, + tags INTEGER NOT NULL ); -COPY movies (id, title, genre, releasedate, showtime, nextshowing, duration, categories, features) FROM STDIN WITH DELIMITER '|'; -6a7837fc-b463-4d32-b628-0f4b3065cb21|Celine et Julie Vont en Bateau|1|1974-10-07|19:35:00|2020-05-22T19:35:00Z|12300000|{"drama","comedy"}|{"hd","hls"} -11daf8c0-11c3-4453-bfe1-cb6e6e2f9115|Duelle|1|1975-09-15|19:20:00|2020-05-27T19:20:00Z|7260000|{"drama"}|{"hd"} -aea9756f-621b-42d5-b130-71f3916c4ba3|L'Amour fou|1|1969-01-15|21:00:00|2020-05-27T21:00:00Z|15120000|{"drama"}|{"hd"} -2ddb041f-86c2-4bd3-848c-990a3862634e|Last Year at Marienbad|1|1961-06-25|20:30:00|2020-05-26T20:30:00Z|5640000|{"drama"}|{"hd"} -8ae5b13b-044c-4ff0-8b71-ccdb7d77cd88|Zazie dans le Métro|3|1960-10-28|20:15:00|2020-05-25T20:15:00Z|5340000|{"drama","comedy"}|{"hd"} -9dce9deb-9188-4cc2-9685-9842b8abdd34|Alphaville|2|1965-05-05|19:45:00|2020-05-19T19:45:00Z|5940000|{"drama","science fiction"}|{"hd"} -1bf00ac6-91ab-4e51-b686-3fd5e2324077|Stalker|1|1979-05-13|15:30:00|2020-05-19T15:30:00Z|9660000|{"drama","science fiction"}|{"hd"} -6a878e06-6563-4a0c-acd9-d28dcfb2e91a|Weekend|3|1967-12-29|22:30:00|2020-05-19T22:30:00Z|6300000|{"drama","comedy"}|{"hd"} -2a40415c-ea6a-413f-bbef-a80ae280c4ff|Daisies|3|1966-12-30|21:30:00|2020-05-15T21:30:00Z|4560000|{"drama","comedy"}|{"hd"} -2f6dcb0a-4122-4a21-a1c6-534744dd6b85|Le Pont du Nord|1|1982-01-13|20:45:00|2020-05-11T20:45:00Z|7620000|{"drama"}|{"hd"} +COPY movies (id, title, genre, releasedate, showtime, nextshowing, duration, categories, features, tags) FROM STDIN WITH DELIMITER '|'; +6a7837fc-b463-4d32-b628-0f4b3065cb21|Celine et Julie Vont en Bateau|1|1974-10-07|19:35:00|2020-05-22T19:35:00Z|12300000|{"drama","comedy"}|{"hd","hls"}|1 +11daf8c0-11c3-4453-bfe1-cb6e6e2f9115|Duelle|1|1975-09-15|19:20:00|2020-05-27T19:20:00Z|7260000|{"drama"}|{"hd"}|1 +aea9756f-621b-42d5-b130-71f3916c4ba3|L'Amour fou|1|1969-01-15|21:00:00|2020-05-27T21:00:00Z|15120000|{"drama"}|{"hd"}|2 +2ddb041f-86c2-4bd3-848c-990a3862634e|Last Year at Marienbad|1|1961-06-25|20:30:00|2020-05-26T20:30:00Z|5640000|{"drama"}|{"hd"}|5 +8ae5b13b-044c-4ff0-8b71-ccdb7d77cd88|Zazie dans le Métro|3|1960-10-28|20:15:00|2020-05-25T20:15:00Z|5340000|{"drama","comedy"}|{"hd"}|3 +9dce9deb-9188-4cc2-9685-9842b8abdd34|Alphaville|2|1965-05-05|19:45:00|2020-05-19T19:45:00Z|5940000|{"drama","science fiction"}|{"hd"}|4 +1bf00ac6-91ab-4e51-b686-3fd5e2324077|Stalker|1|1979-05-13|15:30:00|2020-05-19T15:30:00Z|9660000|{"drama","science fiction"}|{"hd"}|7 +6a878e06-6563-4a0c-acd9-d28dcfb2e91a|Weekend|3|1967-12-29|22:30:00|2020-05-19T22:30:00Z|6300000|{"drama","comedy"}|{"hd"}|2 +2a40415c-ea6a-413f-bbef-a80ae280c4ff|Daisies|3|1966-12-30|21:30:00|2020-05-15T21:30:00Z|4560000|{"drama","comedy"}|{"hd"}|6 +2f6dcb0a-4122-4a21-a1c6-534744dd6b85|Le Pont du Nord|1|1982-01-13|20:45:00|2020-05-11T20:45:00Z|7620000|{"drama"}|{"hd"}|3 \. diff --git a/modules/sql/shared/src/test/scala/SqlCoalesceMapping.scala b/modules/sql/shared/src/test/scala/SqlCoalesceMapping.scala index 64123c5e..baf8bce9 100644 --- a/modules/sql/shared/src/test/scala/SqlCoalesceMapping.scala +++ b/modules/sql/shared/src/test/scala/SqlCoalesceMapping.scala @@ -16,7 +16,7 @@ package grackle.sql.test import grackle.syntax._ -import java.time.ZonedDateTime +import java.time.OffsetDateTime trait SqlCoalesceMapping[F[_]] extends SqlTestMapping[F] { @@ -121,6 +121,6 @@ trait SqlCoalesceMapping[F[_]] extends SqlTestMapping[F] { SqlField("c", cc.c) ) ), - LeafMapping[ZonedDateTime](DateTimeType) + LeafMapping[OffsetDateTime](DateTimeType) ) } diff --git a/modules/sql/shared/src/test/scala/SqlCoalesceSuite.scala b/modules/sql/shared/src/test/scala/SqlCoalesceSuite.scala index 703ea98a..f02925f4 100644 --- a/modules/sql/shared/src/test/scala/SqlCoalesceSuite.scala +++ b/modules/sql/shared/src/test/scala/SqlCoalesceSuite.scala @@ -125,7 +125,7 @@ trait SqlCoalesceSuite extends CatsEffectSuite { } } - test("zoned-date-time coalesced query") { + test("offset-date-time coalesced query") { val query = """ query { r { diff --git a/modules/sql/shared/src/test/scala/SqlComposedWorldMapping.scala b/modules/sql/shared/src/test/scala/SqlComposedWorldMapping.scala index 639523e9..624646b1 100644 --- a/modules/sql/shared/src/test/scala/SqlComposedWorldMapping.scala +++ b/modules/sql/shared/src/test/scala/SqlComposedWorldMapping.scala @@ -210,7 +210,6 @@ class SqlComposedMapping[F[_] : Sync] cities(namePattern: String = "%"): [City!] country(code: String): Country countries: [Country!] - currencies: [Currency!]! } type City { name: String! diff --git a/modules/sql/shared/src/test/scala/SqlCursorJsonMapping.scala b/modules/sql/shared/src/test/scala/SqlCursorJsonMapping.scala index 75881283..12bd7c86 100644 --- a/modules/sql/shared/src/test/scala/SqlCursorJsonMapping.scala +++ b/modules/sql/shared/src/test/scala/SqlCursorJsonMapping.scala @@ -98,8 +98,7 @@ trait SqlCursorJsonMapping[F[_]] extends SqlTestMapping[F] { SqlField("encodedCategories", brands.category, hidden = true), CursorFieldJson("categories", decodeCategories, List("encodedCategories")) ) - ), - PrimitiveMapping(CategoryType) + ) ) override val selectElaborator = SelectElaborator { diff --git a/modules/sql/shared/src/test/scala/SqlEmbeddingMapping.scala b/modules/sql/shared/src/test/scala/SqlEmbeddingMapping.scala index ddb9d227..22b4a48a 100644 --- a/modules/sql/shared/src/test/scala/SqlEmbeddingMapping.scala +++ b/modules/sql/shared/src/test/scala/SqlEmbeddingMapping.scala @@ -43,7 +43,6 @@ trait SqlEmbeddingMapping[F[_]] extends SqlTestMapping[F] { type Query { films: [Film!]! series: [Series!]! - episodes: [Episode!]! } type Film { title: String! diff --git a/modules/sql/shared/src/test/scala/SqlGraphMapping.scala b/modules/sql/shared/src/test/scala/SqlGraphMapping.scala index d33ef393..b668bdda 100644 --- a/modules/sql/shared/src/test/scala/SqlGraphMapping.scala +++ b/modules/sql/shared/src/test/scala/SqlGraphMapping.scala @@ -38,6 +38,7 @@ trait SqlGraphMapping[F[_]] extends SqlTestMapping[F] { schema""" type Query { node(id: Int!): Node + edge(id: Int!): Edge } type Node { id: Int! @@ -60,7 +61,8 @@ trait SqlGraphMapping[F[_]] extends SqlTestMapping[F] { tpe = QueryType, fieldMappings = List( - SqlObject("node") + SqlObject("node"), + SqlObject("edge") ) ), ObjectMapping( @@ -85,5 +87,7 @@ trait SqlGraphMapping[F[_]] extends SqlTestMapping[F] { override val selectElaborator = SelectElaborator { case (QueryType, "node", List(Binding("id", IntValue(id)))) => Elab.transformChild(child => Unique(Filter(Eql(NodeType / "id", Const(id)), child))) + case (QueryType, "edge", List(Binding("id", IntValue(id)))) => + Elab.transformChild(child => Unique(Filter(Eql(EdgeType / "id", Const(id)), child))) } } diff --git a/modules/sql/shared/src/test/scala/SqlInterfacesMapping.scala b/modules/sql/shared/src/test/scala/SqlInterfacesMapping.scala index f524dfc5..0a866850 100644 --- a/modules/sql/shared/src/test/scala/SqlInterfacesMapping.scala +++ b/modules/sql/shared/src/test/scala/SqlInterfacesMapping.scala @@ -26,7 +26,7 @@ import QueryCompiler._ trait SqlInterfacesMapping[F[_]] extends SqlTestMapping[F] { self => - def entityType: Codec + def entityType: TestCodec[EntityType] object entities extends TableDef("entities") { val id = col("id", text) @@ -125,7 +125,6 @@ trait SqlInterfacesMapping[F[_]] extends SqlTestMapping[F] { self => tpe = FilmType, fieldMappings = List( - SqlField("id", entities.id, key = true), SqlField("rating", entities.filmRating), SqlField("label", entities.filmLabel) ) @@ -134,7 +133,6 @@ trait SqlInterfacesMapping[F[_]] extends SqlTestMapping[F] { self => tpe = SeriesType, fieldMappings = List( - SqlField("id", entities.id, key = true), SqlField("numberOfEpisodes", entities.seriesNumberOfEpisodes), SqlObject("episodes", Join(entities.id, episodes.seriesId)), SqlField("label", entities.seriesLabel) diff --git a/modules/sql/shared/src/test/scala/SqlJsonbMapping.scala b/modules/sql/shared/src/test/scala/SqlJsonbMapping.scala index 10363f54..d3d34f42 100644 --- a/modules/sql/shared/src/test/scala/SqlJsonbMapping.scala +++ b/modules/sql/shared/src/test/scala/SqlJsonbMapping.scala @@ -91,7 +91,8 @@ trait SqlJsonbMapping[F[_]] extends SqlTestMapping[F] { fieldMappings = List( SqlField("id", records.id, key = true), - SqlJson("record", records.record) + SqlJson("record", records.record), + SqlJson("nonNullRecord", records.record) ) ), ) diff --git a/modules/sql/shared/src/test/scala/SqlMappingValidatorInvalidMapping.scala b/modules/sql/shared/src/test/scala/SqlMappingValidatorInvalidMapping.scala new file mode 100644 index 00000000..68bb11f8 --- /dev/null +++ b/modules/sql/shared/src/test/scala/SqlMappingValidatorInvalidMapping.scala @@ -0,0 +1,241 @@ +// Copyright (c) 2016-2023 Association of Universities for Research in Astronomy, Inc. (AURA) +// Copyright (c) 2016-2023 Grackle Contributors +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +package grackle.sql.test + +import grackle._ +import grackle.syntax._ + +trait SqlMappingValidatorInvalidMapping[F[_]] extends SqlTestMapping[F] { + object scalars extends TableDef("scalars") { + val boolCol = col("boolCol", bool) + val intCol = col("intCol", int4) + val nullableBoolCol = col("nullableBoolCol", nullable(bool)) + val nullableIntCol = col("nullableIntCol", nullable(int4)) + + val stringsCol = col("stringsCol", list(text)) + val nullableStringsCol = col("nullableStringsCol", nullable(list(text))) + } + + object obj1 extends TableDef("obj1") { + val idCol = col("idCol", int4) + val typeCol = col("typeCol", int4) + val intCol = col("intCol", int4) + } + + object obj2 extends TableDef("obj2") { + val idCol = col("idCol", int4) + val boolCol = col("boolCol", bool) + val stringCol = col("stringCol", text) + } + + object join extends TableDef("join") { + val parentIdCol = col("parentIdCol", int4) + val childIdCol = col("childIdCol", int4) + } + + object subObj1 extends TableDef("subObj1") { + val idCol = col("idCol", int4) + } + + object subObj2 extends TableDef("subObj2") { + val idCol = col("idCol", int4) + val parentIdCol = col("parentIdCol", int4) + val parentBoolCol = col("parentBoolCol", int4) + } + + object subObj3 extends TableDef("subObj3") { + val idCol = col("idCol", int4) + } + + val schema = + schema""" + type Query { + scalars: Scalars + objs: [Intrf] + union: Union + } + + type Scalars { + boolField1: Boolean! + boolField2: Boolean! + nullableBoolField1: Boolean + nullableBoolField2: Boolean + + stringsField: [String]! + nullableStringsField: [String] + + jsonbField: Record + nullableJsonbField: Record! + } + + type Record { + foo: String + } + + interface Intrf { + id: Int! + } + + type Obj1 implements Intrf { + id: Int! + intField: Int! + embedded: Obj3! + sub1: SubObj1 + } + + type Obj2 implements Intrf { + id: Int! + boolField: Boolean! + sub2: SubObj2 + } + + type Obj3 { + stringField: String! + sub3: SubObj3 + } + + type SubObj1 { + id: Int! + } + + type SubObj2 { + id: Int! + } + + type SubObj3 { + id: Int! + } + + union Union = Obj1 | Obj2 + """ + val QueryType = schema.ref("Query") + val ScalarsType = schema.ref("Scalars") + val IntrfType = schema.ref("Intrf") + val Obj1Type = schema.ref("Obj1") + val Obj2Type = schema.ref("Obj2") + val Obj3Type = schema.ref("Obj3") + val UnionType = schema.ref("Union") + val SubObj1Type = schema.ref("SubObj1") + val SubObj2Type = schema.ref("SubObj2") + val SubObj3Type = schema.ref("SubObj3") + + override val typeMappings = + TypeMappings.unsafe( + List( + ObjectMapping( + tpe = QueryType, + fieldMappings = + List( + SqlObject("scalars"), + SqlObject("objs"), + SqlObject("union") + ) + ), + ObjectMapping( + tpe = ScalarsType, + fieldMappings = + List( + SqlField("boolField1", scalars.nullableBoolCol), + SqlField("boolField2", scalars.intCol), + SqlField("nullableBoolField1", scalars.boolCol), + SqlField("nullableBoolField2", scalars.nullableIntCol), + + SqlField("stringsField", scalars.nullableStringsCol), + SqlField("nullableStringsField", scalars.stringsCol), + + SqlJson("jsonbField", scalars.boolCol), + SqlJson("nullableJsonbField", scalars.nullableBoolCol) + ) + ), + SqlInterfaceMapping( + tpe = IntrfType, + discriminator = objectTypeDiscriminator, + fieldMappings = + List( + SqlField("id", obj1.idCol), + SqlField("typeField", obj1.typeCol, hidden = true) + ) + ), + ObjectMapping( + tpe = Obj1Type, + fieldMappings = + List( + SqlField("id", obj1.idCol), + SqlField("intField", obj1.intCol), + SqlObject("embedded"), + SqlObject("sub1", Join(obj1.idCol, join.parentIdCol), Join(join.childIdCol, obj1.idCol)) + ) + ), + ObjectMapping( + tpe = Obj2Type, + fieldMappings = + List( + SqlField("id", obj2.idCol), + SqlField("assoc", obj1.idCol, associative = true, hidden = true), + SqlField("boolField", obj2.boolCol), + SqlObject("sub2", Join(List((obj2.idCol, subObj2.idCol), (obj2.boolCol, subObj3.idCol)))) + ) + ), + ObjectMapping( + tpe = Obj3Type, + fieldMappings = + List( + SqlField("id", obj2.idCol, key = true, hidden = true), + SqlField("stringField", obj2.stringCol), + SqlObject("sub3", Join(Nil)) + ) + ), + SqlUnionMapping( + tpe = UnionType, + discriminator = objectTypeDiscriminator, + fieldMappings = + List( + SqlField("id", obj1.idCol), + SqlField("typeField", obj1.typeCol, hidden = true), + SqlObject("bogus", Nil) + ) + ), + ObjectMapping( + tpe = SubObj1Type, + fieldMappings = + List( + SqlField("id", subObj1.idCol, key = true) + ) + ), + ObjectMapping( + tpe = SubObj2Type, + fieldMappings = + List( + SqlField("id", subObj2.idCol, key = true) + ) + ), + ObjectMapping( + tpe = SubObj3Type, + fieldMappings = + List( + SqlField("id", subObj3.idCol, key = true) + ) + ) + ) + ) + + lazy val objectTypeDiscriminator = new SqlDiscriminator { + def discriminate(c: Cursor): Result[Type] = + Result.failure("discriminator not implemented") + + def narrowPredicate(subtpe: Type): Option[Predicate] = None + } +} diff --git a/modules/sql/shared/src/test/scala/SqlMappingValidatorInvalidSuite.scala b/modules/sql/shared/src/test/scala/SqlMappingValidatorInvalidSuite.scala new file mode 100644 index 00000000..7bfd2450 --- /dev/null +++ b/modules/sql/shared/src/test/scala/SqlMappingValidatorInvalidSuite.scala @@ -0,0 +1,219 @@ +// Copyright (c) 2016-2023 Association of Universities for Research in Astronomy, Inc. (AURA) +// Copyright (c) 2016-2023 Grackle Contributors +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +package grackle.sql.test + +import cats.effect.IO +import cats.implicits._ +import grackle._ +import grackle.sql._ +import munit.CatsEffectSuite +import org.tpolecat.typename.typeName + +trait SqlMappingValidatorInvalidSuite extends CatsEffectSuite { + def mapping: SqlMappingLike[IO] + + lazy val M = mapping + + def check(es: List[ValidationFailure])(expected: PartialFunction[List[ValidationFailure], Unit]): Unit = + es match { + case `expected`(_) => () + case _ => fail(es.foldMap(_.toErrorMessage)) + } + + object INFM { + def unapply(vf: ValidationFailure): Option[(String, String, String, Boolean, String, String, Boolean)] = + vf match { + case M.InconsistentlyNullableFieldMapping(om, fm, field, columnRef, colIsNullable) => + Some((om.tpe.name, fm.fieldName, SchemaRenderer.renderType(field.tpe), field.tpe.isNullable, columnRef.table, columnRef.column, colIsNullable)) + case _ => None + } + } + + object IFLM { + def unapply(vf: ValidationFailure): Option[(String, String, String, String, String, String)] = + vf match { + case M.InconsistentFieldLeafMapping(om, fm, field, columnRef, _) => + Some((om.tpe.name, fm.fieldName, SchemaRenderer.renderType(field.tpe), columnRef.table, columnRef.column, columnRef.scalaTypeName)) + case _ => None + } + } + + object IFTM { + def unapply(vf: ValidationFailure): Option[(String, String, String, String, String, String)] = + vf match { + case M.InconsistentFieldTypeMapping(om, fm, field, columnRef, _) => + Some((om.tpe.name, fm.fieldName, SchemaRenderer.renderType(field.tpe), columnRef.table, columnRef.column, columnRef.scalaTypeName)) + case _ => None + } + } + + object NK { + def unapply(vf: ValidationFailure): Option[String] = + vf match { + case M.NoKeyInObjectTypeMapping(om) => + Some(om.tpe.name) + case _ => None + } + } + + object STM { + def unapply(vf: ValidationFailure): Option[(String, List[String])] = + vf match { + case M.SplitObjectTypeMapping(om, tables) => + Some((om.tpe.name, tables)) + case _ => None + } + } + + object SEM { + def unapply(vf: ValidationFailure): Option[(String, String, String, List[String], List[String])] = + vf match { + case M.SplitEmbeddedObjectTypeMapping(om, fm, com, parentTables, childTables) => + Some((om.tpe.name, fm.fieldName, com.tpe.name, parentTables, childTables)) + case _ => None + } + } + + object SIM { + def unapply(vf: ValidationFailure): Option[(String, List[String], List[String])] = + vf match { + case M.SplitInterfaceTypeMapping(om, intrfs, tables) => + Some((om.tpe.name, intrfs.map(_.tpe.name), tables)) + case _ => None + } + } + + object SUM { + def unapply(vf: ValidationFailure): Option[(String, List[String], List[String])] = + vf match { + case M.SplitUnionTypeMapping(om, members, tables) => + Some((om.tpe.name, members.map(_.tpe.name), tables)) + case _ => None + } + } + + object ND { + def unapply(vf: ValidationFailure): Option[String] = + vf match { + case M.NoDiscriminatorInObjectTypeMapping(om) => + Some(om.tpe.name) + case _ => None + } + } + + object ISMU { + def unapply(vf: ValidationFailure): Option[(String, String)] = + vf match { + case M.IllegalSubobjectInUnionTypeMapping(om, fm) => + Some((om.tpe.name, fm.fieldName)) + case _ => None + } + } + + object ASNK { + def unapply(vf: ValidationFailure): Option[(String, String)] = + vf match { + case M.AssocFieldNotKey(om, fm) => + Some((om.tpe.name, fm.fieldName)) + case _ => None + } + } + + object NHUFM { + def unapply(vf: ValidationFailure): Option[(String, String)] = + vf match { + case M.NonHiddenUnionFieldMapping(om, fm) => + Some((om.tpe.name, fm.fieldName)) + case _ => None + } + } + + object NJC { + def unapply(vf: ValidationFailure): Option[(String, String)] = + vf match { + case M.NoJoinConditions(om, fm) => + Some((om.tpe.name, fm.fieldName)) + case _ => None + } + } + + object IJC { + def unapply(vf: ValidationFailure): Option[(String, String, List[String], List[String])] = + vf match { + case M.InconsistentJoinConditions(om, fm, parents, children) => + Some((om.tpe.name, fm.fieldName, parents, children)) + case _ => None + } + } + + object MJ { + def unapply(vf: ValidationFailure): Option[(String, String, String, String, List[(String, String)])] = + vf match { + case M.MisalignedJoins(om, fm, parent, child, path) => + Some((om.tpe.name, fm.fieldName, parent, child, path)) + case _ => None + } + } + + object TypeNames { + val BooleanTypeName = typeName[Boolean] + val IntTypeName = typeName[Int] + val ListStringTypeName = typeName[List[java.lang.String]] + } + + import TypeNames._ + + test("invalid mapping is invalid") { + val es = M.validate() + + check(es.take(8)) { + case + List( + INFM("Scalars", "boolField1", "Boolean!", false, "scalars", "nullableBoolCol", true), + IFLM("Scalars", "boolField2", "Boolean!", "scalars", "intCol", IntTypeName), + INFM("Scalars", "nullableBoolField1", "Boolean", true, "scalars", "boolCol", false), + IFLM("Scalars", "nullableBoolField2", "Boolean", "scalars", "nullableIntCol", IntTypeName), + INFM("Scalars", "stringsField", "[String]!", false, "scalars", "nullableStringsCol", true), + INFM("Scalars", "nullableStringsField", "[String]", true, "scalars", "stringsCol", false), + IFTM("Scalars", "jsonbField", "Record", "scalars", "boolCol", BooleanTypeName), + IFTM("Scalars", "nullableJsonbField", "Record!", "scalars", "nullableBoolCol", BooleanTypeName), + ) => + } + + check(es.drop(8)) { + case + List( + NK("Scalars"), + ND("Intrf"), + NK("Intrf"), + SEM("Obj1", "embedded", "Obj3", List("obj1"), List("obj2")), + MJ("Obj1", "sub1", "obj1", "subObj1", List(("obj1", "join"), ("join", "obj1"))), + NK("Obj1"), + NJC("Obj3", "sub3"), + IJC("Obj2", "sub2", List("obj2"), List("subObj2", "subObj3")), + SIM("Obj2", List("Obj2", "Intrf"), List("obj2", "obj1")), + STM("Obj2", List("obj2", "obj1")), + ASNK("Obj2", "assoc"), + NK("Obj2"), + ISMU("Union", "bogus"), + NHUFM("Union", "id"), + SUM("Union", List("Obj1", "Obj2"), List("obj1", "obj2")), + ND("Union"), + NK("Union") + ) => + } + } +} diff --git a/modules/sql/shared/src/test/scala/SqlMappingValidatorValidMapping.scala b/modules/sql/shared/src/test/scala/SqlMappingValidatorValidMapping.scala new file mode 100644 index 00000000..8f5bc53b --- /dev/null +++ b/modules/sql/shared/src/test/scala/SqlMappingValidatorValidMapping.scala @@ -0,0 +1,385 @@ +// Copyright (c) 2016-2023 Association of Universities for Research in Astronomy, Inc. (AURA) +// Copyright (c) 2016-2023 Grackle Contributors +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +package grackle.sql.test + +import java.util.UUID + +import cats.Eq +import io.circe.Encoder + +import grackle._ +import grackle.syntax._ + +trait SqlMappingValidatorValidMapping[F[_]] extends SqlTestMapping[F] { + def genre: TestCodec[Genre] + def feature: TestCodec[Feature] + + object scalars extends TableDef("scalars") { + val idCol = col("idCol", int4) + + val boolCol = col("boolCol", bool) + val nullableBoolCol = col("nullableBoolCol", nullable(bool)) + + val textCol = col("textCol", text) + val nullableTextCol = col("nullableTextCol", nullable(text)) + val varcharCol = col("varcharCol", varchar) + val nullableVarcharCol = col("nullableVarcharCol", nullable(varchar)) + val bpcharCol = col("bpcharCol", bpchar(32)) + val nullableBpcharCol = col("nullableBpcharCol", nullable(bpchar(32))) + + val int2Col = col("int2Col", int2) + val nullableInt2Col = col("nullableInt2Col", nullable(int2)) + val int4Col = col("int4Col", int4) + val nullableInt4Col = col("nullableInt4Col", nullable(int4)) + val int8Col = col("int8Col", int8) + val nullableInt8Col = col("nullableInt8Col", nullable(int8)) + + val float4Col = col("float4Col", float4) + val nullableFloat4Col = col("nullableFloat4Col", nullable(float4)) + val float8Col = col("float8Col", float8) + val nullableFloat8Col = col("nullableFloat8Col", nullable(float8)) + + val numericCol = col("numericCol", numeric(10, 2)) + val nullableNumericCol = col("nullableNumericCol", nullable(numeric(10, 2))) + + val uuidcol = col("uuidcol", uuid) + val nullableUuidCol = col("nullableUuidCol", nullable(uuid)) + + val genreCol = col("genreCol", genre) + val nullableGenreCol = col("nullableGenreCol", nullable(genre)) + + val featureCol = col("featureCol", feature) + val nullableFeatureCol = col("nullableFeatureCol", nullable(feature)) + + val featuresCol = col("featuresCol", list(feature)) + val nullableFeatures1Col = col("nullableFeatures1Col", nullable(list(nullable(feature)))) + val nullableFeatures2Col = col("nullableFeatures2Col", list(nullable(feature))) + val nullableFeatures3Col = col("nullableFeatures3Col", nullable(list(feature))) + + val jsonbCol = col("jsonbCol", jsonb) + val nullableJsonbCol = col("nullableJsonbCol", nullable(jsonb)) + } + + object objs extends TableDef("objs") { + val idCol = col("idCol", int4) + val typeCol = col("typeCol", int4) + val intCol = col("intCol", int4) + val boolCol = col("boolCol", bool) + } + + object join extends TableDef("join") { + val parentIdCol = col("parentIdCol", int4) + val childIdCol = col("childIdCol", int4) + } + + object subObj1 extends TableDef("subObj1") { + val idCol = col("idCol", int4) + } + + object subObj2 extends TableDef("subObj2") { + val idCol = col("idCol", int4) + val parentIdCol = col("parentIdCol", int4) + val parentBolCol = col("parentBoolCol", int4) + } + + val schema = + schema""" + type Query { + scalars: Scalars + objs: [Intrf] + union: Union + } + + type Scalars { + boolField: Boolean! + nullableBoolField: Boolean + + textField: String! + nullableTextField: String + varcharField: String! + nullableVarcharField: String + bpcharField: String! + nullableBpcharField: String + + idField: ID! + nullableIdField: ID + + int2Field: Int! + nullableInt2Field: Int + int4Field: Int! + nullableInt4Field: Int + int8Field: Int! + nullableInt8Field: Int + + float4Field: Float! + nullableFloat4Field: Float + float8Field: Float! + nullableFloat8Field: Float + + numericField: Float! + nullablenumericField: Float + + uuidField: UUID! + nullableUuidField: UUID + + genreField: Genre! + nullableGenreField: Genre + + featureField: Feature! + nullableFeatureField: Feature + + featuresField: [Feature!]! + nullableFeatures1Field: [Feature] + nullableFeatures2Field: [Feature]! + nullableFeatures3Field: [Feature!] + + jsonbField: Record! + nullableJsonbField: Record + } + + scalar UUID + + enum Genre { + DRAMA + ACTION + COMEDY + } + enum Feature { + HD + HLS + } + + type Record { + foo: String + } + + interface Intrf { + id: Int! + } + + type Obj1 implements Intrf { + id: Int! + intField: Int! + sub1: SubObj1 + } + + type Obj2 implements Intrf { + id: Int! + boolField: Boolean! + sub2: SubObj2 + } + + type SubObj1 { + id: Int! + } + + type SubObj2 { + id: Int! + } + + + union Union = Obj1 | Obj2 + """ + + val QueryType = schema.ref("Query") + val ScalarsType = schema.ref("Scalars") + val UUIDType = schema.ref("UUID") + val GenreType = schema.ref("Genre") + val FeatureType = schema.ref("Feature") + val IntrfType = schema.ref("Intrf") + val Obj1Type = schema.ref("Obj1") + val Obj2Type = schema.ref("Obj2") + val UnionType = schema.ref("Union") + val SubObj1Type = schema.ref("SubObj1") + val SubObj2Type = schema.ref("SubObj2") + + override val typeMappings = + TypeMappings.unsafe( + List( + ObjectMapping( + tpe = QueryType, + fieldMappings = + List( + SqlObject("scalars"), + SqlObject("objs"), + SqlObject("union") + ) + ), + ObjectMapping( + tpe = ScalarsType, + fieldMappings = + List( + SqlField("id", scalars.idCol, key = true, hidden = true), + + SqlField("boolField", scalars.boolCol), + SqlField("nullableBoolField", scalars.nullableBoolCol), + + SqlField("textField", scalars.textCol), + SqlField("nullableTextField", scalars.nullableTextCol), + SqlField("varcharField", scalars.varcharCol), + SqlField("nullableVarcharField", scalars.nullableVarcharCol), + SqlField("bpcharField", scalars.bpcharCol), + SqlField("nullableBpcharField", scalars.nullableBpcharCol), + + SqlField("idField", scalars.textCol), + SqlField("nullableIdField", scalars.nullableTextCol), + + SqlField("int2Field", scalars.int2Col), + SqlField("nullableInt2Field", scalars.nullableInt2Col), + SqlField("int4Field", scalars.int4Col), + SqlField("nullableInt4Field", scalars.nullableInt4Col), + SqlField("int8Field", scalars.int8Col), + SqlField("nullableInt8Field", scalars.nullableInt8Col), + + SqlField("float4Field", scalars.float4Col), + SqlField("nullableFloat4Field", scalars.nullableFloat4Col), + SqlField("float8Field", scalars.float8Col), + SqlField("nullableFloat8Field", scalars.nullableFloat8Col), + + SqlField("numericField", scalars.numericCol), + SqlField("nullablenumericField", scalars.nullableNumericCol), + + SqlField("uuidField", scalars.uuidcol), + SqlField("nullableUuidField", scalars.nullableUuidCol), + + SqlField("genreField", scalars.genreCol), + SqlField("nullableGenreField", scalars.nullableGenreCol), + + SqlField("featureField", scalars.featureCol), + SqlField("nullableFeatureField", scalars.nullableFeatureCol), + + SqlField("featuresField", scalars.featuresCol), + SqlField("nullableFeatures1Field", scalars.nullableFeatures1Col), + SqlField("nullableFeatures2Field", scalars.nullableFeatures2Col), + SqlField("nullableFeatures3Field", scalars.nullableFeatures3Col), + + SqlJson("jsonbField", scalars.jsonbCol), + SqlJson("nullableJsonbField", scalars.nullableJsonbCol) + ) + ), + SqlInterfaceMapping( + tpe = IntrfType, + discriminator = objectTypeDiscriminator, + fieldMappings = + List( + SqlField("id", objs.idCol, key = true), + SqlField("typeField", objs.typeCol, discriminator = true, hidden = true) + ) + ), + ObjectMapping( + tpe = Obj1Type, + fieldMappings = + List( + SqlField("intField", objs.intCol), + SqlObject("sub1", Join(objs.idCol, join.parentIdCol), Join(join.childIdCol, subObj1.idCol)) + ) + ), + ObjectMapping( + tpe = Obj2Type, + fieldMappings = + List( + SqlField("boolField", objs.boolCol), + SqlObject("sub2", Join(List((objs.idCol, subObj2.idCol), (objs.boolCol, subObj2.parentBolCol)))) + ) + ), + SqlUnionMapping( + tpe = UnionType, + discriminator = objectTypeDiscriminator, + fieldMappings = + List( + SqlField("id", objs.idCol, key = true, hidden = true), + SqlField("typeField", objs.typeCol, discriminator = true, hidden = true) + ) + ), + ObjectMapping( + tpe = SubObj1Type, + fieldMappings = + List( + SqlField("id", subObj1.idCol, key = true) + ) + ), + ObjectMapping( + tpe = SubObj2Type, + fieldMappings = + List( + SqlField("id", subObj2.idCol, key = true) + ) + ), + LeafMapping[UUID](UUIDType), + LeafMapping[Genre](GenreType), + LeafMapping[Feature](FeatureType) + ) + ) + + lazy val objectTypeDiscriminator = new SqlDiscriminator { + def discriminate(c: Cursor): Result[Type] = + Result.failure("discriminator not implemented") + + def narrowPredicate(subtpe: Type): Option[Predicate] = None + } + + sealed trait Genre extends Product with Serializable + object Genre { + case object Drama extends Genre + case object Action extends Genre + case object Comedy extends Genre + + implicit val genreEq: Eq[Genre] = Eq.fromUniversalEquals[Genre] + + def fromString(s: String): Option[Genre] = + s.trim.toUpperCase match { + case "DRAMA" => Some(Drama) + case "ACTION" => Some(Action) + case "COMEDY" => Some(Comedy) + case _ => None + } + + implicit val genreEncoder: io.circe.Encoder[Genre] = + Encoder[String].contramap(_ match { + case Drama => "DRAMA" + case Action => "ACTION" + case Comedy => "COMEDY" + }) + + def fromInt(i: Int): Genre = + (i: @unchecked) match { + case 1 => Drama + case 2 => Action + case 3 => Comedy + } + + def toInt(f: Genre): Int = + f match { + case Drama => 1 + case Action => 2 + case Comedy => 3 + } + } + + sealed trait Feature + object Feature { + case object HD extends Feature + case object HLS extends Feature + + def fromString(s: String): Feature = (s.trim.toUpperCase: @unchecked) match { + case "HD" => HD + case "HLS" => HLS + } + + implicit def featureEncoder: io.circe.Encoder[Feature] = + Encoder[String].contramap(_.toString) + } +} diff --git a/modules/sql/shared/src/test/scala/SqlMappingValidatorValidSuite.scala b/modules/sql/shared/src/test/scala/SqlMappingValidatorValidSuite.scala new file mode 100644 index 00000000..d7c3ed09 --- /dev/null +++ b/modules/sql/shared/src/test/scala/SqlMappingValidatorValidSuite.scala @@ -0,0 +1,36 @@ +// Copyright (c) 2016-2023 Association of Universities for Research in Astronomy, Inc. (AURA) +// Copyright (c) 2016-2023 Grackle Contributors +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +package grackle.sql.test + +import cats.effect.IO +import cats.implicits._ +import grackle.sql._ +import munit.CatsEffectSuite + +trait SqlMappingValidatorValidSuite extends CatsEffectSuite { + def mapping: SqlMappingLike[IO] + + lazy val M = mapping + + test("valid mapping is valid") { + val es = M.validate() + + es match { + case Nil => () + case _ => fail(es.foldMap(_.toErrorMessage)) + } + } +} diff --git a/modules/sql/shared/src/test/scala/SqlMovieMapping.scala b/modules/sql/shared/src/test/scala/SqlMovieMapping.scala index 0955f332..96594925 100644 --- a/modules/sql/shared/src/test/scala/SqlMovieMapping.scala +++ b/modules/sql/shared/src/test/scala/SqlMovieMapping.scala @@ -34,8 +34,9 @@ import QueryCompiler._ trait SqlMovieMapping[F[_]] extends SqlTestMapping[F] { self => - def genre: Codec - def feature: Codec + def genre: TestCodec[Genre] + def feature: TestCodec[Feature] + def tagList: TestCodec[List[String]] object movies extends TableDef("movies") { val id = col("id", uuid) @@ -47,6 +48,7 @@ trait SqlMovieMapping[F[_]] extends SqlTestMapping[F] { self => val duration = col("duration", self.duration) val categories = col("categories", list(varchar)) val features = col("features", list(feature)) + val tags = col("tags", tagList) } val schema = @@ -60,6 +62,7 @@ trait SqlMovieMapping[F[_]] extends SqlTestMapping[F] { self => moviesShownLaterThan(time: Time!): [Movie!]! moviesShownBetween(from: DateTime!, to: DateTime!): [Movie!]! longMovies: [Movie!]! + allMovies: [Movie!]! } scalar UUID scalar Time @@ -86,6 +89,7 @@ trait SqlMovieMapping[F[_]] extends SqlTestMapping[F] { self => duration: Interval! categories: [String!]! features: [Feature!]! + tags: [String!]! } """ @@ -112,7 +116,8 @@ trait SqlMovieMapping[F[_]] extends SqlTestMapping[F] { self => SqlObject("moviesLongerThan"), SqlObject("moviesShownLaterThan"), SqlObject("moviesShownBetween"), - SqlObject("longMovies") + SqlObject("longMovies"), + SqlObject("allMovies") ) ), ObjectMapping( @@ -129,7 +134,8 @@ trait SqlMovieMapping[F[_]] extends SqlTestMapping[F] { self => SqlField("duration", movies.duration), SqlField("categories", movies.categories), SqlField("features", movies.features), - CursorField("isLong", isLong, List("duration"), hidden = true) + CursorField("isLong", isLong, List("duration"), hidden = true), + SqlField("tags", movies.tags) ) ), LeafMapping[UUID](UUIDType), @@ -138,8 +144,7 @@ trait SqlMovieMapping[F[_]] extends SqlTestMapping[F] { self => LeafMapping[OffsetDateTime](DateTimeType), LeafMapping[Duration](IntervalType), LeafMapping[Genre](GenreType), - LeafMapping[Feature](FeatureType), - LeafMapping[List[Feature]](ListType(FeatureType)) + LeafMapping[Feature](FeatureType) ) def nextEnding(c: Cursor): Result[OffsetDateTime] = @@ -288,6 +293,23 @@ trait SqlMovieMapping[F[_]] extends SqlTestMapping[F] { self => Encoder[String].contramap(_.toString) } + object Tags { + val tags = List("tag1", "tag2", "tag3") + + + def fromInt(i: Int): List[String] = { + def getTag(m: Int): List[String] = + if((i&(1 << m)) != 0) List(tags(m)) else Nil + (0 to 2).flatMap(getTag).toList + } + + def toInt(tags: List[String]): Int = { + def getBit(m: Int): Int = + if(tags.contains(tags(m))) 1 << m else 0 + (0 to 2).foldLeft(0)((acc, m) => acc | getBit(m)) + } + } + implicit val localDateOrder: Order[LocalDate] = Order.from(_.compareTo(_)) diff --git a/modules/sql/shared/src/test/scala/SqlMovieSuite.scala b/modules/sql/shared/src/test/scala/SqlMovieSuite.scala index 4d07464f..25856030 100644 --- a/modules/sql/shared/src/test/scala/SqlMovieSuite.scala +++ b/modules/sql/shared/src/test/scala/SqlMovieSuite.scala @@ -203,7 +203,7 @@ trait SqlMovieSuite extends CatsEffectSuite { assertWeaklyEqualIO(res, expected) } - test("query with ZonedDateTime argument") { + test("query with OffsetDateTime argument") { val query = """ query { moviesShownBetween(from: "2020-05-01T10:30:00Z", to: "2020-05-19T18:00:00Z") { @@ -476,4 +476,94 @@ trait SqlMovieSuite extends CatsEffectSuite { assertWeaklyEqualIO(res, expected) } + + test("query with set encoded as a bitfield") { + val query = """ + query { + allMovies { + title + tags + } + } + """ + + val expected = json""" + { + "data" : { + "allMovies" : [ + { + "title" : "Le Pont du Nord", + "tags" : [ + "tag1", + "tag2" + ] + }, + { + "title" : "Last Year at Marienbad", + "tags" : [ + "tag1", + "tag3" + ] + }, + { + "title" : "Daisies", + "tags" : [ + "tag2", + "tag3" + ] + }, + { + "title" : "Celine et Julie Vont en Bateau", + "tags" : [ + "tag1" + ] + }, + { + "title" : "Stalker", + "tags" : [ + "tag1", + "tag2", + "tag3" + ] + }, + { + "title" : "Weekend", + "tags" : [ + "tag2" + ] + }, + { + "title" : "Zazie dans le Métro", + "tags" : [ + "tag1", + "tag2" + ] + }, + { + "title" : "L'Amour fou", + "tags" : [ + "tag2" + ] + }, + { + "title" : "Duelle", + "tags" : [ + "tag1" + ] + }, + { + "title" : "Alphaville", + "tags" : [ + "tag3" + ] + } + ] + } + } + """ + + val res = mapping.compileAndRun(query) + + assertWeaklyEqualIO(res, expected) + } } diff --git a/modules/sql/shared/src/test/scala/SqlRecursiveInterfacesMapping.scala b/modules/sql/shared/src/test/scala/SqlRecursiveInterfacesMapping.scala index 92799d6d..96fde13f 100644 --- a/modules/sql/shared/src/test/scala/SqlRecursiveInterfacesMapping.scala +++ b/modules/sql/shared/src/test/scala/SqlRecursiveInterfacesMapping.scala @@ -23,7 +23,7 @@ import syntax._ import Predicate._ trait SqlRecursiveInterfacesMapping[F[_]] extends SqlTestMapping[F] { self => - def itemType: Codec + def itemType: TestCodec[ItemType] object items extends TableDef("recursive_interface_items") { val id = col("id", text) @@ -88,7 +88,6 @@ trait SqlRecursiveInterfacesMapping[F[_]] extends SqlTestMapping[F] { self => tpe = ItemAType, fieldMappings = List( - SqlField("id", items.id, key = true), SqlObject("nextItem", Join(items.id, nextItems.id), Join(nextItems.nextItem, items.id)) ) ), @@ -96,7 +95,6 @@ trait SqlRecursiveInterfacesMapping[F[_]] extends SqlTestMapping[F] { self => tpe = ItemBType, fieldMappings = List( - SqlField("id", items.id, key = true), SqlObject("nextItem", Join(items.id, nextItems.id), Join(nextItems.nextItem, items.id)) ) ), diff --git a/modules/sql/shared/src/test/scala/SqlTestMapping.scala b/modules/sql/shared/src/test/scala/SqlTestMapping.scala index 9da8febc..8ec9cf08 100644 --- a/modules/sql/shared/src/test/scala/SqlTestMapping.scala +++ b/modules/sql/shared/src/test/scala/SqlTestMapping.scala @@ -15,34 +15,41 @@ package grackle.sql.test +import java.time.{Duration, LocalDate, LocalTime, OffsetDateTime} +import java.util.UUID + +import io.circe.Json import org.tpolecat.sourcepos.SourcePos +import org.tpolecat.typename.TypeName import grackle._ import sql.SqlMappingLike trait SqlTestMapping[F[_]] extends SqlMappingLike[F] { outer => - def bool: Codec - def text: Codec - def varchar: Codec - def bpchar(len: Int): Codec - def int2: Codec - def int4: Codec - def int8: Codec - def float4: Codec - def float8: Codec - def numeric(precision: Int, scale: Int): Codec - - def uuid: Codec - def localDate: Codec - def localTime: Codec - def offsetDateTime: Codec - def duration: Codec - - def jsonb: Codec - - def nullable(c: Codec): Codec - def list(c: Codec): Codec - - def col[T](colName: String, codec: Codec)(implicit tableName: TableName, pos: SourcePos): ColumnRef = - ColumnRef(tableName.name, colName, codec, null, pos) + type TestCodec[T] <: Codec + + def bool: TestCodec[Boolean] + def text: TestCodec[String] + def varchar: TestCodec[String] + def bpchar(len: Int): TestCodec[String] + def int2: TestCodec[Int] + def int4: TestCodec[Int] + def int8: TestCodec[Long] + def float4: TestCodec[Float] + def float8: TestCodec[Double] + def numeric(precision: Int, scale: Int): TestCodec[BigDecimal] + + def uuid: TestCodec[UUID] + def localDate: TestCodec[LocalDate] + def localTime: TestCodec[LocalTime] + def offsetDateTime: TestCodec[OffsetDateTime] + def duration: TestCodec[Duration] + + def jsonb: TestCodec[Json] + + def nullable[T](c: TestCodec[T]): TestCodec[T] + def list[T](c: TestCodec[T]): TestCodec[List[T]] + + def col[T](colName: String, codec: TestCodec[T])(implicit tableName: TableName, typeName: TypeName[T], pos: SourcePos): ColumnRef = + ColumnRef(tableName.name, colName, codec, typeName.value, pos) } diff --git a/profile/src/main/scala/Bench.scala b/profile/src/main/scala/Bench.scala index ad2891ff..091e8861 100644 --- a/profile/src/main/scala/Bench.scala +++ b/profile/src/main/scala/Bench.scala @@ -37,16 +37,16 @@ trait WorldPostgresSchema[F[_]] extends DoobieMapping[F] { val name = col("name", Meta[String]) val continent = col("continent", Meta[String]) val region = col("region", Meta[String]) - val surfacearea = col("surfacearea", Meta[String]) + val surfacearea = col("surfacearea", Meta[Float]) val indepyear = col("indepyear", Meta[Int], true) val population = col("population", Meta[Int]) - val lifeexpectancy = col("lifeexpectancy", Meta[String], true) - val gnp = col("gnp", Meta[String], true) - val gnpold = col("gnpold", Meta[String], true) + val lifeexpectancy = col("lifeexpectancy", Meta[Float], true) + val gnp = col("gnp", Meta[BigDecimal], true) + val gnpold = col("gnpold", Meta[BigDecimal], true) val localname = col("localname", Meta[String]) val governmentform = col("governmentform", Meta[String]) val headofstate = col("headofstate", Meta[String], true) - val capitalId = col("capitalId", Meta[String], true) + val capitalId = col("capitalId", Meta[Int], true) val numCities = col("num_cities", Meta[Int], false) val code2 = col("code2", Meta[String]) } @@ -62,8 +62,8 @@ trait WorldPostgresSchema[F[_]] extends DoobieMapping[F] { object countrylanguage extends TableDef("countrylanguage") { val countrycode = col("countrycode", Meta[String]) val language = col("language", Meta[String]) - val isOfficial = col("isOfficial", Meta[String]) - val percentage = col("percentage", Meta[String]) + val isOfficial = col("isOfficial", Meta[Boolean]) + val percentage = col("percentage", Meta[Float]) } } @@ -101,8 +101,8 @@ trait WorldMapping[F[_]] extends WorldPostgresSchema[F] { indepyear: Int population: Int! lifeexpectancy: Float - gnp: String - gnpold: String + gnp: Float + gnpold: Float localname: String! governmentform: String! headofstate: String @@ -120,62 +120,50 @@ trait WorldMapping[F[_]] extends WorldPostgresSchema[F] { val LanguageType = schema.ref("Language") val typeMappings = - List( - ObjectMapping( - tpe = QueryType, - fieldMappings = List( - SqlObject("cities"), - SqlObject("city"), - SqlObject("country"), - SqlObject("countries"), - SqlObject("language"), - SqlObject("search"), - SqlObject("search2") - ) + TypeMappings( + ObjectMapping(QueryType)( + SqlObject("cities"), + SqlObject("city"), + SqlObject("country"), + SqlObject("countries"), + SqlObject("language"), + SqlObject("search"), + SqlObject("search2") ), - ObjectMapping( - tpe = CountryType, - fieldMappings = List( - SqlField("code", country.code, key = true, hidden = true), - SqlField("name", country.name), - SqlField("continent", country.continent), - SqlField("region", country.region), - SqlField("surfacearea", country.surfacearea), - SqlField("indepyear", country.indepyear), - SqlField("population", country.population), - SqlField("lifeexpectancy", country.lifeexpectancy), - SqlField("gnp", country.gnp), - SqlField("gnpold", country.gnpold), - SqlField("localname", country.localname), - SqlField("governmentform", country.governmentform), - SqlField("headofstate", country.headofstate), - SqlField("capitalId", country.capitalId), - SqlField("code2", country.code2), - SqlField("numCities", country.numCities), - SqlObject("cities", Join(country.code, city.countrycode)), - SqlObject("languages", Join(country.code, countrylanguage.countrycode)) - ), + ObjectMapping(CountryType)( + SqlField("code", country.code, key = true, hidden = true), + SqlField("name", country.name), + SqlField("continent", country.continent), + SqlField("region", country.region), + SqlField("surfacearea", country.surfacearea), + SqlField("indepyear", country.indepyear), + SqlField("population", country.population), + SqlField("lifeexpectancy", country.lifeexpectancy), + SqlField("gnp", country.gnp), + SqlField("gnpold", country.gnpold), + SqlField("localname", country.localname), + SqlField("governmentform", country.governmentform), + SqlField("headofstate", country.headofstate), + SqlField("capitalId", country.capitalId), + SqlField("code2", country.code2), + SqlField("numCities", country.numCities), + SqlObject("cities", Join(country.code, city.countrycode)), + SqlObject("languages", Join(country.code, countrylanguage.countrycode)) ), - ObjectMapping( - tpe = CityType, - fieldMappings = List( - SqlField("id", city.id, key = true, hidden = true), - SqlField("countrycode", city.countrycode, hidden = true), - SqlField("name", city.name), - SqlField("district", city.district), - SqlField("population", city.population), - SqlObject("country", Join(city.countrycode, country.code)), - ) + ObjectMapping(CityType)( + SqlField("id", city.id, key = true, hidden = true), + SqlField("countrycode", city.countrycode, hidden = true), + SqlField("name", city.name), + SqlField("district", city.district), + SqlField("population", city.population), + SqlObject("country", Join(city.countrycode, country.code)), ), - ObjectMapping( - tpe = LanguageType, - fieldMappings = List( - SqlField("language", countrylanguage.language, key = true, associative = true), - SqlField("isOfficial", countrylanguage.isOfficial), - SqlField("percentage", countrylanguage.percentage), - SqlField("countrycode", countrylanguage.countrycode, hidden = true), - SqlObject("countries", Join(countrylanguage.countrycode, country.code)) - ) + ObjectMapping(LanguageType)( + SqlField("language", countrylanguage.language, key = true, associative = true), + SqlField("isOfficial", countrylanguage.isOfficial), + SqlField("percentage", countrylanguage.percentage), + SqlField("countrycode", countrylanguage.countrycode, hidden = true), + SqlObject("countries", Join(countrylanguage.countrycode, country.code)) ) )