Spring Boot с добавлением перехватчиков HTTP-запросов

108

Как правильно добавить перехватчики HttpRequest в приложение весенней загрузки? Я хочу регистрировать запросы и ответы на каждый HTTP-запрос.

Документация по загрузке Spring вообще не затрагивает эту тему. ( http://docs.spring.io/spring-boot/docs/current/reference/htmlsingle/ )

Я нашел несколько веб-примеров того, как сделать то же самое со старыми версиями spring, но они работают с applicationcontext.xml. Пожалуйста помоги.

ришип89
источник
Привет @ riship89 ... Я HandlerInterceptorуспешно реализовал . Работает нормально. Проблема только в том, что некоторые internal HandlerInterceptorгенерируют исключение до того, как оно будет обработано custom HandlerInterceptor. afterCompletion()Метод , который переопределяется вызывается после того , как ошибка возникает при внутренней реализации HandlerInterceptor. У вас есть решение для этого?
Четан Освал

Ответы:

157

Поскольку вы используете Spring Boot, я предполагаю, что вы предпочтете по возможности полагаться на автоконфигурацию Spring. Чтобы добавить дополнительную настраиваемую конфигурацию, такую ​​как ваши перехватчики, просто укажите конфигурацию или компонент WebMvcConfigurerAdapter.

Вот пример класса конфигурации:

@Configuration
public class WebMvcConfig extends WebMvcConfigurerAdapter {

  @Autowired 
  HandlerInterceptor yourInjectedInterceptor;

  @Override
  public void addInterceptors(InterceptorRegistry registry) {
    registry.addInterceptor(...)
    ...
    registry.addInterceptor(getYourInterceptor()); 
    registry.addInterceptor(yourInjectedInterceptor);
    // next two should be avoid -- tightly coupled and not very testable
    registry.addInterceptor(new YourInterceptor());
    registry.addInterceptor(new HandlerInterceptor() {
        ...
    });
  }
}

ПРИМЕЧАНИЕ: не аннотируйте это с помощью @EnableWebMvc, если вы хотите сохранить автоконфигурацию Spring Boots для mvc .

Икумен
источник
Ницца! Выглядит хорошо! Любой пример того, что находится внутри registry.addInterceptor (...)? Просто любопытно узнать образец «...»
riship89
6
используйте аннотацию @Component в yourInjectedInceptor
Паоло
1
@ riship89 Взгляните на это для примера: mkyong.com/spring-mvc/spring-mvc-handler-interceptors-example
Влад Мануэль Муредан
4
Я получаю ошибку The type WebMvcConfigurerAdapter is deprecated. Я использую Spring Web MVC 5.0.6
Джон Хенкель
7
В Spring 5 просто реализуйте WebMvcConfigurer вместо расширения WebMvcConfigurerAdapter. Поскольку интерфейсы Java 8 допускают реализации по умолчанию, больше не требуется использовать адаптер (вот почему он устарел).
edgraaff
89

WebMvcConfigurerAdapterбудет устаревшим в Spring 5. Из его Javadoc :

@deprecated начиная с 5.0 {@link WebMvcConfigurer} имеет методы по умолчанию (что стало возможным благодаря базовой версии Java 8) и может быть реализовано напрямую без необходимости в этом адаптере.

Как указано выше, вам следует реализовать WebMvcConfigurerи переопределить addInterceptorsметод.

@Configuration
public class WebMvcConfig implements WebMvcConfigurer {

    @Override
    public void addInterceptors(InterceptorRegistry registry) {
        registry.addInterceptor(new MyCustomInterceptor());
    }
}
sedooe
источник
10
Ваш ответ неполный, потому что в нем отсутствует реализацияMyCustomInterceptor
peterchaula
33

Чтобы добавить перехватчик в приложение весенней загрузки, выполните следующие действия.

  1. Создать класс перехватчика

    public class MyCustomInterceptor implements HandlerInterceptor{
    
        //unimplemented methods comes here. Define the following method so that it     
        //will handle the request before it is passed to the controller.
    
        @Override
        public boolean preHandle(HttpServletRequest request,HttpServletResponse  response){
        //your custom logic here.
            return true;
        }
    }
  2. Определите класс конфигурации

    @Configuration
    public class MyConfig extends WebMvcConfigurerAdapter{
        @Override
        public void addInterceptors(InterceptorRegistry registry){
            registry.addInterceptor(new MyCustomInterceptor()).addPathPatterns("/**");
        }
    }
  3. Это оно. Теперь все ваши запросы будут проходить через логику, определенную в методе preHandle () MyCustomInterceptor.

сунита
источник
1
Я использовал этот способ для перехвата запросов на регистрацию, поступающих в мое приложение, для выполнения некоторых общих проверок. Но проблема в том, что я получаю getReader() has already been called for this requestошибку. Есть ли более простой способ справиться с этим без использования копии фактического запроса?
vigamage
Когда вызывается предварительный обработчик, тело запроса недоступно, а доступны только параметры. Для проверки тела запроса предпочтительным способом будет использование Aspect J и созданиеAdvice
Hima
12

Поскольку во всех ответах на это используется давно устаревший абстрактный адаптер WebMvcConfigurer вместо WebMvcInterface (как уже отмечалось @sebdooe), вот минимальный рабочий пример для приложения SpringBoot (2.1.4) с перехватчиком:

Minimal.java:

@SpringBootApplication
public class Minimal
{
    public static void main(String[] args)
    {
        SpringApplication.run(Minimal.class, args);
    }
}

MinimalController.java:

@RestController
@RequestMapping("/")
public class Controller
{
    @GetMapping("/")
    @ResponseBody
    public ResponseEntity<String> getMinimal()
    {
        System.out.println("MINIMAL: GETMINIMAL()");

        return new ResponseEntity<String>("returnstring", HttpStatus.OK);
    }
}

Config.java:

@Configuration
public class Config implements WebMvcConfigurer
{
    //@Autowired
    //MinimalInterceptor minimalInterceptor;

    @Override
    public void addInterceptors(InterceptorRegistry registry)
    {
        registry.addInterceptor(new MinimalInterceptor());
    }
}

MinimalInterceptor.java:

public class MinimalInterceptor extends HandlerInterceptorAdapter
{
    @Override
    public boolean preHandle(HttpServletRequest requestServlet, HttpServletResponse responseServlet, Object handler) throws Exception
    {
        System.out.println("MINIMAL: INTERCEPTOR PREHANDLE CALLED");

        return true;
    }

    @Override
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception
    {
        System.out.println("MINIMAL: INTERCEPTOR POSTHANDLE CALLED");
    }

    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception exception) throws Exception
    {
        System.out.println("MINIMAL: INTERCEPTOR AFTERCOMPLETION CALLED");
    }
}

