main.com.netflix.graphql.dgs.internal.ReactiveDataFetcherResultProcessor.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.internal
import com.netflix.graphql.dgs.DgsDataFetchingEnvironment
import com.netflix.graphql.dgs.context.ReactiveDgsContext
import graphql.language.OperationDefinition
import kotlinx.coroutines.CoroutineScope
import kotlinx.coroutines.Dispatchers
import kotlinx.coroutines.flow.Flow
import kotlinx.coroutines.flow.toList
import kotlinx.coroutines.future.future
import kotlinx.coroutines.reactive.asPublisher
import reactor.core.publisher.Flux
import reactor.core.publisher.Mono
import reactor.util.context.Context
class MonoDataFetcherResultProcessor : DataFetcherResultProcessor {
override fun supportsType(originalResult: Any): Boolean = originalResult is Mono<*>
override fun process(
originalResult: Any,
dfe: DgsDataFetchingEnvironment,
): Any {
val mono =
originalResult as? Mono<*>
?: throw IllegalArgumentException(
"Instance passed to ${this::class.qualifiedName} was not a Mono<*>. It was a ${originalResult::class.qualifiedName} instead",
)
return when (dfe.operationDefinition.operation) {
OperationDefinition.Operation.SUBSCRIPTION -> mono
else -> originalResult.contextWrite(reactorContextFrom(dfe)).toFuture()
}
}
}
class FlowDataFetcherResultProcessor : DataFetcherResultProcessor {
override fun supportsType(originalResult: Any): Boolean = originalResult is Flow<*>
override fun process(
originalResult: Any,
dfe: DgsDataFetchingEnvironment,
): Any {
@Suppress("unchecked_cast")
val flow =
originalResult as? Flow
?: throw IllegalArgumentException(
"Instance passed to ${this::class.qualifiedName} was not a Flow<*>. It was a ${originalResult::class.qualifiedName} instead",
)
return when (dfe.operationDefinition.operation) {
OperationDefinition.Operation.SUBSCRIPTION -> flow.asPublisher()
else -> CoroutineScope(Dispatchers.Default).future { flow.toList() }
}
}
}
class FluxDataFetcherResultProcessor : DataFetcherResultProcessor {
override fun supportsType(originalResult: Any): Boolean = originalResult is Flux<*>
override fun process(
originalResult: Any,
dfe: DgsDataFetchingEnvironment,
): Any {
val flux =
originalResult as? Flux<*>
?: throw IllegalArgumentException(
"Instance passed to ${this::class.qualifiedName} was not a Flux<*>. It was a ${originalResult::class.qualifiedName} instead",
)
return when (dfe.operationDefinition.operation) {
OperationDefinition.Operation.SUBSCRIPTION -> flux
else -> flux.contextWrite(reactorContextFrom(dfe)).collectList().toFuture()
}
}
}
private fun reactorContextFrom(dfe: DgsDataFetchingEnvironment) = ReactiveDgsContext.from(dfe)?.reactorContext ?: Context.empty()
© 2015 - 2025 Weber Informatics LLC | Privacy Policy