Dubbo 2.7应用级服务发现踩坑小记

 [[407480]]

背景

本文记录最近一位读者反馈的dubbo 2.7.x中应用级服务发现的问题,关于dubbo应用级服务发现的相关介绍可以参考之前的文章《dubbo应用级服务发现初体验》,这里不再赘述。

站在用户的角度思考问题,与客户深入沟通,找到冷水滩网站设计与冷水滩网站推广的解决方案,凭借多年的经验,让设计与互联网技术结合,创造个性化、用户体验好的作品,建站类型包括:成都网站设计、网站制作、企业官网、英文网站、手机端网站、网站推广、域名注册、网络空间、企业邮箱。业务覆盖冷水滩地区。

读者反馈他们在基于dubbo 2.7应用级服务发现开发dubbo网关,根据文章《dubbo应用级服务发现初体验》写了demo调用时报no provider的错误。

首先觉得他们挺有想法,把dubbo应用级服务发现搬上生产的公司不多。其次当时写文章时测试并没有遇到问题,但本着帮读者解决问题的态度,还是重新写个demo测试下。

问题定位

随手拿了一个平时测试用的dubbo demo工程(注意不是dubbo源码中的demo),发现确实注册不到zookeeper上,接着测试了不同的版本,发现都注册不了,在2.7.5 ~ 2.7.11版本不报错,2.7.12版本会报如下的NPE错误

 
 
 
 
  1. 2021-06-16 13:17:31,086 [Dubbo-framework-scheduler-thread-1] ERROR org.apache.dubbo.config.bootstrap.DubboBootstrap (DubboBootstrap.java:1172) -  [DUBBO] refresh metadata and instance failed, dubbo version: 2.7.12, current host: 172.23.233.52 
  2. java.lang.NullPointerException 
  3.  at org.apache.dubbo.registry.client.metadata.ServiceInstanceMetadataUtils.calInstanceRevision(ServiceInstanceMetadataUtils.java:249) 
  4.  at org.apache.dubbo.registry.client.metadata.ServiceInstanceMetadataUtils.lambda$refreshMetadataAndInstance$6(ServiceInstanceMetadataUtils.java:272) 
  5.  at java.util.ArrayList.forEach(ArrayList.java:1259) 
  6.  at org.apache.dubbo.registry.client.metadata.ServiceInstanceMetadataUtils.refreshMetadataAndInstance(ServiceInstanceMetadataUtils.java:271) 
  7.  at org.apache.dubbo.config.bootstrap.DubboBootstrap.lambda$registerServiceInstance$20(DubboBootstrap.java:1170) 
  8.  at java.util.concurrent.Executors$RunnableAdapter.call(Executors.java:511) 
  9.  at java.util.concurrent.FutureTask.runAndReset(FutureTask.java:308) 
  10.  at java.util.concurrent.ScheduledThreadPoolExecutor$ScheduledFutureTask.access$301(ScheduledThreadPoolExecutor.java:180) 
  11.  at java.util.concurrent.ScheduledThreadPoolExecutor$ScheduledFutureTask.run(ScheduledThreadPoolExecutor.java:294) 
  12.  at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1149) 
  13.  at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:624) 
  14.  at java.lang.Thread.run(Thread.java:748) 

推测服务注册时存在问题,跟着这个错误栈debug,很快就定位到问题

直接导致NPE的是位于org.apache.dubbo.registry.client.AbstractServiceDiscovery#register

