问题
项目使用了nacos 作为配置中心和服务注册中心,使用内嵌容器服务注册正常,将项目打成war包部署到tomcat中无法注册。
经过查看源码发现,nacos注册类NacosAutoServiceRegistration 继承了Spring cloud 中AbstractAutoServiceRegistration 在AbstractAutoServiceRegistration中绑定了一个监听事件,监听内置容器启动完成事件,监听到获取容器端口后向注册中心注册。
@EventListener({WebServerInitializedEvent.class})
public void bind(WebServerInitializedEvent event) {
ApplicationContext context = event.getApplicationContext();
if (!(context instanceof ConfigurableWebServerApplicationContext) || !"management".equals(((ConfigurableWebServerApplicationContext)context).getServerNamespace())) {
this.port.compareAndSet(0, event.getWebServer().getPort());
this.start();
}
}
使用外部容器时,不能监听到事件,所以自动注册失败。
解决方案
springboot 提供 ApplicationRunner 接口 ,是在应用起好之后执行一些初始化动作。通过这个接口我们可以实现启动项目后注册服务。不过使用这种方法,端口号需要在配置文件中配置,如果一个应用部署很多端口,每个应用都要配置,很不方便。
@component
public class NacosConfig implements ApplicationRunner {
@Autowired(required = false)
private NacosAutoServiceRegistration registration;
@Value("${server.port}")
Integer port;
@Override
public void run(ApplicationArguments args) {
if (registration != null && port != null) {
registration.setPort(port);
registration.start();
}
}
}
对此,可以获取外部tomcat 自动设置端口。经过测试,方法可行。
public String getTomcatPort() throws Exception {
MBeanServer beanServer = ManagementFactory.getPlatformMBeanServer();
Set<ObjectName> objectNames = beanServer.queryNames(new ObjectName("*:type=Connector,*"), Query.match(Query.attr("protocol"), Query.value("HTTP/1.1")));
String port = objectNames.iterator().next().getKeyProperty("port");
return port;
}