diff --git a/tools/src/main/scala/caliban/tools/SchemaWriter.scala b/tools/src/main/scala/caliban/tools/SchemaWriter.scala index e7756988f3..d6804fa678 100644 --- a/tools/src/main/scala/caliban/tools/SchemaWriter.scala +++ b/tools/src/main/scala/caliban/tools/SchemaWriter.scala @@ -92,7 +92,15 @@ object SchemaWriter { def isAbstractEffectful(typedef: ObjectTypeDefinition): Boolean = isEffectTypeAbstract && isEffectful(typedef) - def isEffectful(typedef: ObjectTypeDefinition): Boolean = isLocalEffectful(typedef) || isNestedEffectful(typedef) + def isEffectful(typedef: ObjectTypeDefinition): Boolean = + isLocalEffectful(typedef) || isNestedEffectful(typedef) + + def isUnionSiblingAbstractEffectful(typedef: ObjectTypeDefinition): Boolean = + schema.unionTypeDefinitions + .exists(union => + union.memberTypes.contains(typedef.name) && + union.memberTypes.flatMap(schema.objectTypeDefinition).exists(isAbstractEffectful) + ) def isLocalEffectful(typedef: ObjectTypeDefinition): Boolean = hasFieldWithDirective(typedef, LazyDirective) @@ -103,7 +111,7 @@ object SchemaWriter { .exists(t => hasFieldWithDirective(t, LazyDirective)) def generic(op: ObjectTypeDefinition, isRootDefinition: Boolean = false): String = - if ((isRootDefinition && isEffectTypeAbstract) || isAbstractEffectful(op)) + if ((isRootDefinition && isEffectTypeAbstract) || isAbstractEffectful(op) || isUnionSiblingAbstractEffectful(op)) s"[${effect}[_]]" else s"" @@ -179,10 +187,20 @@ object SchemaWriter { } """ - def writeUnions(unions: List[UnionTypeDefinition]): String = - unions.map(x => writeUnionSealedTrait(x)).mkString("\n") + def writeUnions(unions: Map[UnionTypeDefinition, List[ObjectTypeDefinition]]): String = + unions.map { case (unionDecl, unionMembers) => + if (unionMembers.exists(isAbstractEffectful)) { + writeUnionSealedTraitWithAbstractEffect(unionDecl) + } else { + writeUnionSealedTrait(unionDecl) + } + }.mkString("\n") - def writeUnionSealedTrait(union: UnionTypeDefinition): String = + def writeUnionSealedTraitWithAbstractEffect(union: UnionTypeDefinition): String = + s"""${writeTypeAnnotations( + union + )}sealed trait ${union.name}[F[_]] extends scala.Product with scala.Serializable$derivesSchema""" + def writeUnionSealedTrait(union: UnionTypeDefinition): String = s"""${writeTypeAnnotations( union )}sealed trait ${union.name} extends scala.Product with scala.Serializable$derivesSchema""" @@ -417,7 +435,7 @@ object SchemaWriter { .map(union => (union, union.memberTypes.flatMap(schema.objectTypeDefinition))) .toMap - val unions = writeUnions(schema.unionTypeDefinitions) + val unions = writeUnions(unionTypes) val interfacesStr = schema.interfaceTypeDefinitions.map { interface => writeInterface(interface) @@ -432,7 +450,10 @@ object SchemaWriter { ) .map { obj => val extendsInterfaces = obj.implements.map(name => name.name) - val partOfUnionTypes = unionTypes.collect { case (u, os) if os.exists(_.name == obj.name) => u.name } + val partOfUnionTypes = unionTypes.collect { + case (u, members) if members.exists(_.name == obj.name) => + if (members.exists(isAbstractEffectful)) s"${u.name}[F]" else u.name + } writeObject(obj, extend = extendsInterfaces ++ partOfUnionTypes) } .mkString("\n") diff --git a/tools/src/test/resources/snapshots/SchemaWriterSpec/union, abstracted effect type and lazy combination.scala b/tools/src/test/resources/snapshots/SchemaWriterSpec/union, abstracted effect type and lazy combination.scala new file mode 100644 index 0000000000..b37fd47031 --- /dev/null +++ b/tools/src/test/resources/snapshots/SchemaWriterSpec/union, abstracted effect type and lazy combination.scala @@ -0,0 +1,22 @@ +import Types._ + +object Types { + final case class BLazyFieldWithArgsArgs(int: scala.Option[Int]) + final case class A[F[_]](lazyField: F[scala.Option[Int]]) extends U0[F] + final case class B[F[_]](lazyFieldWithArgs: BLazyFieldWithArgsArgs => F[scala.Option[Int]]) extends U0[F] + final case class C[F[_]](field: scala.Option[Int]) extends U0[F] with U1 + final case class D(field: scala.Option[Int]) extends U1 + + sealed trait U0[F[_]] extends scala.Product with scala.Serializable + sealed trait U1 extends scala.Product with scala.Serializable + +} + +object Operations { + + final case class Query[F[_]]( + effectful: F[U0[F]], + pure: F[U1] + ) + +} diff --git a/tools/src/test/scala/caliban/tools/SchemaWriterSpec.scala b/tools/src/test/scala/caliban/tools/SchemaWriterSpec.scala index 4ca18bbab0..71735c700c 100644 --- a/tools/src/test/scala/caliban/tools/SchemaWriterSpec.scala +++ b/tools/src/test/scala/caliban/tools/SchemaWriterSpec.scala @@ -80,6 +80,32 @@ object SchemaWriterSpec extends SnapshotTest { isEffectTypeAbstract = true ) ), + snapshotTest("union, abstracted effect type and lazy combination")( + gen( + """ + type Query { + effectful: U0! + pure: U1! + } + union U0 = A | B | C + union U1 = C | D + type A { + lazyField: Int @lazy + } + type B { + lazyFieldWithArgs(int: Int): Int @lazy + } + type C { + field: Int + } + type D { + field: Int + } + """, + effect = "F", + isEffectTypeAbstract = true + ) + ), snapshotTest("simple mutation with abstracted effect type")( gen( """