com.expediagroup.graphql.federation.validation.validateProvidesDirective.kt Maven / Gradle / Ivy
/*
* Copyright 2019 Expedia, Inc
*
* 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
*
* https://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 com.expediagroup.graphql.federation.validation
import com.expediagroup.graphql.extensions.unwrapType
import com.expediagroup.graphql.federation.directives.PROVIDES_DIRECTIVE_NAME
import com.expediagroup.graphql.federation.extensions.isExtendedType
import graphql.schema.GraphQLFieldDefinition
import graphql.schema.GraphQLObjectType
import graphql.schema.GraphQLTypeReference
// [OK] @provides on base type references valid @external fields on @extend object
// [ERROR] @provides on base type references local object fields
// [ERROR] @provides on base type references local fields on @extends object
// [ERROR] @provides references interface type
// [OK] @provides references list of valid @extend objects
// [ERROR] @provides references @external list field
// [ERROR] @provides references @external interface field
internal fun validateProvidesDirective(federatedType: String, field: GraphQLFieldDefinition): List = when (val returnType = field.type.unwrapType()) {
is GraphQLObjectType -> {
if (!returnType.isExtendedType()) {
listOf("@provides directive is specified on a $federatedType.${field.name} field references local object")
} else {
val returnTypeFields = returnType.fieldDefinitions.associateBy { it.name }
// @provides is applicable on both base and federated types and always references @external fields
validateDirective(
"$federatedType.${field.name}",
PROVIDES_DIRECTIVE_NAME,
field.directivesByName,
returnTypeFields,
true
)
}
}
// skip validation for nested object types as they are still under construction
is GraphQLTypeReference -> emptyList()
else -> listOf("@provides directive is specified on a $federatedType.${field.name} field but it does not return an object type")
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy