Skip to content
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

Provide a configuration property to specify the order of the ServerHttpObservationFilter #35067

Closed
davidmelia opened this issue Apr 19, 2023 · 24 comments
Labels
status: declined A suggestion or change that we don't feel we should currently apply theme: observability Issues related to observability type: enhancement A general enhancement

Comments

@davidmelia
Copy link

davidmelia commented Apr 19, 2023

Hi,

Migrating from Spring Cloud Sleuth to Spring Boot 3 there used to be a property to override the TraceFilter:

spring:
  sleuth:
    web:
      filter-order: -111

which is quite important for us as we have some filters which we need to run before any observation. I have noticed in Spring Boot 3 the order is hardcoded:

public class WebFluxObservationAutoConfiguration {

...

	@Bean
	@ConditionalOnMissingBean
	@Order(Ordered.HIGHEST_PRECEDENCE + 1)
	public ServerHttpObservationFilter webfluxObservationFilter(ObservationRegistry registry,

}

Is it possible to override ServerHttpObservationFilter order (specifically the reactive version)?

Thanks

@spring-projects-issues spring-projects-issues added the status: waiting-for-triage An issue we've not yet triaged label Apr 19, 2023
@wilkinsona
Copy link
Member

wilkinsona commented Apr 19, 2023

we have some filters which we need to run before any observation

Any filters with highest precedence should run before ServerHttpObservationFilter.

Is it possible to override ServerHttpObservationFilter order

There's no property-based support for this at the moment. Without that support you'd have to define your own ServerHttpObservationFilter bean with an appropriate @Order annotation on the @Bean method.

@davidmelia
Copy link
Author

Issue I have is ServerHttpObservationFilter is ordered as Ordered.HIGHEST_PRECEDENCE + 1 and I have three other filters. This gives me the only option to use Ordered.HIGHEST_PRECEDENCE which will randomly be ordered. :-(

I can define my own ServerHttpObservationFilter no problem but a property would be cleaner.

Thanks

@wilkinsona wilkinsona added type: enhancement A general enhancement and removed status: waiting-for-triage An issue we've not yet triaged labels May 9, 2023
@wilkinsona wilkinsona added this to the 3.2.x milestone May 9, 2023
@wilkinsona wilkinsona added the theme: observability Issues related to observability label May 9, 2023
@philwebb philwebb mentioned this issue Jun 7, 2023
31 tasks
@mhalbritter
Copy link
Contributor

The property is named management.observations.http.server.filter.order.

@mhalbritter mhalbritter modified the milestones: 3.2.x, 3.2.0-M1 Jun 13, 2023
@wilkinsona
Copy link
Member

We need to consider the servlet side of things as well. The property should either be applied to the auto-configured webMvcObservationFilter or its name should indicate that it's specific to reactive. Probably the former.

@wilkinsona wilkinsona reopened this Jun 14, 2023
@wilkinsona wilkinsona changed the title Allow override of the order of ServerHttpObservationFilter in WebFluxObservationAutoConfiguration Provide a configuration property to specify the order of the ServerHttpObservationFilter Jun 14, 2023
@mhalbritter
Copy link
Contributor

Thanks Andy, i missed that.

@mhalbritter mhalbritter self-assigned this Jun 14, 2023
@wilkinsona
Copy link
Member

I've just learned about spring-projects/spring-framework@96a429a. We need to review these changes in the context of that deprecation and what it means for the requirement raised here.

/cc @bclozel

@wilkinsona wilkinsona reopened this Jun 19, 2023
@mhalbritter
Copy link
Contributor

We need to set the ObservationRegistry on the WebHttpHandlerBuilder, as described here. We can remove ServerHttpObservationFilter. For servlet applications, nothing changes.

@wilkinsona
Copy link
Member

We need to set the ObservationRegistry on the WebHttpHandlerBuilder, as described here

How would that work for the original requirement where "we have some filters which we need to run before any observation"?

@bclozel
Copy link
Member

bclozel commented Jun 29, 2023

Sorry for the late feedback.

The Servlet Filter abstraction works well in general for Spring MVC, although some people find that limited and prefer a custom Tomcat Valve for observations in order to observe the entire chain, servlet container error processing included. To be accurate, the observation filter should be as early as possible in order to measure most of the processing time. I don't think we have similar properties for other filters in Spring Boot and the general feedback so far was to configure your own filter instance. I'm not against this change, but I guess this would mean that we implement the same thing for other filters.

On the WebFlux side, the WebFilter approach was too limited according to our community. In WebFlux we don't rely on Servlet contracts and there are dedicated SPIs for this. The DispatcherHandler handles HTTP exchanges and delegates to WebFilter and WebExceptionHandler. This means that error handling in WebExceptionHandler happens outside of the scope of WebFilter observations and error logs did not contain the trace context as expected. We fixed that in spring-projects/spring-framework#30013 with a change of infrastructure. Given that WebFlux manages the HTTP contracts, this was the only way to provide the feature to our users.

With the new infrastructure, the observation starts before any webfilter is involved. @davidmelia can you share more about the use case? Maybe there is something we can do to help with that without relying on extra filters?

With that in mind, if we want to provide configuration properties for servlet filters, we probably need a servlet and a reactive namespace to separate concerns are they don't align 100%.

@mhalbritter
Copy link
Contributor

mhalbritter commented Jun 29, 2023

How would that work for the original requirement where "we have some filters which we need to run before any observation"?

I'm afraid it doesn't, like Brian said. It only gets rid of the deprecation.

The changes to get rid of the deprecation are in this branch. However, there are 3 tests in WebFluxObservationAutoConfigurationTests failing. I think this is because the WebClient does something funky or we missed setting up important parts of the WebFlux infrastructure in the test, because it's working in a standalone application.

@wilkinsona
Copy link
Member

I think we should revert the changes that have been made thus far for this issue. The requirement hasn't been raised for servlet apps and, until we hear from @davidmelia, we don't know what, if anything, we might need to do for reactive apps.

@philwebb philwebb added the for: team-meeting An issue we'd like to discuss as a team to make progress label Jun 30, 2023
@davidmelia
Copy link
Author

davidmelia commented Jun 30, 2023

Hi @bclozel @wilkinsona - in terms of my use case we have a few custom filters which need to happen to set up the observation context - this includes extracting the IP address from a custom header and extracting other sources of data. All of this data is integral to our logging context information (N.B I have a custom logging.pattern.level)

Therefore, while I understand the assumption that ServerHttpObservationFilter should go first (well Ordered.HIGHEST_PRECEDENCE + 1) I have a few pre filters that need to run to set up the observation filter.

Thanks

N.B My only use case is reactive.

@mhalbritter
Copy link
Contributor

I think we should revert the changes that have been made thus far for this issue. The requirement hasn't been raised for servlet apps

I reverted the commits in the meantime.

@mhalbritter mhalbritter modified the milestones: 3.2.0-M1, 3.2.x Jun 30, 2023
@bclozel
Copy link
Member

bclozel commented Jul 12, 2023

@davidmelia I'm not really sure I understand the use case here.

If you are trying to have additional metrics/traces tags contributed to the observation, you don't need to set things up before the observation starts. Writing your own ObservationConvention implementation is enough to augment the observations. You can lookup request headers in the observation context.

If your goal is to customize the context propagation, I would need more information.

Maybe you are trying to get ahead of the web observation to pre-populate the MDC with custom values? In this case, I believe you can achieve this by contributing a HttpHandlerDecoratorFactory bean that would do just that. Those are ordered as well and are all processed before the observation starts.

@wilkinsona wilkinsona added status: waiting-for-feedback We need additional information before we can continue and removed for: team-meeting An issue we'd like to discuss as a team to make progress labels Jul 12, 2023
@davidmelia
Copy link
Author

@bclozel I have tried HttpHandlerDecoratorFactory and it works great :-) . This actually removes my need to override this filter order.

@spring-projects-issues spring-projects-issues added status: feedback-provided Feedback has been provided and removed status: waiting-for-feedback We need additional information before we can continue labels Jul 14, 2023
@wilkinsona
Copy link
Member

wilkinsona commented Jul 17, 2023

That's excellent news. Thanks for letting us know, @davidmelia.

I'm going to close this one as we no longer have a need to make the filter order configurable, for either web stack.

@wilkinsona wilkinsona closed this as not planned Won't fix, can't repro, duplicate, stale Jul 17, 2023
@wilkinsona wilkinsona removed this from the 3.2.x milestone Jul 17, 2023
@wilkinsona wilkinsona added status: declined A suggestion or change that we don't feel we should currently apply and removed status: feedback-provided Feedback has been provided labels Jul 17, 2023
@edigu
Copy link

edigu commented Oct 26, 2023

@bclozel I have tried HttpHandlerDecoratorFactory and it works great :-) . This actually removes my need to override this filter order.

