diff --git a/lib/src/main/java/graphql/nadel/validation/NadelAssignableTypeValidation.kt b/lib/src/main/java/graphql/nadel/validation/NadelAssignableTypeValidation.kt index 4b70ea0c1..6a3c01e1a 100644 --- a/lib/src/main/java/graphql/nadel/validation/NadelAssignableTypeValidation.kt +++ b/lib/src/main/java/graphql/nadel/validation/NadelAssignableTypeValidation.kt @@ -2,30 +2,56 @@ package graphql.nadel.validation import graphql.nadel.engine.util.unwrapAll import graphql.schema.GraphQLType -import graphql.schema.GraphQLUnmodifiedType class NadelAssignableTypeValidation internal constructor( private val typeWrappingValidation: NadelTypeWrappingValidation, ) { + context(NadelValidationContext) + fun isOutputTypeAssignable( + overallType: GraphQLType, + underlyingType: GraphQLType, + ): Boolean { + return isTypeAssignable( + suppliedType = underlyingType, + requiredType = overallType, + // Compare underlying type names + suppliedTypeName = underlyingType.unwrapAll().name, + requiredTypeName = getUnderlyingTypeName(overallType.unwrapAll()), + ) + } + + context(NadelValidationContext) + fun isInputTypeAssignable( + overallType: GraphQLType, + underlyingType: GraphQLType, + ): Boolean { + return isTypeAssignable( + suppliedType = overallType, + requiredType = underlyingType, + // Compare underlying type names + suppliedTypeName = getUnderlyingTypeName(overallType.unwrapAll()), + requiredTypeName = underlyingType.unwrapAll().name, + ) + } + context(NadelValidationContext) fun isTypeAssignable( suppliedType: GraphQLType, requiredType: GraphQLType, + suppliedTypeName: String, + requiredTypeName: String, ): Boolean { - val typeWrappingValid = typeWrappingValidation.isTypeWrappingValid( + return suppliedTypeName == requiredTypeName && isTypeWrappingValid(suppliedType, requiredType) + } + + private fun isTypeWrappingValid( + suppliedType: GraphQLType, + requiredType: GraphQLType, + ): Boolean { + return typeWrappingValidation.isTypeWrappingValid( lhs = suppliedType, rhs = requiredType, rule = NadelTypeWrappingValidation.Rule.LHS_MUST_BE_STRICTER_OR_SAME, ) - - return typeWrappingValid && isTypeNameValid(suppliedType.unwrapAll(), requiredType.unwrapAll()) - } - - context(NadelValidationContext) - private fun isTypeNameValid( - overallType: GraphQLUnmodifiedType, - underlyingType: GraphQLUnmodifiedType, - ): Boolean { - return getUnderlyingTypeName(overallType) == underlyingType.name } } diff --git a/lib/src/main/java/graphql/nadel/validation/NadelFieldValidation.kt b/lib/src/main/java/graphql/nadel/validation/NadelFieldValidation.kt index f291d792b..41b217012 100644 --- a/lib/src/main/java/graphql/nadel/validation/NadelFieldValidation.kt +++ b/lib/src/main/java/graphql/nadel/validation/NadelFieldValidation.kt @@ -101,9 +101,9 @@ class NadelFieldValidation internal constructor( } else { // Note: the value comes from the user (overall schema) // So we are supplying the overall argument to the underlying argument - val isArgumentTypeAssignable = assignableTypeValidation.isTypeAssignable( - suppliedType = overallArg.type, - requiredType = underlyingArg.type + val isArgumentTypeAssignable = assignableTypeValidation.isInputTypeAssignable( + overallType = overallArg.type, + underlyingType = underlyingArg.type ) if (isArgumentTypeAssignable) { ok() @@ -190,9 +190,9 @@ class NadelFieldValidation internal constructor( underlyingField: GraphQLFieldDefinition, ): NadelSchemaValidationResult { // Note: the value comes from the underlying schema, so we are supplying the underlying field to the overall field - val isUnderlyingTypeAssignable = assignableTypeValidation.isTypeAssignable( - suppliedType = underlyingField.type, - requiredType = overallField.type, + val isUnderlyingTypeAssignable = assignableTypeValidation.isOutputTypeAssignable( + overallType = overallField.type, + underlyingType = underlyingField.type, ) return if (isUnderlyingTypeAssignable) { diff --git a/lib/src/main/java/graphql/nadel/validation/NadelInputObjectValidation.kt b/lib/src/main/java/graphql/nadel/validation/NadelInputObjectValidation.kt index 2bf057ae1..01febeba9 100644 --- a/lib/src/main/java/graphql/nadel/validation/NadelInputObjectValidation.kt +++ b/lib/src/main/java/graphql/nadel/validation/NadelInputObjectValidation.kt @@ -55,10 +55,15 @@ class NadelInputObjectValidation internal constructor( overallInputField: GraphQLInputObjectField, underlyingInputField: GraphQLInputObjectField, ): NadelSchemaValidationResult { - return if (!assignableTypeValidation.isTypeAssignable(overallInputField.type, underlyingInputField.type)) { - IncompatibleFieldInputType(parent, overallInputField, underlyingInputField) - } else { + val isTypeAssignable = assignableTypeValidation.isInputTypeAssignable( + overallType = overallInputField.type, + underlyingType = underlyingInputField.type + ) + + return if (isTypeAssignable) { ok() + } else { + IncompatibleFieldInputType(parent, overallInputField, underlyingInputField) } } } diff --git a/lib/src/main/java/graphql/nadel/validation/NadelValidationContext.kt b/lib/src/main/java/graphql/nadel/validation/NadelValidationContext.kt index f179c41fe..f189dac1e 100644 --- a/lib/src/main/java/graphql/nadel/validation/NadelValidationContext.kt +++ b/lib/src/main/java/graphql/nadel/validation/NadelValidationContext.kt @@ -69,6 +69,13 @@ fun getHydrationDefinitions( .filterIsInstance() } +context(NadelValidationContext) +fun isRenamed( + container: NadelServiceSchemaElement.Type, +): Boolean { + return getRenamedOrNull(container.overall) != null +} + context(NadelValidationContext) fun isRenamed( container: GraphQLFieldsContainer, diff --git a/lib/src/main/java/graphql/nadel/validation/NadelVirtualTypeValidation.kt b/lib/src/main/java/graphql/nadel/validation/NadelVirtualTypeValidation.kt index a878fbca4..4bab261f4 100644 --- a/lib/src/main/java/graphql/nadel/validation/NadelVirtualTypeValidation.kt +++ b/lib/src/main/java/graphql/nadel/validation/NadelVirtualTypeValidation.kt @@ -62,6 +62,10 @@ class NadelVirtualTypeValidation internal constructor( SKIP -> return ok() } + if (isRenamed(schemaElement)) { + return NadelVirtualTypeIllegalRenameError(schemaElement) + } + if (schemaElement.overall is GraphQLObjectType && schemaElement.underlying is GraphQLObjectType) { return validateType( service = schemaElement.service, @@ -184,19 +188,19 @@ class NadelVirtualTypeValidation internal constructor( } // Note: the value comes from the backing field, and that value needs to fit the virtual field - val isOutputTypeAssignable = assignableTypeValidation.isTypeAssignable( - suppliedType = backingField.type, - requiredType = virtualField.type, + val isOutputTypeAssignable = isOutputTypeAssignable( + backingField = backingField, + virtualField = virtualField, ) return if (isOutputTypeAssignable) { + ok() + } else { NadelVirtualTypeIncompatibleFieldOutputTypeError( parent = parent, virtualField = virtualField, backingField = backingField, ) - } else { - ok() } } @@ -244,11 +248,7 @@ class NadelVirtualTypeValidation internal constructor( virtualFieldArgument: GraphQLArgument, backingFieldArgument: GraphQLArgument, ): NadelSchemaValidationResult { - // Note: the value comes from the virtual field's arg and needs to be assigned to the backing arg - val isInputTypeAssignable = assignableTypeValidation.isTypeAssignable( - suppliedType = virtualFieldArgument.type, - requiredType = backingFieldArgument.type, - ) + val isInputTypeAssignable = isInputTypeAssignable(virtualFieldArgument, backingFieldArgument) return if (isInputTypeAssignable) { ok() @@ -287,4 +287,38 @@ class NadelVirtualTypeValidation internal constructor( } }.toResult() } + + context(NadelValidationContext, NadelVirtualTypeValidationContext) + private fun isInputTypeAssignable( + virtualFieldArgument: GraphQLArgument, + backingFieldArgument: GraphQLArgument, + ): Boolean { + val suppliedType = virtualFieldArgument.type + val requiredType = backingFieldArgument.type + + return assignableTypeValidation.isTypeAssignable( + suppliedType = suppliedType, + requiredType = requiredType, + // Note: we do not check for renames here, types must be used 1-1 + suppliedTypeName = suppliedType.unwrapAll().name, + requiredTypeName = requiredType.unwrapAll().name, + ) + } + + context(NadelValidationContext, NadelVirtualTypeValidationContext) + private fun isOutputTypeAssignable( + backingField: GraphQLFieldDefinition, + virtualField: GraphQLFieldDefinition, + ): Boolean { + val suppliedType = backingField.type + val requiredType = virtualField.type + + return assignableTypeValidation.isTypeAssignable( + suppliedType = suppliedType, + requiredType = requiredType, + // Note: we do not check for renames here, types must be used 1-1 + suppliedTypeName = suppliedType.unwrapAll().name, + requiredTypeName = requiredType.unwrapAll().name, + ) + } } diff --git a/lib/src/main/java/graphql/nadel/validation/NadelVirtualTypeValidationError.kt b/lib/src/main/java/graphql/nadel/validation/NadelVirtualTypeValidationError.kt index c5b54e81f..2b65e9b1c 100644 --- a/lib/src/main/java/graphql/nadel/validation/NadelVirtualTypeValidationError.kt +++ b/lib/src/main/java/graphql/nadel/validation/NadelVirtualTypeValidationError.kt @@ -23,6 +23,15 @@ data class NadelVirtualTypeDuplicationError( get() = type.overall } +data class NadelVirtualTypeIllegalRenameError( + val type: NadelServiceSchemaElement.VirtualType, +) : NadelSchemaValidationError { + override val message: String = "Virtual types cannot be renamed" + + override val subject: GraphQLNamedSchemaElement + get() = type.overall +} + data class NadelVirtualTypeMissingBackingFieldError( val type: NadelServiceSchemaElement.VirtualType, val virtualField: GraphQLFieldDefinition, diff --git a/lib/src/test/kotlin/graphql/nadel/validation/NadelVirtualTypeValidationTest.kt b/lib/src/test/kotlin/graphql/nadel/validation/NadelVirtualTypeValidationTest.kt index 71b8c266b..642950ccd 100644 --- a/lib/src/test/kotlin/graphql/nadel/validation/NadelVirtualTypeValidationTest.kt +++ b/lib/src/test/kotlin/graphql/nadel/validation/NadelVirtualTypeValidationTest.kt @@ -376,7 +376,7 @@ class NadelVirtualTypeValidationTest { } @Test - fun `can reference `() { + fun `can use original field output type in virtual type`() { // Given val fixture = makeFixture( overallSchema = mapOf(