2

什么是资损

资损通常来讲是指支付场景下的资金损失,这里可以从两个维度看 :

  • 用户角度:多扣用户款导致用户资金损失,此问题一般需要通过客服等渠道反馈,可以把多的钱退给用户,但是很大程度上损失了用户体验;
  • 公司角度:主要是多出金、多出货、多充值等情况,一般这种损失很难追回,这就是实打实的产生了资产损失。
    截屏2021-07-04 下午2.41.39.png

比如对于一个电商业务可能涉及到上图各种业务,业务之间都存在下发、回调、消息传递等各种逻辑或者状态同步,如果其中由于一些交互操作丢失导致库存异常、资金结算换算异常、单据流程未结束异常、逻辑触发重复请求,并发控制处理不当等等,最终导致资产或资金的损失。

对于情况预防手段除了事前的一些列严格测试,事后分析的优化补救,也可以通过事中进行监控,得物在这方面就是通过自研的DCheck进行防控。

初识DCheck

此系统由“交易&稳定性”团队主导的,主要是希望及时的发现数据的问题,保障数据稳定的运行,尤其是涉及到资损的场景,为了做到实时有效的监控,在此背景下搭建了准实时核对系统DCheck,此平台基于Mysql的binglog监控,以及MQ的订阅的信息流技术手段,通过配置触发条件、规则和任务运行以及告警,来确保各个业务上下游业务间的状态一致性,进出扣款金额计算后的准确性。

深入DCheck

功能层级

截屏2021-07-04 下午2.42.10.png

架构逻辑

未命名 (2) (1).png

概念

  • 主题:逻辑数据库 或 MQ订阅
  • 事件:Update/ Insert类型操作 或 MQ自定义消
  • 子事件:事件所返回的数据的第一层过滤
  • 脚本池:有filter和check两个继承方法,filter处理二层数据过滤,check业务上下游逻辑判断
  • 规则:触发和检测核心执行部分

DCheck

对于此系统的功能和使用简单演示如下:

核对配置

主题管理

image - 2021-07-04T142758.030.png

  • TOPIC:库名
  • 主题编码:表名
  • 主题名称:中文的表名
  • MQ实例地址: 对于binlog填*

事件管理

image - 2021-07-04T142913.457.png
对于binlog而言,分成了INSERT和UPDATE两种,根据上主题配置自动生成无需创建。

子事件管理

image - 2021-07-04T142938.700.png
清洗的数据的过滤层,此处在规则配置中,返回‘TRUE’的才会进入下一层,如:
if( obj.status.toInteger() == 10000 && (obj.type.toInteger() ==101 || obj.type.toInteger() ==301) ) return 'TRUE';
如果想所有都返回,就直接 return 'TRUE'。

脚本池管理

