-
Notifications
You must be signed in to change notification settings - Fork 38.3k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
@Scheduled task instrumentation does not work for Kotlin suspend functions #32165
Comments
@Scheduled
task instrumentation does not work for Kotlin suspend functions
@Scheduled
task instrumentation does not work for Kotlin suspend functions
@sdeleuze and I investigated this issue and we've found that this is not a simple Spring Framework issue, but rather a broader problem with Kotlin Coroutines and Observability. This issue explored in micrometer-metrics/tracing#174. How this works with plain Spring MVCWith Spring MVC, when an observation is created, we can then use it to open a scope around some code execution. The opening of this scope is handled by the tracing infrastructure and sets up the relevant ThreadLocal and MDC values. The logging statements, when executed, have all the information in context and you can see traceId and spanId. How this works with Reactor Mono and FluxWith Reactor, work can be scheduled on any worker thread so this is not as straightforward. With Kotlin CoroutinesIn this case, the reactor context is propagated by the But unlike Reactor, there is no automatic integration with context propagation, ThreadLocals or the MDC. In fact, it seems Kotlin Coroutines expect users to directly interact with the context to get values, or compose with a I have managed to get this working with a custom function that leverages existing Micrometer infrastructure: import io.micrometer.core.instrument.kotlin.asContextElement
import io.micrometer.observation.Observation
import io.micrometer.observation.contextpropagation.ObservationThreadLocalAccessor
import kotlinx.coroutines.reactor.ReactorContext
import kotlinx.coroutines.withContext
import org.slf4j.LoggerFactory
import org.springframework.scheduling.annotation.Scheduled
import org.springframework.stereotype.Component
import reactor.util.context.ContextView
import java.util.concurrent.TimeUnit
import kotlin.coroutines.CoroutineContext
import kotlin.coroutines.coroutineContext
@Component
class SchedulingService {
companion object {
val logger = LoggerFactory.getLogger(SchedulingService::class.java.name)
}
@Scheduled(fixedDelay = 5, timeUnit = TimeUnit.SECONDS)
suspend fun suspendable() {
withContext(observationContext(coroutineContext)) {
logger.info("Suspendable")
}
}
fun observationContext(context: CoroutineContext) : CoroutineContext {
// get the Reactor context from the CoroutineContext
val contextView = context[ReactorContext]!!.context as ContextView
// this context contains the current observation under this well known key
// because the @Scheduled instrumentation contributed it
val observation = contextView.get(ObservationThreadLocalAccessor.KEY) as Observation
// we can then use this Micrometer context to wrap the execution
// the observation scope and MDC will be taken care of
return observation.observationRegistry.asContextElement()
}
} This is by no means the solution we're advertizing and we're not sure how to tackle this problem at this point. I'm leaving this issue opened for now because we might want to revisit the Coroutine to Publisher arrangement in this case, in order to pass a custom CoroutineContext. But this issue should be mainly discussed and tackled with the Micrometer team. |
I think I've refined a solution that could be contributed to the context-propagation project. This package io.mircrometer.context
import io.micrometer.context.ContextRegistry
import io.micrometer.context.ContextSnapshot
import io.micrometer.context.ContextSnapshotFactory
import kotlinx.coroutines.ThreadContextElement
import kotlinx.coroutines.reactor.ReactorContext
import reactor.util.context.ContextView
import kotlin.coroutines.AbstractCoroutineContextElement
import kotlin.coroutines.CoroutineContext
class PropagationContextElement(private val context: CoroutineContext) : ThreadContextElement<ContextSnapshot.Scope>,
AbstractCoroutineContextElement(Key) {
public companion object Key : CoroutineContext.Key<PropagationContextElement>
val contextSnapshot: ContextSnapshot
get() {
val contextView: ContextView? = context[ReactorContext]?.context
val contextSnapshotFactory =
ContextSnapshotFactory.builder().contextRegistry(ContextRegistry.getInstance()).build()
if (contextView != null) {
return contextSnapshotFactory.captureFrom(contextView)
}
return contextSnapshotFactory.captureAll()
}
override fun restoreThreadContext(context: CoroutineContext, scope: ContextSnapshot.Scope) {
scope.close()
}
override fun updateThreadContext(context: CoroutineContext): ContextSnapshot.Scope {
return contextSnapshot.setThreadLocals()
}
} Using it in the application leverages the context-propagation @Scheduled(fixedDelay = 5, timeUnit = TimeUnit.SECONDS)
suspend fun suspendable() {
withContext(PropagationContextElement(coroutineContext)) {
logger.info("Suspendable")
}
} WDYT @sdeleuze ? |
I think I like the direction taken by this proposal as Spring support for Coroutines is tightly linked to the Reactive support, and the scope of this issue is probably wider than scheduling support and could potentially solved what has been discussed for months in micrometer-metrics/tracing#174. But we should experiment and discuss about where and how this potential feature should be provided and integrated. To make this feature usable, I think we should find a way to configure it automatically. As proposed by @antechrestos in this comment, I am wondering if we could use this kind of facility to provide that support seamlessly at Spring level. For example in if (coroutineContext == null) {
return CoroutinesUtils.invokeSuspendingFunction(method, target, args);
}
else {
return CoroutinesUtils.invokeSuspendingFunction((CoroutineContext) coroutineContext, method, target, args);
} To something like: if (coroutineContext == null) {
return CoroutinesUtils.invokeSuspendingFunction(PropagationContextElement(Dispatchers.getUnconfined()), method, target, args);
}
else {
return CoroutinesUtils.invokeSuspendingFunction(PropagationContextElement((CoroutineContext) coroutineContext), method, target, args);
} and do the same for IMO that could provide the level of integration people expect from Spring and would be consistent with what we do on Reactive side. If provided at context-propagation level, we should make sure this will be provided in a dependency we can use in Spring Java code (potentially using defensive classpath check and nest class to ensure this remains optional). Any thoughts? |
@sdeleuze btw, I guess you meant to write something like this (you need to use
Either way, I would totally love having something like However, If Spring would then add this element to the context in all cases when suspending Kotlin code is called (i.e. WebFlux Controller methods, coRouter, @scheduled etc.) it would hopefully take care of all ThreadLocalAccessors at once. I am wondering if Edit: I realized that this context element may need to be added earlier, as something like CoWebFilter would already require it to be in the context. And as a side note, looks like there are multiple places in Spring Framework which are all creating "an initial" coroutine context, which right now happens to be just consist of |
I have discussed this with the Micrometer team and finding a proper home for
|
Hello, I seem to have found another context propagation issue when using proxied beans. @GetMapping("test")
@ResponseStatus(OK
suspend fun test() {
CoroutineScope(SupervisorJob()).launch(observationRegistry.asContextElement()) {
proxiedService.doSomethingSuspend()
}
} And class ProxiedServiceImpl(private val webClient: WebClient) : ProxiedService {
override suspend fun doSomethingSuspend() {
webClient.get()
.uri("https://www.google.fr")
.awaitExchange {
println(MDC.getCopyOfContextMap())
}
}
} and it is proxied using @Bean
fun autoProxyCreator() = BeanNameAutoProxyCreator().apply {
setBeanNames("proxiedService")
setInterceptorNames("dummyInterceptor")
}
@Bean
fun dummyInterceptor(observationRegistry: ObservationRegistry) = MethodInterceptor { invocation -> invocation.proceed() } Then the MDC context will be empty in the If the service is not a proxy, it works fine. Also if the service is not called inside the coroutine, but rather directly inside the controller method, it works with the proxy too. So I wonder if there is not an issue with Spring AOP with observability. |
@grassehh I'm just curious if this is because Could you try grabbing the |
Doesn't seem to work. You can checkout my sample here. The thing is that if you use the Alternatively like I said, if you pass the parent |
I'm closing this issue for now, in favor of Kotlin/kotlinx.coroutines#4187. We can reopen this issue if we need to reconsider. |
Spring docs for Scheduled tasks instrumentation states:
This function get an automatic observation:
but this suspend function does not:
I use Spring Boot 3.2.2 and I've also tried 3.2.3-SNAPSHOT and 3.3.0-M1.
build.gradle.kts (shortened):
Application code:
When I run the application I get the following output:
There we can se that "65b9e9b65a4ecd1702feecf2dbdd6be4,02feecf2dbdd6be4" means that the function "nonSuspendable" gets an observation but "suspendable" doesn't.
Regards Peter
The text was updated successfully, but these errors were encountered: