动态多数据源使用seata实现分布式事务

一、背景

目前公司很多应用因为历史原因,一个应用访问多个数据库进行插入和更新操作,这就可能产生数据一致性问题,同时应用如果跨服务的调用也可能会产生事务问题。

目前应用是采用dynamic-datasource-spring-boot-starter做多数据源控制的。而seata是一款开源的分布式事务框架。我们了解到dynamic-datasource-spring-boot-starter的新版本已经支持基于seata的分布式事务了,而官网的例子基本都是标准的单数据源的整合,下面我们分别对dynamic-datasource-spring-boot-starter、seata以及它们的整合的进行功能使用实践。

二、dynamic-datasource

dynamic-datasource-spring-boot-starter 是一个基于springboot的快速集成多数据源的启动器。相关的特性如下,更多的信息可以查看github官网。我们这里主要用到了它对seata数据源的支持和动态切换数据源的特性。

官网地址:https://github.com/baomidou/dynamic-datasource-spring-boot-starter

  1. 支持 数据源分组 ,适用于多种场景 纯粹多库 读写分离 一主多从混合模式。

  2. 支持数据库敏感配置信息加密 ENC()。

  3. 支持每个数据库独立初始化表结构schema和数据库database。

  4. 支持无数据源启动,支持懒加载数据源(需要的时候再创建连接)。

  5. 支持自定义注解 ,需继承DS(3.2.0+)。

  6. 提供并简化对Druid,HikariCp,BeeCp,Dbcp2的快速集成。

  7. 提供对Mybatis-Plus,Quartz,ShardingJdbc,P6sy,Jndi等组件的集成方案。

  8. 提供自定义数据源来源 方案(如全从数据库加载)。

  9. 提供项目启动后动态增加移除数据源方案。

  10. 提供Mybatis环境下的 纯读写分离方案。

  11. 提供使用 spel动态参数 解析数据源方案。内置spel,session,header,支持自定义。

  12. 支持多层数据源嵌套切换 。(ServiceA >>> ServiceB >>> ServiceC)。

  13. 提供基于seata的分布式事务方案。

  14. 提供本地多数据源事务方案。

三、seata介绍

Seata 是一款开源的分布式事务解决方案,致力于提供高性能和简单易用的分布式事务服务。Seata主打AT模式。AT模式的机制如下:

  1. 一阶段:业务数据和回滚日志记录在同一个本地事务中提交,释放本地锁和连接资源。

  2. 二阶段:提交异步化,非常快速地完成。回滚通过一阶段的回滚日志进行反向补偿。

四、动手实践

在实际使用上面的框架过程中,会遇到各式各样的场景,遇到各种问题。实践出真知,下面我们模拟简单的业务实际动手来检测下他们的功能吧!

4.1 业务流程图

image

如上图所示,我们模拟一个下单的业务,主要的业务流程:

  1. postman使用http请求下单服务

  2. 下单服务访问订单库产生下单记录、访问信用分库校验和扣减信用分

  3. 访问库存库,扣减库存

  4. 如果下单成功,最终会返回true

4.2 准备环境

  1. 数据库:seata_storage(库存数据库)、seata_order(订单数据库)、seata_credit(信用分数据库)、seata(seata数据库)

  2. 应用:order-service(下单服务)、storage-service(库存服务)、seata tc(分布式事务协调器)

  3. 配置中心: Nacos

  4. seata服务端:本次实践的seata版本为v1.4.2

4.2.1 seata的安装

具体的安装步骤见官网:http://seata.io/zh-cn/docs/ops/deploy-guide-beginner.html

  1. 下载tc包,https://github.com/seata/seata/releases/tag/v1.4.2

  2. tc包解压后,修改register.conf,进行nacos地址,名称空间等配置。

  3. nacos配置项,可以将通过脚本:https://github.com/seata/seata/tree/1.4.2/script/config-center/nacos自动上传修改;也可以自行在nacos上手动添加配置项;该配置项主要为seata集群数据库地址和一些开关等;

  4. 创建seata数据库,sql脚本见:https://github.com/seata/seata/tree/1.4.2/script/server/db

  5. 启动seata服务端,使用解压包中的seataserver.sh启动,可在nacos查看启动的seata集群

踩坑:nacos如果使用带有权限的版本,密码不要带有特殊字符,否则在启动时会一直报403错误,因为从seata的服务端request到nacos时,密码特殊字符被转义传递从而导致错误。

4.2.2 业务应用的配置订单应用(接入了两个数据库):