@davidmelia it would be great if you could share your solution involving HttpHandlerDecoratorFactory to shed some light for others.

I have a similar use case: extracting a special header value from the the request (using a very high order once per request filter - request has to be rejected in case of an invalid value) where the value is vital for the rest of the lifecycle. I have been setting the value in MDC there to make it available in logs, and also wanted to make it globally available for all metrics.

I read the thread and learned that very high precedence is intentional here.

what would you recommend for this use case? How HttpHandlerDecoratorFactory helped?

@bclozel
Copy link
Member

bclozel commented Oct 26, 2023

@edigu your use case might be a bit different actually.

  1. Is the application reactive or Servlet based?
  2. Which Spring Boot version are you using?
  3. If the request is rejected because of an invalid value, should it still record an observation (metrics/tracing) for this?

Maybe you can share those details in a StackOverflow question and point us to it? A closed issue isn't the right place to get help on a different matter.

@edigu
Copy link

edigu commented Oct 27, 2023

@bclozel

  1. It is Servlet based
  2. Using boot 3.1.x
  3. No I track invalid values in a different way. Observation is not a must.

Yes you are right, will try SO in worst case. Wanted to get feedback since the use case is quite similar.

@davidmelia
Copy link
Author

@edigu my use-case is to set up the log context which i used to do using web filters without realising I could simply use HttpHandlerDecoratorFactory. Below is an example. I can have multiple factories for the different log context concerns

