Модульное тестирование сервлетов

У меня проблема с тестированием сервлетов. Я думаю, это произошло из-за того, что я использую список контекстов для инициализации своего дао, но я не могу это исправить. Приложение работает, но тесты не работают.

Один из моих сервлетов, который показывает список пользователей:

public class AllUsersServlet extends HttpServlet {
private static final String LIST_OF_USERS = "/WEB-INF/view//usersList.jsp";
private static final String DAO = "userDao";
private static final String USERS = "users";
private UserDaoImpl userDao;

@Override
public void init() throws ServletException {
    userDao = (UserDaoImpl) getServletContext().getAttribute(DAO);
}

@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws IOException, ServletException {
    List<User> users = userDao.getAll();
    req.getServletContext().setAttribute(USERS, users);
    req.getRequestDispatcher(LIST_OF_USERS).forward(req, resp);
}

@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
    super.doPost(req, resp);
}
}

Один из моих модульных тестов, проверяющих AllUsersServlet:

public class AllUsersServletTest {
private static final String PAGE = "/WEB-INF/view//list.jsp";

@Test
public void whenDoGetThenReturnListPage() throws IOException, ServletException {
    AllUsersServlet allUsersServlet = new AllUsersServlet();

    HttpServletRequest request = mock(HttpServletRequest.class);
    HttpServletResponse response = mock(HttpServletResponse.class);
    RequestDispatcher dispatcher = mock(RequestDispatcher.class);
    ServletContext context = mock(ServletContext.class);

    when(request.getServletContext()).thenReturn(context);
    when(request.getRequestDispatcher(PAGE)).thenReturn(dispatcher);

    allUsersServlet.doGet(request, response);

    verify(request, times(1)).getRequestDispatcher(PAGE);
    verify(request, never()).getSession();
    verify(dispatcher).forward(request, response);
}
}

Итак, когда я начинаю свой тест, он не работает с нулевым указателем, stacktrace:

java.lang.NullPointerException
at ru.skilanov.controller.AllUsersServlet.doGet(AllUsersServlet.java:55)
at ru.skilanov.controller.AllUsersServletTest.whenDoGetThenReturnListPage(AllUsersServletTest.java:40)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:498)
at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:50)
at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12)
at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:47)
at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:17)
at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:325)
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:78)
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:57)
at org.junit.runners.ParentRunner$3.run(ParentRunner.java:290)
at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:71)
at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:288)
at org.junit.runners.ParentRunner.access$000(ParentRunner.java:58)
at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:268)
at org.junit.runners.ParentRunner.run(ParentRunner.java:363)
at org.junit.runner.JUnitCore.run(JUnitCore.java:137)
at com.intellij.junit4.JUnit4IdeaTestRunner.startRunnerWithArgs(JUnit4IdeaTestRunner.java:68)
at com.intellij.rt.execution.junit.IdeaTestRunner$Repeater.startRunnerWithArgs(IdeaTestRunner.java:47)
at com.intellij.rt.execution.junit.JUnitStarter.prepareStreamsAndStart(JUnitStarter.java:242)
at com.intellij.rt.execution.junit.JUnitStarter.main(JUnitStarter.java:70)


Process finished with exit code -1

Я подозреваю, что строка 55 AllUsersServlet - это строка userDao.getAll() ...? Вы издеваетесь над контекстом, но не предоставляете реализацию userDao, и метод init, вероятно, никогда не вызывается. Размещено как комментарий, а не как ответ, так как я догадываюсь о вашей нумерации строк.

Dave 07.05.2018 12:01

Да: List <User> users = userDao.getAll ();

Sam Kilanoff 07.05.2018 12:03
Пользовательский скаляр GraphQL
Пользовательский скаляр GraphQL
Листовые узлы системы типов GraphQL называются скалярами. Достигнув скалярного типа, невозможно спуститься дальше по иерархии типов. Скалярный тип...
Как вычислять биты и понимать побитовые операторы в Java - объяснение с примерами
Как вычислять биты и понимать побитовые операторы в Java - объяснение с примерами
В компьютерном программировании биты играют важнейшую роль в представлении и манипулировании данными на двоичном уровне. Побитовые операции...
Поднятие тревоги для долго выполняющихся методов в Spring Boot
Поднятие тревоги для долго выполняющихся методов в Spring Boot
Приходилось ли вам сталкиваться с требованиями, в которых вас могли попросить поднять тревогу или выдать ошибку, когда метод Java занимает больше...
Полный курс Java для разработчиков веб-сайтов и приложений
Полный курс Java для разработчиков веб-сайтов и приложений
Получите сертификат Java Web и Application Developer, используя наш курс.
0
2
676
3
Перейти к ответу Данный вопрос помечен как решенный

Ответы 3

Когда инициализируется поле userDao? В методе init() вроде. Метод init() вызывается контейнерным механизмом, когда ваш сервлет развертывается в контейнере. Когда в вашем тесте вызывается метод init()?

(Примечание: контейнерный механизм делает много всего для инициализации среды, в которой работает сервлет. Вам нужно будет выполнить все это, если вы хотите запускать тесты вне контейнера.)

Ответ принят как подходящий

После разъяснения вашей нумерации строк в комментариях ответ довольно очевиден - добавьте сеттер в свой AllUsersServlet - что-то вроде:

default void setUserDao(UsersDao dao) {...}

Затем в вашем тесте перед вызовом метода сервлета doGet() предоставьте настроенный макет через сеттер. Тестовый код должен выглядеть примерно так:

@Test
public void whenDoGetThenReturnListPage() throws IOException, ServletException {
AllUsersServlet allUsersServlet = new AllUsersServlet();

HttpServletRequest request = mock(HttpServletRequest.class);
HttpServletResponse response = mock(HttpServletResponse.class);
RequestDispatcher dispatcher = mock(RequestDispatcher.class);
ServletContext context = mock(ServletContext.class);
UserDaoImpl mockDao = mock(UserDaoImpl.class);

when(mockDao.getAll()).thenReturn(new ArrayList<User>()); // change the list as appropriate for your tests
when(request.getServletContext()).thenReturn(context);
when(request.getRequestDispatcher(PAGE)).thenReturn(dispatcher);

allUsersServlet.setUserDaoImpl(mockDao);
allUsersServlet.doGet(request, response);

verify(request, times(1)).getRequestDispatcher(PAGE);
verify(request, never()).getSession();
verify(dispatcher).forward(request, response);

}

Это должно устранить ваше исключение NullPointerException.

Простите, я не понимаю. Я добавил сеттер в свой сервлет, тогда я должен издеваться над своим дао следующим образом: UserDaoImpl userDao = mock (UserDaoImpl.class); и установите это dao в тесте, например: allUsersServlet.setUserDao (userDao) ;?

Sam Kilanoff 07.05.2018 12:19

Ответ отредактирован, я буду недоступен в течение нескольких часов, но если вам все еще непонятно, оставьте еще один комментарий, и я расскажу подробнее.

Dave 07.05.2018 12:25

Спасибо, сейчас тесты запущены.

Sam Kilanoff 07.05.2018 12:34

Полагаю, похоже на

when(request.getRequestDispatcher(PAGE)).thenReturn(dispatcher);

вы должны издеваться над ответом на List<User> users = userDao.getAll();

у тебя может быть что-то вроде

List<User> users = new ArrayList<User>();
when(userDao.getAll()).thenReturn(users);

Думаю, это должно решить вашу проблему. У вас также есть возможность добавить некоторых пользователей в список перед возвращением.

Другие вопросы по теме