6
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

プライム・ブレインズAdvent Calendar 2024

Day 19

Spring Bootを使ったRESTAPIで、共通的な例外処理を実装する

Last updated at Posted at 2024-12-18

はじめに

ここのところ、新規で開発をするための準備として共通的に実装すべきことについて整理を続けています。
ここでは、例外処理について、改めて整理をします。

環境

Spring Boot
Lombok

ControllerAdviceと、ExceptionHandler

Spring Bootでは、ControllerAdviceアノテーションをクラスに付与し、そのメソッドにExceptionHanlderアノテーションを付与することで、特定の例外を捕捉してその場合の処理を記載する仕組みがあります。
ここで、Runtime例外のサブクラスを捕捉することにより、各レイヤーで発生する例外をすべて処理することができます。

今回はこれを、利用してやってみようと思います。
また、HandlerExceptionResolverインタフェースをインプリメントしたクラスでもエラーの捕捉は可能です。
これは、上記のExceptionHandlerで定義されていない例外を捕捉するのに利用するようです。

やってみる

例外クラス

まずは例外クラスを作ります。
ExceptionHandlerで捕捉するために、個別の例外を作ります。
プロジェクト共通的なものは基底クラスに押し込んで、用途別にクラスを作ります。

image.png

今回は適当な名前で作成してますが、用途別にわかりやすいネーミングにするのがよいでしょう。

HogeFugaException.java
import java.util.Arrays;
import java.util.List;

import lombok.Getter;

public abstract class HogeFugaException extends RuntimeException {

	@Getter
	private List<ErrorEntry> errors;

	public HogeFugaException(Throwable cause, List<ErrorEntry> errors) {
		super(cause);
		this.errors = errors;
	}

	public HogeFugaException(Throwable cause, ErrorEntry error) {
		super(cause);
		this.errors = Arrays.asList(error);
	}

	public HogeFugaException(List<ErrorEntry> errors) {
		this.errors = errors;
	}

	public HogeFugaException(ErrorEntry error) {
		this.errors = Arrays.asList(error);
	}

}

HogeException.java
import java.util.List;

public class HogeException extends HogeFugaException {

	public HogeException(Throwable cause, List<ErrorEntry> errors) {
		super(cause, errors);
	}

	public HogeException(Throwable cause, ErrorEntry error) {
		super(cause, error);
	}

	public HogeException(List<ErrorEntry> errors) {
		super(errors);
	}

	public HogeException(ErrorEntry error) {
		super(error);
	}
}
FugaException.java
import java.util.List;

public class FugaException extends HogeFugaException {

	public FugaException(Throwable cause, List<ErrorEntry> errors) {
		super(cause, errors);
	}

	public FugaException(Throwable cause, ErrorEntry error) {
		super(cause, error);
	}

	public FugaException(List<ErrorEntry> errors) {
		super(errors);
	}

	public FugaException(ErrorEntry error) {
		super(error);
	}
}

なお、エラーの情報は、エラーコード、エラーメッセージを持つ以下のクラスで管理します。

ErrorEntry.java
import lombok.Getter;

@Getter
public class ErrorEntry {

	private String id;
	private String message;

	private ErrorEntry() {
	}

	private ErrorEntry(String id, String message) {
		this.id = id;
		this.message = message;
	}

	public static ErrorEntry createErrorEntry(String id, String message) {
		return new ErrorEntry(id, message);
	}
}

レスポンスクラス

ここでは、RESTAPIのレスポンスクラスを定義します。
正常時のクラスと、エラー時のクラスを定義しておきます。

TestResponse.java
import lombok.Data;

@Data
public class TestResponse {
	long id;
	String name;
}
ErrorResponse.java
import java.util.ArrayList;
import java.util.List;

import lombok.Data;
import lombok.Getter;

@Getter
public class ErrorResponse {
	List<ErrorInfo> errors;

	@Data
	class ErrorInfo {
		String messageId;
		String message;
	}

	private ErrorResponse(HogeFugaException e) {
		List<ErrorInfo> errors = new ArrayList<>();

		for (ErrorEntry entry : e.getErrors()) {
			ErrorInfo info = new ErrorInfo();
			info.setMessageId(entry.getId());
			info.setMessage(entry.getMessage());

			errors.add(info);
		}
		this.errors = errors;
	}

