Spring MockMVC - 如何模拟在控制器之外运行的自定义验证器

led*_*led 3 validation spring unit-testing spring-test spring-test-mvc

@UsernameAlreadyExists
private String username;
Run Code Online (Sandbox Code Playgroud)

我创建了一个自定义验证器,以确保在提交帐户创建表单时应用程序会捕获重复的用户名。

当我使用 MockMVC 对帐户创建控制器进行单元测试时,它失败了,因为验证器依赖于服务,所以我得到空指针异常。

如何模拟验证器或此验证器所依赖的服务?我无法弄清楚如何进行这项工作,因为控制器不明确依赖于验证器,它在控制器之外运行。

tom*_*zyk 8

在不加载整个 Spring 上下文的情况下使用“standaloneSetup”进行测试时,验证由框架调用在内部触发SpringWebConstraintValidatorFactory。为了连接到该流,您需要SpringWebConstraintValidatorFactory在“mockMvc”中设置新实例。
不幸的是,没有一种简单干净的方法可以做到这一点。您必须子类化SpringWebConstraintValidatorFactory并设置您的实例LocalValidatorFactoryBean,然后可以在 mockMvc 中设置实例。但是LocalValidatorFactoryBean会需要ApplicationContext。下面是一个例子:



    public class TestConstrainValidationFactory extends SpringWebConstraintValidatorFactory {

        private final WebApplicationContext ctx;

        private boolean isValid = false;

        public TestConstrainValidationFactory(WebApplicationContext ctx) {
            this.ctx = ctx;
        }

        @Override
        public < T extends ConstraintValidator<?, ?>> T getInstance(Class key) {
            ConstraintValidator instance = super.getInstance(key);
            if (instance instanceof DeviceValidator) {
                DeviceValidator deviceValidator = (DeviceValidator) instance;
                deviceValidator.setYourAutowiredField((String id, String type) -> isValid); //change that to suit your needs
                instance = deviceValidator;
            }
            return (T) instance;
        }

        @Override
        protected WebApplicationContext getWebApplicationContext() {
            return ctx;
        }

    }

Run Code Online (Sandbox Code Playgroud)

将其连接到 mockMvc 的示例



    @RunWith(SpringJUnit4ClassRunner.class)
    @SpringApplicationConfiguration(classes = MockServletContext.class)
    @WebAppConfiguration()
    public class DevicesControllerTest {
        @Autowired
        private MockServletContext servletContext;
        @InjectMocks
        private DevicesController devicesController;
        private TestConstrainValidationFactory constraintFactory;

        @Before
        public void setUp() {
            MockitoAnnotations.initMocks(this);

            final GenericWebApplicationContext context = new GenericWebApplicationContext(servletContext);
            final ConfigurableListableBeanFactory beanFactory = ((ConfigurableApplicationContext) context).getBeanFactory();
            beanFactory.registerSingleton(DeviceValidator.class.getCanonicalName(), new DeviceValidator());
            context.refresh();

            LocalValidatorFactoryBean validatorFactoryBean = new LocalValidatorFactoryBean();
            validatorFactoryBean.setApplicationContext(context);
            constraintFactory = new TestConstrainValidationFactory(context);
            validatorFactoryBean.setConstraintValidatorFactory(constraintFactory);
            validatorFactoryBean.setProviderClass(HibernateValidator.class);
            validatorFactoryBean.afterPropertiesSet();

            this.mockMvc = MockMvcBuilders
                    .standaloneSetup(devicesController)
                    .setValidator(validatorFactoryBean)
                    .setHandlerExceptionResolvers()
                    .build();
        }
    }

Run Code Online (Sandbox Code Playgroud)

拥有后,ReflectionTestUtils.setField(constraintFactory, "isValid", false);将按预期工作,您可以通过工厂在验证器中设置字段。
查看问题上下文spring-projects/spring-test-mvc/issues