服务摘流核心逻辑

当某个服务挂掉后,可以通过下发命令,使某个服务实例摘流,其核心逻辑如下 :


import com.alibaba.fastjson.JSON;
import com.***.flow.dto.ServiceFlowMqCallBackDTO;
import com.***.advance.flow.dto.ServiceFlowMqExchangeDTO;
import com.***.advance.flow.enums.ServiceStatusEnum;
import com.***.advance.flow.utils.InetUtil;
import com.***.base.json.JsonUtil;
import com.***.base.object.AskBeanUtil;
import com.***.communicate.async2mq.component.ServiceFlowMQProducer;
import com.***.rocketmq.enums.XXXMqEnums;
import lombok.extern.slf4j.Slf4j;
import org.apache.commons.lang3.StringUtils;
import org.apache.curator.x.discovery.ServiceDiscovery;
import org.apache.dubbo.common.extension.ExtensionLoader;
import org.apache.dubbo.common.utils.NetUtils;
import org.apache.dubbo.registry.Registry;
import org.apache.dubbo.registry.RegistryFactory;
import org.apache.dubbo.rpc.model.ApplicationModel;
import org.apache.dubbo.rpc.model.ProviderModel;
import org.apache.dubbo.rpc.model.ServiceRepository;
import org.apache.rocketmq.client.consumer.DefaultMQPushConsumer;
import org.apache.rocketmq.client.consumer.MQPushConsumer;
import org.apache.rocketmq.client.consumer.listener.ConsumeConcurrentlyStatus;
import org.apache.rocketmq.client.consumer.listener.MessageListenerConcurrently;
import org.apache.rocketmq.client.exception.MQClientException;
import org.apache.rocketmq.client.producer.DefaultMQProducer;
import org.apache.rocketmq.common.message.MessageExt;
import org.apache.rocketmq.common.protocol.heartbeat.MessageModel;
import org.apache.rocketmq.remoting.RPCHook;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.context.annotation.Bean;
import org.springframework.stereotype.Component;
import org.springframework.util.CollectionUtils;

import javax.annotation.PostConstruct;
import javax.annotation.Resource;
import java.io.IOException;
import java.net.InetAddress;
import java.util.*;

@Slf4j
@Component
public class ServiceFlowCommandMqListener {

    private final String SERVICE_FLOW_COMMAND_MQ_CONSUMER_BEAN_ID = "serviceFlowCommandMQConsumer";

    private final String SERVICE_FLOW_COMMAND_MQ_PRODUCER = "serviceFlowCommandMQProducer";
    private final String SERVICE_FLOW_COMMAND_MQ_CONSUMER_LISTENER_BEAN_ID = "serviceFlowCommandMQConsumerListener";

    /**
     * 执行后回执消息tag
     */
    public final String SERVICE_FLOW_CALL_BACK_TAG = "service_flow_call_back";
    public final String SERVICE_FLOW_CALL_COMMAND = "service_flow_command";

    @Value("${spring.application.name:}")
    private String APPLICATION_NAME;

    @Autowired(required = false)
    private ServiceDiscovery serviceDiscovery;

    @Autowired(required = false)
    private Map<String, MQPushConsumer> mqConsumerMap;

    @Value("${spring.cloud.zookeeper.discovery.preferIpAddress:true}")
    private Boolean preferIpAddress;

    @Autowired
    private InetUtil inetUtil;

    @PostConstruct
    public void init() {
        getProviderIPTag4DubboApplication();
        getProviderIPTag4HttpApplication();
    }

    public String getProviderIPTag4DubboApplication() {

        InetAddress inetAddress = NetUtils.getLocalAddress();
        if(Objects.isNull(inetAddress)){
            return StringUtils.EMPTY;
        }

        log.info("getProviderIPTag4DubboApplication, ip={}", inetAddress.getHostAddress());
        String ip  = inetAddress.getHostAddress();
        String ipTag = APPLICATION_NAME.replaceAll("-dev|-test|-pre|-prd|-grey", "") + "_" + ip.replace(".", "");
        log.info("getProviderIPTag4DubboApplication, ipTag={}", ipTag);
        return ipTag;
    }

    public String getProviderIPTag4HttpApplication() {
        String ip = getInstanceHost4HttpApplication();
        if(Objects.isNull(ip)){
            log.warn("getProviderIPTag4HttpApplication, get ip failed");
            return StringUtils.EMPTY;
        }

        log.info("getProviderIPTag4HttpApplication, ip={}", ip);
        String ipTag = APPLICATION_NAME.replaceAll("-dev|-test|-pre|-prd|-grey", "") + "_" + ip.replace(".", "");
        log.info("getProviderIPTag4HttpApplication, ipTag={}", ipTag);
        return ipTag;
    }

    private String getInstanceHost4HttpApplication() {
        InetUtil.HostInfo hostInfo = inetUtil.findFirstNonLoopbackHostInfo();
        log.info("getInstanceHost4HttpApplication, hostInfo:{}", JsonUtil.toDefaultString(hostInfo));
        String instanceHost = hostInfo.getHostname();
        String instanceIpAddress = hostInfo.getIpAddress();
        return this.preferIpAddress && org.springframework.util.StringUtils.hasText(instanceIpAddress) ? instanceIpAddress : instanceHost;
    }

    @Bean(name = SERVICE_FLOW_COMMAND_MQ_PRODUCER, initMethod = "start", destroyMethod = "shutdown")
    public ServiceFlowMQProducer getServiceFlowCallBackMQProducer(@Value("${rocketmq.namesrv}") String rocketMqNameSrv, RPCHook rpcHook) throws MQClientException{
        DefaultMQProducer producer = new DefaultMQProducer(DxyMqEnums.SendGroup.SERVICE_FLOW_COMMAND_CALL_BACK_MSG.name(), rpcHook);
        producer.setNamesrvAddr(rocketMqNameSrv);
        producer.setVipChannelEnabled(false);
        return new ServiceFlowMQProducer(producer, SERVICE_FLOW_CALL_BACK_TAG);
    }

    @Resource(name = SERVICE_FLOW_COMMAND_MQ_PRODUCER)
    private ServiceFlowMQProducer serviceFlowMQProducer;


    @Bean(name = SERVICE_FLOW_COMMAND_MQ_CONSUMER_BEAN_ID, initMethod = "start", destroyMethod = "shutdown")
    public DefaultMQPushConsumer serviceFlowCommandMQConsumer(@Qualifier(SERVICE_FLOW_COMMAND_MQ_CONSUMER_LISTENER_BEAN_ID) MessageListenerConcurrently serviceFlowCommandChangeMsgListenerConcurrently,
                                                                 @Value("${rocketmq.namesrv}") String rocketMqNameSrv, RPCHook rpcHook) throws MQClientException {
        DefaultMQPushConsumer consumer = new DefaultMQPushConsumer(rpcHook);
        consumer.setNamesrvAddr(rocketMqNameSrv);
        consumer.setVipChannelEnabled(false);
        consumer.setConsumeTimeout(120_000L);
        consumer.setMessageModel(MessageModel.BROADCASTING);
        //注意如果是消费顺序消息,单线程才能保证消息的顺序到达和顺序消费
        consumer.setConsumeThreadMin(1);
        consumer.setConsumeThreadMax(10);
        consumer.setConsumeMessageBatchMaxSize(1);
        consumer.setInstanceName(UUID.randomUUID().toString());
        consumer.setConsumerGroup(DxyMqEnums.ConsumeGroup.SERVICE_FLOW_COMMAND_MSG.name());
        consumer.subscribe(DxyMqEnums.Topic.SERVICE_FLOW_COMMAND_MSG.name(), SERVICE_FLOW_CALL_COMMAND);
        consumer.registerMessageListener(serviceFlowCommandChangeMsgListenerConcurrently);
        return consumer;
    }

