/mq-idempotent

mq幂等框架开箱即用(支持RocketMQ、RabbitMQ、Kafka)mq idempotent framework (support RocketMQ, RabbitMQ, Kafka)

Primary LanguageJavaApache License 2.0Apache-2.0

img.png

mq-Idempotent

通用的mq消息幂等去重框架,开箱即用,支持主流所有mq
General mq message idempotent deduplication framework, out of the box, supports all mainstream mq

English

  1. 需要项目是Springboot项目
  2. 原理很简单基于Spring AOP + Redis做的
  3. 现在暂时只支持aliyun ons-client、RocketMQ Client、Spring Boot RabbitMQ.不支持的可以自己实现MessageConverter去定义,非常轻量方便
  4. 由于目前源代码非常轻量,所以不引用jar直接copy源代码到项目中使用也是可以的

由于本地没有rabbitmq、Kafka等,但是需要支持也非常方便,只需要实现MessageConverter 接口,后面有详细说明

分支说明

  • master: 最新稳定代码
  • develop: 正在开发中的代码

使用

必要条件

  1. redis redis使用的是redission
  2. springboot
  3. jdk8+

1. 添加依赖

rocketmq-client
  • maven
<dependency>
  <groupId>io.github.weihubeats</groupId>
  <artifactId>mq-idempotent-spring-boot-starter</artifactId>
  <version>1.1.6</version>
</dependency>
  • gradle
implementation 'io.github.weihubeats:mq-idempotent-spring-boot-starter:1.1.6'
aliyun ons-client
<dependency>
<groupId>io.github.weihubeats</groupId>
<artifactId>mq-idempotent-spring-boot-starter-aliyun-rocketmq</artifactId>
<version>1.1.6</version>
</dependency>
  • gradle
implementation 'io.github.weihubeats:mq-idempotent-spring-boot-starter-aliyun-rocketmq:1.1.6'

2. 在需要幂等的方法上添加注解

@Idempotent
    @Idempotent
    public void testConsumer(Message message) {
        String msg = new String(message.getBody());
        System.out.println("消息id " + message.getMsgID());
        System.out.println("消息key " + message.getKey());
        System.out.println("消费成功, msg " + msg);
    }

注意事项,由于是基于AOP实现的,所以需要注意AOP失效场景导致的问题 添加住的方法的参数必须是 Message,方法返回值必须是void 或者 boolean 因为aop要处理重复消费直接返回 true 或者 null 更详细实用方式请参考模块 wh-mq-Idempotent-samples

例子参考

使用例子请参考 wh-mq-Idempotent-samples 模块

自定义配置

  • 基于配置文件自动配置
idempotent:
  redisKey: "mq:unique:"
  redisValue : "ss"
  tryLockTime: 2
  redisTimeOut: 3
  • 基于JavaBean
    @Bean
    public IdempotentConfig idempotentConfig() {
        IdempotentConfig idempotentConfig = new IdempotentConfig();
        // 去重 redis key名 默认 mq::unique::
        idempotentConfig.setRedisKey(redisKey);
        // 去重redis value 默认 s
        idempotentConfig.setRedisValue(redisValue);
        // 去重redis尝试获取锁等待时间 默认1s 单位秒
        idempotentConfig.setTryLockTime(tryLockTime);
        // 设置处理成功消息存放redis时间 默认 3天
        idempotentConfig.setRedisTimeOut(redisTimeOut);
        return idempotentConfig;

    }

自定义mq去重

引入依赖

<dependency>
<groupId>io.github.weihubeats</groupId>
<artifactId>mq-idempotent-core</artifactId>
<version>1.1.6</version>
</dependency>

实现接口 MessageConverter.java 例如 支持rocketMQ

@Component
public class RocketMQMessageConverter implements MessageConverter<MessageExt> {


    @Override
    public String getUniqueKey(MessageExt messageExt, String field, Method method, Object[] args) {
        return StringUtils.isNoneBlank(messageExt.getKeys()) ? messageExt.getKeys() :messageExt.getMsgId();
    }
}

模块说明

  • wh-core 核心实现
  • wh-mq-rocketmq rocketmq幂等核心实现
  • wh-mq-aliyun-rocketmq 阿里云client幂等核心实现
  • wh-mq-Idempotent-samples 使用例子

设计思路

img.png

目前redis实现的大致思路如下

  1. 消费者获取到MQ消费信息
  2. 基于配置的MQ消息中的业务唯一键去reids(Mysql) 判断是否已消费
  3. 如果没有消费则加锁防止并发问题,加锁成功则消费,失败则返回消费失败让MQ重新投递,因为这里防止第一个抢到锁的线程执行失败,所以不能直接返回成功,需要后面的任务重新投递到MQ重新消费
  4. 执行业务代码成功后写入消费成功(写入redis或更新Mysql)
  5. 释放锁

发布版本

  • 1.0.4:支持阿里云RocketMQ Client
  • 1.0.5:新增支持开源RocketMQ Client,新增自动化配置 IdempotentConfig.java
  • 1.0.6:优化重复消费重复投递问题,优化代码结构
  • 1.0.7:优化redis key 过期时间
  • 1.0.8:优化AOP实现方式
  • 1.1.9:添加对Spring Boot RabbitMQ支持
  • 1.1.0-Release:优化消息存储细节,自定义消息转换策略添加methdo、args
  • 1.1.1-Release: 修复分布式锁异常问题
  • 1.1.2-Release: 修复分布式锁异常问题
  • 1.1.4-Release: 添加事务幂等处理,添加飞书报警,添加自定义spi异常报警

未来版本

  1. 支持kafka
  2. 支持Mysql去重
  3. 网络两将军问题处理

正在开发中。。。。。。

期待你的加入

作者微信: 作者微信

鸣谢

感谢 JetBrains 提供的免费开源 License:

图片引用自lets-mica