Почему mockito сообщает об ошибке на thenReturn() в java 7, но не в java 6
Mockito сообщает о незавершенной ошибке обрушения при издевательском client.getPrograms()
, который должен возвращать SortedSet<Program>
. Интересная часть заключается в том, что он делает это только при использовании Java 7, а не при использовании Java 6.
Здесь код, который вызывает ошибку при издевательском client.getPrograms()
:
private void prepareScheduleChangePreconditions() {
Client client = mock(Client.class);
TimeTable tt = BuilderUtil.buildTable(AcceleratedScheduleTimeTable.Schedule.NORMAL, "08:00");
when(clientRepository.findByCode(anyString())).thenReturn(client);
//Error is reported for next line of code
when(client.getPrograms()).thenReturn(new TreeSet<Program>(Collections.singleton(program)));
when(event.getTimeTable()).thenReturn(tt);
}
Здесь вывод ошибки:
Tests in error:
testExampleScheduleChangeNotify1(com.example.service.impl.ExampleServiceImplTest):
Unfinished stubbing detected here:
-> at com.example.service.impl.ExampleServiceImplTest.prepareScheduleChangePreconditions(ExampleServiceImplTest.java:134)
E.g. thenReturn() may be missing.
Examples of correct stubbing:
when(mock.isOk()).thenReturn(true);
when(mock.isOk()).thenThrow(exception);
doThrow(exception).when(mock).someVoidMethod();
Hints:
1. missing thenReturn()
2. you are trying to stub a final method, you naughty developer!
Метод не является окончательным. Любая помощь или подсказки будут оценены.
ОБНОВЛЕНИЕ 1
В соответствии с запросом Mike B мне удалось выделить это в более простой тестовый пример, который не работает в Java 7.
@RunWith(MockitoJUnitRunner.class)
public class MockitoTest {
@Mock
Program program;
private void preparePreconditions() {
Client client = mock(Client.class);
when(client.getPrograms()).thenReturn(new TreeSet<Program>(Collections.singleton(program)));
}
public static class Client {
public SortedSet<Program> getPrograms() {
return new TreeSet<Program>();
}
}
public static class Program implements Comparable<Program> {
private String name;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
@Override
public int compareTo(Program program) {
return 0;
}
}
@Test
public void test() {
preparePreconditions();
}
}
ОБНОВЛЕНИЕ 2
Как ни странно, он работает, если я так делаю:
TreeSet<Program> programs = new TreeSet<Program>();
programs.add(program);
when(client.getPrograms()).thenReturn(programs);
Ответы
Ответ 1
Я считаю, что то, что на самом деле происходит здесь, - это более одного вызова метода mock в одиночном выражении when(...).then(...)
.
Рассмотрим следующий пример:
when(program.getName()).thenReturn(String.valueOf(program.compareTo(null)));
Он возвращает то же исключение, что и у вас. Это связано с тем, что в mock используются два метода: getName()
и compareTo()
в одном выражении when(...).thenReturn(...)
.
Также в этой странице (документация mockito) Вы можете прочитать, что:
По умолчанию для всех методов, возвращающих значение, mock возвращает null, пустую коллекцию или соответствующее примитивное/примитивное значение обертки (например: 0, false,... для int/Integer, boolean/Boolean,...),
Поэтому mockito должен иметь некоторый механизм для обнаружения того, что делать (возврат), для которого вызов на каком-то издеваемом объекте.
В вашем примере второй вызов выполняется оператором new TreeSet<>(Collections.singleton(program))
, потому что конструктор TreeSet использует метод compareTo()
вашего макета.
Кажется, что реализация TreeSet изменилась с java 6 на java 7. Вот почему она могла работать раньше.
Ответ 2
Это на самом деле не отвечает на ваш вопрос, но помните о главной директиве насмешки:
- Ролевые роли, а не объекты
На практике это означает, что высмеивание класса (т.е. не интерфейса) является мигающим красным предупреждающим знаком о том, что что-то не так с вашим дизайном.
Если у вас нет контроля над кодом, с которым вы взаимодействуете (например, это сторонняя библиотека или устаревшее программное обеспечение), мне нравится обернуть класс в интерфейс и издеваться над этим. В качестве бонуса это также позволяет вам переименовать неправильно названные методы (и классы) в другой код. Из вашего примера, какие "программы" возвращают клиент? Это набор активных программ, запущенных программ, что? Предположим, что это набор активных программ:
public interface Programs {
SortedSet<Program> active();
}
class SimpleClientWrapper implements Programs {
private final Client wrapped;
SimpleClientWrapper(Client c) { wrapped = c; }
public SortedSet<Program> active() { return wrapped.getPrograms(); }
}
Если это похоже на слишком много работы (и я признаю, что иногда это слишком много), другая возможность, если метод (ы), который вы хотите высмеять, не является окончательным, заключается в том, чтобы временно переопределить их в вашем тесте:
Client client = new Client() {
@Override
public SortedSet<Program> getPrograms() {
return new TreeSet<Program>(Collections.singleton(program));
}
};
Я иногда использую этот подход для переопределения обработчиков ошибок и т.д.