работает как рекламируется

Результат будет примерно таким:

> Task :Minimal.main()

  .   ____          _            __ _ _
 /\\ / ___'_ __ _ _(_)_ __  __ _ \ \ \ \
( ( )\___ | '_ | '_| | '_ \/ _` | \ \ \ \
 \\/  ___)| |_)| | | | | || (_| |  ) ) ) )
  '  |____| .__|_| |_|_| |_\__, | / / / /
 =========|_|==============|___/=/_/_/_/
 :: Spring Boot ::        (v2.1.4.RELEASE)

2019-04-29 11:53:47.560  INFO 4593 --- [           main] io.minimal.Minimal                       : Starting Minimal on y with PID 4593 (/x/y/z/spring-minimal/build/classes/java/main started by x in /x/y/z/spring-minimal)
2019-04-29 11:53:47.563  INFO 4593 --- [           main] io.minimal.Minimal                       : No active profile set, falling back to default profiles: default
2019-04-29 11:53:48.745  INFO 4593 --- [           main] o.s.b.w.embedded.tomcat.TomcatWebServer  : Tomcat initialized with port(s): 8080 (http)
2019-04-29 11:53:48.780  INFO 4593 --- [           main] o.apache.catalina.core.StandardService   : Starting service [Tomcat]
2019-04-29 11:53:48.781  INFO 4593 --- [           main] org.apache.catalina.core.StandardEngine  : Starting Servlet engine: [Apache Tomcat/9.0.17]
2019-04-29 11:53:48.892  INFO 4593 --- [           main] o.a.c.c.C.[Tomcat].[localhost].[/]       : Initializing Spring embedded WebApplicationContext
2019-04-29 11:53:48.893  INFO 4593 --- [           main] o.s.web.context.ContextLoader            : Root WebApplicationContext: initialization completed in 1269 ms
2019-04-29 11:53:49.130  INFO 4593 --- [           main] o.s.s.concurrent.ThreadPoolTaskExecutor  : Initializing ExecutorService 'applicationTaskExecutor'
2019-04-29 11:53:49.375  INFO 4593 --- [           main] o.s.b.w.embedded.tomcat.TomcatWebServer  : Tomcat started on port(s): 8080 (http) with context path ''
2019-04-29 11:53:49.380  INFO 4593 --- [           main] io.minimal.Minimal                       : Started Minimal in 2.525 seconds (JVM running for 2.9)
2019-04-29 11:54:01.267  INFO 4593 --- [nio-8080-exec-1] o.a.c.c.C.[Tomcat].[localhost].[/]       : Initializing Spring DispatcherServlet 'dispatcherServlet'
2019-04-29 11:54:01.267  INFO 4593 --- [nio-8080-exec-1] o.s.web.servlet.DispatcherServlet        : Initializing Servlet 'dispatcherServlet'
2019-04-29 11:54:01.286  INFO 4593 --- [nio-8080-exec-1] o.s.web.servlet.DispatcherServlet        : Completed initialization in 19 ms
MINIMAL: INTERCEPTOR PREHANDLE CALLED
MINIMAL: GETMINIMAL()
MINIMAL: INTERCEPTOR POSTHANDLE CALLED
MINIMAL: INTERCEPTOR AFTERCOMPLETION CALLED
Ксенонит
источник
Но для этого вам потребуется реализовать все методы из WebMvcConfigurer, верно?
Стивен
Нет, это интерфейс (Java 8) с пустыми реализациями по умолчанию
Том
9

У меня была такая же проблема, что WebMvcConfigurerAdapter устарел. Когда я искал примеры, я почти не нашел реализованного кода. Вот фрагмент рабочего кода.

создать класс, расширяющий HandlerInterceptorAdapter

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.stereotype.Component;
import org.springframework.web.servlet.handler.HandlerInterceptorAdapter;

import me.rajnarayanan.datatest.DataTestApplication;
@Component
public class EmployeeInterceptor extends HandlerInterceptorAdapter {
    private static final Logger logger = LoggerFactory.getLogger(DataTestApplication.class);
    @Override
    public boolean preHandle(HttpServletRequest request, 
            HttpServletResponse response, Object handler) throws Exception {

            String x = request.getMethod();
            logger.info(x + "intercepted");
        return true;
    }

}

затем реализовать интерфейс WebMvcConfigurer

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Configuration;
import org.springframework.web.servlet.config.annotation.InterceptorRegistry;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;

import me.rajnarayanan.datatest.interceptor.EmployeeInterceptor;
@Configuration
public class WebMvcConfig implements WebMvcConfigurer {
    @Autowired
    EmployeeInterceptor employeeInterceptor ;

    @Override
    public void addInterceptors(InterceptorRegistry registry){
        registry.addInterceptor(employeeInterceptor).addPathPatterns("/employee");
    }
}
user2532195
источник
4
как можно переопределить только один метод в интерфейсе без проблем с компиляцией?
xetra11
1
@ xetra11 Я также пытаюсь понять, можем ли мы реализовать только один метод вместо всех других методов, которые не используются в этом случае. Является ли это возможным? Вы поняли это?
user09
3
@arjun Остальные реализованы как методы по умолчанию благодаря Java 8. Это рассуждение удобно документировать в устаревшем классе.
Боб
8

Вы также можете рассмотреть возможность использования библиотеки SpringSandwich с открытым исходным кодом, которая позволяет вам напрямую аннотировать в контроллерах Spring Boot, какие перехватчики применять, почти так же, как вы аннотируете свои маршруты URL-адресов.

Таким образом, не будет плавающих строк, подверженных опечаткам - методы и аннотации классов SpringSandwich легко выдерживают рефакторинг и дают понять, что и где применяется. (Раскрытие: я автор).

http://springsandwich.com/

Магнус
источник
Выглядит потрясающе! Я создал проблему с просьбой о том, чтобы SpringSandwich был доступен в maven central, чтобы упростить использование для проектов, которые создаются под CI или которые развертываются через Heroku.
Брендон Дуган,
Отлично. Доступен ли он в центральном репозитории maven? Повторное ведение блога springsandwich.com на моем веб-сайте со ссылкой на ваш репозиторий git и ссылкой
Арпан Дас,
1
SpringSandwich теперь в Maven Central
Магнус
0

Ниже представлена ​​реализация, которую я использую для перехвата каждого HTTP-запроса до его отправки и ответа, который возвращается. Благодаря этой реализации у меня также есть единственная точка, где я могу передать любое значение заголовка с запросом.

public class HttpInterceptor implements ClientHttpRequestInterceptor {
private Logger logger = LoggerFactory.getLogger(this.getClass());
@Override
public ClientHttpResponse intercept(
        HttpRequest request, byte[] body,
        ClientHttpRequestExecution execution
) throws IOException {
    HttpHeaders headers = request.getHeaders();
    headers.add("Accept", MediaType.APPLICATION_JSON_UTF8_VALUE);
    headers.add("Content-Type", MediaType.APPLICATION_JSON_VALUE);
    traceRequest(request, body);
    ClientHttpResponse response = execution.execute(request, body);
    traceResponse(response);
    return response;
}

private void traceRequest(HttpRequest request, byte[] body) throws IOException {
    logger.info("===========================Request begin======================================");
    logger.info("URI         : {}", request.getURI());
    logger.info("Method      : {}", request.getMethod());
    logger.info("Headers     : {}", request.getHeaders() );
    logger.info("Request body: {}", new String(body, StandardCharsets.UTF_8));
    logger.info("==========================Request end=========================================");
}

private void traceResponse(ClientHttpResponse response) throws IOException {
    logger.info("============================Response begin====================================");
    logger.info("Status code  : {}", response.getStatusCode());
    logger.info("Status text  : {}", response.getStatusText());
    logger.info("Headers      : {}", response.getHeaders());
    logger.info("=======================Response end===========================================");
}}

Ниже представлен компонент шаблона Rest Template

@Bean
public RestTemplate restTemplate(HttpClient httpClient)
{
    HttpComponentsClientHttpRequestFactory requestFactory =
            new HttpComponentsClientHttpRequestFactory();
    requestFactory.setHttpClient(httpClient);
    RestTemplate restTemplate=  new RestTemplate(requestFactory);
    List<ClientHttpRequestInterceptor> interceptors = restTemplate.getInterceptors();
    if (CollectionUtils.isEmpty(interceptors))
    {
        interceptors = new ArrayList<>();
    }
    interceptors.add(new HttpInterceptor());
    restTemplate.setInterceptors(interceptors);

    return restTemplate;
}
Сильвестр
источник
1
Хорошо, но на самом деле вы здесь сделали перехватчик ДЛЯ RestTemplate (т.е. когда ВЫ выполняете HTTP-вызовы) ... а не перехватчик для контроллеров Spring REST (т.е. когда HTTP-вызовы выполняются против вашего приложения Spring / REST-webservices).
maxxyme