在<=2.7.11版本中

 
 
 
 
  1. @Override 
  2. public final void register(ServiceInstance serviceInstance) throws RuntimeException { 
  3.     this.serviceInstance = serviceInstance; 
  4.     doRegister(serviceInstance); 

而在2.7.12版本中代码顺序被调整成了

 
 
 
 
  1. @Override 
  2. public final void register(ServiceInstance serviceInstance) throws RuntimeException { 
  3.    doRegister(serviceInstance); 
  4.     this.serviceInstance = serviceInstance; 

为什么调整了代码顺序导致了报错?

追踪下来发现NPE的来源是this.serviceInstance为null,原先的代码先对其赋值再执行doRegister。调整过后的代码先执行doRegister再赋值,然而在执行diRegister时抛出异常了,不幸的是这个异常被吃掉了,doRegister的实现如下

 
 
 
 
  1. @Override 
  2. public final void register(ServiceInstance serviceInstance) throws RuntimeException { 
  3.  
  4.     assertDestroyed(REGISTER_ACTION); 
  5.     assertInitialized(REGISTER_ACTION); 
  6.  
  7.     executeWithEvents( 
  8.             of(new ServiceInstancePreRegisteredEvent(serviceDiscovery, serviceInstance)), 
  9.             () -> serviceDiscovery.register(serviceInstance), 
  10.             of(new ServiceInstanceRegisteredEvent(serviceDiscovery, serviceInstance)) 
  11.     ); 

而这个executeWithEvents会将异常以事件的形式发出去

 
 
 
 
  1. protected final void executeWithEvents(Optional beforeEvent, 
  2.                                        ThrowableAction action, 
  3.                                        Optional afterEvent) { 
  4.     beforeEvent.ifPresent(this::dispatchEvent); 
  5.     try { 
  6.         action.execute(); 
  7.     } catch (Throwable e) { 
  8.         dispatchEvent(new ServiceDiscoveryExceptionEvent(this, serviceDiscovery, e)); 
  9.     } 
  10.     afterEvent.ifPresent(this::dispatchEvent); 

然而这个事件丢出去之后并没有被处理,也就是说这个异常被吃掉了。这也就是为什么之前的dubbo版本没有抛出异常,也不能注册上服务。

这个异常是什么?

 
 
 
 
  1. va.lang.NoClassDefFoundError: org/apache/curator/x/discovery/ServiceDiscovery 

其实只是少引入了一个依赖。加入以下以下就能解决这个问题

 
 
 
 
  1.  
  2.     org.apache.dubbo 
  3.     dubbo-registry-zookeeper 
  4.     ${version} 
  5.  

为什么当时写文章时没有遇到这个问题?

恰好本次调试是基于一个demo工程,《dubbo应用级服务发现初体验》文章是直接在dubbo源码中修改demo,源码中已经把依赖都引入了

更进一步

其实是个小问题,但对用户来说挺困惑的,为什么没有报错但也没法注册服务?如果不是2.7.12有个附带的报错,可能排查起来更加困难。

于是提了个issue和社区的朋友交流下,得出的结论是2.7.x的应用级服务发现不再维护,3.x会继续维护。

https://github.com/apache/dubbo/issues/8061

提个issue也是让有问题的用户能搜索到,少走弯路。

顺便也提了个PR,加一行log,方便直观的发现这个问题。

https://github.com/apache/dubbo/pull/8066

新版本(>=2.7.13)如果有朋友再遇到这个问题,会直接打印出错误,就像这样

 
 
 
 
  1. 2021-06-16 16:58:02,210 [main] ERROR org.apache.dubbo.registry.client.EventPublishingServiceDiscovery (EventPublishingServiceDiscovery.java:287) -  [DUBBO] Execute action throws and dispatch a ServiceDiscoveryExceptionEvent, dubbo version: 2.7.12, current host: 172.23.233.52 
  2. java.lang.BootstrapMethodError: java.lang.NoClassDefFoundError: org/apache/curator/x/discovery/ServiceDiscovery 
  3.  at org.apache.dubbo.registry.zookeeper.ZookeeperServiceDiscovery.doRegister(ZookeeperServiceDiscovery.java:92) 
  4.  at org.apache.dubbo.registry.client.AbstractServiceDiscovery.register(AbstractServiceDiscovery.java:33) 
  5.  at org.apache.dubbo.registry.client.EventPublishingServiceDiscovery.lambda$register$0(EventPublishingServiceDiscovery.java:159) 
  6.  at org.apache.dubbo.registry.client.EventPublishingServiceDiscovery.executeWithEvents(EventPublishingServiceDiscovery.java:285) 
  7.  at org.apache.dubbo.registry.client.EventPublishingServiceDiscovery.register(EventPublishingServiceDiscovery.java:157) 
  8.  at org.apache.dubbo.config.bootstrap.DubboBootstrap.lambda$doRegisterServiceInstance$21(DubboBootstrap.java:1192) 
  9.  at java.util.ArrayList.forEach(ArrayList.java:1259) 
  10.   ... 

既然2.7.x的应用级服务发现不再更新,下次写一篇分析3.0版本的应用级服务发现源码的文章吧~

当前名称:Dubbo 2.7应用级服务发现踩坑小记
分享网址:http://www.shufengxianlan.com/qtweb/news35/148885.html

网站建设、网络推广公司-创新互联,是专注品牌与效果的网站制作,网络营销seo公司;服务项目有等

广告

声明:本网站发布的内容(图片、视频和文字)以用户投稿、用户转载内容为主,如果涉及侵权请尽快告知,我们将会在第一时间删除。文章观点不代表本网站立场,如需处理请联系客服。电话:028-86922220;邮箱:631063699@qq.com。内容未经允许不得转载,或转载时需注明来源: 创新互联