【问题标题】:Jersey - How to mock serviceJersey - 如何模拟服务
【发布时间】:2015-02-10 08:41:13
【问题描述】:

我正在使用“Jersey 测试框架”对我的 Web 服务进行单元测试。

这是我的资源类:

import javax.ws.rs.GET;
import javax.ws.rs.Path;
import javax.ws.rs.Produces;

// The Java class will be hosted at the URI path "/helloworld" 

@Path("/helloworld") 
public class class HelloWorldResource {

    private SomeService service;

    @GET 
    @Produces("text/plain")
    public String getClichedMessage() {
        // Return some cliched textual content
        String responseFromSomeService = service.getSomething();
        return responseFromSomeService;
    }
}

如何在单元测试中模拟 SomeService ?

【问题讨论】:

  • 只需使用像 EasyMock 或 JMock 这样的模拟框架。稍后我将尝试通过示例提供答案。
  • 我不确定您要在这里说明什么?我见过有人使用@Inject CDI 作为依赖注入,在我们的测试中,我们可以使用 Mockito.* 来模拟它们。但是我遇到了 CDI + JAX-RS Jersey 的问题。如果您能提供一个例子,我将不胜感激?谢谢
  • 您使用的是哪个 Jersey 版本?

标签: java web-services unit-testing jersey jax-rs


【解决方案1】:

请参阅下面的更新:您不需要Factory


如果您使用 Jersey 2,一种解决方案是使用 Custom Injection and Lifecycle Management 功能(使用 HK2 - Jersey dist 附带)。当然,还需要一个模拟框架。我将使用 Mockito。

首先使用模拟实例创建一个工厂:

public static interface GreetingService {
    public String getGreeting(String name);
}

public static class MockGreetingServiceFactory 
                                     implements Factory<GreetingService> {
    @Override
    public GreetingService provide() {
        final GreetingService mockedService
                = Mockito.mock(GreetingService.class);
        Mockito.when(mockedService.getGreeting(Mockito.anyString()))
                .thenAnswer(new Answer<String>() {
                    @Override
                    public String answer(InvocationOnMock invocation) 
                                                      throws Throwable {
                        String name = (String)invocation.getArguments()[0];
                        return "Hello " + name;
                    }

                });
        return mockedService;
    }

    @Override
    public void dispose(GreetingService t) {}
}

然后使用AbstractBinder将工厂绑定到接口/服务类,并注册绑定器。 (上面的链接都有描述):

@Override
public Application configure() {
    AbstractBinder binder = new AbstractBinder() {
        @Override
        protected void configure() {
            bindFactory(MockGreetingServiceFactory.class)
                               .to(GreetingService.class);
        }
    };
    ResourceConfig config = new ResourceConfig(GreetingResource.class);
    config.register(binder);
    return config;
}

看起来很多,但这只是一种选择。我对测试框架不太熟悉,或者它是否具有用于注入的模拟功能。

这是完整的测试:

import javax.inject.Inject;
import javax.ws.rs.GET;
import javax.ws.rs.Path;
import javax.ws.rs.Produces;
import javax.ws.rs.QueryParam;
import javax.ws.rs.client.Client;
import javax.ws.rs.client.ClientBuilder;
import javax.ws.rs.core.Application;
import javax.ws.rs.core.MediaType;
import javax.ws.rs.core.Response;
import org.glassfish.hk2.api.Factory;
import org.glassfish.hk2.utilities.binding.AbstractBinder;
import org.glassfish.jersey.server.ResourceConfig;
import org.glassfish.jersey.test.JerseyTest;
import org.junit.Assert;
import org.junit.Test;
import org.mockito.Mockito;
import org.mockito.invocation.InvocationOnMock;
import org.mockito.stubbing.Answer;

public class ServiceMockingTest extends JerseyTest {

    @Path("/greeting")
    public static class GreetingResource {

        @Inject
        private GreetingService greetingService;

        @GET
        @Produces(MediaType.TEXT_PLAIN)
        public String getGreeting(@QueryParam("name") String name) {
            return greetingService.getGreeting(name);
        }
    }

    public static interface GreetingService {
        public String getGreeting(String name);
    }
    
    public static class MockGreetingServiceFactory 
                                  implements Factory<GreetingService> {
        @Override
        public GreetingService provide() {
            final GreetingService mockedService
                    = Mockito.mock(GreetingService.class);
            Mockito.when(mockedService.getGreeting(Mockito.anyString()))
                    .thenAnswer(new Answer<String>() {
                        @Override
                        public String answer(InvocationOnMock invocation) 
                                                       throws Throwable {
                            String name = (String)invocation.getArguments()[0];
                            return "Hello " + name;
                        }

                    });
            return mockedService;
        }

        @Override
        public void dispose(GreetingService t) {}
    }

    @Override
    public Application configure() {
        AbstractBinder binder = new AbstractBinder() {
            @Override
            protected void configure() {
                bindFactory(MockGreetingServiceFactory.class)
                        .to(GreetingService.class);
            }
        };
        ResourceConfig config = new ResourceConfig(GreetingResource.class);
        config.register(binder);
        return config;
    }
    
    @Test
    public void testMockedGreetingService() {
        Client client = ClientBuilder.newClient();
        Response response = client.target("http://localhost:9998/greeting")
                .queryParam("name", "peeskillet")
                .request(MediaType.TEXT_PLAIN).get();
        Assert.assertEquals(200, response.getStatus());
        
        String msg = response.readEntity(String.class);
        Assert.assertEquals("Hello peeskillet", msg);
        System.out.println("Message: " + msg);
        
        response.close();
        client.close();
       
    }
}

此测试的依赖项:

<dependency>
    <groupId>org.glassfish.jersey.test-framework.providers</groupId>
    <artifactId>jersey-test-framework-provider-grizzly2</artifactId>
    <version>2.13</version>
</dependency>
<dependency>
    <groupId>org.mockito</groupId>
    <artifactId>mockito-all</artifactId>
    <version>1.9.0</version>
</dependency>

更新

所以在大多数情况下,您真的不需要Factory。您可以简单地将模拟实例与其合同绑定:

@Mock
private Service service;

@Override
public ResourceConfig configure() {
    MockitoAnnotations.initMocks(this);
    return new ResourceConfig()
        .register(MyResource.class)
        .register(new AbstractBinder() {
            @Override
            protected configure() {
                bind(service).to(Service.class);
            }
        });
}

@Test
public void test() {
    when(service.getSomething()).thenReturn("Something");
    // test
}

简单多了!

【讨论】:

  • GreetingResource.java 中有什么?
  • 不知何故,在我的情况下,模拟并不成功。在您编写的单元测试中,它没有模拟 .getGreeting 方法。
  • 查看整个测试。 GreetingResource 是资源类。在我编写的代码中,它 模拟了GreetingServiceGreetingService 是一个简单的接口,它被注入到GreetingResource 中。 Factory 创建模拟实例,绑定器将工厂绑定到GreetingService,这样当使用GreetingService 时,模拟实例将被注入。请运行整个测试。如果它没有被嘲笑,那么你将看不到任何结果。
  • 在测试中,你会看到(按此顺序-请仔细检查)(1)GreetingResource(Jax-RS 资源类)(2)一个简单的接口GreetingService(即是注入的内容)(3)Factory 类,它模拟了GreetingService。 (4)AbstractBinder将工厂绑定到GreetingService的应用配置。 (5) 测试。如果您对测试有其他不明白的地方,请告诉我。它对我来说很好,它应该如何
  • 只添加了一个依赖项:模拟工具。如果您只使用 Jersey 并且不想在其上添加其他工具,则此答案是一个不错的答案。
【解决方案2】:

下面是我使用 Jersey 2.20、Spring 4.1.4 RELEASE、Mockito 1.10.8 和 TestNG 6.8.8 的方法。

@Test
public class CasesResourceTest extends JerseyTestNg.ContainerPerMethodTest {

    @Mock
    private CaseService caseService;

    @Mock
    private CaseConverter caseConverter;    

    @Mock
    private CaseRepository caseRepository;

    private CasesResource casesResource;

    @Override
    protected Application configure() {

        MockitoAnnotations.initMocks(this);
        casesResource = new CasesResource();

        AbstractBinder binder = new AbstractBinder() {

            @Override
            protected void configure() {
                bindFactory(new InstanceFactory<CaseConverter>(caseConverter)).to(CaseConverter.class);
                bindFactory(new InstanceFactory<CaseService>(caseService)).to(CaseService.class);
            }
        };

        return new ResourceConfig()
            .register(binder)
            .register(casesResource)
            .property("contextConfigLocation", "solve-scm-rest/test-context.xml");
    }

    public void getAllCases() throws Exception {

        when(caseService.getAll()).thenReturn(Lists.newArrayList(new solve.scm.domain.Case()));
        when(caseConverter.convertToApi(any(solve.scm.domain.Case.class))).thenReturn(new Case());

        Collection<Case> cases = target("/cases").request().get(new GenericType<Collection<Case>>(){});

        verify(caseService, times(1)).getAll();
        verify(caseConverter, times(1)).convertToApi(any(solve.scm.domain.Case.class));

        assertThat(cases).hasSize(1);
    }
}

你还需要这个类来简化上面的绑定代码:

public class InstanceFactory<T> implements Factory<T> {

    private T instance;

    public InstanceFactory(T instance) {
        this.instance = instance;
    }

    @Override
    public void dispose(T t) {
    }

    @Override
    public T provide() {
        return instance;
    }

}

编辑为公关。要求。这是我的 test-context.xml 的内容:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
        xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">

</beans>

事实证明,我的 test-context.xml 没有实例化任何 bean,也没有扫描任何包,事实上,它根本没有做任何事情。我想我只是把它放在那里以备不时之需。

【讨论】:

  • 优雅的解决方案,节省了我很多时间,因为没有太多关于此的信息......
  • 你的 test-context.xml 文件中有什么?
  • 您可以更进一步,为服务的资源添加一个设置器并以这种方式注入它。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2015-09-19
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2023-03-03
相关资源
最近更新 更多