所有被执行脚本使用的是groovy进行编写的,主要是BaseScript实现filter和check两个方法,内部可以参考脚本库:

  • Filter经过事件过滤后需要脚本再进行二次过滤的操作,主要是一些不能简单从donCleanData进行判断,需要进行正向逆向获取其他数据,或是一些早为复杂的逻辑。
  • Check验证逻辑处理方法,主要是就是进行上下游同步状态验证,复杂结果计算(尤其是进出帐扣款等方面)比较。
    代码例子:

    import com.alibaba.fastjson.JSON; 
    import com.alibaba.fastjson.JSONObject; 
    import groovy.util.logging.Slf4j; 
    import org.springframework.stereotype.Service; 
     
    import javax.annotation.Resource; 
     
    /** 
     * DCheck:冷静期内-平台客服取消订单-退买家支付金额 
     */ 
    @Slf4j 
    @Service 
    class CheckRefundPayForLess30min implements BaseScript { 
     
      @Resource 
      private OrderDevOpsApi orderDevOpsApi; 
     
      @Resource
      DCheckApi dCheckApi; 
     
      @Resource 
      private PayServiceFeignClient payServiceFeignClient 
     
      String logTag = "TAG_CheckCrossAndOverSeaRefundPayForLess30min:{}" 
     
      // 1.关单时间-支付时间<30分钟 
      @Override 
      boolean filter(JSONObject doneCleanData) { 
          // 查询支付时间 
          String unionId = doneCleanData.getString("order_no"); 
          String payTime = getOrderData(unionId,"payTime", DevOpsSceneEnum.FORWARD_PAY); 
          long modifyTime = doneCleanData.getDate("modify_time").getTime();  
          long diffTime = modifyTime - Long.valueOf(payTime) 
          if (diffTime < 30 * 60 * 1000){ 
              return true; 
          } 
          log.info(logTag,"===>有符合数据进入Check") 
          return false; 
      } 
     
      @Override 
      String check(JSONObject doneCleanData) { 
          String subOrderNo = doneCleanData.getString("sub_order_no"); 
          Result<List<String>> listResult = dCheckApi.queryPayNoBySubOrderNo(subOrderNo); 
          if(listResult == null || listResult.getData() == null) { 
              return "根据正向查询接口通过子订单号查询支付流水号数据为空"; 
          } 
          if(listResult.getData().size() > 1){ 
              return "根据正向查询接口通过子订单号查询支付流水号多条数据,请查看是否需要优化逻辑"; 
          } 
          
          String outPayNo = listResult.getData().get(0); 
          RefundQueryRequest refundQueryRequest = new RefundQueryRequest(); 
          refundQueryRequest.setPayLogNum(outPayNo); 
          Result<List<RefundBillDTO>>  resp = payServiceFeignClient.queryRefundsByPayLogNum(refundQueryRequest); 
          // 判断支付查询数据是否为空,如果为空直接报数据错误,以及是否查询到了多条数据 
          if (resp == null || resp.getData() == null) { 
              return "上游数据为空:支付退款查询(根据支付流水号)"; 
          } 
          else if (resp.getData().size() != 1) { 
              return "上游数据为多条请确认逻辑:支付退款查询(根据支付流水号)"; 
          } 
     
          // 检查点逻辑判断1: 状态为打款成功 
          if (resp.getData().get(0).getStatus() !=2 ){ 
              return "校验支付打款状态非2"; 
          } 
     
          // 逻辑校验点2:交易退款和RPC查询的金额一致,否则告警 
          if (resp.getData().get(0).getAmount() != doneCleanData.getLong("amount")) { 
              return "校验交易退款金额和支付打款金额不一致"; 
          } 
          return "SUCCESS"; 
      } 
     
      // 数据库查询对应字段值 
      String getOrderData(String unionNo,String key,DevOpsSceneEnum devOpsSceneEnum){ 
          // 内部方法省略.... 
          return value; 
      } 
    } 

规则配置

image - 2021-07-04T143059.264.png
规则配置为上述所有基础配置的组合以及真正的运行核心,主要两大块,基础信息和降级策略:
基础信息:子事件(支持搜索和多个选中)+ 脚本类(搜索选择) = 触发和执行逻辑,其他为辅助配置信息按照各自域和需求配置
降级策略

  • 采样百分比:线上流量采样百分比,前期测试或者对业务有很大影响的需要控制流量,不能为100%。
  • 首次延迟时间:触发执行延迟时间,在业务流程,数据同步有些会有些延迟,为了避免由于延迟导致的状态不同步问题,建议设置一定的延迟比例,一般10秒左右。
  • 最大超时时间和生效时间:规则有效时间配置。

工具使用

核对异常

针对check异常的数据,一般首先会发到配置的告警飞书群和配置的个人,点击即可跳转到此页面,主要看错误的具体数据,经确认后是脚本或者部分数据问题的,优化后“重发”可以标记为处理,如果是确定是问题的,则定位“资损”问题。

Mock

由于本地脚本调用一些RPC接口,目前还没有好的办法能在本地进行debug,所以就需要通过先配置规则后,使用mock进行调试,主要用到的是规则调试,选择指定的规则,搜索或者创造符合dcheck场景的json格式的请求参数,提交请求查看响应结果即可。

