我发现一个Web应用程序运行良好。现在我正在尝试为它编写单元测试。我的网络应用程序有以下conversionService
<bean id="conversionService" class="org.springframework.context.support.ConversionServiceFactoryBean">
<property name="converters">
<list>
<bean class="....Class1ToStringConverter"/>
<bean class="....StringToClass1Converter"/>
</list>
</property>
</bean>
<mvc:annotation-driven conversion-service="conversionService" />
这很好,当我向提出请求时
/somepath/{class1-object-string-representation}/xx
一切都按预期进行(字符串被解释为Class1对象)。
我的问题是试图给我的控制器写一个单元测试。conversionService只是没有使用,spring只是告诉我
Cannot convert value of type [java.lang.String] to required type [Class1]: no matching editors or conversion strategy found
到目前为止我的测试:
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations = {"file:src/main/webapp/WEB-INF/applicationContext.xml", "file:src/main/webapp/WEB-INF/jpm-servlet.xml"})
@WebAppConfiguration()
public class GeneralTest {
@Autowired
private WebApplicationContext ctx;
private MockMvc mockMvc;
private TestDAO testDAO = org.mockito.Mockito.mock(TestDAO.class);
@Before
public void setUp() throws Exception {
Mockito.reset(testDAO);
mockMvc = MockMvcBuilders.webAppContextSetup(ctx).build();
}
@Test
public void testList() throws Exception {
final Test first = new Test(1L, "Hi", 10, new Date(), true);
final Test second = new Test(2L, "Bye", 50, new Date(), false);
first.setTest(second);
when(testDAO.list()).thenReturn(Arrays.asList(first, second));
mockMvc.perform(get("/jpm/class1-id1"))
.andExpect(status().isOk())
.andExpect(view().name("list"))
.andExpect(forwardedUrl("/WEB-INF/jsp/list.jsp"));
}
我缺少什么?感谢
模拟转换器像这样,
GenericConversionService conversionService = new GenericConversionService();
conversionService.addConverter(new StringToClass1Converter());
Deencapsulation.setField(FIXTURE, conversionService);
这篇文章有点过时了,但在搜索这个问题时,它仍然是谷歌的首批结果之一。
因此,这里是SpringFramework5的更新。
当您像Spring文档中所记录的那样配置WebMvc上下文时,您可以编写这样的内容:
@Configuration
@EnableWebMvc
public class WebConfig implements WebMvcConfigurer {
@Override
public void addFormatters(FormatterRegistry registry) {
// ...
}
}
但是,这不是在JUnit上下文中加载的!天知道为什么。但是您需要这样声明您的配置类:
@Configuration
public class WebConfig extends WebMvcConfigurationSupport {
@Override
public void addFormatters(FormatterRegistry registry) {
// ...
}
}
因此,如上所述,删除@EnableWebMvc
注释并从另一个类进行扩展。你可能不需要改变其他任何事情。然后在单元测试中也会调用addFormatters
方法。
这是非常不直观和奇怪的。但这是真的。您可以使用断点调试spring测试源代码,并看到WebMvcConfigurer
接口的所有其他方法都被调用了,但addFormatters
没有。也许他们只是忘记了叫它,或者他们有其他"原因"。通过从WebMvcConfigurationSupport
扩展,每个方法都可以根据需要进行调用,JUnit测试就会成功。
特别是,这个代码最终会被执行:
@Bean
public FormattingConversionService mvcConversionService() {
FormattingConversionService conversionService = new DefaultFormattingConversionService();
addFormatters(conversionService);
return conversionService;
}
老实说,在实现Spring文档中描述的接口时,我不知道到底是什么失败了。
我遇到了同样的问题。如果你试图测试单个控制器,你可以尝试以下操作:
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations = {"file:src/main/webapp/WEB-INF/applicationContext.xml", "file:src/main/webapp/WEB-INF/jpm-servlet.xml"})
@WebAppConfiguration()
public class GeneralTest {
@Autowired
private WebApplicationContext ctx;
@Autowired
private FormattingConversionServiceFactoryBean conversionService;
private MockMvc mockMvc;
@Mock
private TestDAO testDAO;
/* The following assumes you are injecting your DAO into your controller
* If you are using a service layer (most likely), you should
* inject your DAO into your service and your service into your controller.
*/
@InjectMocks
private YourControllerClass controllerToTest;
@Before
public void setUp() throws Exception {
MockitoAnnotations.initMocks(this);
//get the conversion service from the factory bean
FormattingConversionService cs = conversionService.getObject();
Mockito.reset(testDAO);
//setup MockMvc using the conversion service
mockMvc = MockMvcBuilders.standaloneSetup(controllerToTest)
.setConversionService(cs)
.build();
}
@Test
public void testList() throws Exception {
final Test first = new Test(1L, "Hi", 10, new Date(), true);
final Test second = new Test(2L, "Bye", 50, new Date(), false);
first.setTest(second);
when(testDAO.list()).thenReturn(Arrays.asList(first, second));
mockMvc.perform(get("/jpm/class1-id1"))
.andExpect(status().isOk())
.andExpect(view().name("list"))
.andExpect(forwardedUrl("/WEB-INF/jsp/list.jsp"));
}
希望能有所帮助!
我意识到这是一个旧线程,但和我一样,在花了几个小时解决为什么他们的自定义转换器没有被调用后,将来还会有其他人遇到这个问题。
@Mariano D'Ascanio提出的解决方案在根本没有控制器的情况下是不够的,至少没有你写的控制器,比如当你使用Spring JPA时。MockMvcBuilders.standaloneSetup
要求至少有一个控制器传递给构造函数,所以在这种情况下不能使用它。解决这个问题的方法是注入org.springframework.core.convert.converter.ConverterRegistry
,或者更好的是,它是org.springframework.core.convert.converter.FormatterRegistry
的子类,然后在@PostContruct
方法中,注册您的自定义转换器/格式化程序,如下所示:
@PostConstruct
void init() {
formatterRegistry.removeConvertible(String.class, OffsetDateTime.class);
formatterRegistry.addFormatter(customOffsetDateTimeFormatter);
formatterRegistry.addConverter(customOffsetDateTimeConverter);
}
诀窍是使用名称而不是类型来注入ConverterRegistry
,因为对于web测试,有两个转换器注册表,默认注册表和mvc注册表。Spring测试使用默认的转换器,所以这就是您需要的转换器。
// GOTCHA ALERT: There's also a mvcConversionService; tests DO NOT use that
@Resource(name = "defaultConversionService")
private FormatterRegistry formatterRegistry;
希望这能有所帮助。