Spring 3.1 WebApplicationInitializer和Embedded Jetty 8 AnnotationConfiguration

Dun*_*can 33 java jetty spring-mvc embedded-jetty jetty-8

我正在尝试使用Spring 3.1和嵌入式Jetty 8服务器创建一个没有任何XML配置的简单webapp.

但是,我正在努力让Jetty认识到我对Spring WebApplicationInitializer接口的实现.

项目结构:

src
 +- main
     +- java
     |   +- JettyServer.java
     |   +- Initializer.java
     | 
     +- webapp
         +- web.xml (objective is to remove this - see below).
Run Code Online (Sandbox Code Playgroud)

上面的Initializer类是WebApplicationInitializer的一个简单实现:

import javax.servlet.ServletContext;
import javax.servlet.ServletException;

import org.springframework.web.WebApplicationInitializer;

public class Initializer implements WebApplicationInitializer {

    @Override
    public void onStartup(ServletContext servletContext) throws ServletException {
        System.out.println("onStartup");
    }
}
Run Code Online (Sandbox Code Playgroud)

同样JettyServer是嵌入式Jetty服务器的简单实现:

import org.eclipse.jetty.annotations.AnnotationConfiguration;
import org.eclipse.jetty.server.Server;
import org.eclipse.jetty.webapp.Configuration;
import org.eclipse.jetty.webapp.WebAppContext;

public class JettyServer {

    public static void main(String[] args) throws Exception { 

        Server server = new Server(8080);

        WebAppContext webAppContext = new WebAppContext();
        webAppContext.setResourceBase("src/main/webapp");
        webAppContext.setContextPath("/");
        webAppContext.setConfigurations(new Configuration[] { new AnnotationConfiguration() });
        webAppContext.setParentLoaderPriority(true);

        server.setHandler(webAppContext);
        server.start();
        server.join();
    }
}
Run Code Online (Sandbox Code Playgroud)

我的理解是,在启动时,Jetty将使用AnnotationConfiguration来扫描ServletContainerInitializer的注释实现; 它应该找到初始化器并将其连接到......

但是,当我启动Jetty服务器(从Eclipse中)时,我在命令行上看到以下内容:

2012-11-04 16:59:04.552:INFO:oejs.Server:jetty-8.1.7.v20120910
2012-11-04 16:59:05.046:INFO:/:No Spring WebApplicationInitializer types detected on classpath
2012-11-04 16:59:05.046:INFO:oejsh.ContextHandler:started o.e.j.w.WebAppContext{/,file:/Users/duncan/Coding/spring-mvc-embedded-jetty-test/src/main/webapp/}
2012-11-04 16:59:05.117:INFO:oejs.AbstractConnector:Started SelectChannelConnector@0.0.0.0:8080
Run Code Online (Sandbox Code Playgroud)

重要的是这个:

No Spring WebApplicationInitializer types detected on classpath
Run Code Online (Sandbox Code Playgroud)

请注意,src/main/java在Eclipse中定义为源文件夹,因此应该在类路径上.另请注意,Dynamic Web Module Facet设置为3.0.

我确信有一个简单的解释,但我很难看到树木的木头!我怀疑关键是以下行:

...
webAppContext.setResourceBase("src/main/webapp");
...
Run Code Online (Sandbox Code Playgroud)

这对于使用web.xml的2.5 servlet是有意义的(见下文),但使用AnnotationConfiguration时它应该是什么?

注意:如果我将配置更改为以下内容,则所有内容都会正常启动:

...
webAppContext.setConfigurations(new Configuration[] { new WebXmlConfiguration() });
...
Run Code Online (Sandbox Code Playgroud)

在这种情况下,它在src/main/webapp下找到web.xml,并使用它以常规方式使用DispatcherServletAnnotationConfigWebApplicationContext连接 servlet (完全绕过上面的WebApplicationInitializer实现).

这感觉非常类似于类路径问题,但我很难理解Jetty如何将自己与WebApplicationInitializer的实现联系起来 - 任何建议都将是最受欢迎的!

有关信息,我使用以下内容:

Spring 3.1.1 Jetty 8.1.7 STS 3.1.0

Ste*_*vie 26

问题是Jetty的AnnotationConfiguration类不扫描类路径上的非jar资源(WEB-INF/classes下除外).

它发现我WebApplicationInitializer,如果我注册一个子类的AnnotationConfiguration它覆盖configure(WebAppContext)扫描主机类路径中,除了容器和WEB-INF位置.

大多数子类是(遗憾地)从父级复制粘贴.这包括:

  • parseHostClassPath在configure方法的末尾添加一个额外的解析call();
  • parseHostClassPath这在很大程度上是由复制粘贴方法 AnnotationConfigurationparseWebInfClasses;
  • getHostClassPathResource从类加载器中获取第一个非jar URL 的方法(至少对我来说,这是eclipse中我的类路径的文件url).

我使用稍微不同版本的Jetty(8.1.7.v20120910)和Spring(3.1.2_RELEASE),但我想相同的解决方案将起作用.

编辑:我在github中创建了一个工作示例项目,并进行了一些修改(下面的代码在Eclipse中可以正常工作,但在使用阴影jar时运行时却不行) - https://github.com/steveliles/jetty-embedded-spring-mvc-noxml

在OP的JettyServer类中,必要的更改将替换第15行:

webAppContext.setConfigurations (new Configuration []
{
        new AnnotationConfiguration() 
        {
            @Override
            public void configure(WebAppContext context) throws Exception
            {
                boolean metadataComplete = context.getMetaData().isMetaDataComplete();
                context.addDecorator(new AnnotationDecorator(context));   

                AnnotationParser parser = null;
                if (!metadataComplete)
                {
                    if (context.getServletContext().getEffectiveMajorVersion() >= 3 || context.isConfigurationDiscovered())
                    {
                        parser = createAnnotationParser();
                        parser.registerAnnotationHandler("javax.servlet.annotation.WebServlet", new WebServletAnnotationHandler(context));
                        parser.registerAnnotationHandler("javax.servlet.annotation.WebFilter", new WebFilterAnnotationHandler(context));
                        parser.registerAnnotationHandler("javax.servlet.annotation.WebListener", new WebListenerAnnotationHandler(context));
                    }
                }

                List<ServletContainerInitializer> nonExcludedInitializers = getNonExcludedInitializers(context);
                parser = registerServletContainerInitializerAnnotationHandlers(context, parser, nonExcludedInitializers);

                if (parser != null)
                {
                    parseContainerPath(context, parser);
                    parseWebInfClasses(context, parser);
                    parseWebInfLib (context, parser);
                    parseHostClassPath(context, parser);
                }                  
            }

            private void parseHostClassPath(final WebAppContext context, AnnotationParser parser) throws Exception
            {
                clearAnnotationList(parser.getAnnotationHandlers());
                Resource resource = getHostClassPathResource(getClass().getClassLoader());                  
                if (resource == null)
                    return;

                parser.parse(resource, new ClassNameResolver()
                {
                    public boolean isExcluded (String name)
                    {           
                        if (context.isSystemClass(name)) return true;                           
                        if (context.isServerClass(name)) return false;
                        return false;
                    }

                    public boolean shouldOverride (String name)
                    {
                        //looking at webapp classpath, found already-parsed class of same name - did it come from system or duplicate in webapp?
                        if (context.isParentLoaderPriority())
                            return false;
                        return true;
                    }
                });

                //TODO - where to set the annotations discovered from WEB-INF/classes?    
                List<DiscoveredAnnotation> annotations = new ArrayList<DiscoveredAnnotation>();
                gatherAnnotations(annotations, parser.getAnnotationHandlers());                 
                context.getMetaData().addDiscoveredAnnotations (annotations);
            }

            private Resource getHostClassPathResource(ClassLoader loader) throws IOException
            {
                if (loader instanceof URLClassLoader)
                {
                    URL[] urls = ((URLClassLoader)loader).getURLs();
                    for (URL url : urls)
                        if (url.getProtocol().startsWith("file"))
                            return Resource.newResource(url);
                }
                return null;                    
            }
        },
    });
Run Code Online (Sandbox Code Playgroud)

更新:Jetty 8.1.8引入了与上述代码不兼容的内部更改.对于8.1.8,以下似乎有效:

