Sentinel?Gateway自定義限流返回結(jié)果方式
Sentinel Gateway自定義限流返回結(jié)果
sentinel在1.6.0后就引入了Sentinel API Gateway Adapter Common,適配了Gateway。
以前sentinel自定義限流結(jié)果是通過實(shí)現(xiàn)BlockExceptionHandler接口進(jìn)行自定義限流返回結(jié)果,但是如果是使用了spring-cloud-alibaba-sentinel-gateway的依賴進(jìn)行限流的話,使用之前的接口是不起作用的。
首先是我本地項(xiàng)目的環(huán)境,適配的是springboot 2.3.12.RELEASE版本,cloud版本為Hoxton.SR12。
<dependency>
<groupId>com.alibaba.cloud</groupId>
<artifactId>spring-cloud-starter-alibaba-sentinel</artifactId>
<version>2.2.7.RELEASE</version>
</dependency>
<dependency>
<groupId>com.alibaba.cloud</groupId>
<artifactId>spring-cloud-alibaba-sentinel-gateway</artifactId>
<version>2.2.7.RELEASE</version>
</dependency>通過SentinelSCGAutoConfiguration配置類,我們可以看到主要有兩種方法可以實(shí)現(xiàn)自定義:配置文件配置和自定義代碼配置。
1、配置文件配置
在application.yml加入以下配置:
spring:
cloud:
sentinel:
scg:
fallback:
mode: response
responseBody: "{\"code\":\"429\",\"msg\":\"請(qǐng)求太多了\"}"對(duì)應(yīng)的配置類為FallbackProperties,以下是詳細(xì)的配置:
/** * The fallback mode for sentinel spring-cloud-gateway. choose `redirect` or * `response`. */ private String mode; /** * Redirect Url for `redirect` mode. */ private String redirect; /** * Response Body for `response` mode. */ private String responseBody; /** * Response Status for `response` mode. */ private Integer responseStatus = HttpStatus.TOO_MANY_REQUESTS.value(); /** * Content-Type for `response` mode. */ private String contentType = MediaType.APPLICATION_JSON.toString();
可以通過配置看到根據(jù)mode的類型處理模式有兩種:response和redirect。在SentinelSCGAutoConfiguration中對(duì)應(yīng)的代碼在 initFallback() 方法。
response:
如果設(shè)置mode為response則返回contentType 類型的responseBody,響應(yīng)碼為responseStatus 。
redirect:
如果設(shè)置mode為redirect,則限流后自動(dòng)跳轉(zhuǎn)某個(gè)頁面,頁面地址為redirect。
2、自定義代碼配置
通過Sentinel wiki 我們可以看到:

