Java报告推送失败补偿机制;钉钉群通知消息核心代码

简介: Java报告推送失败补偿机制,超过次数后使用钉钉通知开发自动补偿实现:要求方法调用的过程中,失败的时候,系统有办法进行自动重试,重试达到一定次数后,钉钉通知开发。实现设计:注解,反射,定时任务

Java报告推送失败补偿机制,超过次数后使用钉钉通知开发

自动补偿实现:

要求方法调用的过程中,失败的时候,系统有办法进行自动重试,重试达到一定次数后,钉钉通知开发。

实现设计:注解,反射,定时任务

自动一般涉及到定时任务。

钉钉群通知消息核心代码

钉钉群通知消息核心代码

application-local.yml

自动一般涉及到定时任务。
钉钉群通知消息核心代码
钉钉群通知消息核心代码
在这里插入图片描述
application-local.yml
messa

钉钉订单监听消息

@Slf4j
@Component
@SuppressWarnings("PMD")
public class DingOrderDistributeMessageListener implements MessageListener {
    @Autowired
    private MessageConfig messageConfig;
    @Override
    public Action consume(Message message, ConsumeContext consumeContext) {
        try {
            boolean notifyOpen = messageConfig.getNotifyDistributeOpen();
            if (!notifyOpen) {
                log.info("订单分配通知开关已关闭/未配置 messageOpen:{}", notifyOpen);
                return Action.CommitMessage;
            }
            // 订单钉钉通知群url
            String dingWebHookUrl = messageConfig.getDingDistributeUrl();
            // 密钥
            String orderSecret = messageConfig.getDingDistributeSecret();
            // 钉钉消息体
            String messageBody = messageConfig.getDingDistributeMessageBody();
            if (StringUtils.isEmpty(messageBody) || StringUtils.isEmpty(dingWebHookUrl)
                    || StringUtils.isEmpty(orderSecret)) {
                log.info("发送订单分配钉钉消息, 消息配置项未配置");
                return Action.CommitMessage;
            }
            // 请求体
            String requestBody = new String(message.getBody(), StandardCharsets.UTF_8);
            // 报告消息体
            DingOrderDistributeMessage orderMessage = JSONObject.parseObject(requestBody,
                    DingOrderDistributeMessage.class);
            // 请求消息
            String sendMessage = String.format(messageBody, "\n", orderMessage.getCode() + "\n",
                    orderMessage.getDeviceType() + "\n", orderMessage.getAppVersion() + "\n",
                    orderMessage.getGoodsName() + "\n", orderMessage.getPayAmount() + "\n",
                    orderMessage.getPayType() + "\n", orderMessage.getPlaceTime() + "\n",
                    orderMessage.getDistributeTime() + "\n", orderMessage.getAdminUserName());
            List<String> mobileList = new ArrayList<>();
            if (StringUtils.isNotEmpty(orderMessage.getAdminUserPhone())) {
                mobileList.add(orderMessage.getAdminUserPhone());
            }
            String dingMessage = DingMessageUtil.generateMessageBody(sendMessage, false, mobileList);
            // 发送钉钉消息
            DingMessageUtil.sendMessage(orderSecret, dingMessage, dingWebHookUrl);
            return Action.CommitMessage;
        } catch (Exception e) {
            log.error("发送报告钉钉消息出现异常:{}", e);
            return Action.ReconsumeLater;
        }
    }
}

MessageConfig 是个实体类

@ConfigurationProperties(prefix = "message.config")
@Component
@Data
public class MessageConfig {
    @ApiModelProperty("钉钉报告机密钥")
    private String dingReportSecret;
    @ApiModelProperty("钉钉报告地址")
    private String dingReportUrl;
    @ApiModelProperty("钉钉报告密钥")
    private String dingOrderSecret;
    @ApiModelProperty("钉钉订单消息图片地址")
    private String dingOrderPicUrl;
    @ApiModelProperty("钉钉订单消息头部")
    private String dingOrderTitle;
    @ApiModelProperty("订单钉钉报告地址")
    private String dingOrderUrl;
    @ApiModelProperty("订单路由地址")
    private String orderRouteUrl;
    @ApiModelProperty("钉钉订单消息体")
    private String dingOrderMessageBody;
    @ApiModelProperty("钉钉报告消息体")
    private String dingReportMessageBody;
    @ApiModelProperty("消息开关")
    private Boolean notifyOpen;
    @ApiModelProperty("钉钉日报")
    private String dingDailyReportMessageBody;
    @ApiModelProperty("订单分配通知开关")
    private Boolean notifyDistributeOpen;
    @ApiModelProperty("分配订单通知密钥")
    private String dingDistributeSecret;
    @ApiModelProperty("分配通知地址")
    private String dingDistributeUrl;
    @ApiModelProperty("订单分配消息主体")
    private String dingDistributeMessageBody;
}

