1 General Description
In Web development with the separation of the front and back ends, it is necessary to unify the return result format of the back-end interface and handle global exceptions uniformly for the convenience of development. Common unified formats are as follows:
{" status ":" 100 ", "message" : "successful", "data" : "hello, world"}Copy the code
In order to avoid the encapsulation of returned results and handling of global exceptions by each microservice itself, the above functions can be extracted as public services, and other services can be directly used by public services.
Interception of all @RestControllerAdvice via @RestControllerAdvice and ResponseBodyAdvice interface to encapsulate return data when the interface call is normal. The @RestControllerAdvice annotation and the @ExceptionHandler annotation encapsulate the results of the specified exception type when an exception is called by the interface. This article explains step-by-step how to set up a public service and reference it in other services.
2 public service construction
2.1 The service is self-built and self-used
Resources: How does SpringBoot unify the backend return format? This is how old birds play!
2.1.1 Define the return result format
@Data public class ResultData<T> { private int status; private String message; private T data; private long timestamp ; public ResultData (){ this.timestamp = System.currentTimeMillis(); } public static <T> ResultData<T> success(T data) { ResultData<T> resultData = new ResultData<>(); resultData.setStatus(ReturnCode.RC100.getCode()); resultData.setMessage(ReturnCode.RC100.getMessage()); resultData.setData(data); return resultData; } public static <T> ResultData<T> fail(int code, String message) { ResultData<T> resultData = new ResultData<>(); resultData.setStatus(code); resultData.setMessage(message); return resultData; }}Copy the code
/* Define status code */ public enum ReturnCode {/** Succeeded **/ RC100(100," succeeded "), /** failed **/ RC999(999," Failed "), /** service traffic limiting **/ RC200(200," Service traffic limiting is enabled, please try again later!" ), /** service degraded **/ RC201(201," Service degraded protection enabled, please try again later!" ), /** Hotspot parameter limit **/ RC202(202," Hotspot parameter limit, please try again later!" ), /** system rule does not meet **/ RC203(203," system rule does not meet requirements, please try again later! ), /** Authorization rule failed **/ RC204(204," Authorization rule failed, please try again later!" ), /**access_denied**/ RC403(403," no access permission, please contact administrator to grant access permission "), /**access_denied**/ RC401(401," anonymous access error "), RC500(500," system exception, Please try again later "), INVALID_TOKEN(2001," Access token invalid "), ACCESS_DENIED(2003," No permission to access this resource "), CLIENT_AUTHENTICATION_FAILED(1001," client authentication failed "), USERNAME_OR_PASSWORD_ERROR(1002," user name or password error "), UNSUPPORTED_GRANT_TYPE(1003, "unsupported authentication mode "); /** Private final int code; /** Custom description **/ private final String message; ReturnCode(int code, String message){ this.code = code; this.message = message; } public int getCode() { return code; } public String getMessage() { return message; }}Copy the code
2.1.2 Returned Result of Abnormal Encapsulation Interface
@slf4j @RestControllerAdvice Public class RestExceptionHandler {/** * Default global exception handling. */ @ExceptionHandler(Exception.class) @ResponseStatus(HttpStatus.INTERNAL_SERVER_ERROR) public ResultData<String> Exception (exception e) {log.error(" ex={}", LLDB message (), e); return ResultData.fail(ReturnCode.RC500.getCode(),e.getMessage()); }}Copy the code
2.1.3 The Result is Returned if the Encapsulation interface is normal
Implemented with the @RestControllerAdvice annotation and ResponseBodyAdvice interface.
@RestControllerAdvice is an enhancement of the @RestController annotation that implements three functions: global exception handling, global data binding, and global data preprocessing.
ResponseBodyAdvice intercepts the return value of the Controller method and processes the return value/response body.
@restControllerAdvice // Intercept the return value of the Controller method, process the return value/response body uniformly, Public class RestResponseAdvice implements ResponseBodyAdvice<Object> {@autoWired private ObjectMapper objectMapper; */ @override public Boolean supports(MethodParameter, Class<? extends HttpMessageConverter<? >> aClass) { return true; */ @sneakythrows @Override public Object beforeBodyWrite(Object O, MethodParameter MethodParameter, MediaType mediaType, Class<? extends HttpMessageConverter<? >> aClass, ServerHttpRequest serverHttpRequest, ServerHttpResponse serverHttpResponse) { if(o instanceof String){ return objectMapper.writeValueAsString(ResultData.success(o)); } if (o instanceof ResultData) {// If (o instanceof ResultData) {return o; } return ResultData.success(o); }}Copy the code
2.1.4 Verifying in the current service
According to the processing in section 2.1.1 to 2.1.3, the returned data format of unified interfaces and encapsulation of global exceptions have been completed. Two interfaces are directly created in the current service for verification.
@RestController
@RequestMapping("/api/demo")
public class RestDemo {
@GetMapping("/hello")
public String Hello() {
return "hello, world!";
}
@GetMapping("/wrong")
public int Wrong() {
return 3/0;
}
}
Copy the code
Project Structure:
The result of a normal interface call:
Result of interface call exception:
2.2 Installation of public services
References:
Maven build life cycle
How can springBoot projects be packaged for reference to other projects
In order to reference the public services built in Section 2.1 as Maven dependencies in other services, the public services need to be installed in a local Maven repository or deployed to a remote repository. This is demonstrated by installing to a local repository.
Spring Boot is a jar package that can be run in the fat JAR format (the source code of the generated JAR is in the boot-INF directory). It is not a traditional JAR package. To do this, modify the pom.xml configuration file for the public service as follows:
<build> <plugins> <! -- <plugin> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-maven-plugin</artifactId> <configuration> <excludes> <exclude> <groupId>org.projectlombok</groupId> <artifactId>lombok</artifactId> </exclude> </excludes> </configuration> </plugin>--> <plugin> <groupId>org.apache.maven.plugins</groupId> < artifactId > maven -- the compiler plugin < / artifactId > < configuration > < source > 1.8 < / source > < target > 1.8 < / target > </configuration> </plugin> </plugins> </build>Copy the code
Run the maven project installation command MVN clean install -dmaven.test. skip=true in the Terminal of IDEA.
3 Public service reference
Set up another SpringBoot Web business service that references the common service set up in Section 2 for the purpose of unifying the return result format of the interface.
3.1 Introduction of public service dependencies
The pom.xml part of the public service is configured as follows:
< the groupId > com. Example < / groupId > < artifactId > common - advice < / artifactId > < version > 0.0.1 - the SNAPSHOT < / version > <name>common-advice</name> <description>Public service that wraps interface results</description> <properties> < Java version > 1.8 < / Java version > < / properties >Copy the code
Add a dependency to the business service’s POM.xml:
< the dependency > < groupId > com. Example < / groupId > < artifactId > common - advice < / artifactId > < version > 0.0.1 - the SNAPSHOT < / version > </dependency>Copy the code
3.2 Component Scanning Path Add a public service
Add the package path of the public service to the ComponentScan path of the service (add the @componentscan configuration in the service entry). Otherwise, the configuration class modified by the @restcontrolleradvice annotation cannot take effect.
@ControllerAdvic encapsulates the @ControllerAdvice annotation. @ControllerAdvice encapsulates the @Component annotation. Components marked with the @RestControllerAdvic annotation can be loaded using the @ComponentScan configuration.
@ComponentScan({"com.example.demorest", "com.example.common.advice"})
@SpringBootApplication
public class DemorestApplication {
public static void main(String[] args) {
SpringApplication.run(DemorestApplication.class, args);
}
}
Copy the code
3.3 Writing interface authentication
@RestController
@RequestMapping("/api/original")
public class RestOriginal {
@GetMapping("/success")
public String GetSuccess() {
return "hello, don!";
}
@GetMapping("/error")
public int GetError() {
return 6/0;
}
}
Copy the code
Project Structure:
Result of normal interface access:
Results of abnormal interface access:
4 Supplementary Notes
The ResponseBodyAdvice interface and ExceptionHandler annotation are both applied after the controller interface method body has been executed.