##########服务注册
spring.cloud.nacos.discovery.server-addr=localhost:8848
spring.cloud.nacos.discovery.username=nacos_test
spring.cloud.nacos.discovery.password=nacos_test 
##########seata相关
seata.enabled=true 
seata.tx-service-group=my_test_tx_group
seata.enable-auto-data-source-proxy=false
seata.config.type=nacos 
seata.config.nacos.data-id=seataServer.properties 
seata.config.nacos.server-addr=localhost:8848
seata.config.nacos.application=seata-server 
seata.config.nacos.namespace=7322f40d-74de-4679-ad4c-c3dff499cd98
seata.config.nacos.group=SEATA_GROUP 
seata.config.nacos.username=seata 
seata.config.nacos.password=seata 
seata.registry.type=nacos 
seata.registry.nacos.server-addr=localhost:8848
seata.registry.nacos.namespace=7322f40d-74de-4679-ad4c-c3dff499cd98 
seata.registry.nacos.group=SEATA_GROUP 
seata.registry.nacos.username=seata 
seata.registry.nacos.password=seata 
logging.level.io.seata = debug
########数据源1(主数据源,订单库) 
spring.datasource.dynamic.primary=master
spring.datasource.dynamic.seata=true
spring.datasource.dynamic.datasource.master.driver-class-name=com.mysql.jdbc.Driver
spring.datasource.dynamic.datasource.master.url=jdbc:mysql://localhost:3306/seata_order?useUnicode=true&characterEncoding=utf-8&useSSL=false
spring.datasource.dynamic.datasource.master.username=seata
spring.datasource.dynamic.datasource.master.password=seata
#########数据源2(信用分库)
spring.datasource.dynamic.datasource.credit.driver-class-name=com.mysql.jdbc.Driver
spring.datasource.dynamic.datasource.credit.url=jdbc:mysql://localhost:3306/seata_credit?useUnicode=true&characterEncoding=utf-8&useSSL=false
spring.datasource.dynamic.datasource.credit.username=seata 
spring.datasource.dynamic.datasource.credit.password=seata

库存应用(一个数据库):

##########服务注册 
spring.cloud.nacos.discovery.server-addr=localhost:8848 
spring.cloud.nacos.discovery.username=nacos_test 
spring.cloud.nacos.discovery.password=nacos_test
####seata相关
seata.enabled=true seata.tx-service-group=my_test_tx_group 
seata.enable-auto-data-source-proxy=false 
seata.config.type=nacos
seata.config.nacos.data-id=seataServer.properties 
seata.config.nacos.server-addr=localhost:8848 
seata.config.nacos.application=seata-server 
seata.config.nacos.namespace=7322f40d-74de-4679-ad4c-c3dff499cd98
seata.config.nacos.group=SEATA_GROUP 
seata.config.nacos.username=seata 
seata.config.nacos.password=seata 
seata.registry.type=nacos 
seata.registry.nacos.server-addr=localhost:8848 
seata.registry.nacos.namespace=7322f40d-74de-4679-ad4c-c3dff499cd98 
seata.registry.nacos.group=SEATA_GROUP 
seata.registry.nacos.username=seata
seata.registry.nacos.password=seata 
logging.level.io.seata = debug
######数据源
spring.datasource.dynamic.primary=master 
spring.datasource.dynamic.seata=true 
spring.datasource.dynamic.datasource.master.driver-class-name=com.mysql.jdbc.Driver 
spring.datasource.dynamic.datasource.master.url=jdbc:mysql://localhost:3306/seata_storage?useUnicode=true&characterEncoding=utf-8&useSSL=false
spring.datasource.dynamic.datasource.master.username=seata
spring.datasource.dynamic.datasource.master.password=seata

pom依赖:

<dependency>
  <artifactId>dynamic-datasource-spring-boot-starter</artifactId>
  <groupId>com.baomidou</groupId>
  <version>3.2.1</version>
</dependency>
<dependency>
  <groupId>com.alibaba.cloud</groupId>
  <artifactId>spring-cloud-starter-alibaba-seata</artifactId>
  <exclusions>
  <exclusion>
  <groupId>io.seata</groupId>
  <artifactId>seata-spring-boot-starter</artifactId>
  </exclusion>
  </exclusions>
</dependency>
<dependency>
  <groupId>io.seata</groupId>
  <artifactId>seata-spring-boot-starter</artifactId>
  <version>1.4.2</version>
</dependency>

4.3 开始验证

下面我们分多种情况来验证这种多数据源的事务是否有效,再加上SEATA来控制事务会产生什么效果。 4.3.1 同应用多库多表调用逻辑为:订单服务向订单库插入订单记录,接着向信用分库扣除信用分,暂时不访问库存服务1)订单controller的代码:

@RequestMapping("/orderByCredit")
public Boolean orderByCredit(String userId, @Nullable String requestId, int count) {
    //①下单,并产生下单日志
    orderService.onlyOrder(userId, "product-1", requestId, count);
    //② 访问信用库,校验并扣减信用分
    creditService.checkHasCredit(userId, count);
    return true;
}

2)使用多数据源@DS注解访问credit库修改数据

@DS("credit")
public void checkHasCredit(String userId, Integer val){
    Credit credit = creditDAO.selectByPrimaryKey(userId);
    if (credit == null) {
      throw new RuntimeException("无信用,无法购买");
    }
    credit.setCredit(credit.getCredit() - val);
    creditDAO.updateByPrimaryKey(credit);
    if (credit.getCredit() <= 0) {
      throw new RuntimeException("信用分不足,无法购买");
    }
}

3)使用postman请求使得,新增订单记录成功,并操作第二个数据库信用分不足报错,该情况按照预期,事务不会生效,订单库插入数据成功,接口也报错了。

