Spring BootがHTTPリクエストインターセプターを追加する


107

Spring BootアプリケーションにHttpRequestインターセプターを追加する正しい方法は何ですか?私がやりたいことは、すべてのhttp要求の要求と応答をログに記録することです。

Spring bootのドキュメントには、このトピックはまったく含まれていません。(http://docs.spring.io/spring-boot/docs/current/reference/htmlsingle/

以前のバージョンのSpringで同じことを行う方法に関するWebサンプルをいくつか見つけましたが、それらはapplicationcontext.xmlで動作します。助けてください。


こんにちは@ riship89 ...私はHandlerInterceptor正常に実装しました。正常に動作しています。問題は、internal HandlerInterceptorで処理される前に例外をスローするものだけcustom HandlerInterceptorです。afterCompletion()エラーがHandlerInterceptorの内部実装によってスローされた後、オーバーライドされるメソッドが呼び出されます。これに対する解決策はありますか?
Chetan Oswal

回答:


155

Spring Bootを使用しているので、可能な場合はSpringの自動構成に依存することをお勧めします。インターセプターなどのカスタム構成を追加するには、の構成またはBeanを指定するだけです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() {
        ...
    });
  }
}

注: mvcのSpring Bootsの自動構成を保持する場合は、@ EnableWebMvcで注釈を付けないでください


いいね!いいね!registry.addInterceptor(...)の内部にあるものの例はありますか?「...」のサンプルを知りたいだけです
riship89

6
yourInjectedInceptorで@Componentアノテーションを使用
Paolo Biavati

1
@ riship89これを例にチェックアウトしてください:mkyong.com/spring-mvc/spring-mvc-handler-interceptors-example
Vlad Manuel

4
エラーが発生しますThe type WebMvcConfigurerAdapter is deprecated。私はSpring Web MVC 5.0.6を使用しています
John Henckel

7
Spring 5では、WebMvcConfigurerAdapterを拡張する代わりに、WebMvcConfigurerを実装するだけです。Java 8インターフェースではデフォルトの実装が許可されているため、アダプターを使用する必要はありません(そのため、アダプターは非推奨になっています)。
edgraaff 2018年

88

WebMvcConfigurerAdapterそのことから春5で廃止される予定のJavadoc

@ 5.0で非推奨{@link WebMvcConfigurer}にはデフォルトのメソッドがあり(Java 8ベースラインによって可能になりました)、このアダプターを必要とせずに直接実装できます。

上で述べたように、あなたがすべきことはメソッドの実装WebMvcConfigurerとオーバーライドaddInterceptorsです。

@Configuration
public class WebMvcConfig implements WebMvcConfigurer {

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

10
実装が欠落しているため、回答は不完全ですMyCustomInterceptor
peterchaula

33

インターセプターをSpring Bootアプリケーションに追加するには、次の操作を行います

  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. それでおしまい。これで、すべてのリクエストがMyCustomInterceptorのpreHandle()メソッドで定義されたロジックを通過します。


1
この方法に従って、いくつかの一般的な検証を行うために、アプリケーションに送信されるサインアップ要求をインターセプトしました。しかし、問題は私がgetReader() has already been called for this requestエラーを受け取ることです。実際のリクエストのコピーを使用せずにこれを回避する簡単な方法はありますか?
vigamage 2017

プリハンドラーが呼び出されると、リクエスト本文は使用できませんが、パラメーターのみが使用できます。リクエスト本文で検証を行うには、アスペクトJを使用して作成することをおAdvice
Hima

12

これに対するすべての応答は、WebMvcInterface(@sebdooeですでに指摘されている)ではなく、現在は非推奨となっている抽象WebMvcConfigurerアダプターを使用しているため、Interceptorを使用した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)インターフェース
Tom

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");
    }
}

4
コンパイルの問題なしに、インターフェイスの1つのメソッドのみをオーバーライドするにはどうすればよいですか?
xetra11 2017年

1
@ xetra11また、この場合に使用されない他のすべてのメソッドではなく、1つのメソッドのみを実装できるかどうかを確認しようとしています。出来ますか?あなたはそれを理解しましたか?
user09

3
@arjunその他はJava 8のおかげでデフォルトのメソッドとして実装されています。この推論は、都合の良いことに、非推奨のクラスで文書化されています。
ボブは

7

また、URLルートに注釈を付けるのと同じように、インターセプターを適用するSpring Bootコントローラーに直接注釈を付けることができるオープンソースのSpringSandwichライブラリの使用を検討することもできます。

そうすれば、誤字がちな文字列が浮かぶことはありません。SpringSandwichのメソッドとクラスのアノテーションは、簡単にリファクタリングを乗り越えて、どこに何が適用されているかを明確にします。(開示:私は著者です)。

http://springsandwich.com/


いいね!CIで構築されているプロジェクトやHeroku経由でデプロイされているプロジェクトで使いやすくするために、Maven CentralでSpringSandwichを使用できるように要求する問題を作成しました。
Brendon Dugan 2017

すごい。Maven Central Repositoryで利用できますか?再-ブログあなたのgitリポジトリとリファレンスのリファレンスを使って私のウェブサイトでspringsandwich.comを再度追加
Arpan Das

1
SpringSandwichは現在Maven Centralにあります
Magnus

1

以下は、各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===========================================");
}}

以下は、残りのテンプレートBeanです。

@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;
}
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.