    @Bean(name = SERVICE_FLOW_COMMAND_MQ_CONSUMER_LISTENER_BEAN_ID)
    public MessageListenerConcurrently serviceFlowCommandChangeMsgListenerConcurrently() {
        return (msgs, context) -> {
            if(CollectionUtils.isEmpty(msgs)){
                log.info("serviceFlowCommandChangeMsgListenerConcurrently skipped for msgList is empty.");
                return ConsumeConcurrentlyStatus.CONSUME_SUCCESS;
            }
            for (MessageExt message : msgs) {
                final String msgStr = new String(message.getBody());
                ServiceFlowMqExchangeDTO mqExchangeDTO = JSON.parseObject(msgStr, ServiceFlowMqExchangeDTO.class);
                log.info("serviceFlowCommandChangeMsgListenerConcurrently come, msg:{}, mqExchangeDTO={}", msgStr, mqExchangeDTO);
                if(Objects.isNull(mqExchangeDTO)){
                    continue;
                }
                String uniqueKey = mqExchangeDTO.getApplicationName() + "_" + mqExchangeDTO.getIp().replace(".", "");
                // 不是同一个实例不处理
                if (mqExchangeDTO.getHttpApplication()) {
                    if(!Objects.equals(uniqueKey, getProviderIPTag4HttpApplication())) {
                        continue;
                    }
                } else {
                    if(!Objects.equals(uniqueKey, getProviderIPTag4DubboApplication())) {
                        continue;
                    }
                }

                if (serviceDiscovery != null) {
                    httpHandle(mqExchangeDTO.getStatus());
                }
                dubboHandle(mqExchangeDTO.getStatus());
                if (mqExchangeDTO.getHandleMq()) {
                    mqHandle(mqExchangeDTO.getStatus());
                }
                ServiceFlowMqCallBackDTO mqCallBackDTO = AskBeanUtil.copyProperties(mqExchangeDTO, new ServiceFlowMqCallBackDTO(), false);
                mqCallBackDTO.setExecuteStatus(1);

                serviceFlowMQProducer.sendFlowBackInfo(DxyMqEnums.Topic.SERVICE_FLOW_COMMAND_MSG, SERVICE_FLOW_CALL_BACK_TAG, JSON.toJSONString(mqCallBackDTO));
            }
            return ConsumeConcurrentlyStatus.CONSUME_SUCCESS;
        };
    }



    void httpHandle(Integer status) {
        if (Objects.equals(status, ServiceStatusEnum.ON_LINE.getCode())){
            //http服务摘流
            try {
                log.warn("http service register started");
                serviceDiscovery.start();
                log.warn("http service register finished");
            } catch (Exception e) {
                log.warn("http service register exception", e);
            }
        } else if (Objects.equals(status, ServiceStatusEnum.OFF_LINE.getCode())){
            //http服务恢复流量
            try {
                log.warn("http service unregister started");
                serviceDiscovery.close();
                log.warn("http service unregister started");
            } catch (IOException e) {
                log.warn("http service unregister exception", e);
            }
        }
    }

    void dubboHandle(Integer status){
        RegistryFactory registryFactory = ExtensionLoader.getExtensionLoader(RegistryFactory.class).getAdaptiveExtension();
        if (registryFactory == null) {
            return;
        }
        ServiceRepository serviceRepository = ApplicationModel.getServiceRepository();
        if (serviceRepository == null) {
            return;
        }
        Collection<ProviderModel> providerModelList = serviceRepository.getExportedServices();
        if (CollectionUtils.isEmpty(providerModelList)) {
            return;
        }
        if(Objects.equals(status, ServiceStatusEnum.ON_LINE.getCode())){
            log.warn("Dubbo service register started");
            for (ProviderModel providerModel : providerModelList) {
                List<ProviderModel.RegisterStatedURL> statedUrls = providerModel.getStatedUrl();
                for (ProviderModel.RegisterStatedURL statedURL : statedUrls) {
                    if (!statedURL.isRegistered()) {
                        Registry registry = registryFactory.getRegistry(statedURL.getRegistryUrl());
                        registry.register(statedURL.getProviderUrl());
                        statedURL.setRegistered(true);
                    }
                }
            }
            log.warn("Dubbo service register finished");

        }
        if(Objects.equals(status, ServiceStatusEnum.OFF_LINE.getCode())) {
            log.warn("Dubbo service unregister started");
            for (ProviderModel providerModel : providerModelList) {
                List<ProviderModel.RegisterStatedURL> statedUrls = providerModel.getStatedUrl();
                for (ProviderModel.RegisterStatedURL statedURL : statedUrls) {
                    if (statedURL.isRegistered()) {
                        Registry registry = registryFactory.getRegistry(statedURL.getRegistryUrl());
                        registry.unregister(statedURL.getProviderUrl());
                        statedURL.setRegistered(false);
                    }
                }
            }
            log.warn("Dubbo service unregister finished");
        }

    }

