模拟存储库返回null

时间:2016-03-15 01:39:35

标签: java unit-testing spring-mvc mockito

我使用springMVC并且我想测试我的服务,但我的@Mock存储库不能正常工作,所以我永远不会得到我的用户,因为我的存储库为null,我该如何解决这个问题?我在注释中做错了吗?

存储库接口:

package br.com.api.model.data;

import br.com.api.model.resources.User;

import java.util.List;

public interface UserDao {
    User findByLoginAndPassword(String login, String password);
    List<User> listAll();
}

存储库实施:

package br.com.api.model.repository;
import br.com.api.model.data.UserDao;
import br.com.api.model.resources.User;
import org.springframework.stereotype.Repository;

import java.util.ArrayList;
import java.util.List;


@Repository
public class UserRepository implements UserDao {

    public User findByLoginAndPassword(String login, String password) {
        System.out.println("lets find:");
        User userMatch = new User();
        List<User> users = usersMockup();

        for (User user : users) {
            if(user.getLogin().equals(login)
                    && user.getPassword().equals(password)){
                userMatch = user;
            }
        }

        return userMatch;
    }

    public List<User> listAll() {
        List<User> users = usersMockup();

        return users;
    }


    private List<User> usersMockup(){
        List<User> users = new ArrayList<User>();

        User first = new User();
        first.setLogin("first");
        first.setPassword("teste");

        User scnd = new User();
        scnd.setLogin("second");
        scnd.setPassword("teste");

        users.add(first);
        users.add(scnd);

        return users;
    }
}

这是我的服务:

package br.com.api.model.services;


import br.com.api.model.data.UserDao;
import br.com.api.model.resources.User;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;

@Service
public class UserService {

    @Autowired
    private UserDao userRepository;

    public User findUser(String login, String password) {
        User userMatch;
        userMatch = userRepository.findByLoginAndPassword(login, password);

        return  userMatch;
    }
}

这是我的测试:

package br.com.api.model.services;


import br.com.api.model.repository.UserRepository;
import br.com.api.model.resources.User;
import org.junit.runner.RunWith;
import org.mockito.MockitoAnnotations;
import org.mockito.Mock;
import org.junit.Before;
import org.junit.Test;
import org.mockito.InjectMocks;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;

import static org.mockito.Mockito.when;

@RunWith(SpringJUnit4ClassRunner.class)
public class UserServiceTest {

    @Mock
    UserDao userRepository;

    @InjectMocks
    UserService userService;

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

    @Test
    public void findUser(){

        User user;
        user = userService.findUser("first", "teste");

        assertNotNull(user.getLogin());

    }

}  

3 个答案:

答案 0 :(得分:16)

我相信你错过了单元测试和嘲笑的整个想法。

  1. 当您对UserService进行单元测试时,您不希望使用真正的UserRepository实现。
  2. 您嘲笑了UserRepository,您不希望模拟对象立即表现为真实对象。你需要弥补它的行为(即顽固)。
  3. 您应该很少需要在单元测试中使用Spring Runner。
  4. 为了决定模拟对象的行为,你必须知道被测系统(SUT,在你的情况下是UserService)与其依赖项({{1})的预期交互。 })

    在你的情况下,测试看起来应该是(没有编译,只是告诉你这个想法)

    UserRepository

答案 1 :(得分:3)

您也可以传递名称参数,例如

 @MockBean(name="userRepository")
 UserDao userRepository;

答案 2 :(得分:3)

您需要先修改模拟顺序,然后使用@InjectMocks,然后使用@Mock,然后再添加测试类名称 @ExtendWith(MockitoExtension.class)

我遇到了同样的问题,但是在执行了上述步骤之后,它对我有用