	private ErrorResponse(String id, String message) {
		List<ErrorInfo> errors = new ArrayList<>();

		ErrorInfo info = new ErrorInfo();
		info.setMessageId(id);
		info.setMessage(message);

		errors.add(info);
		this.errors = errors;
	}

	public static ErrorResponse createErrorResponse(HogeFugaException e) {

		return new ErrorResponse(e);
	}

	public static ErrorResponse createErrorResponse(String id, String message) {

		return new ErrorResponse(id, message);
	}
}

コントローラークラス

次にコントローラークラスを定義します。
ここではあえて、例外が発生するような実装も作っています。

TestRestController.java
import java.util.ArrayList;
import java.util.List;

import com.example.demo.exception.ErrorEntry;
import com.example.demo.exception.FugaException;
import com.example.demo.exception.HogeException;
import com.example.demo.model.response.TestResponse;

@RestController
public class TestRestController {

	@GetMapping("/test")
	public TestResponse test() {
		TestResponse obj = new TestResponse();
		obj.setId(1);
		obj.setName("test");
		return obj;
	}

	@GetMapping("/test1")
	public TestResponse error() {

		throw new HogeException(ErrorEntry.createErrorEntry("1234", "エラーテスト"));
	}

	@GetMapping("/test2")
	public TestResponse errors() {

		List<ErrorEntry> errors = new ArrayList<>();
		errors.add(ErrorEntry.createErrorEntry("1111", "エラーテスト"));
		errors.add(ErrorEntry.createErrorEntry("2222", "エラーメッセージ"));

		throw new FugaException(errors);
	}

	@GetMapping("/test3")
	public TestResponse unkownError() {
		throw new IllegalArgumentException();
	}
}

エラー処理を行うクラス

GlobalExceptionHandler.java
import org.springframework.http.HttpStatus;
import org.springframework.web.bind.annotation.ControllerAdvice;
import org.springframework.web.bind.annotation.ExceptionHandler;
import org.springframework.web.bind.annotation.ResponseBody;
import org.springframework.web.bind.annotation.ResponseStatus;
import org.springframework.web.context.request.WebRequest;

@ControllerAdvice
public class GlobalExceptionHandler {

    // HogeExceptionが発生した場合は、このメソッドで捕捉
	@ExceptionHandler({ HogeException.class })
	@ResponseStatus(HttpStatus.BAD_REQUEST)
	@ResponseBody
	public ErrorResponse handleException(HogeException e, WebRequest req) {
		ErrorResponse response = ErrorResponse.createErrorResponse(e);
		return response;
	}

    // FugaExceptionが発生した場合は、このメソッドで捕捉
	@ExceptionHandler({ FugaException.class })
	@ResponseStatus(HttpStatus.INTERNAL_SERVER_ERROR)
	@ResponseBody
	public ErrorResponse handleException(FugaException e, WebRequest req) {
		ErrorResponse response = ErrorResponse.createErrorResponse(e);
		return response;
	}

    // HogeException、FugaException以外の例外が発生した場合は、このメソッドで捕捉
	@ExceptionHandler({ Exception.class })
	@ResponseStatus(HttpStatus.INTERNAL_SERVER_ERROR)
	@ResponseBody
	public ErrorResponse handleException(Exception e, WebRequest req) {
		ErrorResponse response = ErrorResponse.createErrorResponse("9999","想定外のエラー");
		return response;
	}
}

実行結果

無事、例外の捕捉ができました。

正常

image.png

HogeExceptionの場合

image.png

FugaExceptionの場合

image.png

それ以外の場合

image.png

おわりに

今回はControllerAdviceとExceptionHandlerで実装を行いましたが、ExceptionHandlerで、Exceptionクラスを捕捉する方法以外に、HandlerExceptionResolverインタフェースをインプリメントする方法もあるようです。

どちらがいいかはわかりませんが、ErrorとかThrowableとかを考えると、HandlerExceptionResolverインタフェースをインプリメントしたほうがいいかもしれません(機会があれば試してみます)

6
1
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
6
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?