public class MyHttpHandlerDecoratorFactory implements HttpHandlerDecoratorFactory {

  @Override
  public HttpHandler apply(HttpHandler httpHandler) {
    return (request, response) -> {
      log.debug("enter filter()");
      // extract stuff and decide how you want to add this to the log context
      log.debug("end filter()");
      return httpHandler.handle(mutatedRequest, response);
    };
  }

}

@AutoConfiguration
public class HttpHandlerDecorationAuthConfiguration {

  @Bean
  MyHttpHandlerDecoratorFactory myHttpHandlerDecoratorFactory() {
    MyHttpHandlerDecoratorFactory httpHandlerDecorator = new MyHttpHandlerDecoratorFactory();
    return httpHandlerDecorator;
  }
}

@krzyk
Copy link
Contributor

krzyk commented Oct 7, 2024

I have similar usecase (but for Servlet code, actually I don't get why the distinction, both cases might need the same) - described in https://stackoverflow.com/questions/79062775/adding-keyvalues-from-custom-filters-to-observationconvention

Basically I need a way to inject values into observation later in the lifecycle, after another filter has been executed (it injects values that I need to observer). I would also be able to get this data from Spring Security (which works in together with the custom filter), but again, I'm not sure how I could that. This is to allow a metric where I could see which kind of user groups (taken from either filter or spring security) use given URIs.

@bclozel
Copy link
Member

bclozel commented Oct 7, 2024

@krzyk I have asked clarifying questions on your SO post.

@edigu
Copy link

edigu commented Oct 19, 2024

I have similar usecase (but for Servlet code, actually I don't get why the distinction, both cases might need the same) - described in https://stackoverflow.com/questions/79062775/adding-keyvalues-from-custom-filters-to-observationconvention

Basically I need a way to inject values into observation later in the lifecycle, after another filter has been executed (it injects values that I need to observer). I would also be able to get this data from Spring Security (which works in together with the custom filter), but again, I'm not sure how I could that. This is to allow a metric where I could see which kind of user groups (taken from either filter or spring security) use given URIs.

@krzyk it seems the SO link is broken.

@bclozel
Copy link
Member

bclozel commented Oct 19, 2024

@edigu the question was deleted by its author so I guess @krzyk found a solution.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
status: declined A suggestion or change that we don't feel we should currently apply theme: observability Issues related to observability type: enhancement A general enhancement
Projects
None yet
Development

No branches or pull requests

8 participants