Spring MVC - ловить исключения, сгенерированные на странице просмотра
Вот сценарий, контроллер возвращает ModelAndView, и при рендеринге страницы представления JSP выдается исключение, причина в том, что один из кодов сообщения не найден.
org.apache.jasper.JasperException: org.springframework.context.NoSuchMessageException: No message found under code 'Diff.userform.password' for locale 'en_US'. org.apache.jasper.servlet.JspServletWrapper.handleJspException(JspServletWrapper.java:549) org.apache.jasper.servlet.JspServletWrapper.service(JspServletWrapper.java:470) org.apache.jasper.servlet.JspServlet.serviceJspFile(JspServlet.java:390) org.apache.jasper.servlet.JspServlet.service(JspServlet.java:334) javax.servlet.http.HttpServlet.service(HttpServlet.java:722)
Возникает исключительная ситуация и отображается страница ошибки HTTP 500 напрямую.
1. проблема
Объявлен глобальный обработчик исключений, но все еще не удается пойматьNoSuchMessageException
?
@ControllerAdvice public class GlobalExceptionHandler { private final Logger logger = LoggerFactory.getLogger(GlobalExceptionHandler.class); @ExceptionHandler(value = Exception.class) public ModelAndView defaultErrorHandler(HttpServletRequest req, Exception e) throws Exception { logger.error("[URL] : {}", req.getRequestURL(), e); ModelAndView mav = new ModelAndView(); mav.addObject("exception", e); mav.addObject("url", req.getRequestURL()); mav.setViewName("error""); return mav; } }
2. Решение
Нет, вы не можете поймать исключение, которое выдается страницей просмотра. @ExceptionHandler
используется только для перехвата исключений, создаваемых классом Controller.
Лучшее, что вы можете сделать, это определить страницу ошибки вweb.xml
web.xml
500 /WEB-INF/views/jsp/error.jsp /WEB-INF/views/jsp/error.jsp
Note
Подскажите, есть ли у вас решение получше :)
3. Фильтр сервлетов
Кроме того, вы можете создать фильтр и перехватывать все запросы в вашем веб-приложении. Если выдается какое-либо исключение, войдите в систему или делайте что хотите.
package com.example.form.web; import java.io.IOException; import javax.servlet.Filter; import javax.servlet.FilterChain; import javax.servlet.FilterConfig; import javax.servlet.ServletException; import javax.servlet.ServletRequest; import javax.servlet.ServletResponse; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.web.context.support.WebApplicationContextUtils; import com.example.form.service.UserService; public class ErrorHandleFilter implements Filter { private final Logger logger = LoggerFactory.getLogger(ErrorHandleFilter.class); //Your custom bean to handle error ErrorHandlerSpecialForce errHandle; @Override public void destroy() { // ... } @Override public void init(FilterConfig filterConfig) throws ServletException { //Get bean from Spring container errHandle = (ErrorHandlerSpecialForce) WebApplicationContextUtils .getRequiredWebApplicationContext(filterConfig.getServletContext()) .getBean("errorHandlerSpecialForce"); } @Override public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException { try { // your code, do whatever errHandle.handle(request, response); chain.doFilter(request, response); } catch (Exception ex) { logger.error("Error : {}", ex); //log it request.setAttribute("errorMessage", ex); request.getRequestDispatcher("/WEB-INF/views/jsp/error.jsp") .forward(request, response); } } }
Чтобы зарегистрировать фильтр выше, объявляет его вweb.xml
web.xml
org.springframework.web.context.ContextLoaderListener errorHandlerFilter com.example.form.web.ErrorHandleFilter errorHandlerFilter /*
Или объявляет это в классе инициализатора.
package com.example.form.config.servlet3; import javax.servlet.Filter; import org.springframework.web.servlet.support.AbstractAnnotationConfigDispatcherServletInitializer; import com.example.form.web.ErrorHandleFilter; public class MyWebInitializer extends AbstractAnnotationConfigDispatcherServletInitializer { //... @Override protected Filter[] getServletFilters() { return new Filter[]{new ErrorHandleFilter()}; } }