微服务实际去调用这钉钉消息

    public void sendOrderDistributeMessage(Order order, String adminUserName, String adminUserPhone,
                                           Date distributeTime) {
        log.info("发送订单分配钉钉消息 order:{}, adminUserName:{}, adminUserPhone:{}, distributeTime:{}", order,
                adminUserName, adminUserPhone, distributeTime);
        // 适配部分下单在支付通知才回填支付方式
        Order dbOrder = orderService.getByOrderNo(order.getOrderNo());
        OrderItemGoods goods = orderService.getOrderItemGoodsByOrderNo(order.getOrderNo());
        DingOrderDistributeMessage orderMessage = new DingOrderDistributeMessage();
        orderMessage.setGoodsName(goods.getGoodsName());
        String suffix = "元";
        String payType = dbOrder.getPayTypeChannel().getName();
        PayTypeChannel payTypeChannel = dbOrder.getPayTypeChannel();
        switch (payTypeChannel) {
            case APPLE_IN:
                payType = "苹果支付";
                break;
            case WX_MWEB:
            case WX_APP:
            case WX_JSAPI:
            case WX_MICROPAY:
            case WX_NATIVE:
                payType = "微信";
                break;
            case SSCM:
                suffix = "上上贝";
                payType = "上上贝";
                break;
            case ALIPAY_PC:
            case ALIPAY_QR:
            case ALIPAY_WAP:
            case ALIPAY_MOBILE:
                payType = "支付宝";
                break;
            default:
                break;
        }
        orderMessage.setPayType(payType);
        orderMessage.setPayAmount(AmountUtils.changeFen2Yuan(order.getPayAmount()) + suffix);
        orderMessage.setPlaceTime(DateUtils.formatYYYYMMDDHHMMSS(order.getPlaceTime()));
        User user = userRemoteService.getById(order.getPlaceUserId()).getData();
        orderMessage.setUserName(user.getUsername());
        orderMessage.setCode(user.getCode());
        orderMessage.setOrderNo(order.getOrderNo());
        List<UserChannel> userChannelList = userRemoteService.getUserChannelList().getData();
        Map<String, String> userChannelMap = userChannelList.stream()
                .collect(Collectors.toMap(UserChannel::getChannelCode, UserChannel::getChannelName));
        orderMessage.setDeviceType(userChannelMap.get(user.getChannelCode()));
        orderMessage.setAppVersion(user.getVersionName());
        orderMessage.setDistributeTime(DateUtils.formatYYYYMMDDHHMMSS(distributeTime));
        orderMessage.setAdminUserName(adminUserName);
        orderMessage.setAdminUserPhone(adminUserPhone);
        mqSend.sendDingOrderDistributeMessage(orderMessage);
    }
    /**
     *  发送钉钉订单分配消息
     * @param orderMessage
     */
    void sendDingOrderDistributeMessage(DingOrderDistributeMessage orderMessage);
    @Override
    public void sendDingOrderDistributeMessage(DingOrderDistributeMessage orderMessage) {
        String message = JSON.toJSONString(orderMessage);
        LOGGER.info("sendPayOrderMessage take success send message to mq message = {}", message);
        producerUtil.sendAsyncMsg(orderConfig.getOrderTopic(), orderConfig.getOrderTopicDistributeDingTag(),
                message, orderMessage.getOrderNo());
    }