    void mqHandle(Integer status){
        if (CollectionUtils.isEmpty(mqConsumerMap)) {
            log.warn("mqHandle skipped for mqConsumerMap is empty");
            return;
        }
        if(Objects.equals(status, ServiceStatusEnum.ON_LINE.getCode())){
            //RocketMQ Consumer 摘流
            log.warn("RocketMQ Consumer suspend started");
            mqConsumerMap.forEach((name, consumer) -> {
                DefaultMQPushConsumer defaultMQPushConsumer = (DefaultMQPushConsumer) consumer;
                if (!defaultMQPushConsumer.getConsumerGroup().equals(DxyMqEnums.ConsumeGroup.SERVICE_FLOW_COMMAND_MSG.name())) {
                    consumer.suspend();
                }
            });
            log.warn("RocketMQ Consumer suspend finished");
        }
        if(Objects.equals(status, ServiceStatusEnum.OFF_LINE.getCode())){
            //RocketMQ Consumer 恢复流量
            log.warn("RocketMQ Consumer resume started");
            mqConsumerMap.forEach((name, consumer) -> {
                DefaultMQPushConsumer defaultMQPushConsumer = (DefaultMQPushConsumer) consumer;
                if (!defaultMQPushConsumer.getConsumerGroup().equals(DxyMqEnums.ConsumeGroup.SERVICE_FLOW_COMMAND_MSG.name())) {
                    consumer.resume();
                }
            });
            log.warn("RocketMQ Consumer resume finished");
        }

    }
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
完整版:https://download.csdn.net/download/qq_27595745/89522468 【课程大纲】 1-1 什么是java 1-2 认识java语言 1-3 java平台的体系结构 1-4 java SE环境安装和配置 2-1 java程序简介 2-2 计算机中的程序 2-3 java程序 2-4 java类库组织结构和文档 2-5 java虚拟机简介 2-6 java的垃圾回收器 2-7 java上机练习 3-1 java语言基础入门 3-2 数据的分类 3-3 标识符、关键字和常量 3-4 运算符 3-5 表达式 3-6 顺序结构和选择结构 3-7 循环语句 3-8 跳转语句 3-9 MyEclipse工具介绍 3-10 java基础知识章节练习 4-1 一维数组 4-2 数组应用 4-3 多维数组 4-4 排序算法 4-5 增强for循环 4-6 数组和排序算法章节练习 5-0 抽象和封装 5-1 面向过程的设计思想 5-2 面向对象的设计思想 5-3 抽象 5-4 封装 5-5 属性 5-6 方法的定义 5-7 this关键字 5-8 javaBean 5-9 包 package 5-10 抽象和封装章节练习 6-0 继承和多态 6-1 继承 6-2 object类 6-3 多态 6-4 访问修饰符 6-5 static修饰符 6-6 final修饰符 6-7 abstract修饰符 6-8 接口 6-9 继承和多态 章节练习 7-1 面向对象的分析与设计简介 7-2 对象模型建立 7-3 类之间的关系 7-4 软件的可维护与复用设计原则 7-5 面向对象的设计与分析 章节练习 8-1 内部类与包装器 8-2 对象包装器 8-3 装箱和拆箱 8-4 练习题 9-1 常用类介绍 9-2 StringBuffer和String Builder类 9-3 Rintime类的使用 9-4 日期类简介 9-5 java程序国际化的实现 9-6 Random类和Math类 9-7 枚举 9-8 练习题 10-1 java异常处理 10-2 认识异常 10-3 使用try和catch捕获异常 10-4 使用throw和throws引发异常 10-5 finally关键字 10-6 getMessage和printStackTrace方法 10-7 异常分类 10-8 自定义异常类 10-9 练习题 11-1 Java集合框架和泛型机制 11-2 Collection接口 11-3 Set接口实现类 11-4 List接口实现类 11-5 Map接口 11-6 Collections类 11-7 泛型概述 11-8 练习题 12-1 多线程 12-2 线程的生命周期 12-3 线程的调度和优先级 12-4 线程的同步 12-5 集合类的同步问题 12-6 用Timer类调度任务 12-7 练习题 13-1 Java IO 13-2 Java IO原理 13-3 流类的结构 13-4 文件流 13-5 缓冲流 13-6 转换流 13-7 数据流 13-8 打印流 13-9 对象流 13-10 随机存取文件流 13-11 zip文件流 13-12 练习题 14-1 图形用户界面设计 14-2 事件处理机制 14-3 AWT常用组件 14-4 swing简介 14-5 可视化开发swing组件 14-6 声音的播放和处理 14-7 2D图形的绘制 14-8 练习题 15-1 反射 15-2 使用Java反射机制 15-3 反射与动态代理 15-4 练习题 16-1 Java标注 16-2 JDK内置的基本标注类型 16-3 自定义标注类型 16-4 对标注进行标注 16-5 利用反射获取标注信息 16-6 练习题 17-1 顶目实战1-单机版五子棋游戏 17-2 总体设计 17-3 代码实现 17-4 程序的运行与发布 17-5 手动生成可执行JAR文件 17-6 练习题 18-1 Java数据库编程 18-2 JDBC类和接口 18-3 JDBC操作SQL 18-4 JDBC基本示例 18-5 JDBC应用示例 18-6 练习题 19-1 。。。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值