SpanCustomizingHandlerInterceptor.java
/*
* Copyright 2013-2021 the original author or authors.
*
* 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
*
* https://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 org.springframework.cloud.sleuth.instrument.web.mvc;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.cloud.sleuth.SpanCustomizer;
import org.springframework.cloud.sleuth.instrument.web.servlet.TracingFilter;
import org.springframework.lang.Nullable;
import org.springframework.web.servlet.HandlerInterceptor;
import org.springframework.web.servlet.ModelAndView;
/**
* Adds application-tier data to an existing http span via {@link HandlerParser}. This
* also sets the request property "http.route" so that it can be used in naming the http
* span.
*
* <p>
* Use this when you start traces at the servlet layer via {@link TracingFilter}.
*/
public final class SpanCustomizingHandlerInterceptor implements HandlerInterceptor {
/**
* Redefined from HandlerMapping.BEST_MATCHING_PATTERN_ATTRIBUTE added in Spring 3.
*/
static final String BEST_MATCHING_PATTERN_ATTRIBUTE = "org.springframework.web.servlet.HandlerMapping.bestMatchingPattern";
@Autowired(required = false)
HandlerParser handlerParser = new HandlerParser();
SpanCustomizingHandlerInterceptor() { // hide the ctor so we can change later if
// needed
}
/**
* Parses the request and sets the "http.route" attribute so that the
* {@link TracingFilter} can read it.
*/
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object o) {
Object span = request.getAttribute(SpanCustomizer.class.getName());
if (span instanceof SpanCustomizer) {
setHttpRouteAttribute(request);
handlerParser.preHandle(request, o, (SpanCustomizer) span);
}
return true;
}
@Override
public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler,
ModelAndView modelAndView) {
Object span = request.getAttribute(SpanCustomizer.class.getName());
if (span instanceof SpanCustomizer) {
handlerParser.postHandle(request, handler, modelAndView, (SpanCustomizer) span);
}
}
/** Sets the "error" attribute so that the {@link TracingFilter} can read it. */
@Override
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler,
Exception ex) {
Object span = request.getAttribute(SpanCustomizer.class.getName());
if (span instanceof SpanCustomizer) {
setErrorAttribute(request, ex);
}
}
/**
* Sets the "error" attribute if not already set, so that the {@link TracingFilter}
* can read it.
*/
static void setErrorAttribute(HttpServletRequest request, @Nullable Exception ex) {
if (ex != null && request.getAttribute("error") == null) {
request.setAttribute("error", ex);
}
}
/**
* Sets the "http.route" attribute from {@link #BEST_MATCHING_PATTERN_ATTRIBUTE} so
* that the {@link TracingFilter} can read it.
*/
static void setHttpRouteAttribute(HttpServletRequest request) {
Object httpRoute = request.getAttribute(BEST_MATCHING_PATTERN_ATTRIBUTE);
request.setAttribute("http.route", httpRoute != null ? httpRoute.toString() : "");
}
}