使用spring cloud alibaba sentinel遇到的坑

背景

feign是spring cloud提供的一个声明式伪http客户端,让调用远程服务就像调用本地服务一样。为了进行服务容错,所以开启了feign关于sentinel的支持,在本地开发环境,有A、B两个服务,分别已启动,A通过feign调用B服务的某个方法,可以正常返回结果。然后将B服务停掉,也可以正常进入A服务提供的容错处理类对应的方法。然后通过jenkins打完A服务的可运行jar包,部署到测试环境,此时B服务并没有部署到测试环境,按理说此时A去调用B服务,应该给出服务容错的相关提示信息,但是却给出:org.springframework.web.util.NestedServletException: Handler dispatch failed; nested exception is java.lang.NoClassDefFoundError: Could not initialize class com.alibaba.csp.sentinel.context.ContextUtil。

分析

看到NoClassDefFoundError,排查步骤:
① ContextUtil是否有在项目里或者依赖的jar包里
② 是否jar包依赖冲突导致
经过上面两个步骤的排查,返现jar包依赖正常,ContextUtil就在sentinel-core里;后来发现使用jenkins打包,然后到目标服务器启动A服务对应的jar包,使用的普通用户:比如appuser。后面手动使用root用户启动A服务的jar包,发现正常进入服务容错了。这时候毕本可以定位应该是因为用户权限的问题导致。后面还是用普通用户通过远程debug jar包才找到问题,看到输出了一条关键的日志,但是不用debug却没有这条日志:Permission denied /home/appuser/logs/csp/sentinel-record.log.2021-03-18.0.lck。
然后到/home/appuser目录下,发现有logs目录(很早以前就创建),但归属所属用户和组都是root,修改该目录所属用户和组为appuser。修改完成,使用appuser用户重新启动A服务,正常进入服务容错,不再出现:Could not initialize class com.alibaba.csp.sentinel.context.ContextUtil。

小结

将服务部署到linux服务器,需要注意当前用户是否对目录具有可读可写可执行的权限,以免出现像上述这样的错误,因为使用sentinel,会往用户自己的home目录写数据

远程debug方法:
1)目标服务器启动服务:java -Xdebug -Xrunjdwp:transport=dt_socket,address=5005,server=y,suspend=n -jar xxx.jar --spring.profiles.active=test
2)本地IDEA使用remote启动,打断点
3)访问服务器上的API,即可远程debug

你可能感兴趣的:(java)