main.com.netflix.graphql.dgs.DgsDataFetchingEnvironment.kt Maven / Gradle / Ivy
/*
* Copyright 2022 Netflix, 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
*
* 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 com.netflix.graphql.dgs
import com.netflix.graphql.dgs.context.DgsContext
import com.netflix.graphql.dgs.exceptions.MultipleDataLoadersDefinedException
import com.netflix.graphql.dgs.exceptions.NoDataLoaderFoundException
import com.netflix.graphql.dgs.internal.utils.DataLoaderNameUtil
import graphql.schema.DataFetchingEnvironment
import org.dataloader.DataLoader
import org.springframework.context.ApplicationContext
import org.springframework.context.ConfigurableApplicationContext
import org.springframework.core.type.StandardMethodMetadata
class DgsDataFetchingEnvironment(
private val dfe: DataFetchingEnvironment,
private val ctx: ApplicationContext,
) : DataFetchingEnvironment by dfe {
fun getDfe(): DataFetchingEnvironment = this.dfe
fun getDgsContext(): DgsContext = DgsContext.from(this)
/**
* Get the value of the current object to be queried.
*
* @throws IllegalStateException if called on the root query
*/
fun getSourceOrThrow(): T = getSource() ?: throw IllegalStateException("source is null")
fun getDataLoader(loaderClass: Class<*>): DataLoader {
val annotation = loaderClass.getAnnotation(DgsDataLoader::class.java)
val loaderName =
if (annotation != null) {
DataLoaderNameUtil.getDataLoaderName(loaderClass, annotation)
} else {
val loaders = loaderClass.fields.filter { it.isAnnotationPresent(DgsDataLoader::class.java) }
if (loaders.isEmpty()) {
// annotation is not on the class, but potentially on the Bean definition
tryGetDataLoaderFromBeanDefinition(loaderClass)
} else {
if (loaders.size > 1) throw MultipleDataLoadersDefinedException(loaderClass)
val loaderField = loaders.firstOrNull() ?: throw NoDataLoaderFoundException(loaderClass)
val theAnnotation = loaderField.getAnnotation(DgsDataLoader::class.java)
theAnnotation.name
}
}
return getDataLoader(loaderName) ?: throw NoDataLoaderFoundException("DataLoader with name $loaderName not found")
}
private fun tryGetDataLoaderFromBeanDefinition(loaderClass: Class<*>): String {
var name = loaderClass.simpleName
if (ctx is ConfigurableApplicationContext) {
val beansOfType = ctx.beanFactory.getBeansOfType(loaderClass)
if (beansOfType.isEmpty()) {
throw NoDataLoaderFoundException(loaderClass)
}
if (beansOfType.size > 1) {
throw MultipleDataLoadersDefinedException(loaderClass)
}
val beanName = beansOfType.keys.first()
val beanDefinition = ctx.beanFactory.getBeanDefinition(beanName)
if (beanDefinition.source is StandardMethodMetadata) {
val methodMetadata = beanDefinition.source as StandardMethodMetadata
val method = methodMetadata.introspectedMethod
val methodAnnotation = method.getAnnotation(DgsDataLoader::class.java)
name = DataLoaderNameUtil.getDataLoaderName(loaderClass, methodAnnotation)
}
}
return name
}
/**
* Check if an argument is explicitly set using "argument.nested.property" or "argument->nested->property" syntax.
* Note that this requires String splitting which is expensive for hot code paths.
* Use the isArgumentSet(String...) as a faster alternative.
*/
fun isNestedArgumentSet(path: String): Boolean {
val pathParts = path.split(".", "->").map { s -> s.trim() }
return isArgumentSet(*pathParts.toTypedArray())
}
/**
* Check if an argument is explicitly set.
* For complex object arguments, use the isArgumentSet("root", "nested", "property") syntax.
*/
fun isArgumentSet(vararg path: String): Boolean = isArgumentSet(path.asSequence())
private fun isArgumentSet(keys: Sequence): Boolean {
var args: Map<*, *> = dfe.executionStepInfo.arguments
var value: Any?
for (key in keys) {
// Explicitly check contains to support explicit null values
if (!args.contains(key)) return false
value = args[key]
if (value !is Map<*, *>) {
return true
}
args = value
}
return true
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy