Play 2.4

时间:2015-08-29 19:01:13

标签: java unit-testing guice playframework-2.4

所以我一直试图通过文档来解决这个问题,但是我没有到达任何地方。

我在一个创建存储库对象的服务类中设置了一些简单的DI绑定。简单。但是,当我在测试模式下运行它时,@ Inject什么都不做,并且从不实例化存储库对象。

@Inject
TagRepository tagRepository;

所以在它使用的那一行,在测试模式下,我们当然得到一个NullPointerException

tagRepository.tagExistsByName(tag);

这就像我的测试一样:

[error] Test services.TagsServiceTest.testAddNewTag failed: java.lang.NullPointerException: null, took 0.097 sec
[error]     at services.TagService.tagExists(TagService.java:27)
[error]     at services.TagService.addNewTag(TagService.java:18)
[error]     at services.TagsServiceTest.testAddNewTag(TagsServiceTest.java:29)

我的问题是,如何配置我的应用程序以在测试模式下使用Guice注射器?我的控制器没有这个问题,因为实际上正在向他们发出请求,设置了完整的应用程序。

我应该提到的一点是,我正在使用提供商为测试提供我的应用程序。我应该使用Guice应用程序构建器吗?如果是这样,那会去哪儿?在这方面,剧本文档并不是很有用。这是提供者

@Override
protected FakeApplication provideFakeApplication() {
    return new FakeApplication(new java.io.File("."), Helpers.class.getClassLoader(), ImmutableMap.of("play.http.router", "router.Routes"), new ArrayList<String>(), null);
}

更新:

以下是基于以下建议的更新

在我的BaseTest类中

    @Override
    protected Application provideApplication() {
        return new GuiceApplicationBuilder().in(Mode.TEST).build();
    }

然后在服务测试类中

    @Before
    public void beforeTest() {
        Injector injector = new GuiceInjectorBuilder().bindings(bind(TagService.class).toInstance(new TagService())).injector();
        tagService = injector.instanceOf(TagService.class);
    }

但是,我仍然得到空指针异常,因为TagRepository没有被注入。

解答:

我在想这个有点不对劲。如果您使用需要注入的对象设置注入器,然后从中创建一个实例,您将不再获得NullPointerExceptions

@Before
public void beforeTest() {
    Injector injector = new GuiceInjectorBuilder().bindings(bind(TagRepository.class).toInstance(new TagRepository())).injector();
    tagService = injector.instanceOf(TagService.class);
}

1 个答案:

答案 0 :(得分:6)

如果您正在WithApplication展开,则可以覆盖protected Application provideApplication()以返回使用Application构建的GuiceApplicationBuilder

例如(基于this code)您可以创建应用,添加或覆盖绑定等,并设置模式。如果你使类成为抽象,它将自动与所有子类一起工作。

public abstract class AbstractFakeApplicationTest extends WithApplication
{
    private static final Logger LOGGER = LoggerFactory.getLogger(AbstractFakeApplicationTest.class);

    @Override
    protected Application provideApplication()
    {
        return new GuiceApplicationBuilder().in(Mode.TEST)
                                            .build();
    }

    @Override
    public void startPlay()
    {
        super.startPlay();
        // mock or otherwise provide a context
        Http.Context.current.set(new Http.Context(1L,
                                                  Mockito.mock(RequestHeader.class),
                                                  Mockito.mock(Http.Request.class),
                                                  Collections.<String, String>emptyMap(),
                                                  Collections.<String, String>emptyMap(),
                                                  Collections.<String, Object>emptyMap()));
    }

    public Http.Context context()
    {
        return Http.Context.current.get();
    }
}

子类然后只是扩展此类并正常测试 - 所有DI都应该像正常运行应用程序时那样发生。

您可以看到它的各种示例here

这给出了您需要做的基本概要。希望https://playframework.com/documentation/2.4.x/JavaTestingWithGuice的文档现在会更有意义。