三種Sentinel自定義異常,你用過(guò)幾種?
Spring Cloud Alibaba Sentinel 是目前主流并開(kāi)源的流量控制和系統(tǒng)保護(hù)組件,它提供了強(qiáng)大的限流、熔斷、熱點(diǎn)限流、授權(quán)限流和系統(tǒng)保護(hù)及監(jiān)控等功能。使用它可以輕松的保護(hù)我們微服務(wù),在高并發(fā)環(huán)境下的正常運(yùn)行。
那么,當(dāng)程序觸發(fā)了限流和熔斷規(guī)則時(shí),如何自定義返回的異常信息呢?這是我們接下來(lái)要解決的問(wèn)題。
一、概述
Spring Cloud Alibaba Sentinel 有以下 3 種自定義異常的實(shí)現(xiàn)方式:
- 自定義局部異常
- 自定義(Sentinel)全局異常
- 自定義系統(tǒng)異常
以上這 3 種實(shí)現(xiàn)方式,都可以重新定義 Sentinel 的異常返回信息,它們的具體實(shí)現(xiàn)如下。
二、自定義局部異常
自定義局部異常是在使用 @SentinelResource 注解時(shí),直接定義的 blockHandler 異常方法,如下代碼所示:
@SentinelResource(value = "/user/getuser",
blockHandler = "myBlockHandler")
@RequestMapping("getuser")
public String getUser(Integer uid) {
return "User:" + uid;
}
/**
* 定義限流/熔斷等異常
*/
public String myBlockHandler(Integer uid, BlockException e) {
String msg = "未知異常";
if (e instanceof FlowException) {
msg = "請(qǐng)求被限流了";
} else if (e instanceof ParamFlowException) {
msg = "請(qǐng)求被熱點(diǎn)參數(shù)限流";
} else if (e instanceof DegradeException) {
msg = "請(qǐng)求被降級(jí)了";
} else if (e instanceof AuthorityException) {
msg = "沒(méi)有權(quán)限訪問(wèn)";
}
return msg;
}
注意事項(xiàng)
在定義 blockHandler 方法時(shí),需要注意以下 3 個(gè)問(wèn)題:
- 自定義的 blockHandler 方法的返回值,必須要和原方法(使用 @SentinelResource 注解修飾的方法)的返回值保持一致。
- 自定義的 blockHandler 方法的參數(shù)必須和原方法參數(shù)保持一致。
- 自定義的 blockHandler 方法的方法參數(shù)中必須包含 BlockException 參數(shù)。
如果不滿足以上事項(xiàng)中的任何一項(xiàng),那么就不能正常匹配到自定義的 blockHandler 方法,并且程序也會(huì)報(bào)錯(cuò)。
三、自定義全局異常
自定義 Sentinel 全局異常需要實(shí)現(xiàn) BlockExceptionHandler 類,并重寫(xiě) handle 方法,如下代碼所示:
import com.alibaba.csp.sentinel.adapter.spring.webmvc.callback.BlockExceptionHandler;
import com.alibaba.csp.sentinel.slots.block.BlockException;
import com.alibaba.csp.sentinel.slots.block.authority.AuthorityException;
import com.alibaba.csp.sentinel.slots.block.degrade.DegradeException;
import com.alibaba.csp.sentinel.slots.block.flow.FlowException;
import com.alibaba.csp.sentinel.slots.block.flow.param.ParamFlowException;
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;
import org.springframework.http.HttpStatus;
import org.springframework.stereotype.Component;
@Component
public class SentinelExceptionHandler implements BlockExceptionHandler {
@Override
public void handle(HttpServletRequest request, HttpServletResponse response, BlockException e) throws Exception {
String msg = "未知異常";
int status = HttpStatus.TOO_MANY_REQUESTS.value();
if (e instanceof FlowException) {
msg = "請(qǐng)求被限流了";
} else if (e instanceof ParamFlowException) {
msg = "請(qǐng)求被熱點(diǎn)參數(shù)限流";
} else if (e instanceof DegradeException) {
msg = "請(qǐng)求被降級(jí)了";
} else if (e instanceof AuthorityException) {
msg = "沒(méi)有權(quán)限訪問(wèn)";
status = HttpStatus.UNAUTHORIZED.value();
}
response.setContentType("application/json;charset=utf-8");
response.setStatus(status);
response.getWriter().println("{\"msg\": " + msg + ", \"code\": " + status + "}");
}
}
自定義 Sentinel 全局異常是在執(zhí)行 Sentinel 控制臺(tái)設(shè)置的限流和熔斷異常時(shí),執(zhí)行的全局自定義異常方法。
但是,如果是程序中出現(xiàn)的 Sentinel 報(bào)錯(cuò)信息,例如使用熱點(diǎn)限流時(shí),因?yàn)橐浜鲜褂?@SentinelResource 注解時(shí),此時(shí)只自定義了 value 屬性,未定義局部 blockHandler 方法,此時(shí)系統(tǒng)就會(huì)報(bào)錯(cuò),但這個(gè)時(shí)候并不會(huì)執(zhí)行 Sentinel 全局自定義異常,而是程序報(bào)錯(cuò),此時(shí)就需要使用系統(tǒng)自定義異常來(lái)重新定義異常信息了。
四、自定義系統(tǒng)異常
自定義系統(tǒng)異常需要新建一個(gè)異常類,并且使用 @RestControllerAdvice 注解修飾此類,并配合 @ExceptionHandler 注解來(lái)完成全局系統(tǒng)異常的獲取和定義,具體實(shí)現(xiàn)代碼如下:
import com.alibaba.csp.sentinel.slots.block.authority.AuthorityException;
import com.alibaba.csp.sentinel.slots.block.degrade.DegradeException;
import com.alibaba.csp.sentinel.slots.block.flow.FlowException;
import com.alibaba.csp.sentinel.slots.block.flow.param.ParamFlowException;
import org.springframework.http.HttpStatus;
import org.springframework.web.bind.annotation.ExceptionHandler;
import org.springframework.web.bind.annotation.ResponseBody;
import org.springframework.web.bind.annotation.RestControllerAdvice;
import java.util.HashMap;
import java.util.Map;
@RestControllerAdvice
public class CustomExceptionHandler {
/**
* 限流全局異常
*/
@ExceptionHandler(FlowException.class)
public Map handlerFlowException(){
return new HashMap(){{
put("code", HttpStatus.TOO_MANY_REQUESTS.value());
put("msg", "被限流");
}};
}
/**
* 熔斷全局異常
*/
@ExceptionHandler(DegradeException.class)
public Map handlerDegradeException(){
return new HashMap(){{
put("code", HttpStatus.TOO_MANY_REQUESTS.value());
put("msg", "被熔斷");
}};
}
/**
* 熱點(diǎn)限流異常
*/
@ExceptionHandler(ParamFlowException.class)
public Map handlerparamFlowException(){
return new HashMap(){{
put("code", HttpStatus.TOO_MANY_REQUESTS.value());
put("msg", "熱點(diǎn)限流");
}};
}
/**
* Sentinel 權(quán)限攔截全局異常
*/
@ExceptionHandler(AuthorityException.class)
@ResponseBody
public Map handlerAuthorityException(){
return new HashMap(){{
put("code", HttpStatus.UNAUTHORIZED.value());
put("msg", "暫無(wú)權(quán)限");
}};
}
}
此時(shí),只要是系統(tǒng)中出現(xiàn)的 Sentinel 報(bào)錯(cuò)信息,都會(huì)被此方法所捕獲,并通過(guò)自定義的代碼完成自定義異常信息的返回。
小結(jié)
Sentinel 有 3 種自定義異常的實(shí)現(xiàn):自定義局部異常、自定義(Sentinel)全局異常、自定義系統(tǒng)異常。自定義局部異常作用范圍比較小,需要給每個(gè)資源單獨(dú)設(shè)置才行;而自定義全局異常作用范圍比較大,但如果是程序報(bào)錯(cuò),也不會(huì)執(zhí)行其方法,所以需要配合系統(tǒng)異常同時(shí)來(lái)完成自定義異常的返回。
PS:如果這 3 種自定義異常同時(shí)存在,那么它的執(zhí)行優(yōu)先級(jí)是:自定義局部異常 > 自定義全局異常 > 自定義系統(tǒng)異常。