注:订单库订单表插入成功,订单日志失败也不会回滚4)加上spring的@Transactional注解,下单请求

@RequestMapping("/orderByCredit")
@Transactional
public Boolean orderByCredit(String userId, @Nullable String requestId, int count) {
    //①下单,并产生下单日志
    orderService.onlyOrder(userId, "product-1", requestId, count);
    //② 访问信用库,校验并扣减信用分
    creditService.checkHasCredit(userId, count);
    return true;
}

这种情况产生找不到表的报错,操作②的@DS("credit")注解失效,没有按照预期访问credit库,访问到操作①的库了,但是操作①插入的数据回滚了。

注:订单库订单表插入成功,订单日志插入失败也会回滚。

小结: 所以在这里我们稍微总结下,在一个方法中,如果操作了多个数据源,不能在外面使用@Transactional来进行事务控制,会使得@DB注解切换数据源不生效,产生报错,但是不报错的部分事务有效,因为还没提交。如何来对这种操作多数据源的情况进行事务控制呢,那就需要使用分布式事务SEATA5)加上seata的@GlobalTransactional注解,下单请求

@RequestMapping("/orderByCredit")
@GlobalTransactional
public Boolean orderByCredit(String userId, @Nullable String requestId, int count) {
    //① 下单,并产生下单日志
    orderService.onlyOrder(userId, "product-1", requestId, count);
    //② 访问信用库,校验并扣减信用分
    creditService.checkHasCredit(userId, count);
    return true;
}

在seata可用的情况下,步骤①,步骤②任意的地方出现错误,均可以事务回滚。

6)同时加上seata的@GlobalTransactional注解和Spring的@Transactional,下单请求 @Transactional会导致切换数据库异常。

4.3.2 跨应用多库多表

调用逻辑为:

订单服务向订单库插入订单记录,接着向信用分库扣除信用分,远程调用库存服务扣减库存。

1)下订单,使得扣减库存报错

订单服务

@RequestMapping("/orderByCredit")
@GlobalTransactional
public Boolean orderByCredit(String userId, @Nullable String requestId,@Nullable String requestId2,  int count) {
    //① 下单,并产生下单日志
    orderService.onlyOrder(userId, "product-1", requestId, count);
    //② 访问直连信用库,校验并扣减信用分
    creditService.checkHasCredit(userId, count);
    //③ 调用库存服务,扣减库存
    storageFeignClient.deductFlow("product-1", count, requestId2);
    return true;
}

库存服务

public void deductFlow(String commodityCode, int count, String requestId) throws InterruptedException {
    Storage storage = this.deduct(commodityCode, count);
    StorageFlow storageFlow = new StorageFlow();
    storageFlow.setStorageId(storage.getId());
    storageFlow.setFlowId(StringUtils.isEmpty(requestId) ? String.valueOf(System.currentTimeMillis()): requestId);
    storageFlowDAO.insert(storageFlow);
}

库存服务中的任意地方报错,调用链的数据库操作均会回滚,@GlobalTransactional注解会将seata申请的xid通过request传播下去,被调用的服务如果接入seata将会组成一个完整的分布式事务。

4.3.3 跨服务调用吃掉异常

1)如下代码所示,下单服务访问了自身应用的数据库,feign远程调用库存服务,但是catch了异常。

@RequestMapping("/orderByCredit")
@GlobalTransactional
public Boolean orderByCredit(String userId, @Nullable String requestId,@Nullable String requestId2,  int count) {
    //① 下单,并产生下单日志
    orderService.onlyOrder(userId, "product-1", requestId, count);
    //② 访问直连信用库,校验并扣减信用分
    creditService.checkHasCredit(userId, count);
    try {
        //③ 调用库存服务,扣减库存
        storageFeignClient.deductFlow("product-1", count, requestId2);
    } catch (Exception e){
        e.printStackTrace();
    }
}

2)使得接口请求,①、②步骤正常请求,③步骤出现内部数据库主键异常错误。查看结果,全局事务最终commit了,说明seata没有感知异常。

五、总结

  1. 在需要分布式事务的方法入口使用@GlobalTransactional即可实现分布式事务

  2. 多数据源不能在方法外层加@Transactional,这样会导致切换库异常;可直接使用@GlobalTransactional达到事务效果,因为本来这就是分布式事务场景。

  3. 全局事务最终决议是由全局事务入口应用(TM)决定的,即使下游事务节点发生了异常,只要TM没有catch到异常,就不会全局回滚。即决议是由TM发出的,并不是有些文档所说由TC决议的。

  4. 分析:tm决议简单快速,性能相对tc来说,tc压力被各个tm所分散

  5. 跨服务调用需要将全局事务id传递下去,seata已经封装了相关代码,需要依赖相关的包,如spring-cloud-starter-alibaba-seata


关注IT巅峰技术,私信作者,获取以下2021全球架构师峰会PDF资料。

网关系统实践
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,558评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,002评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,024评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,144评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,255评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,295评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,068评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,478评论 1 305
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,789评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,965评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,649评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,267评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,982评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,223评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,800评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,847评论 2 351

推荐阅读更多精彩内容