How to test a controller with constructor injection by MockMvc

这一生的挚爱 提交于 2019-12-14 00:52:41

问题


I have a controller with constructor injection

@RestController
@RequestMapping("/user")
public class MainController {

    private final UserMapper userMapper; // autowired by constructor below

    public MainController(UserMapper userMapper) {
        this.userMapper = userMapper;
    }

    @RequestMapping("/getChannels")
    public String index() {
        LoginUser user = userMapper.getUserByName("admin");
        return "Channels: " + user.getChannels();
    }
}

It's a simple class which is working fine. However when I tried to run a JUnit testing with below class I got an error.

@RunWith(SpringJUnit4ClassRunner.class)
@WebAppConfiguration
public class MainControllerTest {

    private MockMvc mvc;
    private final UserMapper userMapper;

    public MainControllerTest(UserMapper userMapper) {
        this.userMapper = userMapper;
    }

    @Before
    public void setUp() throws Exception {
        mvc = MockMvcBuilders.standaloneSetup(new MainController(userMapper)).build();
    }

    ......

The error is:

java.lang.Exception: Test class should have exactly one public zero-argument constructor

I was confused by above error message that how could I inject the userMapper with a zero-argument constructor? I know it's possible to add the @Autowired for userMapper in MainController however the field injection is not recommended. Please could anybody guide me a suitable way for both constructor injection and MockMvc testing. Thanks.


回答1:


Other answers talk about using annotations, but here your problem doesn't have any relation to using annotations. keep in mind as spring 4.3+ you don't need to annotate constructor for dependencies, see more here.

In fact You don't need to try to simulate constructor injection in your Test class (MainControllerTest). All you need is to declaring UserMapper as spring component in your application context and in your test class it will be auto injected in your controller as your running application.

Whats your error mean: All of Junit Test classes as error message says should have exactly one public zero-argument constructor that's because Junit Test suites in cases like your scenario doesn't know how to instantiate Test class.




回答2:


Use @Autowire on the field. Field injection is not recommended in production code, because it makes reasoning messy (what was injected where and why). But test contexts (and especially those like this one) are simple, so there the pitfall of hard reasoning is not there.

@Autowired
private final UserMapper userMapper;

public MainControllerTest() { //remove me, implicit
}

This change will fix the problem, as jUnit will have a constructor it understands and will be able to instantiate the test class.




回答3:


You re missing @Autowired for the MainController constructor and always use the constructor injection as shown below to resolve the issue:

@RestController
@RequestMapping("/user")
public class MainController {

    private final UserMapper userMapper;

    @Autowired
    public MainController(UserMapper userMapper) {
        this.userMapper = userMapper;
    }

    //add methods here
}

It is NOT a best practice to use the field injection, look here for more details.

Also, to fix the issue, you need to update your Test class as shown below:

@RunWith(SpringJUnit4ClassRunner.class)
@WebAppConfiguration
public class MainControllerTest {

    private MockMvc mvc;

    @Mock
    private UserMapper userMapper;

    @InjectMocks
    private MainController mainController = new MainController(userMapper);

    @Before
    public void setUp() throws Exception {
        MockitoAnnotations.initMocks(this);
        mvc = MockMvcBuilders.standaloneSetup(mainController)..build();
    }

    ......
}


来源:https://stackoverflow.com/questions/42884466/how-to-test-a-controller-with-constructor-injection-by-mockmvc

易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!