Guice和Wicket:使用SessionScoped注入

时间:2013-10-28 15:30:42

标签: java servlets wicket guice wicket-tester

我有一个使用Guice [v3]的Wicket [v6]应用程序 - 我现在已经使用依赖注入存储库操作,我想将它用于使用会话范围的服务(每个用户的会话一个)。我在这里阅读了官方文档,各种博客文章和问题,但我不确定我是否使用了正确的方法。

我有两个问题: 1.我使用正确的方法吗? 2.在依赖SessionScoped注入的类上运行TestNG测试需要什么特殊的东西吗?

我的设置: web.xml中:

<filter>
    <filter-name>guiceFilter</filter-name>
    <filter-class>com.google.inject.servlet.GuiceFilter</filter-class>
</filter>    
<filter-mapping>
    <filter-name>guiceFilter</filter-name>
    <url-pattern>/*</url-pattern>
</filter-mapping>
<listener>
    <listener-class>com.xxx.CustomServletConfig</listener-class>

MyApplication init:

@Override
protected void init()
{
    super.init();
    getResourceSettings().setResourcePollFrequency(null);
    getMarkupSettings().setStripWicketTags(true);
    getDebugSettings().setDevelopmentUtilitiesEnabled(true);
    GuiceComponentInjector injector = new GuiceComponentInjector(this, new WebModule(), new GuiceModule());;
}

CustomServletConfig:

public class CustomServletConfig  extends GuiceServletContextListener {

    @Override
    protected Injector getInjector() {
        return Guice.createInjector(new GuiceModule(), new WebModule());
    }

WebModule:

public static class WebModule extends ServletModule {

    @Override
    protected void configureServlets() {
        bind(WebApplication.class).toProvider(WicketGuiceAppProvider.class).asEagerSingleton();  

        bind(IUserService.class).to(UserService.class).in(ServletScopes.SESSION);

        Map<String, String> params = new HashMap<String, String>();    
        params.put(WicketFilter.FILTER_MAPPING_PARAM, "/*");  

        filter("/*").through(WicketGuiceFilter.class, params);  
    }
}

在示例页面中,我有:

@Inject
IUserService userService

...

userService.doSomething

在单元测试期间的userService.doSomething我收到Guice OutOfScopeException,指向我在ServletModule中的绑定:  自定义提供程序中的错误,com.google.inject.OutOfScopeException?:无法访问作用域对象。我们目前不在HTTP Servlet请求中,或者您可能忘记应用com.google.inject.servlet.GuiceFilter?作为此请求的servlet过滤器。

我的配置是否正常,我需要以不同方式运行单元测试(我只是使用WicketTester启动我的应用程序),还是我的设计有问题?

1 个答案:

答案 0 :(得分:3)

这是非常常见的错误。

ServletScopesRequestScopes中的所有实体都应以Providers传递。

所以你的代码应该是:

@Inject
Provider<IUserService> userServiceProvider

public IUserService getUserService() {
  userServiceProvider.get(); 
}

为什么这样?!只要你在Stage.DEVELOPMENT中使用它就一切都很好,并且不会急切地创建父类。如果将父类绑定为asEagerSingleton或切换到Stage.PRODUCTION,则会在启动时急切地创建类。否则,只有在访问它们时才会以懒惰方式创建它们(很可能在第一次请求期间)。

你的问题出现了。您的WebApplication在启动时急切地初始化。然后guice尝试注入所有子依赖项,并在IUserService中找到SessionScope字段注入。问题是你目前不在GuiceFilter里面并且没有请求,因此guice无法确定当前会话或创建新会话。所以这些范围无法达成。您当前在ContextListener,并且您的应用程序急切地被实例化。如果由于延迟加载而只使用Singleton而不是asEagerSingleton,那么一切都会很好。

无论如何,将Session和Request范围对象作为提供者传递是最佳实践。您可以详细了解Providers hereScopes here (there is also nice table with eager vs. lazy loading comparsion)