Spring Cloud Zuul自定義過(guò)濾器的實(shí)現(xiàn)
構(gòu)建Zuul自定義過(guò)濾器,限制ip頻繁請(qǐng)求
自定義zuul過(guò)濾器其實(shí)很簡(jiǎn)單
1. 首先pom文件得先引入zuul依賴
<dependency> <groupId>org.springframework.cloud</groupId> <artifactId>spring-cloud-starter-netflix-zuul</artifactId> </dependency>
2. 創(chuàng)建一個(gè)類,繼承自ZuulFilter
import com.netflix.zuul.ZuulFilter;
import com.netflix.zuul.exception.ZuulException;
import org.springframework.stereotype.Component;
/**
* 構(gòu)建zuul自定義過(guò)濾器
*/
@Component
public class MyFilter extends ZuulFilter {
/**
* 定義過(guò)濾器的類型
* pre:在請(qǐng)求被路由之前執(zhí)行
* route:在路由請(qǐng)求的時(shí)候執(zhí)行
* post:請(qǐng)求路由以后執(zhí)行
* error:處理請(qǐng)求時(shí)發(fā)生錯(cuò)誤的時(shí)候執(zhí)行
*
* @return 過(guò)濾器的類型
*/
@Override
public String filterType() {
return "pre";
}
/**
* 過(guò)濾器執(zhí)行的順序,配置多個(gè)有順序的過(guò)濾
* 執(zhí)行順序從小到大
*
* @return 執(zhí)行順序
*/
@Override
public int filterOrder() {
return 1;
}
/**
* 是否開啟過(guò)濾器
* true:開啟
* false:禁用
*
* @return 是否開啟過(guò)濾器
*/
@Override
public boolean shouldFilter() {
return true;
}
/**
* 過(guò)濾器的業(yè)務(wù)實(shí)現(xiàn)
*
* @return null 沒(méi)有意義
* @throws ZuulException 異常信息
*/
@Override
public Object run() throws ZuulException {
System.out.println("per zuul filter...");
return null;
}
}
自定義類上需要加上 @Component 注解
a. filterType()方法,定義過(guò)濾器的類型,返回的就是字符串,有以下4種類型
- pre:在請(qǐng)求被路由之前執(zhí)行
- route:在路由請(qǐng)求的時(shí)候執(zhí)行
- post:請(qǐng)求路由以后執(zhí)行
- error:處理請(qǐng)求時(shí)發(fā)生錯(cuò)誤的時(shí)候執(zhí)行
b. filterOrder()方法,過(guò)濾器執(zhí)行的順序
c. shouldFilter()方法,是否開啟過(guò)濾器,true開啟,false不開啟
d. run()方法,過(guò)濾器的業(yè)務(wù)實(shí)現(xiàn),在這里寫實(shí)現(xiàn)邏輯的具體代碼
3. 限制ip頻繁請(qǐng)求,示例代碼
import com.imooc.grace.result.GraceJsonResult;
import com.imooc.grace.result.ResponseStatusEnum;
import com.imooc.utils.IPUtil;
import com.imooc.utils.JsonUtils;
import com.imooc.utils.RedisOperator;
import com.netflix.zuul.ZuulFilter;
import com.netflix.zuul.context.RequestContext;
import com.netflix.zuul.exception.ZuulException;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.http.MediaType;
import org.springframework.stereotype.Component;
import javax.servlet.http.HttpServletRequest;
/**
* 限制ip頻繁請(qǐng)求
*/
@Component
public class BlackIpFilter extends ZuulFilter {
/**
* ip連續(xù)請(qǐng)求的次數(shù)
*/
private static final int CONTINUE_COUNTS = 10;
/**
* ip判斷的時(shí)間間隔,單位秒
*/
private static final int TIME_INTERVAL = 10;
/**
* 限制的時(shí)間,單位秒
*/
private static final int LIMIT_TIMES = 15;
@Autowired
private RedisOperator redisOperator;
@Override
public String filterType() {
return "pre";
}
@Override
public int filterOrder() {
// 這里設(shè)置為2,上面那個(gè)過(guò)濾器設(shè)置為1,則執(zhí)行順序?yàn)?1->2,大家可以測(cè)試一下
return 2;
}
@Override
public boolean shouldFilter() {
return true;
}
@Override
public Object run() throws ZuulException {
// 獲取上下文對(duì)象
RequestContext currentContext = RequestContext.getCurrentContext();
HttpServletRequest request = currentContext.getRequest();
// 獲取ip
String requestIp = IPUtil.getRequestIp(request);
// 判斷該ip在10秒內(nèi)請(qǐng)求次數(shù)是否超過(guò)10次,超過(guò)則限制該ip15秒內(nèi)不能訪問(wèn),15秒后再放行
final String ipRedisKey = "zuul-ip:" + requestIp;
final String ipRedisLimitKey = "zuul-ip-limit:" + requestIp;
// 獲取當(dāng)前ip這個(gè)key的剩余時(shí)間
long limitLeftTime = redisOperator.ttl(ipRedisLimitKey);
// 判斷該ip是否還有剩余時(shí)間
if (limitLeftTime > 0) {
stopRequest(currentContext);
return null;
}
// 在redis中累加ip的請(qǐng)求次數(shù)
long requestCounts = redisOperator.increment(ipRedisKey, 1);
if (requestCounts == 1) {
redisOperator.expire(ipRedisKey, TIME_INTERVAL);
}
if (requestCounts > CONTINUE_COUNTS) {
// 限制ip訪問(wèn)
redisOperator.set(ipRedisLimitKey, ipRedisLimitKey, LIMIT_TIMES);
stopRequest(currentContext);
}
return null;
}
private void stopRequest(RequestContext context) {
// 停止zuul繼續(xù)向下路由,禁止請(qǐng)求通信
context.setSendZuulResponse(false);
// 返回響應(yīng)碼200
context.setResponseStatusCode(200);
// TODO 要返回提示的json內(nèi)容(可以自定義任何響應(yīng)內(nèi)容)
// 例如 {"status":544,"msg":"請(qǐng)求過(guò)于頻繁,請(qǐng)稍后再試","success":false,"data":null}
String result = "json內(nèi)容";
// 設(shè)置返回內(nèi)容
context.setResponseBody(result);
// 設(shè)置編碼
context.getResponse().setCharacterEncoding("utf-8");
// 設(shè)置返回內(nèi)容格式為json
context.getResponse().setContentType(MediaType.APPLICATION_JSON_VALUE);
}
}
這里使用了redis來(lái)記錄ip請(qǐng)求次數(shù)和控制時(shí)間間隔
獲取ip工具類 IPUtil
import javax.servlet.http.HttpServletRequest;
/**
* 獲取ip工具類
*/
public class IPUtil {
/**
* 獲取請(qǐng)求IP:
* 用戶的真實(shí)IP不能使用request.getRemoteAddr()
* 這是因?yàn)榭赡軙?huì)使用一些代理軟件,這樣ip獲取就不準(zhǔn)確了
* 此外我們?nèi)绻褂昧硕嗉?jí)(LVS/Nginx)反向代理的話,ip需要從X-Forwarded-For中獲得第一個(gè)非unknown的IP才是用戶的有效ip。
*/
public static String getRequestIp(HttpServletRequest request) {
String ip = request.getHeader("x-forwarded-for");
if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
ip = request.getHeader("Proxy-Client-IP");
}
if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
ip = request.getHeader("WL-Proxy-Client-IP");
}
if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
ip = request.getHeader("HTTP_CLIENT_IP");
}
if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
ip = request.getHeader("HTTP_X_FORWARDED_FOR");
}
if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
ip = request.getRemoteAddr();
}
return ip;
}
}
到此這篇關(guān)于Spring Cloud Zuul自定義過(guò)濾器的實(shí)現(xiàn)的文章就介紹到這了,更多相關(guān)Spring Cloud Zuul過(guò)濾器內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
關(guān)于Java利用反射實(shí)現(xiàn)動(dòng)態(tài)運(yùn)行一行或多行代碼
這篇文章主要介紹了關(guān)于Java利用反射實(shí)現(xiàn)動(dòng)態(tài)運(yùn)行一行或多行代碼,借鑒了別人的方法和書上的內(nèi)容,最后將題目完成了,和大家一起分享以下解決方法,需要的朋友可以參考下2023-04-04
Java容器ArrayList知識(shí)點(diǎn)總結(jié)
本篇文章給大家分享了Java容器ArrayList的相關(guān)知識(shí)點(diǎn),對(duì)此有需要的朋友可以跟著學(xué)習(xí)參考下。2018-05-05
Java實(shí)現(xiàn)FTP上傳到服務(wù)器
這篇文章主要為大家詳細(xì)介紹了Java實(shí)現(xiàn)FTP上傳到服務(wù)器,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-09-09
springboot?集成identityserver4身份驗(yàn)證的過(guò)程解析
這篇文章主要介紹了springboot?集成identityserver4身份驗(yàn)證的相關(guān)知識(shí),本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友參考下吧2024-01-01
Javabean轉(zhuǎn)換成json字符并首字母大寫代碼實(shí)例
這篇文章主要介紹了javabean轉(zhuǎn)成json字符并首字母大寫代碼實(shí)例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-02-02
Springboot內(nèi)置tomcat配置虛擬路徑過(guò)程解析
這篇文章主要介紹了Springboot內(nèi)置tomcat配置虛擬路徑過(guò)程解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-04-04
Spring+Http請(qǐng)求+HttpClient實(shí)現(xiàn)傳參
這篇文章主要介紹了Spring+Http請(qǐng)求+HttpClient實(shí)現(xiàn)傳參,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-03-03