所以我們可以通
過GatewayCallbackManager的 setBlockHandler() 方法來實(shí)現(xiàn)注冊(cè)我們自定義的異常處理類,需要實(shí)現(xiàn)BlockRequestHandler接口,默認(rèn)的異常處理類是:DefaultBlockRequestHandler。
首先自定義一個(gè)異常處理類:
import com.alibaba.csp.sentinel.adapter.gateway.sc.callback.BlockRequestHandler;
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;
import org.springframework.http.HttpStatus;
import org.springframework.http.InvalidMediaTypeException;
import org.springframework.http.MediaType;
import org.springframework.web.reactive.function.server.ServerResponse;
import org.springframework.web.server.ServerWebExchange;
import reactor.core.publisher.Mono;
import java.net.URI;
import java.util.List;
import static org.springframework.web.reactive.function.BodyInserters.fromValue;
/**
* @author TYH
* @Description: Sentinel異常處理類
* @date 2022/3/8 14:34
*/
public class SentinelBlockRequestHandler implements BlockRequestHandler {
@Override
public Mono<ServerResponse> handleRequest(ServerWebExchange exchange, Throwable t) {
// 判斷是否是html訪問,如果是則轉(zhuǎn)發(fā)url
if (acceptsHtml(exchange)) {
return htmlErrorResponse();
}
// 如果是接口訪問,則返回提示
return ServerResponse.status(HttpStatus.TOO_MANY_REQUESTS)
.contentType(MediaType.APPLICATION_JSON_UTF8)
.body(fromValue(new ResponseData(HttpStatus.TOO_MANY_REQUESTS.value(),"請(qǐng)求太多了")));
}
private Mono<ServerResponse> htmlErrorResponse() {
String url="http://www.baidu.com";
URI uri =URI.create(url);
return ServerResponse.temporaryRedirect(uri).build();
}
private boolean acceptsHtml(ServerWebExchange exchange) {
try {
List<MediaType> acceptedMediaTypes = exchange.getRequest().getHeaders().getAccept();
acceptedMediaTypes.remove(MediaType.ALL);
MediaType.sortBySpecificityAndQuality(acceptedMediaTypes);
return acceptedMediaTypes.stream()
.anyMatch(MediaType.TEXT_HTML::isCompatibleWith);
} catch (InvalidMediaTypeException ex) {
return false;
}
}
/**
* 定義返回的實(shí)體類,字段根據(jù)需要添加
*/
@Data
@AllArgsConstructor
@NoArgsConstructor
static class ResponseData {
private int code;
private String msg;
}
}然后再調(diào)用GatewayCallbackManager的方法進(jìn)行注冊(cè):
import com.alibaba.csp.sentinel.adapter.gateway.sc.callback.GatewayCallbackManager;
import lombok.extern.slf4j.Slf4j;
import org.springframework.boot.CommandLineRunner;
import org.springframework.context.annotation.Configuration;
/**
* @author TYH
* @Description:
* @date 2022/3/8 15:02
*/
@Slf4j
@Configuration
public class InitConfig implements CommandLineRunner {
@Override
public void run(String... args) throws Exception {
log.info("初始化限流handler");
GatewayCallbackManager.setBlockHandler(new SentinelBlockRequestHandler());
}
}
然后就實(shí)現(xiàn)了接口訪問限流返回固定值,html訪問限流跳轉(zhuǎn)頁面的效果。
總結(jié)
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
詳解設(shè)計(jì)模式中的proxy代理模式及在Java程序中的實(shí)現(xiàn)
代理模式主要分為靜態(tài)代理和動(dòng)態(tài)代理,使客戶端方面的使用者通過設(shè)置的代理來操作對(duì)象,下面來詳解設(shè)計(jì)模式中的proxy代理模式及在Java程序中的實(shí)現(xiàn)2016-05-05
Netty學(xué)習(xí)教程之Netty與Marshalling結(jié)合發(fā)送對(duì)象
Netty是由JBOSS提供的一個(gè)Java開源框架,之前已經(jīng)給大家簡(jiǎn)單介紹了一些基礎(chǔ)與使用,下面這篇文章主要給大家介紹了關(guān)于Netty與Marshalling結(jié)合發(fā)送對(duì)象的相關(guān)資料,需要的朋友可以參考借鑒,下面來一起看看吧。2017-05-05
文件上傳SpringBoot后端MultipartFile參數(shù)報(bào)空問題的解決辦法
這篇文章主要介紹了文件上傳SpringBoot后端MultipartFile參數(shù)報(bào)空問題的解決辦法,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-11-11
Spring Cloud入門系列服務(wù)提供者總結(jié)
這篇文章主要介紹了Spring Cloud入門系列之服務(wù)提供者總結(jié),服務(wù)提供者使用Eureka Client組件創(chuàng)建 ,創(chuàng)建完成以后修改某文件,具體操作方法及實(shí)例代碼跟隨小編一起看看吧2021-06-06
JAVA 數(shù)據(jù)結(jié)構(gòu)鏈表操作循環(huán)鏈表
這篇文章主要介紹了JAVA 數(shù)據(jù)結(jié)構(gòu)鏈表操作循環(huán)鏈表的相關(guān)資料,需要的朋友可以參考下2016-10-10

