升级阿里云RDS实例报 CommunicationsException 问题排查

@[toc]

现象

目前营销数据库使用的是阿里云 MYSQL 数据库,近期需要支持 DBA 对阿里云 RDS 进行升级,升级公告如下:
升级阿里云RDS实例报 CommunicationsException 问题排查_第1张图片
凌晨 5 点 10 分进行升级后开始日志报错,错误信息如下:
升级阿里云RDS实例报 CommunicationsException 问题排查_第2张图片

问题分析与结论

一般出现 duridCommunicationsException 原因是由于连接池获取已经失效的连接导致。从 mysql 上来说:

Mysql服务器默认的“wait_timeout”是8小时(也就是默认的值默认是28800秒),即一个connection空闲超过8个小时,Mysql将自动断开该connection,通俗的讲就是一个连接在8小时内没有活动,就会自动断开该连接。而连接池却认为该连接还是有效的(因为并未校验连接的有效性),当应用申请使用该连接时,就会导致上面的报错。

从 druid 配置属性上和连接失效的属性有以下几个:

  • minEvictableIdleTimeMillis:连接空闲时间大于该值并且池中空闲连接大于 minIdle 则关闭该连接。配置中采用的默认值 30 分钟。
  • maxEvictableIdleTimeMillis:连接空闲时间大于该值,不管 minIdle 都关闭该连接。配置中采用的默认值 7 个小时。
  • maxidle:目前 druid 已经废弃了 maxidle 属性,使用 maxActive 和 minIdel 来代替,参考https: //github.com/alibaba/druid/wiki/FAQ,从 druid 配置上看,minIdle 是 50,maxActive 是 400(代码上设置不合理,可调整)。

即理论上 druid 默认连接失效的最大 7 个小时也小于数据库 8 个小时,不会存在配置层面上获取到失效的连接。再从公告上看,有说明:“在迁移过程中会有 30 秒的连接闪断”,说明 rds 闪断后原有的的数据库连接 都已经失效了,而 smc 系统中 druid 连接池还保持着有效状态,所以会导致com.mysql.jdbc.exceptions.jdbc4. CommunicationsException 发生。

解决方案

直接思路时在 druid 线程拿到连接后校验一下就 ok 了,涉及到相关属性有 test- WhileIdle、testOnBorrow,意义如下:

  • testWhileIdle:如果为 true(默认 true),当应用向连接池申请连接,并且 testOnBorrow 为 false 时,连接池将会判断连接是否处于空闲状态,如果是,则验证这条连接是否可用(高效,推荐使用)。
  • testOnBorrow:(默认 false)申请连接时进行连接有效性检查(低效,影响性能),与 testWhileIdle 是差不多的,都是在获取连接的时候测试连接的有效性,如果两者都为 true,则 testOnBorrow 优先 级高,则不会使用到 testWhileIdle。
  • validationQuery:在连接池返回连接给调用者前用来对连接进行验证的查询 SQL,要求为一条查询语句(开启 testWhileIdle 必须要设置 validationQuery)。

druid 配置中增加:

spring.datasource.druid.test-on-borrow=false
spring.datasource.druid.test-while-idle=true
spring.datasource.druid.validation-query=SELECT 1

此时还有一个疑问,在没有配置上述属性的情况下,testWhileIdlel 默认为 true,testOnBorrow 默认 为 false,validationQuery 默认为空,为什么没有强制设置 validationQuery 呢?此时要分 2 个阶段说明问题:

  • durid 初始化阶段:只打了 error,并未中断初始化过程
  • 获取连接过程:校验了 minEvictableIdleTimeMillis 和 maxEvictableIdleTimeMillis,发现 validation- Query 为 null 直接返回了,所以上述配置只需要增加 validationQuery 的配置也能解决问题。

欢迎关注微信公众号:方辰的博客
升级阿里云RDS实例报 CommunicationsException 问题排查_第3张图片

你可能感兴趣的:(java)