Browse Source

Merge branch '6.1.x'

pull/32992/head
Brian Clozel 2 years ago
parent
commit
24997b3356
  1. 50
      spring-web/src/main/java/org/springframework/web/filter/ServerHttpObservationFilter.java
  2. 17
      spring-web/src/test/java/org/springframework/web/filter/ServerHttpObservationFilterTests.java

50
spring-web/src/main/java/org/springframework/web/filter/ServerHttpObservationFilter.java

@ -21,9 +21,12 @@ import java.util.Optional;
import io.micrometer.observation.Observation; import io.micrometer.observation.Observation;
import io.micrometer.observation.ObservationRegistry; import io.micrometer.observation.ObservationRegistry;
import jakarta.servlet.AsyncEvent;
import jakarta.servlet.AsyncListener;
import jakarta.servlet.FilterChain; import jakarta.servlet.FilterChain;
import jakarta.servlet.RequestDispatcher; import jakarta.servlet.RequestDispatcher;
import jakarta.servlet.ServletException; import jakarta.servlet.ServletException;
import jakarta.servlet.ServletRequest;
import jakarta.servlet.http.HttpServletRequest; import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse; import jakarta.servlet.http.HttpServletResponse;
@ -94,11 +97,6 @@ public class ServerHttpObservationFilter extends OncePerRequestFilter {
return Optional.ofNullable((ServerRequestObservationContext) request.getAttribute(CURRENT_OBSERVATION_CONTEXT_ATTRIBUTE)); return Optional.ofNullable((ServerRequestObservationContext) request.getAttribute(CURRENT_OBSERVATION_CONTEXT_ATTRIBUTE));
} }
@Override
protected boolean shouldNotFilterAsyncDispatch() {
return false;
}
@Override @Override
@SuppressWarnings("try") @SuppressWarnings("try")
protected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain filterChain) protected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain filterChain)
@ -115,8 +113,12 @@ public class ServerHttpObservationFilter extends OncePerRequestFilter {
throw ex; throw ex;
} }
finally { finally {
// Only stop Observation if async processing is done or has never been started. // If async is started, register a listener for completion notification.
if (!request.isAsyncStarted()) { if (request.isAsyncStarted()) {
request.getAsyncContext().addListener(new ObservationAsyncListener(observation));
}
// Stop Observation right now if async processing has not been started.
else {
Throwable error = fetchException(request); Throwable error = fetchException(request);
if (error != null) { if (error != null) {
observation.error(error); observation.error(error);
@ -152,13 +154,43 @@ public class ServerHttpObservationFilter extends OncePerRequestFilter {
} }
@Nullable @Nullable
private Throwable unwrapServletException(Throwable ex) { static Throwable unwrapServletException(Throwable ex) {
return (ex instanceof ServletException) ? ex.getCause() : ex; return (ex instanceof ServletException) ? ex.getCause() : ex;
} }
@Nullable @Nullable
private Throwable fetchException(HttpServletRequest request) { static Throwable fetchException(ServletRequest request) {
return (Throwable) request.getAttribute(RequestDispatcher.ERROR_EXCEPTION); return (Throwable) request.getAttribute(RequestDispatcher.ERROR_EXCEPTION);
} }
private static class ObservationAsyncListener implements AsyncListener {
private final Observation currentObservation;
public ObservationAsyncListener(Observation currentObservation) {
this.currentObservation = currentObservation;
}
@Override
public void onStartAsync(AsyncEvent event) {
}
@Override
public void onTimeout(AsyncEvent event) {
this.currentObservation.stop();
}
@Override
public void onComplete(AsyncEvent event) {
this.currentObservation.stop();
}
@Override
public void onError(AsyncEvent event) {
this.currentObservation.error(unwrapServletException(event.getThrowable()));
this.currentObservation.stop();
}
}
} }

17
spring-web/src/test/java/org/springframework/web/filter/ServerHttpObservationFilterTests.java

@ -54,6 +54,11 @@ class ServerHttpObservationFilterTests {
private ServerHttpObservationFilter filter = new ServerHttpObservationFilter(this.observationRegistry); private ServerHttpObservationFilter filter = new ServerHttpObservationFilter(this.observationRegistry);
@Test
void filterShouldNotProcessAsyncDispatch() {
assertThat(this.filter.shouldNotFilterAsyncDispatch()).isTrue();
}
@Test @Test
void filterShouldFillObservationContext() throws Exception { void filterShouldFillObservationContext() throws Exception {
this.filter.doFilter(this.request, this.response, this.mockFilterChain); this.filter.doFilter(this.request, this.response, this.mockFilterChain);
@ -64,7 +69,7 @@ class ServerHttpObservationFilterTests {
assertThat(context.getCarrier()).isEqualTo(this.request); assertThat(context.getCarrier()).isEqualTo(this.request);
assertThat(context.getResponse()).isEqualTo(this.response); assertThat(context.getResponse()).isEqualTo(this.response);
assertThat(context.getPathPattern()).isNull(); assertThat(context.getPathPattern()).isNull();
assertThatHttpObservation().hasLowCardinalityKeyValue("outcome", "SUCCESS"); assertThatHttpObservation().hasLowCardinalityKeyValue("outcome", "SUCCESS").hasBeenStopped();
} }
@Test @Test
@ -121,6 +126,16 @@ class ServerHttpObservationFilterTests {
assertThat(this.response.getHeader("X-Trace-Id")).isEqualTo("badc0ff33"); assertThat(this.response.getHeader("X-Trace-Id")).isEqualTo("badc0ff33");
} }
@Test
void shouldCloseObservationAfterAsyncCompletion() throws Exception {
this.request.setAsyncSupported(true);
this.request.startAsync();
this.filter.doFilter(this.request, this.response, this.mockFilterChain);
this.request.getAsyncContext().complete();
assertThatHttpObservation().hasLowCardinalityKeyValue("outcome", "SUCCESS").hasBeenStopped();
}
private TestObservationRegistryAssert.TestObservationRegistryAssertReturningObservationContextAssert assertThatHttpObservation() { private TestObservationRegistryAssert.TestObservationRegistryAssertReturningObservationContextAssert assertThatHttpObservation() {
return TestObservationRegistryAssert.assertThat(this.observationRegistry) return TestObservationRegistryAssert.assertThat(this.observationRegistry)
.hasObservationWithNameEqualTo("http.server.requests").that(); .hasObservationWithNameEqualTo("http.server.requests").that();

Loading…
Cancel
Save