问题

最近接手了一个旧的项目,在服务器上成功运行后想把部分服务在本地启动进行调试,启动过程中出现了一个错误

2021-06-05 17:51:53.776 WARN  org.springframework.boot.actuate.redis.RedisHealthIndicator [AbstractHealthIndicator.java:87] ## Redis health check failed
org.springframework.data.redis.RedisConnectionFailureException: Cannot get Jedis connection; nested exception is redis.clients.jedis.exceptions.
JedisConnectionException: Failed connecting to host localhost:6379
	at org.springframework.data.redis.connection.jedis.JedisConnectionFactory
	.fetchJedisConnector(JedisConnectionFactory.java:281)
	at org.springframework.data.redis.connection.jedis.JedisConnectionFactory
	.getConnection(JedisConnectionFactory.java:464)

错误信息的主要内容就是Failed connecting to host localhost:6379
关于这个错误网上有一些内容,基本都是在本地部署的redis,解决方案通常是

  • 打开redis服务
  • 关闭redis保护模式

但是在我遇到的场景中,把nacos中所有和redis相关的配置全都指定成服务器的ip地址后,仍然出现了localhost,说明是一个配置的错误,没有把redis服务器的值设置成服务器的ip。

解决方案

最终的解决方法
配置中加入spring.redis.host=服务器ip就可以解决这个问题了。
因为这是一个旧的项目,里面配置项很多,刚开始重新部署的时候只把配置中有的ip全局替换了,没有考虑到原来程序运行的时候一直没有配置redis的ip地址,而是一直使用的缺省值,在项目本地调试的时候就遇到了问题。

排查过程

错误日志中已经写到了
JedisConnectionFactory.fetchJedisConnector(JedisConnectionFactory.java:281)这行代码出现了问题
先打开JedisConnectionFactory看看代码。
在这里插入图片描述
这段代码先创建了Jedis对象,再调用connect方法,可以在connect方法上打个断点。
在这里插入图片描述
这里很明显,还没有调用connect方法,在创建Jedis对象后host的值已经是localhost了。点进去看看createJedis方法。
在这里插入图片描述
这里很明显调用了一个getHostName方法,这个方法返回的应该是localhost,点进去继续看。
在这里插入图片描述

这个方法从standaloneConfig对象中取出了hostname的值,而standaloneConfigJedisConnectionFactory的一个对象,在这个对象上打个断点看看这个对象是通过哪个方法进来的。
在这里插入图片描述
进入断点后发现是在构造函数中传入了standaloneConfig对象覆盖了默认值,再向上一层,进入了JedisConnectionConfiguration类中,是这个类调用了JedisConnectionFactory的构造方法。
在这里插入图片描述
调用的时候直接通过getStandaloneConfig方法取到对象,继续进入。
在这里插入图片描述
发现是在properties对象中取到了host的值
在这里插入图片描述
properties是在RedisConnectionConfiguration的构造函数中传入的。
点进RedisProerties类中看一下
在这里插入图片描述
发现整个过程中host的源头就是spring.redis.host这个配置的值,因为原来拿到的nacos配置文件中没有,加上对redis不太熟悉,就一直没有想到这个问题。不过通过一层一层分析还是找到了原因。

Logo

华为开发者空间,是为全球开发者打造的专属开发空间,汇聚了华为优质开发资源及工具,致力于让每一位开发者拥有一台云主机,基于华为根生态开发、创新。

更多推荐