在用shiro做登陆的时候,如果访问的地址没有登陆,会自定义返回到Web工程根目录下的"/login.jsp"页面,而这个不是我想要的,所以需要自定义登陆页面。
shiroFilterFactoryBean.setLoginUrl("/h/login");
通过这样的设置,就能让未登录的用户,跳转到自定义的登陆页面了。
但是,这里主要讲的是前后端分离情况下,前端通过ajax的形式来访问数据,后端java只作为了一个restful的形式返回数据,那么,在跳转就不适合了。所以我们需要做页面跳转,而是直接返回json格式的数组。
首先,建一个过滤器ShiroLoginFilter,内容如下:
package crd.student.api.config; import com.alibaba.fastjson.JSONObject; import crd.student.api.reponse.Result; import org.apache.shiro.web.filter.authc.FormAuthenticationFilter; import javax.servlet.ServletRequest; import javax.servlet.ServletResponse; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.io.IOException; public class ShiroLoginFilter extends FormAuthenticationFilter { /** * 如果isAccessAllowed返回false 则执行onAccessDenied * @param request * @param response * @param mappedValue * @return */ @Override protected boolean isAccessAllowed(ServletRequest request, ServletResponse response, Object mappedValue) { if (request instanceof HttpServletRequest) { if (((HttpServletRequest) request).getMethod().toUpperCase().equals("OPTIONS")) { return true; } } return super.isAccessAllowed(request, response, mappedValue); } /** * 在访问controller前判断是否登录,返回json,不进行重定向。 * * @param request * @param response * @return true-继续往下执行,false-该filter过滤器已经处理,不继续执行其他过滤器 * @throws Exception */ @Override protected boolean onAccessDenied(ServletRequest request, ServletResponse response) throws IOException { HttpServletResponse httpServletResponse = (HttpServletResponse) response; //这里是个坑,如果不设置的接受的访问源,那么前端都会报跨域错误,因为这里还没到corsConfig里面 httpServletResponse.setHeader("Access-Control-Allow-Origin", ((HttpServletRequest) request).getHeader("Origin")); httpServletResponse.setHeader("Access-Control-Allow-Credentials", "true"); httpServletResponse.setCharacterEncoding("UTF-8"); httpServletResponse.setContentType("application/json"); Result result = new Result(-10000, "请先登陆"); httpServletResponse.getWriter().write(JSONObject.toJSON(result).toString()); return false; } }
然后在ShiroConfig中,shirFilter方法中加入:
//用户未登录不进行跳转,而是自定义返回json数据 Map<String, Filter> filters = shiroFilterFactoryBean.getFilters();//获取filters filters.put("authc", new ShiroLoginFilter());//将自定义 的FormAuthenticationFilter注入shiroFilter中 // 如果不设置默认会自动寻找Web工程根目录下的"/login.jsp"页面 // 注释掉 // shiroFilterFactoryBean.setLoginUrl("/h/login");
这样,没有登陆那么就会返回json了。