利用断路器状态统计sentinel熔断次数

最近项目需要sentinel熔断时记录熔断的次数,在经过一阵搜索后决定利用断路器的状态变化来实现此功能
然而,遇到了这样的一个情况,断路器的状态在第一次熔断时正常从close–>open,但在后续(熔断时间内blocked或者熔断结束后重新开启断路器)断路器的状态变化均为half open–>open,导致无法区分究竟是熔断时间内还是再次被熔断了,因此使用了一个flag来做标记实现了功能。

本方法利用了断路器的状态统计熔断次数,代码修改自chatgpt

代码如下:

/**
 * @ClassName: SentinelBlockHandler
 * @Description: sentinel统一异常处理
 * @author: liuzijian @ haiyisoft
 * @Date: 2023/11/7 11:54
 * @version: 1.0
 **/
@Component
@Slf4j
public class SentinelBlockHandler implements BlockExceptionHandler {
   

    private boolean flag = false;
    @Resource
    private ServerDegradeLogsRepository serverDegradeLogsRepository;
    @Value("${server.port}")
    private Integer serverPort;

    private SentinelBlockHandler() {
   
        // 私有构造方法,防止外部实例化
    }

    public static SentinelBlockHandler getInstance() {
   
        return SingletonHolder.INSTANCE;
    }

    private static class SingletonHolder {
   
        private static final SentinelBlockHandler INSTANCE = new SentinelBlockHandler();
    }

    static {
   
        registerStateChangeObserver();
    }

    private static void registerStateChangeObserver() {
   
        EventObserverRegistry.getInstance().addStateChangeObserver("logging",
                (prevState, newState, rule, snapshotValue) -> {
   
                    List<CircuitBreakerStateChangeObserver> observers = EventObserverRegistry.getInstance().getStateChangeObservers();
                    if (newState == CircuitBreaker.State.OPEN) {
   
                        getInstance().flag = true;
                        System.err.println(String.format("===> resource: %s, %s -> OPEN at %d, snapshotValue=%.2f",
                                rule.getResource(), prevState.name(), TimeUtil.currentTimeMillis(), snapshotValue));
                    } else {
   
                        System.err.println(String.format("===> resource: %s, %s -> %s at %d",
                                rule.getResource(), prevState.name(), newState.name(), TimeUtil.currentTimeMillis()));
                    }
                });
    }

    @Override
    public void handle(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, BlockException e) throws Exception {
   
        /**
         * 请求被限流了
         */
        if (e instanceof FlowException){
   
            httpServletResponse.setStatus(429);
            ResponseService.httpResponse(httpServletResponse, ResponseStatus.SENTINEL_FLOW_ERROR);
            log.error("请求被限流了:{}",e.getMessage());
        }
        /**
         * 请求被熔断降级了
         */
        else if (e instanceof DegradeException) {
   
            System.out.println("进入熔断异常");
            if (getInstance().flag){
   
                InetAddress addr = InetAddress.getLocalHost();
                String degradeText = "请求被熔断降级了:{" + e.getMessage() + "}";
                ServerDegradeLogs serverDegradeLogs = new ServerDegradeLogs(addr.getHostAddress(), this.serverPort, degradeText);
                serverDegradeLogsRepository.save(serverDegradeLogs);
                getInstance().flag = false;
            }
            httpServletResponse.setStatus(429);
            ResponseService.httpResponse(httpServletResponse, ResponseStatus.SENTINEL_DEGRADE_ERROR);
            log.error("请求被熔断降级了:{}",e.getMessage());
        }
        /**
         * 热点被限流了
         */
        else if (e instanceof ParamFlowException) {
   
            httpServletResponse.setStatus(429);
            ResponseService.httpResponse(httpServletResponse, ResponseStatus.SENTINEL_PARAM_FLOW_ERROR);
            log.error("热点被限流了:{}",e.getMessage());
        }
        /**
         * 请求没有权限
         */
        else if (e instanceof AuthorityException) {
   
            httpServletResponse.setStatus(401);
            ResponseService.httpResponse(httpServletResponse, ResponseStatus.UNAUTHORIZED);
            log.error("请求没有权限:{}",e.getMessage());
        }
    }
}

参考链接:
断路器的使用:https://github.com/alibaba/Sentinel/issues/1939

相关推荐

  1. 利用断路状态统计sentinel熔断次数

    2023-12-13 22:30:02       46 阅读
  2. 学习分享-断路Hystrix与Sentinel的区别

    2023-12-13 22:30:02       11 阅读

最近更新

  1. TCP协议是安全的吗?

    2023-12-13 22:30:02       18 阅读
  2. 阿里云服务器执行yum,一直下载docker-ce-stable失败

    2023-12-13 22:30:02       19 阅读
  3. 【Python教程】压缩PDF文件大小

    2023-12-13 22:30:02       18 阅读
  4. 通过文章id递归查询所有评论(xml)

    2023-12-13 22:30:02       20 阅读

热门阅读

  1. Json格式化

    2023-12-13 22:30:02       46 阅读
  2. ThreadLocal和InheritableThreadLocal

    2023-12-13 22:30:02       36 阅读
  3. vue3父子传值实现弹框功能

    2023-12-13 22:30:02       37 阅读
  4. CentOS 7 部署pure-ftp

    2023-12-13 22:30:02       45 阅读
  5. LeetCode //C - 1. Two Sum

    2023-12-13 22:30:02       38 阅读
  6. 【编码器】位置管理器BoosterPack

    2023-12-13 22:30:02       38 阅读
  7. 重载和重写的区别

    2023-12-13 22:30:02       40 阅读
  8. ARM day8

    ARM day8

    2023-12-13 22:30:02      35 阅读