package com.alibaba.csp.sentinel.dashboard.controller;
|
|
|
import java.util.Date;
|
import java.util.List;
|
|
import com.alibaba.csp.sentinel.dashboard.auth.AuthAction;
|
import com.alibaba.csp.sentinel.dashboard.auth.AuthService.PrivilegeType;
|
import com.alibaba.csp.sentinel.dashboard.controller.base.BaseRuleController;
|
import com.alibaba.csp.sentinel.dashboard.repository.rule.RuleRepository;
|
import com.alibaba.csp.sentinel.dashboard.rule.DynamicRuleProvider;
|
import com.alibaba.csp.sentinel.dashboard.rule.DynamicRulePublisher;
|
import com.alibaba.csp.sentinel.slots.block.RuleConstant;
|
import com.alibaba.csp.sentinel.slots.block.degrade.circuitbreaker.CircuitBreakerStrategy;
|
import com.alibaba.csp.sentinel.util.StringUtil;
|
|
import com.alibaba.csp.sentinel.dashboard.datasource.entity.rule.DegradeRuleEntity;
|
import com.alibaba.csp.sentinel.dashboard.domain.Result;
|
|
import org.slf4j.Logger;
|
import org.slf4j.LoggerFactory;
|
import org.springframework.beans.factory.annotation.Autowired;
|
import org.springframework.beans.factory.annotation.Qualifier;
|
import org.springframework.web.bind.annotation.DeleteMapping;
|
import org.springframework.web.bind.annotation.GetMapping;
|
import org.springframework.web.bind.annotation.PathVariable;
|
import org.springframework.web.bind.annotation.PostMapping;
|
import org.springframework.web.bind.annotation.PutMapping;
|
import org.springframework.web.bind.annotation.RequestBody;
|
import org.springframework.web.bind.annotation.RequestMapping;
|
import org.springframework.web.bind.annotation.RestController;
|
|
/**
|
* 降级规则控制器
|
*
|
* @author zyf
|
* @date 2022-04-13
|
*/
|
@RestController
|
@RequestMapping("/degrade")
|
public class DegradeController extends BaseRuleController {
|
|
private final Logger logger = LoggerFactory.getLogger(DegradeController.class);
|
|
@Autowired
|
private RuleRepository<DegradeRuleEntity, Long> repository;
|
@Autowired
|
@Qualifier("degradeRuleNacosProvider")
|
private DynamicRuleProvider<List<DegradeRuleEntity>> ruleProvider;
|
@Autowired
|
@Qualifier("degradeRuleNacosPublisher")
|
private DynamicRulePublisher<List<DegradeRuleEntity>> rulePublisher;
|
|
@GetMapping("/rules.json")
|
@AuthAction(PrivilegeType.READ_RULE)
|
public Result<List<DegradeRuleEntity>> apiQueryMachineRules(String app, String ip, Integer port) {
|
if (StringUtil.isEmpty(app)) {
|
return Result.ofFail(-1, "app can't be null or empty");
|
}
|
if (StringUtil.isEmpty(ip)) {
|
return Result.ofFail(-1, "ip can't be null or empty");
|
}
|
if (port == null) {
|
return Result.ofFail(-1, "port can't be null");
|
}
|
try {
|
List<DegradeRuleEntity> rules = ruleProvider.getRules(app);
|
rules = repository.saveAll(rules);
|
return Result.ofSuccess(rules);
|
} catch (Throwable throwable) {
|
logger.error("queryApps error:", throwable);
|
return Result.ofThrowable(-1, throwable);
|
}
|
}
|
|
@PostMapping("/rule")
|
@AuthAction(PrivilegeType.WRITE_RULE)
|
public Result<DegradeRuleEntity> apiAddRule(@RequestBody DegradeRuleEntity entity) {
|
Result<DegradeRuleEntity> checkResult = checkEntityInternal(entity);
|
if (checkResult != null) {
|
return checkResult;
|
}
|
Date date = new Date();
|
entity.setGmtCreate(date);
|
entity.setGmtModified(date);
|
try {
|
entity = repository.save(entity);
|
publishRules(entity.getApp());
|
} catch (Throwable t) {
|
logger.error("Failed to add new degrade rule, app={}, ip={}", entity.getApp(), entity.getIp(), t);
|
return Result.ofThrowable(-1, t);
|
}
|
return Result.ofSuccess(entity);
|
}
|
|
@PutMapping("/rule/{id}")
|
@AuthAction(PrivilegeType.WRITE_RULE)
|
public Result<DegradeRuleEntity> apiUpdateRule(@PathVariable("id") Long id,
|
@RequestBody DegradeRuleEntity entity) {
|
if (id == null || id <= 0) {
|
return Result.ofFail(-1, "id can't be null or negative");
|
}
|
DegradeRuleEntity oldEntity = repository.findById(id);
|
if (oldEntity == null) {
|
return Result.ofFail(-1, "Degrade rule does not exist, id=" + id);
|
}
|
entity.setApp(oldEntity.getApp());
|
entity.setIp(oldEntity.getIp());
|
entity.setPort(oldEntity.getPort());
|
entity.setId(oldEntity.getId());
|
Result<DegradeRuleEntity> checkResult = checkEntityInternal(entity);
|
if (checkResult != null) {
|
return checkResult;
|
}
|
|
entity.setGmtCreate(oldEntity.getGmtCreate());
|
entity.setGmtModified(new Date());
|
try {
|
entity = repository.save(entity);
|
publishRules(entity.getApp());
|
} catch (Throwable t) {
|
logger.error("Failed to save degrade rule, id={}, rule={}", id, entity, t);
|
return Result.ofThrowable(-1, t);
|
}
|
return Result.ofSuccess(entity);
|
}
|
|
@DeleteMapping("/rule/{id}")
|
@AuthAction(PrivilegeType.DELETE_RULE)
|
public Result<Long> delete(@PathVariable("id") Long id) {
|
if (id == null) {
|
return Result.ofFail(-1, "id can't be null");
|
}
|
|
DegradeRuleEntity oldEntity = repository.findById(id);
|
if (oldEntity == null) {
|
return Result.ofSuccess(null);
|
}
|
|
try {
|
repository.delete(id);
|
publishRules(oldEntity.getApp());
|
} catch (Throwable throwable) {
|
logger.error("Failed to delete degrade rule, id={}", id, throwable);
|
return Result.ofThrowable(-1, throwable);
|
}
|
return Result.ofSuccess(id);
|
}
|
|
private void publishRules(/*@NonNull*/ String app) throws Exception {
|
List<DegradeRuleEntity> rules = repository.findAllByApp(app);
|
rulePublisher.publish(app, rules);
|
//延迟加载
|
delayTime();
|
}
|
|
private <R> Result<R> checkEntityInternal(DegradeRuleEntity entity) {
|
if (StringUtil.isBlank(entity.getApp())) {
|
return Result.ofFail(-1, "app can't be blank");
|
}
|
if (StringUtil.isBlank(entity.getIp())) {
|
return Result.ofFail(-1, "ip can't be null or empty");
|
}
|
if (entity.getPort() == null || entity.getPort() <= 0) {
|
return Result.ofFail(-1, "invalid port: " + entity.getPort());
|
}
|
if (StringUtil.isBlank(entity.getLimitApp())) {
|
return Result.ofFail(-1, "limitApp can't be null or empty");
|
}
|
if (StringUtil.isBlank(entity.getResource())) {
|
return Result.ofFail(-1, "resource can't be null or empty");
|
}
|
Double threshold = entity.getCount();
|
if (threshold == null || threshold < 0) {
|
return Result.ofFail(-1, "invalid threshold: " + threshold);
|
}
|
Integer recoveryTimeoutSec = entity.getTimeWindow();
|
if (recoveryTimeoutSec == null || recoveryTimeoutSec <= 0) {
|
return Result.ofFail(-1, "recoveryTimeout should be positive");
|
}
|
Integer strategy = entity.getGrade();
|
if (strategy == null) {
|
return Result.ofFail(-1, "circuit breaker strategy cannot be null");
|
}
|
if (strategy < CircuitBreakerStrategy.SLOW_REQUEST_RATIO.getType()
|
|| strategy > RuleConstant.DEGRADE_GRADE_EXCEPTION_COUNT) {
|
return Result.ofFail(-1, "Invalid circuit breaker strategy: " + strategy);
|
}
|
if (entity.getMinRequestAmount() == null || entity.getMinRequestAmount() <= 0) {
|
return Result.ofFail(-1, "Invalid minRequestAmount");
|
}
|
if (entity.getStatIntervalMs() == null || entity.getStatIntervalMs() <= 0) {
|
return Result.ofFail(-1, "Invalid statInterval");
|
}
|
if (strategy == RuleConstant.DEGRADE_GRADE_RT) {
|
Double slowRatio = entity.getSlowRatioThreshold();
|
if (slowRatio == null) {
|
return Result.ofFail(-1, "SlowRatioThreshold is required for slow request ratio strategy");
|
} else if (slowRatio < 0 || slowRatio > 1) {
|
return Result.ofFail(-1, "SlowRatioThreshold should be in range: [0.0, 1.0]");
|
}
|
} else if (strategy == RuleConstant.DEGRADE_GRADE_EXCEPTION_RATIO) {
|
if (threshold > 1) {
|
return Result.ofFail(-1, "Ratio threshold should be in range: [0.0, 1.0]");
|
}
|
}
|
return null;
|
}
|
}
|