webAppContext.setConfigurations (new Configuration []
    {
        // This is necessary because Jetty out-of-the-box does not scan
        // the classpath of your project in Eclipse, so it doesn't find
        // your WebAppInitializer.
        new AnnotationConfiguration() 
        {
            @Override
            public void configure(WebAppContext context) throws Exception {
                   boolean metadataComplete = context.getMetaData().isMetaDataComplete();
                   context.addDecorator(new AnnotationDecorator(context));   


                   //Even if metadata is complete, we still need to scan for ServletContainerInitializers - if there are any
                   AnnotationParser parser = null;
                   if (!metadataComplete)
                   {
                       //If metadata isn't complete, if this is a servlet 3 webapp or isConfigDiscovered is true, we need to search for annotations
                       if (context.getServletContext().getEffectiveMajorVersion() >= 3 || context.isConfigurationDiscovered())
                       {
                           _discoverableAnnotationHandlers.add(new WebServletAnnotationHandler(context));
                           _discoverableAnnotationHandlers.add(new WebFilterAnnotationHandler(context));
                           _discoverableAnnotationHandlers.add(new WebListenerAnnotationHandler(context));
                       }
                   }

                   //Regardless of metadata, if there are any ServletContainerInitializers with @HandlesTypes, then we need to scan all the
                   //classes so we can call their onStartup() methods correctly
                   createServletContainerInitializerAnnotationHandlers(context, getNonExcludedInitializers(context));

                   if (!_discoverableAnnotationHandlers.isEmpty() || _classInheritanceHandler != null || !_containerInitializerAnnotationHandlers.isEmpty())
                   {           
                       parser = createAnnotationParser();

                       parse(context, parser);

                       for (DiscoverableAnnotationHandler h:_discoverableAnnotationHandlers)
                           context.getMetaData().addDiscoveredAnnotations(((AbstractDiscoverableAnnotationHandler)h).getAnnotationList());      
                   }

            }

            private void parse(final WebAppContext context, AnnotationParser parser) throws Exception
            {                   
                List<Resource> _resources = getResources(getClass().getClassLoader());

                for (Resource _resource : _resources)
                {
                    if (_resource == null)
                        return;

                    parser.clearHandlers();
                    for (DiscoverableAnnotationHandler h:_discoverableAnnotationHandlers)
                    {
                        if (h instanceof AbstractDiscoverableAnnotationHandler)
                            ((AbstractDiscoverableAnnotationHandler)h).setResource(null); //
                    }
                    parser.registerHandlers(_discoverableAnnotationHandlers);
                    parser.registerHandler(_classInheritanceHandler);
                    parser.registerHandlers(_containerInitializerAnnotationHandlers);

                    parser.parse(_resource, 
                                 new ClassNameResolver()
                    {
                        public boolean isExcluded (String name)
                        {
                            if (context.isSystemClass(name)) return true;
                            if (context.isServerClass(name)) return false;
                            return false;
                        }

                        public boolean shouldOverride (String name)
                        {
                            //looking at webapp classpath, found already-parsed class of same name - did it come from system or duplicate in webapp?
                            if (context.isParentLoaderPriority())
                                return false;
                            return true;
                        }
                    });
                }
            }

            private List<Resource> getResources(ClassLoader aLoader) throws IOException
            {
                if (aLoader instanceof URLClassLoader)
                {
                    List<Resource> _result = new ArrayList<Resource>();
                    URL[] _urls = ((URLClassLoader)aLoader).getURLs();                      
                    for (URL _url : _urls)
                        _result.add(Resource.newResource(_url));

                    return _result;
                }
                return Collections.emptyList();                 
            }
        }
    });
Run Code Online (Sandbox Code Playgroud)

  • @RyanStewart:我刚刚提交了它:https://bugs.eclipse.org/bugs/show_bug.cgi?id = 404176 (3认同)

小智 14

通过向AnnotationConfiguration明确提供我想要加载的实现类(MyWebApplicationInitializerImpl),我能够以更简单但更有限的方式解决:

webAppContext.setConfigurations(new Configuration[] {
    new WebXmlConfiguration(),
    new AnnotationConfiguration() {
        @Override
        public void preConfigure(WebAppContext context) throws Exception {
            MultiMap<String> map = new MultiMap<String>();
            map.add(WebApplicationInitializer.class.getName(), MyWebApplicationInitializerImpl.class.getName());
            context.setAttribute(CLASS_INHERITANCE_MAP, map);
            _classInheritanceHandler = new ClassInheritanceHandler(map);
        }
    }
});
Run Code Online (Sandbox Code Playgroud)


Jan*_*Jan 6

Jetty 9.0.1包含一个增强功能,允许在容器类路径上扫描非jar资源(即类)的注释.有关如何使用它,请参阅以下问题的评论#5:

https://bugs.eclipse.org/bugs/show_bug.cgi?id=404176#c5

一月

  • 是否有使用war文件的示例?使用.*和.*/classes /.*不起作用:| 而且我真的没有jetty_home,因为这是嵌入式码头. (2认同)