如何使用Headers模拟HttpServletRequest?

Sat*_*Sat 10 java unit-testing mockito

我正在使用Mockito和Junit来测试应用程序,我需要在模拟时向HttpServletRequest添加标头.这是我第一次使用模拟概念来测试应用程序.在使用这个模拟概念时,我们如何设置标题来请求对象?

Java代码:

@Produces({ MediaType.APPLICATION_JSON })
@Path("/devices")
public class DvrRestService {
    private static final Logger logger = LoggerFactory.getLogger(DvrRestService.class);

    private DvrMiddleService dvrMiddleService;

    @Inject
    public DvrRestService(DvrMiddleService dvrMiddleService) {
        this.dvrMiddleService = dvrMiddleService;
    }

    @GET 
    @Path("/{deviceId}/metadata")
    public Response getDeviceMetadata(@Context HttpServletRequest request, @PathParam("deviceId") String deviceId,
        @RequiredSession final Session session) {

        try {
            public static String[] REQUEST_HEADERS = { "if-none-match" };
            List<String> requiredHeaders = Lists.newArrayList(REQUEST_HEADERS);
            Map<String, String> headers = new HashMap<String, String>();

            Enumeration<String> headerNames = request.getHeaderNames();
            while (headerNames.hasMoreElements()) { // here gettting NullPointerException

                String headerName = headerNames.nextElement();
                if (requiredHeaders.contains(headerName.toLowerCase())) {
                    String value = request.getHeader(headerName);
                    if (value != null) {
                        headers.put(headerName, value);
                        System.out.println("headerName: " + headerName + ", Value: " + value);
                    }
                }
            }
            DvrResponse response = dvrMiddleService.getDeviceMetadata(deviceId.toUpperCase(), getHeaders(request));
            return processResponse(response.statusCode, response.getResponse(), DeviceMetadataResponse.class,
                response.getHeaders());
        } catch (Exception e) {
            return processErrorResponse(e, new DeviceMetadataResponse(), logger);
        }

    }
}
Run Code Online (Sandbox Code Playgroud)

这是我的测试代码:

public class DvrRestServiceTest {

    static DvrMiddleService dms;
    static HttpServletRequest request;
    static Session session;
    static DvrRestService drs;
    public static final String DeviceId = "000004D42070";
    @BeforeClass
    public static void init(){
        dms = mock(DvrMiddleService.class);
        request = mock(HttpServletRequest.class);
        session = mock(Session.class);
        drs = new DvrRestService(dms);
    }
    @Test
    public void getDeviceMetadataTest(){
        Response rs = drs.getDeviceMetadata(request, DeviceId, session);
        assertEquals(Response.Status.OK, rs.getStatus());
    }
}
Run Code Online (Sandbox Code Playgroud)

Sub*_*mal 13

作为校长的起点和演示,您可以从以下片段开始.

// define the headers you want to be returned
Map<String, String> headers = new HashMap<>();
headers.put(null, "HTTP/1.1 200 OK");
headers.put("Content-Type", "text/html");

// create an Enumeration over the header keys
Iterator<String> iterator = headers.keySet().iterator();
Enumeration headerNames = new Enumeration<String>() {
    @Override
    public boolean hasMoreElements() {
        return iterator.hasNext();
    }

    @Override
    public String nextElement() {
        return iterator.next();
    }
};

// mock HttpServletRequest
HttpServletRequest request = mock(HttpServletRequest.class);
// mock the returned value of request.getHeaderNames()
when(request.getHeaderNames()).thenReturn(headerNames);

System.out.println("demonstrate output of request.getHeaderNames()");
while (headerNames.hasMoreElements()) {
    System.out.println("header name: " + headerNames.nextElement());
}

// mock the returned value of request.getHeader(String name)
doAnswer(new Answer<String>() {
    @Override
    public String answer(InvocationOnMock invocation) throws Throwable {
        Object[] args = invocation.getArguments();
        return headers.get((String) args[0]);
    }
}).when(request).getHeader("Content-Type");

System.out.println("demonstrate output of request.getHeader(String name)");
String headerName = "Content-Type";
System.out.printf("header name: [%s]   value: [%s]%n", 
        headerName, request.getHeader(headerName));
}
Run Code Online (Sandbox Code Playgroud)

产量

demonstrate output of request.getHeaderNames()
header name: null
header name: Content-Type

demonstrate output of request.getHeader(String name)
header name: [Content-Type]   value: [text/html]
Run Code Online (Sandbox Code Playgroud)

  • 通过调用 java.util.Collections#enumeration 创建枚举不是更短吗? (2认同)
  • 哇伙计!!谢谢!!我从“.m2”中删除了 jar 并再次下载。那行得通。 (2认同)

M. *_*tin 12

对于HttpServletRequest,我建议使用功能齐全的模拟类型,而不是直接使用 Mockito 模拟来模拟它。为此,图书馆spring-test设有:MockHttpServletRequest

@BeforeClass
public static void init(){
    // ...

    MockHttpServletRequest mockRequest = new MockHttpServletRequest();
    mockRequest.addHeader("Content-Type", "text/html");
    mockRequest.addHeader("if-none-match", "*");
    mockRequest.addHeader("customHeader", "customValue");

    this.request = mockRequest;
}
Run Code Online (Sandbox Code Playgroud)

基本原理

HttpServletRequest是一个复杂的接口,包含 20 多个方法,并且它们之间有明确定义的相互作用。使用库中的功能齐全的模拟类型HttpServletRequest可以简化模拟,无需仔细模拟您正在使用的方法。

这种方法的一个优点是,面对未来使用类上的其他方法获取相同信息的重构,它更具弹性。在检索 中的“if-none-match”标头的情况下HttpServletRequest,我看到三种可以合法用于检索标头的不同方法:getHeader(String name)getHeaders(String name)getHeaderNames()getHeader此外, 和的参数都不getHeaders区分大小写(“if-none-match”、“If-None-Match”等返回相同的结果),因此任何可能的参数大小写都是正确的。很可能通过直接模拟来支持这一点,尽管它涉及额外的样板代码,使测试变得复杂并且使其不那么明显。

MockHttpServletRequest库中的类模拟spring-test此接口,并允许通过简单的 API 设置标头和其他值。虽然该库是为测试Spring应用程序而设计的,但该类MockHttpServletRequest独立于任何特定于 Spring 的功能,并且即使应用程序不使用 Spring,它本身也应该完全可用。