首页 > 技术文章 > Apollo报错找不到apollo.meta的问题解决方案

slankka 2019-11-01 15:27 原文

问题描述

Apollo报错,找不到apoll.meta,但是明明配置了apollo-env.properties到apollo-client内了。

apollo-env.properties

pro.meta=http://apollo.xxxx.com:81

问题分析

因公司内部使用的 ctrip Apollo用了较早的版本,期间经过一次升级,而我算是Apollo的忠实用户,使用时间超过两年。
因此,我恰好知道,Apollo稍早的版本,没有将Apollo-client发布到 中央仓库,因为 Apollo-client的内部包含了编译期间指定的apollo-env.properties文件,
而为了发布到中央仓库,不能使用此方式将自己公司的meta地址放入其中,因此Apollo增加了一种配置方式:通过JVM参数等方式指定 apollo 的meta Server地址。

公司Maven仓库内的apollo-client内是含有apollo-env.properties。因此原则上Apollo不需要再配置。

而为什么报错呢?

问题原因

Apollo的 Meta 地址获取逻辑,采用JAVA的SPI实现

package com.ctrip.framework.apollo.core.spi;

import com.ctrip.framework.apollo.core.enums.Env;

/**
 * @since 1.0.0
 */
public interface MetaServerProvider extends Ordered {

  /**
   * Provide the Apollo meta server address, could be a domain url or comma separated ip addresses, like http://1.2.3.4:8080,http://2.3.4.5:8080.
   * <br/>
   * In production environment, we suggest using one single domain like http://config.xxx.com(backed by software load balancers like nginx) instead of multiple ip addresses
   */
  String getMetaServerAddress(Env targetEnv);
}

而该接口有两个实现

默认实现:

public class DefaultMetaServerProvider implements MetaServerProvider

第二个实现:(旧版本)

public class LegacyMetaServerProvider implements MetaServerProvider

正常情况下,apollo应该使用二者之一的Provider。

如果JVM启动的时候,使用了DefaultMetaServerProvider,这种情况没有配置JVM参数 apollo.meta,那么将报错,找不到meta server。

那为什么公司的仓库内 apollo-client 有apollo-env.properties 但是没有使用LegacyMetaServerProvider ?

问题解决

因为踩坑次数多了,自然想到去jar包内看看,经过检查发现 jar内的META-INF确实有apollo-env.properties。
问题就出在,apollo-core-1.3.0-xxx.jar!/META-INF/services/com.ctrip.framework.apollo.core.spi.MetaServerProvider内是:

com.ctrip.framework.apollo.internals.DefaultMetaServerProvider


apollo-client-1.3.0-xxx.jar!/META-INF/services/com.ctrip.framework.apollo.core.spi.MetaServerProvider内是:

com.ctrip.framework.apollo.core.internals.LegacyMetaServerProvider

最终原因在于,maven打包工具有 maven-jar-plugin,maven-assembly-plugin, maven-shaded-plugin。

如果使用spring-boot-maven-plugin,虽然内置有maven-shaded-plugin,但是他根本不知道apollo的。

这些情况最终导致了不确定性,JVM加载的Jar包顺序不一样,或者打包的顺序,导致一个SPI文件被另一个文件覆盖掉了,因此丢失了一个provider实现。

但是只有maven-shaded-plugin,可以经过配置TRANSFORMER,将这两种合并到一个com.ctrip.framework.apollo.core.spi.MetaServerProvider文件内。

最终解决方案

保持apollo-client和apollo-core包内 的/META-INF/services/com.ctrip.framework.apollo.core.spi.MetaServerProvider
有两个Provider,分别写一行。
或者联系相关人员,将这两个文件的内容保持一致,并给出适用的使用方案。例如jar包内附带了pro.meta等变量的时候,提供LegacyMetaServerProvider。
如果没有附带,则要求使用人员自行配置 apollo.meta。

刚好这篇文章解决了我另一篇同样是关于Apollo和Dubbo集成的文章中遇到的问题。

解决Dubbo 2.7.3版本使用ConfigCenterConfig集成Apollo No Provider found的问题

推荐阅读