情况
我正在开发一个使用 Java 的URLClassLoader和ServiceLoader来加载 jar 文件的应用程序。在这些 jar 文件中是实现我的接口的提供程序。文件夹结构如oracle 在这篇文章中所述,这意味着:
- 该接口与插件中实现该接口的类(com.x.projectname.plugin.IInterface.java)位于同一目录中。在插件中,接口和实现它的类都在com.x.projectname.plugin文件夹中。
- 在插件中有一个resources.META-INF.services目录,其中有一个文件:com.x.projectname.plugin.IInterface,内容如下:com.x.projectname.plugin.ClassThatImplementsIInterface。
在我的本地机器上运行时(使用 Oracle JDK 1.8 162 和 OpenJDK 1.8 171 测试),插件加载良好,应用程序可以根据需要使用插件。
问题
在 docker 容器中运行主应用程序时,它无法加载所需的插件。Docker 容器在机器上安装了一个文件夹,其中安装了插件。该应用程序的docker镜像使用openjdk:8-jdk-alpine,但无论我是否使用alpine版本,问题仍然存在。
尝试使用Serviceloader.load()加载插件时(参见ServiceLoader.load(InterfaceName.class, ClassLoader loader)
它崩溃并出现以下错误:Provider com.x.projectname.plugin.InterfaceImplementingClass not a subtype。
这是堆栈跟踪的相关部分:
Caused by: java.util.ServiceConfigurationError: com.x.projectname.plugin.IInterface: Provider com.x.projectname.plugin.ImplementingClass not a subtype
at java.util.ServiceLoader.fail(ServiceLoader.java:239) ~[na:1.8.0_151]
at java.util.ServiceLoader.access$300(ServiceLoader.java:185) ~[na:1.8.0_151]
at java.util.ServiceLoader$LazyIterator.nextService(ServiceLoader.java:376) ~[na:1.8.0_151]
at java.util.ServiceLoader$LazyIterator.next(ServiceLoader.java:404) ~[na:1.8.0_151]
at java.util.ServiceLoader$1.next(ServiceLoader.java:480) ~[na:1.8.0_151]
at com.x.projectname.loader.PluginLoader.loadPlugins(PluginLoader.java:118) ~[classes!/:na]
at com.x.projectname.loader.PluginLoader.initializePluginLoading(PluginLoader.java:67) ~[classes!/:na]
at com.x.projectname.service.PluginService.<init>(PluginService.java:37) ~[classes!/:na]
at sun.reflect.NativeConstructorAccessorImpl.newInstance0(Native Method) ~[na:1.8.0_151]
at sun.reflect.NativeConstructorAccessorImpl.newInstance(NativeConstructorAccessorImpl.java:62) ~[na:1.8.0_151]
at sun.reflect.DelegatingConstructorAccessorImpl.newInstance(DelegatingConstructorAccessorImpl.java:45) ~[na:1.8.0_151]
任何想法为什么它在 Docker 容器中运行时不会加载插件?
编辑 1:Dockerfile 和 docker run 命令
FROM openjdk:8-jdk-alpine
ARG JAR_FILE
ADD ${JAR_FILE} /CICD-dashboard.jar
# Remote debugging port for intelliJ == address
EXPOSE 50505
ENTRYPOINT [ "java", "-Xrunjdwp:transport=dt_socket,address=50505,suspend=n,server=y", "-jar", "/X-project.jar"]
码头工人运行命令:
docker run -d -v /home/folder/pluginfolder:/pluginfolder -p=50505:50505 image-name