Skip to content

Commit

Permalink
Fix stuff
Browse files Browse the repository at this point in the history
  • Loading branch information
gnawf committed Nov 27, 2024
1 parent eb7bc7d commit a0a7211
Show file tree
Hide file tree
Showing 6 changed files with 104 additions and 29 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -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
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -93,9 +93,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()
Expand Down Expand Up @@ -182,9 +182,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) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -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)
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -69,6 +69,13 @@ fun getHydrationDefinitions(
.filterIsInstance<NadelHydrationDefinition>()
}

context(NadelValidationContext)
fun isRenamed(
container: NadelServiceSchemaElement.Type,
): Boolean {
return getRenamedOrNull(container.overall) == null
}

context(NadelValidationContext)
fun isRenamed(
container: GraphQLFieldsContainer,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -40,6 +40,10 @@ class NadelVirtualTypeValidation internal constructor(
return ok()
}

if (isRenamed(schemaElement)) {
return NadelVirtualTypeIllegalRenameError(schemaElement)
}

if (schemaElement.overall is GraphQLObjectType && schemaElement.underlying is GraphQLObjectType) {
return validateType(
service = schemaElement.service,
Expand Down Expand Up @@ -162,9 +166,9 @@ 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) {
Expand Down Expand Up @@ -222,11 +226,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()
Expand Down Expand Up @@ -265,4 +265,32 @@ class NadelVirtualTypeValidation internal constructor(
}
}.toResult()
}

context(NadelValidationContext, NadelVirtualTypeValidationContext)
private fun isInputTypeAssignable(
virtualFieldArgument: GraphQLArgument,
backingFieldArgument: GraphQLArgument,
): Boolean {
// Note: we do not check for renames here, types must be used 1-1
return assignableTypeValidation.isTypeAssignable(
suppliedType = virtualFieldArgument.type,
suppliedTypeName = virtualFieldArgument.type.unwrapAll().name,
requiredType = backingFieldArgument.type,
requiredTypeName = backingFieldArgument.type.unwrapAll().name,
)
}

context(NadelValidationContext, NadelVirtualTypeValidationContext)
private fun isOutputTypeAssignable(
backingField: GraphQLFieldDefinition,
virtualField: GraphQLFieldDefinition,
): Boolean {
// Note: we do not check for renames here, types must be used 1-1 unless they are virtual types
return assignableTypeValidation.isTypeAssignable(
suppliedType = backingField.type,
suppliedTypeName = backingField.type.unwrapAll().name,
requiredType = virtualField.type,
requiredTypeName = virtualField.type.unwrapAll().name,
)
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,15 @@ data class NadelVirtualTypeIllegalTypeError(
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,
Expand Down

0 comments on commit a0a7211

Please sign in to comment.