相关文章
|
11天前
|
XML 安全 Java
Java反射机制:解锁代码的无限可能
Java 反射(Reflection)是Java 的特征之一,它允许程序在运行时动态地访问和操作类的信息,包括类的属性、方法和构造函数。 反射机制能够使程序具备更大的灵活性和扩展性
19 5
Java反射机制:解锁代码的无限可能
|
5天前
|
安全 IDE Java
Java反射Reflect机制详解
Java反射(Reflection)机制是Java语言的重要特性之一,允许程序在运行时动态地获取类的信息,并对类进行操作,如创建实例、调用方法、访问字段等。反射机制极大地提高了Java程序的灵活性和动态性,但也带来了性能和安全方面的挑战。本文将详细介绍Java反射机制的基本概念、常用操作、应用场景以及其优缺点。 ## 基本概念 ### 什么是反射 反射是一种在程序运行时动态获取类的信息,并对类进行操作的机制。通过反射,程序可以在运行时获得类的字段、方法、构造函数等信息,并可以动态调用方法、创建实例和访问字段。 ### 反射的核心类 Java反射机制主要由以下几个类和接口组成,这些类
15 2
|
7天前
|
jenkins Java 测试技术
如何使用 Jenkins 自动发布 Java 代码,通过一个电商公司后端服务的实际案例详细说明
本文介绍了如何使用 Jenkins 自动发布 Java 代码,通过一个电商公司后端服务的实际案例,详细说明了从 Jenkins 安装配置到自动构建、测试和部署的全流程。文中还提供了一个 Jenkinsfile 示例,并分享了实践经验,强调了版本控制、自动化测试等关键点的重要性。
33 3
|
12天前
|
存储 安全 Java
系统安全架构的深度解析与实践:Java代码实现
【11月更文挑战第1天】系统安全架构是保护信息系统免受各种威胁和攻击的关键。作为系统架构师,设计一套完善的系统安全架构不仅需要对各种安全威胁有深入理解,还需要熟练掌握各种安全技术和工具。
41 10
|
8天前
|
分布式计算 Java MaxCompute
ODPS MR节点跑graph连通分量计算代码报错java heap space如何解决
任务启动命令:jar -resources odps-graph-connect-family-2.0-SNAPSHOT.jar -classpath ./odps-graph-connect-family-2.0-SNAPSHOT.jar ConnectFamily 若是设置参数该如何设置
|
6天前
|
Java
Java代码解释++i和i++的五个主要区别
本文介绍了前缀递增(++i)和后缀递增(i++)的区别。两者在独立语句中无差异,但在赋值表达式中,i++ 返回原值,++i 返回新值;在复杂表达式中计算顺序不同;在循环中虽结果相同但使用方式有别。最后通过 `Counter` 类模拟了两者的内部实现原理。
Java代码解释++i和i++的五个主要区别
|
10天前
|
存储 缓存 安全
🌟Java零基础:深入解析Java序列化机制
【10月更文挑战第20天】本文收录于「滚雪球学Java」专栏,专业攻坚指数级提升,希望能够助你一臂之力,帮你早日登顶实现财富自由🚀;同时,欢迎大家关注&&收藏&&订阅!持续更新中,up!up!up!!
18 3
|
10天前
|
安全 Java UED
深入理解Java中的异常处理机制
【10月更文挑战第25天】在编程世界中,错误和意外是不可避免的。Java作为一种广泛使用的编程语言,其异常处理机制是确保程序健壮性和可靠性的关键。本文通过浅显易懂的语言和实际示例,引导读者了解Java异常处理的基本概念、分类以及如何有效地使用try-catch-finally语句来处理异常情况。我们将从一个简单的例子开始,逐步深入到异常处理的最佳实践,旨在帮助初学者和有经验的开发者更好地掌握这一重要技能。
16 2
|
12天前
|
Java 数据库连接 开发者
Java中的异常处理机制####
本文深入探讨了Java语言中异常处理的核心概念,通过实例解析了try-catch语句的工作原理,并讨论了finally块和throws关键字的使用场景。我们将了解如何在Java程序中有效地管理错误,提高代码的健壮性和可维护性。 ####
|
14天前
|
安全 Java 程序员
深入浅出Java中的异常处理机制
【10月更文挑战第20天】本文将带你一探Java的异常处理世界,通过浅显易懂的语言和生动的比喻,让你在轻松阅读中掌握Java异常处理的核心概念。我们将一起学习如何优雅地处理代码中不可预见的错误,确保程序的健壮性和稳定性。准备好了吗?让我们一起踏上这段旅程吧!
22 6