[tomcat-jdbc]Can‘t call commit when autocommit=true

Can't call commit when autocommit=true

问题

使用tomcat-jdbc时出现Can't call commit when autocommit=true,错误信息如下

log 复制代码
### Error committing transaction.  Cause: java.sql.SQLException: Can't call commit when autocommit=true
### Cause: java.sql.SQLException: Can't call commit when autocommit=true
; uncategorized SQLException for SQL []; SQL state [null]; error code [0]; Can't call commit when autocommit=true; nested exception is java.sql.SQLException: Can't call commit when autocommit=true] with root cause

java.sql.SQLException: Can't call commit when autocommit=true

分析

这个错误的意思是:自动提交的情况下不需要再执行commit操作。通常我们会把autocommit设置为true,这样非事务的操作都是自动提交的,只有开启事务的时候才需要把autocommit设置为false,并在事务结束的时候执行commit/rollback操作,然后再把autocommit设置为true。也就是说执行某个非事务操作完成之后执行了非预期的commit操作才会导致以上的异常,应该有两个检测autocommit的位置状态不一致导致的。

抛出Can't call commit when autocommit=true这个异常的代码在mysql-connector-java-5.1.42.jarcom.mysql.jdbc.ConnectionImpl中,使用mybatis获取数据库连接的org.mybatis.spring.transaction.SpringManagedTransaction.openConnection的地方通过this.connection.getAutoCommit()获取autoCommit的值,并在commit()通过this.connection != null && !this.isConnectionTransactional && !this.autoCommit判断是否需要在数据库连接上执行最终的提交this.connection.commit(),也就是这个地方的状态判断和数据库连接上的状态不一致导致执行了非预期的commit()操作。

通过调试跟踪最终确定是因为org.apache.tomcat.jdbc.pool.interceptor.ConnectionState缓存了状态,并且在异常的情况下和连接上的状态不一致导致,

其大概的逻辑代码(非实际代码)如下

java 复制代码
class ConnectionState {
  boolean getAutoCommit() {
    if (autoCommit == null) {
      autoCommit = connection.getAutoCommit();
    }
    return autoCommit;
  }
  void setAutoCommit(boolean value) {
    connection.setAutoCommit(value);
    this.autoCommit = value;
  }
}

以上代码中只要connection.setAutoCommit出现异常就会导致状态不一致,缓存的逻辑需要调整成

java 复制代码
class ConnectionState {
  void setAutoCommit(boolean value) {
    try {
      connection.setAutoCommit(value);
      this.autoCommit = value;
    } catch (Exception e) {
      this.autoCommit = null; // 异常时清空,以便重新获取最新状态
    }
  }
}

复现

知道了问题的原因,如何复现这个问题呢?

我们只需要在setAutoCommit(true)的时候模拟异常即可,为了更贴近实际应用的情况,需要模拟服务端断开连接,因此需要准备两个账户,一个用于执行业务代码,另一个用于Kill连接来模拟断开连接。

sql 复制代码
CREATE TABLE `test`(
    id INT(11) AUTO_INCREMENT NOT NULL,
    create_time TIMESTAMP,
    update_time TIMESTAMP,
    PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_bin;

CREATE USER 'test-tomcat-jdbc'@'localhost' IDENTIFIED BY 'test';
GRANT ALL ON test.* TO 'test-tomcat-jdbc'@'localhost';

CREATE USER 'root-tomcat-jdbc'@'localhost' IDENTIFIED BY 'test';
GRANT ALL ON *.* TO 'root-tomcat-jdbc'@'localhost';

FLUSH PRIVILEGES;

为了使用ConnectionState需要在业务的数据库连接池的配置jdbc-interceptors中包含ConnectionState,正常来说如果连接被断开之后,连接会被剔除掉(tomcat-jdbc的连接池没有剔除导致一直不可用),因此还需要在数据库连接的url中增加autoReconnect=true确保断开后自动连接避免被剔除。

有了已经配置,运行应用之后请求新增接口之后就会触发Kill。

sh 复制代码
$ curl http://127.0.0.1:8080/test/add

此时会发现日志中有Communications link failure的错误信息,但是后续的检测数据库连接又是正常的,因为autoReconnect=true自动连接成功了。

但是再次请求则会报Can't call commit when autocommit=true的错误了

sh 复制代码
$ curl http://127.0.0.1:8080/test/get
{"timestamp":1690722907640,"status":500,"error":"Internal Server Error","exception":"org.springframework.jdbc.UncategorizedSQLException","message":"\n### Error committing transaction.  Cause: java.sql.SQLException: Can't call commit when autocommit=true\n### Cause: java.sql.SQLException: Can't call commit when autocommit=true\n; uncategorized SQLException for SQL []; SQL state [null]; error code [0]; Can't call commit when autocommit=true; nested exception is java.sql.SQLException: Can't call commit when autocommit=true","path":"/test/get"}

测试时只需要修改数据库的IP和端口即可,并通过配置以下不同的参数模拟断开连接的时机

yaml 复制代码
# 模拟连接到只读库
mock-readonly: false
# 开始事务后检测是否只读时断开连接
kill-condition: session.tx_read_only
# 开始事务并提交后重置自动提交时断开连接
#kill-condition: autocommit=1
#kill-condition: not-kill

结论

暂时不要使用ConnectionState或者使用其他数据库连接池。

复现代码lab-tomcat-jdbc-autocommit

相关推荐
uup20 分钟前
Java 中 ArrayList 线程安全问题
java
uup21 分钟前
Java 中日期格式化的潜在问题
java
老华带你飞30 分钟前
海产品销售系统|海鲜商城购物|基于SprinBoot+vue的海鲜商城系统(源码+数据库+文档)
java·前端·数据库·vue.js·论文·毕设·海鲜商城购物系统
2401_8370885035 分钟前
Redisson的multilock原理
java·开发语言
今天你TLE了吗41 分钟前
Stream流学习总结
java·学习
⑩-1 小时前
基于Redis Lua脚本的秒杀系统
java·redis
0和1的舞者2 小时前
《网络编程核心概念与 UDP Socket 组件深度解析》
java·开发语言·网络·计算机网络·udp·socket
稚辉君.MCA_P8_Java2 小时前
Gemini永久会员 Java动态规划
java·数据结构·leetcode·排序算法·动态规划
oioihoii2 小时前
C++语言演进之路:从“C with Classes”到现代编程基石
java·c语言·c++
N***73852 小时前
SQL锁机制
java·数据库·sql