这里有个问题,因为dcheck内部逻辑对一些系统异常脚本做了统一处理,很多时候没法看到具体的原因,就是失败或者逻辑外通过,这就需要需要在脚本中多加些打印日志,然后通过日志平台去查看具体逻辑问题。

一些使用技巧

规则配置技巧

  • 规则中事件是可以多选的,所以对于事件不同,但check相同或者相似的脚本处理逻辑可以归并,减少规则维护量。
  • 触发数据可以在事件进行配置的,尽量少用脚本filter处理,代码一般处理需要非触发数据外逻辑。

Groovy的闭包使用

脚本数据处理有会有很多list,key-value的处理,可以充分利用groovy的闭包特性,大大简化java语言的复杂处理逻辑。

举例:如果如下数据格式的结果返回objectList:

[{id=10086, refundNo=RE10086, orderNo=100888, userId=15206, bizType=110, payTool=0, payStatus=404,amount=100, feature=, isDel=0, createTime=2021-05-11 21:39:34.000, modifyTime=2021-05-11 21:39:34.000, moneyFrom=1, currency=, countryCode=},{id=10087, refundNo=RE10087, orderNo=100999, userId=15206, bizType=202, payTool=0, payStatus=404, amount=400, feature=, isDel=0, createTime=2021-05-11 21:39:34.000, modifyTime=2021-05-11 21:39:34.000, moneyFrom=1, countryCode=}] 
  1. filter进行条件判断,可以使用any
    def filterResult = objectList.any{it.bizType in [110,119] && it.payStatus == 404};
    return filterResult
  2. check获取某个符合条件的值,可以使用find
    def mount = objectList.find{ it.type=5}.amount

更多groovy特性可以参考文章:https://www.jianshu.com/p/5d3...

一些平台不足

  1. 没有比较方便的调试方式

    目前本地脚本库没有可以运行调试的环境,虽然有Mock工具,但也是需要先配置事件,上传脚本配置后才能进行调试,同时脚本逻辑问题也需要去日志不断改脚本的加日志的方式去查看,另外最后如果在测试环境了进行了调试通过,还要去线环境再重复配置一遍流程。

    建议:线上在子事件、脚本、规则增加的页面就增加一个Debug按钮,可以直接通过给定参数或者抓取符合条件的一条数据进行调试给出结果,最好也能给出这部分日志。

  2. 脚本池Filter和check方法放在一起会有很多冗余

    在开发脚本和配置中发现其实好多filter逻辑相同,抑或check逻辑相同,但由于脚本中放在一起,就会产生交叉的逻辑编写,无法做到有效的公共剥离。

    建议:Filter 和 check 拆分,并且作为公共的池,规则中单独配置,更可以选择模式为引用或者导入,导入的支持主要方便逻辑大部分相同,个别参数不相同下快速修改配置规则上线。

  3. 真的问题出现时没有熔断机制

    虽然目前整个平台的实际还待观察,但其中后续如果平台效果是很好的,真的发生大批量数据问题或者资金损失问题,平台的机制也只是发生问题的警告,然后技术介入,本质上还是后置的处理的方式,如一开始讲的这种后置的处理,并不能达到及时止损的目的。

    建议:后续在平台更加准确时,应该考虑与关键域熔断联动,及时制止损失的产生。

  4. 平台应该增上下线开关

    目前规则的执行不执行,只能通过编辑控制流量0,或是执行时间来关闭,也没有批量的,操作上有些不便。

    建议:增加上下线开关,增加批量操作如:流量大小,开关,警告人等

  5. 平台可以考虑增加流量的动态判断机制

    由于好多check点是通过各个域的接口进行操作,在流量大的时候,尤其是关键的业务,可能对业务产品很大影响,或者在某接口兼容不好,系统异常导致大量非常规一场报错的时候,各方人会炸掉。

    建议:规则分等级,核心业务可以做高级配置,遇到上述问题触发自动调节降流量,对于问题已恢复可自动增加流量。

文|大奇

关注得物技术,携手走向技术的云端


得物技术
854 声望1.5k 粉丝