使用 Jetty 9 作为 Java 9 模块给出 ClassNotFoundException

Using Jetty 9 as Java 9 Modules gives ClassNotFoundException

我将 jetty 9.4.8.v20171121 与 Java 9 一起使用,这是我加载的 jetty 模块列表:

jetty-server-9.4.8.v20171121.jar
jetty-servlet-9.4.8.v20171121.jar
jetty-servlets-9.4.8.v20171121.jar 
jetty-util-9.4.8.v20171121.jar   
jetty-xml-9.4.8.v20171121.jar 
jetty-http-9.4.8.v20171121.jar   
jetty-io-9.4.8.v20171121.jar 
jetty-security-9.4.8.v20171121.jar   
jetty-webapp-9.4.8.v20171121.jar

这是我启动 Jetty 的代码:

Server server = new Server( 8080 );
WebAppContext webapp = new WebAppContext();
webapp.setContextPath( "/" );
File warFile = new File(
        "/home/Sam/WebServer/jar/org.test.site.fend-0.1.0-SNAPSHOT.war" );
if (!warFile.exists())
{
    throw new RuntimeException( "Unable to find WAR File: "
            + warFile.getAbsolutePath() );
}
webapp.setWar( warFile.getAbsolutePath() );
webapp.setExtractWAR(true);
webapp.setAttribute(
        "org.eclipse.jetty.server.webapp.ContainerIncludeJarPattern",
        ".*/[^/]*servlet-api-[^/]*\.jar$|.*/javax.servlet.jsp.jstl-.*\.jar$|.*/[^/]*taglibs.*\.jar$" );
server.setHandler( webapp );
server.start();
server.dumpStdErr();

这就是我得到的:

2018-02-06 17:18:03:111 [main] DEBUG org.eclipse.jetty.util.component.AbstractLifeCycle - starting o.e.j.w.WebAppContext@659499f1{/,null,UNAVAILABLE}{/home/Sam/WebServer/jar/org.test.site.fend-0.1.0-SNAPSHOT.war}
2018-02-06 17:18:03:115 [main] WARN org.eclipse.jetty.webapp.WebAppContext - Failed startup of context o.e.j.w.WebAppContext@659499f1{/,null,UNAVAILABLE}{/home/Sam/WebServer/jar/org.test.site.fend-0.1.0-SNAPSHOT.war}
java.lang.ClassNotFoundException: org.eclipse.jetty.webapp.WebInfConfiguration
    at java.base/jdk.internal.loader.BuiltinClassLoader.loadClass(BuiltinClassLoader.java:582)
    at java.base/jdk.internal.loader.ClassLoaders$AppClassLoader.loadClass(ClassLoaders.java:185)
    at java.base/java.lang.ClassLoader.loadClass(ClassLoader.java:496)
    at jetty.util@9.4.8.v20171121/org.eclipse.jetty.util.Loader.loadClass(Loader.java:65)
    at jetty.webapp@9.4.8.v20171121/org.eclipse.jetty.webapp.WebAppContext.loadConfigurations(WebAppContext.java:1035)
    at jetty.webapp@9.4.8.v20171121/org.eclipse.jetty.webapp.WebAppContext.preConfigure(WebAppContext.java:473)
    at jetty.webapp@9.4.8.v20171121/org.eclipse.jetty.webapp.WebAppContext.doStart(WebAppContext.java:544)
    at jetty.util@9.4.8.v20171121/org.eclipse.jetty.util.component.AbstractLifeCycle.start(AbstractLifeCycle.java:68)
    at jetty.util@9.4.8.v20171121/org.eclipse.jetty.util.component.ContainerLifeCycle.start(ContainerLifeCycle.java:133)
    at jetty.server@9.4.8.v20171121/org.eclipse.jetty.server.Server.start(Server.java:418)
    at jetty.util@9.4.8.v20171121/org.eclipse.jetty.util.component.ContainerLifeCycle.doStart(ContainerLifeCycle.java:107)
    at jetty.server@9.4.8.v20171121/org.eclipse.jetty.server.handler.AbstractHandler.doStart(AbstractHandler.java:113)
    at jetty.server@9.4.8.v20171121/org.eclipse.jetty.server.Server.doStart(Server.java:385)
    at jetty.util@9.4.8.v20171121/org.eclipse.jetty.util.component.AbstractLifeCycle.start(AbstractLifeCycle.java:68)

如何解决?

Jetty 9.x 的 Jars(无论子版本)不是 JPMS modules

为了让 Jetty 正确支持 JPMS,需要对包装进行彻底的更改,并且肯定会影响 Jetty 代码库的主要部分。

注意:如果您希望看到 Jetty 完全符合 JPMS,请对问题发表评论 https://github.com/eclipse/jetty.project/issues/2189

在 Jetty 中的 JPMS 工作能够真正进行之前,还有一些事情需要解决。

  1. JEP238(多版本 Jars)如何与 ServiceLoader 行为一起工作? (提示:目前没有)
  2. 注释/字节码扫描如何在模块环境中工作? (目前的想法是,我们需要实现与 JPMS 相同的可见性规则,并且只扫描从相关方范围内可以看到的内容:webapp 或服务器)

至于 Jetty 何时支持 JPMS,Jetty 不太可能出现这种情况 9.x,曾针对 Jetty 10.x 进行了简短讨论,但确定支持 Servlet 4.0,javax.websocket 1.1 和 Java 8 仍然很重要。

一旦Java 8 的生命周期完全结束,将现有稳定分支迁移到 JPMS 的过程就有效了。 (如果发生这种情况是另一个问题)。

我们正在密切关注规范组的未决问题,并将 Java 11 (LTS) 版本作为 JPMS 支持的良好候选者。这也可能与(目前理论上的)码头 11.x 重合。