1

在实际应用程序中,我使用org.springframework.web.multipart.commons.CommonsMultipartResolver我定义maxUploadSize参数的位置。所以我想MaxUploadSizeExceededException使用 spring mvc 测试来测试捕获。

为此,我org.springframework.web.multipart.support.MultipartFilter在 Java 中添加:

@Autowired
private WebApplicationContext wac;

@Autowired
private MultipartFilter multipartFilter;

private MockMvc mockMvc;

@Before
public void init() {
    mockMvc = MockMvcBuilders.webAppContextSetup(wac)
        .addFilter(multipartFilter).build();
}

在测试conext(基于xml的配置):

<bean id="multipartFilter" class="org.springframework.web.multipart.support.MultipartFilter">
    <property name="multipartResolverBeanName" value="MyMultipartResolverBeanName" />
</bean>

MyMultipartResolverBeanName bean 在现实生活中定义,使用 maven-surefire-plugin 添加到测试上下文中。

我的测试就像在教程中一样:

@Test(expected = MaxUploadSizeExceededException.class)
public void testCertSizeInvalid() throws Exception {
    mockMvc.perform(fileUpload(SUBMIT_URL)
        .file(new MockMultipartFile("fileName", new byte[MAX_UPLOAD_SIZE + 10]))
    );
}

运行时我得到原因:org.apache.commons.fileupload.FileUploadException:请求被拒绝,因为没有找到多部分边界org.apache.commons.fileupload.FileUploadBase

据我了解,在现实生活中,浏览器在 Content-Type 标头中设置了唯一的边界,并将正文中的部分与此边界分开,但 spring 模拟测试在发送模拟多部分请求时不会设置边界。

所以我发现org.springframework.http.converter.FormHttpMessageConverter,创建org.springframework.test.web.servlet.request.RequestPostProcessor转换正文和 Content-Type 标头以更正(带边界)并将后处理器添加到请求生成器的位置 -测试通过了!.

也许有人知道更好或更简单的方法来为 Content-Type 标头和模拟请求正文添加边界?!

PS
如果没有更好的方法 - 我可以在此处附上我的 RequestPostProcessor
使用的版本:

  • 弹簧测试3.2.0
  • commons-fileupload-1.2.2
4

1 回答 1

1

我找到了一种很好的干净方法来测试 MaxUploadSizeExceededException

它涉及使用 Mockito 监视 MultipartResolver 以强制抛出异常。

  • 使用内部 @Configuration 类,连接一个 Mockito Spy 的多部分解析器。
  • 将解析器自动连接到您的课程中
  • 对于 MaxUploadSizeExceededException 测试用例,存根解析器以引发异常,然后执行内容类型为 MediaType.MULTIPART_FORM_DATA 的发布

这是代码。

@RunWith(SpringJUnit4ClassRunner.class)
@WebAppConfiguration
@ContextConfiguration(classes = {SpringMVCConfiguration.class, UploadTest.TestConfiguration.class})
@SuppressWarnings("unchecked")
public final class UploadTest {

    public static final String MERCHANT_AUTH = "MERCHANT_AUTH";
    public static final String USER = "user";

    @Configuration
    public static class TestConfiguration {
        @Bean
        public MultipartResolver multipartResolver() {
            CommonsMultipartResolver resolver = new CommonsMultipartResolver();
            resolver.setMaxUploadSize(500 * 1028);
            return Mockito.spy(resolver);
        }
    }

    @Autowired
    private WebApplicationContext m_webApplicationContext;

    @Autowired
    private CommonsMultipartResolver m_multipartResolver;

    private MockMvc m_mockMvc;

    @Before
    public void setUp() throws Exception {
        m_mockMvc = MockMvcBuilders.webAppContextSetup(m_webApplicationContext).build();
    }

    @Test
    public void testName() throws Exception {
        Mockito.reset(m_multipartResolver);
    }

    @Test
    public void testUpload_FileTooLarge() throws Exception {

        /*
        Force multipart resolver to throw file too large exception
        */
        doThrow(new MaxUploadSizeExceededException(100)).when(m_multipartResolver).resolveMultipart(
            any(HttpServletRequest.class));

        /*
            Do a multipart post. Cannot use fileUpload as that skips calling the MultipartResolver
         */
        m_mockMvc.perform(post("/upload/")
                .contentType(MediaType.MULTIPART_FORM_DATA)
                .accept(MediaType.APPLICATION_JSON))
            .andExpect(status().isRequestEntityTooLarge())
            .andExpect(content().contentTypeCompatibleWith(MediaType.APPLICATION_JSON));

    }
}    
于 2013-11-08T02:07:45.670 回答