From dfc769152826d7cd5e2f639cc387f4c655d6dc9b Mon Sep 17 00:00:00 2001 From: Miles Sabin Date: Mon, 29 Apr 2024 12:21:15 +0100 Subject: [PATCH] Reworked mapping management and validation + The collection of type mappings has moved to a new first class TypeMappings container type. There is an implicit conversion from List[TypeMapping] to TypeMappings, so this is a source compatible change (this will probably be deprecated in a later release). Most mapping validtion logic has been moved to TypeMappings, apart from mapping-specific rules (eg. for SqlMapping) which are delegated to the relevant Mapping subtype at the granularity of validity for individual type and field mappings. + Added MappingPredicate as an extensible mechanism for matching mappings to paths. TypeMatch replaces the existing Type linked association, PrefixedTypeMatch replaces the PrefixedMapping wrapper, and PathMatch corresponds to the the SwitchTypeMapping used by Gemini/Aura. + Mapping validation is now performed by default unless explicitly disabled. Validation is deferred until the mappings compiler is first referenced (to avoid init-order issues, object initialization errors and poor interatactions of the latter with munit-cats-effect) and is performed exactly once. Applications which have at least one guery unit test will trigger validation failures automatically at test time. + The validation algorithm has been reworked to start from the GraphQL schema and generate an exhaustive set of paths which are used to determine the relevant mappings to check against GraphQL and DB schema types. The support both traversal through mappings which are partly implicit (eg. the Circe and GenericMappings) and also accomodate mappings which are guarded by MappingPredicates. + The base mapping validator now also reports unused (ie. not reachable via any valid path in the GraphQL schema) type and field mappings. + The SQL mapping validator tests for, + Consistency of nullability and Scala type between GraphQL and SQL. + Leaf/ObjectMapping consistency with GraphQL leaf or non-leaf types. + Objects, interfaces and unions being nested within single a single DB table. + Associative fields also being keys. + Union field mappings must be hidden and leaf. + Embedded subobjects must be nested in their parent object table. + Joins must have at least one join condition. + Parallel joins must relate the same tables. + Serial joins must chain correctly. + Type mappings with non-trivial predicates are now indexed. + LeafMapping now supports MappingPrediates and are indexed along with ObjectMappings. + Built-in Scalar types now have explicit LeafMappings and can now be specialized for particular contexts via MappingPredicates. + Added a subtree Boolean attribute to FieldMapping to signal that a field value can represent structured result subtrees which are not explicitly mapped at all levels. This allows mapping validation to include the Circe and Generic mappings which implicitly map subtrees. + The ValueObjectMapping has a new on constructor and the withParent initializer method, which was present on all mappings has been removed and replaced by the ValueFieldMapping specific unwrap. + The IntrospectionMapping now uses the new mapping API. + All tests have been updated so validate under the new scheme. + The Doobie and Skunk test suites now have typed codecs which can capture the metadata needed for validation. + The tutorial and demo and profile projects have been updated to the new API. + Added schema validation checks to ensure interfaces are non-cyclic. + Added forUnderlyingNamed convenience method to Context. + ValidationFailure split out to a separate file. + The sql classifier has been moved from ValidationFailure to a SqlValidationFailure subtype. + GraphQL types menioned in ValidationFailure error messages are now correctly rendered uing GraphQL syntax. + ComposedMapping has been split out to a separate file. + Various signatures previously using List now use Seq. --- .../scala/demo/starwars/StarWarsMapping.scala | 16 +- .../main/scala/demo/world/WorldMapping.scala | 94 +-- .../circe/src/main/scala/circemapping.scala | 6 +- .../src/test/scala/CircePrioritySuite.scala | 2 +- modules/core/src/main/scala/compiler.scala | 6 +- .../core/src/main/scala/composedmapping.scala | 47 ++ modules/core/src/main/scala/cursor.scala | 6 + .../core/src/main/scala/introspection.scala | 190 ++--- modules/core/src/main/scala/mapping.scala | 795 +++++++++++++++--- .../src/main/scala/mappingvalidator.scala | 303 ------- modules/core/src/main/scala/schema.scala | 59 +- .../src/main/scala/validationfailure.scala | 81 ++ .../core/src/main/scala/valuemapping.scala | 63 +- .../test/scala/compiler/CompilerSuite.scala | 5 +- .../src/test/scala/compiler/TestMapping.scala | 2 +- .../directives/DirectiveValidationSuite.scala | 2 +- .../introspection/IntrospectionSuite.scala | 90 +- .../scala/mapping/MappingValidatorSuite.scala | 350 ++++++-- .../src/test/scala/schema/SchemaSuite.scala | 60 ++ .../test/scala/starwars/StarWarsData.scala | 2 +- .../test/scala/starwars/StarWarsSuite.scala | 10 - .../subscription/SubscriptionSuite.scala | 2 +- .../src/main/scala/DoobieMapping.scala | 1 + .../src/test/scala/DoobieDatabaseSuite.scala | 46 +- .../src/test/scala/DoobieSuites.scala | 28 +- .../src/main/scala-2/genericmapping2.scala | 2 +- .../src/main/scala-3/genericmapping3.scala | 2 +- .../src/main/scala/genericmapping.scala | 11 +- .../src/test/scala/SkunkDatabaseSuite.scala | 60 +- .../js-jvm/src/test/scala/SkunkSuites.scala | 28 +- .../shared/src/main/scala/SkunkMapping.scala | 1 + .../shared/src/main/scala/SqlMapping.scala | 667 ++++++++++++++- .../src/main/scala/SqlMappingValidator.scala | 111 --- .../sql/shared/src/main/scala/SqlModule.scala | 2 + .../shared/src/test/resources/db/movies.sql | 25 +- .../src/test/scala/SqlCoalesceMapping.scala | 4 +- .../src/test/scala/SqlCoalesceSuite.scala | 2 +- .../test/scala/SqlComposedWorldMapping.scala | 1 - .../src/test/scala/SqlCursorJsonMapping.scala | 3 +- .../src/test/scala/SqlEmbeddingMapping.scala | 1 - .../src/test/scala/SqlGraphMapping.scala | 6 +- .../src/test/scala/SqlInterfacesMapping.scala | 4 +- .../src/test/scala/SqlJsonbMapping.scala | 3 +- .../SqlMappingValidatorInvalidMapping.scala | 241 ++++++ .../SqlMappingValidatorInvalidSuite.scala | 219 +++++ .../SqlMappingValidatorValidMapping.scala | 385 +++++++++ .../scala/SqlMappingValidatorValidSuite.scala | 36 + .../src/test/scala/SqlMovieMapping.scala | 34 +- .../shared/src/test/scala/SqlMovieSuite.scala | 92 +- .../scala/SqlRecursiveInterfacesMapping.scala | 4 +- .../src/test/scala/SqlTestMapping.scala | 55 +- profile/src/main/scala/Bench.scala | 112 ++- 52 files changed, 3337 insertions(+), 1040 deletions(-) create mode 100644 modules/core/src/main/scala/composedmapping.scala delete mode 100644 modules/core/src/main/scala/mappingvalidator.scala create mode 100644 modules/core/src/main/scala/validationfailure.scala delete mode 100644 modules/sql/shared/src/main/scala/SqlMappingValidator.scala create mode 100644 modules/sql/shared/src/test/scala/SqlMappingValidatorInvalidMapping.scala create mode 100644 modules/sql/shared/src/test/scala/SqlMappingValidatorInvalidSuite.scala create mode 100644 modules/sql/shared/src/test/scala/SqlMappingValidatorValidMapping.scala create mode 100644 modules/sql/shared/src/test/scala/SqlMappingValidatorValidSuite.scala 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..3b06aecf 100644 --- a/modules/core/src/main/scala/mapping.scala +++ b/modules/core/src/main/scala/mapping.scala @@ -16,10 +16,11 @@ 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} @@ -33,6 +34,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 +42,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 +104,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 +123,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 +133,591 @@ 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 */ + 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 +737,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 +804,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 +850,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 +876,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 +889,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 +910,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 +928,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 +950,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 +970,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 +1045,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)) ) )