A lightweight powerful flow control component enabling reliability and monitoring for microservices. (輕量級的流量控制、熔斷降級 Java 庫).中文文檔html
雪崩效應又稱cascading failure(級聯故障),指基礎服務故障致使上層服務故障而且故障像雪球同樣越滾越大。java
<!--sentinel-->
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-sentinel</artifactId>
</dependency>
複製代碼
無
複製代碼
無
複製代碼
打開全部端點查看是否生效git
# actuator
management:
endpoints:
web:
exposure:
# 生產不能所有放開且需配置安全措施
include: '*'
複製代碼
出現如下內容證實整合成功了github
下載咱們pom文件定義的相同版本的Sentinel。下載地址web
#啓動
java -jar XXX.jar
複製代碼
spring:
cloud:
sentinel:
transport:
# 指定sentinel控制檯地址
dashboard: localhost:8080
複製代碼
流控規則spring
直接apache
關聯:關聯資源達到閾值,限流本身json
鏈路:指定鏈路上的流量api
// 定義一個common服務
package com.virgo.user.service;
import com.virgo.entity.TblCar;
/**
* @author zhaozha
* @date 2019/10/15 下午4:27
*/
public interface CommonService {
TblCar common();
}
package com.virgo.user.service;
import com.alibaba.csp.sentinel.annotation.SentinelResource;
import com.virgo.entity.TblCar;
import org.springframework.stereotype.Service;
/**
* @author zhaozha
* @date 2019/10/15 下午4:28
*/
@Service
public class CommonServiceImpl implements CommonService {
@Override
@SentinelResource("common")
public TblCar common() {
return TblCar.builder().id(1L).build();
}
}
// controller層同時調用這個服務
@GetMapping("/test/a")
public String testA() {
commonServiceImpl.common();
return "testA";
}
@GetMapping("/test/b")
public String testB() {
commonServiceImpl.common();
return "testB";
}
複製代碼
降級規則安全
熱點規則
系統規則
受權規則
微服務註冊/心跳到控制檯,控制檯經過api獲取/推送消息
spring:
cloud:
sentinel:
transport:
控制檯地址
dashboard:
和控制檯通訊的ip
client-ip:
和控制通訊端口(默認8719,若是已經佔用,+1直到找到)
port:
心跳毫秒
heartbeat-interval-ms:
複製代碼
配置項 | 默認值 | 描述 |
---|---|---|
server.port | 8080 | 指定端口 |
csp.sentinel.dashboard.server | localhost:8080 | 指定地址 |
project.name | - | - |
sentinel.dashboard.auth.username[1.6] | sentinel | Dashboard登錄帳號 |
sentinel.dashboard.auth.password[1.6] | sentinel | Dashboard登錄密碼 |
server.servlet.session.timeout[1.6] | 30分鐘 | 登錄session過時時間(7200:7200秒/60m:60分鐘) |
@GetMapping("/test/sentinel/api")
public String testSentinelApi(@RequestParam(required = false) String a) {
String resourceName = "test-sentinel-api";
ContextUtil.enter(resourceName, "test");
Entry entry = null;
try {
entry = SphU.entry(resourceName);
if (StringUtils.isBlank(a)) {
throw new IllegalArgumentException("參數不可爲空");
}
return a;
} catch (BlockException e) {
return "限流/降級了";
} catch (IllegalArgumentException e) {
Tracer.trace(e);
return "參數非法";
} finally {
if (entry != null) {
entry.exit();
}
ContextUtil.exit();
}
}
複製代碼
經過@SentinelResource簡化代碼
package com.virgo.user.sentinel;
import com.alibaba.csp.sentinel.slots.block.BlockException;
/**
* @author zhaozha
* @date 2019/10/16 下午3:11
*/
public class ControllerBlockHandlerClass {
/**
* 處理限流/降級
*
* @param a
* @param e
* @return
*/
public static String block(String a, BlockException e) {
return "降級/限流了";
}
}
@GetMapping("/test/sentinel/api")
@SentinelResource(
value = "test-sentinel-api",
blockHandler = "block",
blockHandlerClass = com.virgo.user.sentinel.ControllerBlockHandlerClass.class
)
public String testSentinelApi(@RequestParam(required = false) String a) {
if (StringUtils.isBlank(a)) {
throw new IllegalArgumentException("參數不可爲空");
}
return a;
}
複製代碼
@Bean
@LoadBalanced
@SentinelRestTemplate
public RestTemplate restTemplate(){
return new RestTemplate();
}
複製代碼
resttemplate:
sentinel:
enabled:
複製代碼
feign:
#feign整合sentinel
sentinel
enable: true
複製代碼
自定義返回報文
package com.virgo.user.feignclient;
import com.virgo.entity.TblCar;
import com.virgo.user.configuration.GlobalFeignConfiguration;
import com.virgo.user.feignclient.fallback.LockCenterFeignClientFallback;
import org.springframework.cloud.openfeign.FeignClient;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PathVariable;
/**
* @author zhaozha
* @date 2019/10/11 下午12:52
*/
@FeignClient(name = "lock-center", configuration = GlobalFeignConfiguration.class,fallback = LockCenterFeignClientFallback.class)
public interface LockCenterFeignClient {
@GetMapping("/lock/test/{id}")
TblCar findById(@PathVariable(value="id") Long id);
}
package com.virgo.user.feignclient.fallback;
import com.virgo.entity.TblCar;
import com.virgo.user.feignclient.LockCenterFeignClient;
import org.springframework.stereotype.Component;
/**
* @author zhaozha
* @date 2019/10/16 下午3:36
*/
@Component
public class LockCenterFeignClientFallback implements LockCenterFeignClient {
@Override
public TblCar findById(Long id) {
return TblCar.builder().level(1).build();
}
}
複製代碼
package com.virgo.user.feignclient;
import com.virgo.entity.TblCar;
import com.virgo.user.configuration.GlobalFeignConfiguration;
import com.virgo.user.feignclient.fallbackFactory.LockCenterFeignClientFallbackFactory;
import org.springframework.cloud.openfeign.FeignClient;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PathVariable;
/**
* @author zhaozha
* @date 2019/10/11 下午12:52
*/
@FeignClient(name = "lock-center", configuration = GlobalFeignConfiguration.class,fallbackFactory = LockCenterFeignClientFallbackFactory.class)
public interface LockCenterFeignClient {
@GetMapping("/lock/test/{id}")
TblCar findById(@PathVariable(value="id") Long id);
}
package com.virgo.user.feignclient.fallbackFactory;
import com.virgo.entity.TblCar;
import com.virgo.user.feignclient.LockCenterFeignClient;
import feign.hystrix.FallbackFactory;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Component;
/**
* @author zhaozha
* @date 2019/10/16 下午4:02
*/
@Component
@Slf4j
public class LockCenterFeignClientFallbackFactory implements FallbackFactory<LockCenterFeignClient> {
@Override
public LockCenterFeignClient create(Throwable throwable) {
return new LockCenterFeignClient() {
@Override
public TblCar findById(Long id) {
log.info("限流/降級",throwable);
return TblCar.builder().level(1).build();
}
};
}
}
複製代碼
使用方式 | 使用方式 | 使用方法 |
---|---|---|
編碼方式 | API | try...catch...finally |
註解方式 | @SentinelResource | blockHandler/fallback |
RestTemplate | @SentinelRestTemplate | blockHandler/fallback |
Feign | feign.sentinel.enabled | fallback/fallbackFactory |
阿里雲服務(AHAS)
package com.virgo.user.sentinel;
import com.alibaba.csp.sentinel.adapter.servlet.callback.UrlBlockHandler;
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 com.alibaba.csp.sentinel.slots.system.SystemBlockException;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.virgo.dto.CommonResult;
import org.springframework.stereotype.Component;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
/**
* @author zhaozha
* @date 2019/10/17 上午9:49
*/
@Component
public class VirgoUrlBlockHandler implements UrlBlockHandler {
@Override
public void blocked(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, BlockException e) throws IOException {
// 統一返回對象
CommonResult commonResult = null;
// 流控
if(e instanceof FlowException){
commonResult = CommonResult.builder().status(100).msg("流控異常").build();
}
// 降級
else if(e instanceof DegradeException){
commonResult = CommonResult.builder().status(101).msg("降級異常").build();
}
// 熱點
else if(e instanceof ParamFlowException){
commonResult = CommonResult.builder().status(102).msg("熱點異常").build();
}
// 系統
else if(e instanceof SystemBlockException){
commonResult = CommonResult.builder().status(102).msg("系統異常").build();
}
// 受權
else if (e instanceof AuthorityException){
commonResult = CommonResult.builder().status(102).msg("受權異常").build();
}
httpServletResponse.setStatus(500);
httpServletResponse.setCharacterEncoding("utf-8");
httpServletResponse.setHeader("Content-Type", "application/json;charset=utf-8");
httpServletResponse.setContentType("application/json;charset=utf-8");
new ObjectMapper()
.writeValue(
httpServletResponse.getWriter(),
commonResult
);
}
}
複製代碼
package com.virgo.user.sentinel;
import com.alibaba.csp.sentinel.adapter.servlet.callback.RequestOriginParser;
import org.apache.commons.lang3.StringUtils;
import org.springframework.stereotype.Component;
import javax.servlet.http.HttpServletRequest;
/**
* @author zhaozha
* @date 2019/10/17 上午11:51
*/
@Component
public class MyRequestOriginParser implements RequestOriginParser {
@Override
public String parseOrigin(HttpServletRequest httpServletRequest) {
// todo 改爲header
String origin = httpServletRequest.getParameter("origin");
if (StringUtils.isBlank(origin)) {
throw new IllegalArgumentException("origin must be specified");
}
return origin;
}
}
複製代碼
package com.virgo.user.sentinel;
import com.alibaba.csp.sentinel.adapter.servlet.callback.UrlCleaner;
import org.apache.commons.lang.math.NumberUtils;
import org.springframework.stereotype.Component;
import java.util.Arrays;
/**
* @author zhaozha
* @date 2019/10/17 下午12:05
*/
@Component
public class VirgoUrlCleaner implements UrlCleaner {
@Override
public String clean(String originUrl) {
// todo 多參數
String[] split = originUrl.split("/");
return Arrays.stream(split)
.map(string -> {
if (NumberUtils.isNumber(string)) {
return "{number}";
}
return string;
})
.reduce((a, b) -> a + "/" + b)
.orElse("");
}
}
複製代碼