项目中经常能碰到有些接口是需要用户登录之后才能访问的,而有些的是不要做登录校验的;
当然可以使用到一些权限管理的框架;但是有些简单的业务功能的就可以通过自定义注解+自定义的拦截器设置指定的消息内容;
项目demo下载
主要通过添加在指定的方法判断接口访问是不是需要登录
/***
* 设置拦截器通过注解设定某种属性
* @author kanghanyu
* 自定义的注解内容设定对应的内容信息
*
*/
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface AuthPassport {
/**
* 是否需要登录
* @return
*/
boolean auth() default false;
}
/**
* 自定义拦截器的 通过获取请求接口自定义的注解判定是否需要登录返回对应的信息
* @author khy
* @createTime 2019年10月9日上午11:37:02
*/
@Configuration
public class AnnotationInterceptor extends WebMvcConfigurerAdapter {
public void addInterceptors(InterceptorRegistry registry) {
registry.addInterceptor(new HandlerInterceptorAdapter() {
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)
throws Exception {
// 验证用户权限
if (handler.getClass().isAssignableFrom(HandlerMethod.class)) {
AuthPassport authPassport = ((HandlerMethod) handler).getMethodAnnotation(AuthPassport.class);
if (authPassport != null) {
boolean auth = authPassport.auth();
//如果是需要登录校验的
if(auth){
String userName = AnnotationController.userName;
String password = AnnotationController.password;
if(StringUtils.isAnyBlank(userName,password)){
auth = false;
}
if(!userName.equals(password) || !userName.equals(AnnotationController.LOGIN_FLAG)){
auth = false;
}
if(!auth){
JSONObject json = new JSONObject();
json.put("code", "10000");
json.put("msg", "用户未登录不能访问");
response.setCharacterEncoding("UTF-8");
response.setContentType("application/json;charset=utf-8");
response.getWriter().write(JSON.toJSONString(json));
return false;
}
}
}
}
return true;
}
@Override
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler,
Exception ex) throws Exception {
if (ex != null) {
Enumeration<String> enu = request.getParameterNames();
Map<String, Object> param = new HashMap<String, Object>();
while (enu.hasMoreElements()) {
String paraName = (String) enu.nextElement();
param.put(paraName, request.getParameter(paraName));
}
} else {
super.afterCompletion(request, response, handler, ex);
}
}
@Override
public void afterConcurrentHandlingStarted(HttpServletRequest request, HttpServletResponse response,
Object handler) throws Exception {
super.afterConcurrentHandlingStarted(request, response, handler);
}
}).addPathPatterns("/annotation/**");//拦截的请求路径地址内容
}
}
@RestController
@RequestMapping("/annotation")
public class AnnotationController {
private static final Logger khy = LoggerFactory.getLogger("KHY");
public static final String LOGIN_FLAG = "admin";
public static String userName = "test";
public static String password = "test";
/**
* 这个是模拟登录的接口内容
* http://localhost:8080/annotation/login?userName=admin&password=admin
* @param userName
* @param password
* @return
*/
@RequestMapping("/login")
public String login(String userName,String password){
if(StringUtils.isNotBlank(userName)){
this.userName = userName;
}
if(StringUtils.isNotBlank(password)){
this.password = password;
}
return "登录成功";
}
/**
* 这个是模拟保存需要登录
* http://localhost:8080/annotation/save?userName=admin&password=admin
* @param userName
* @param password
* @return
*/
@AuthPassport(auth=true)
@RequestMapping("/save")
public String save(String userName,String password){
khy.debug("保存用户userName={},password={}",userName,password);
return "登录成功";
}
/**
* 这个是模拟查询接口不需要登录
* http://localhost:8080/annotation/query?userName=admin&password=admin
* @param userName
* @param password
* @return
*/
@AuthPassport(auth=false)
@RequestMapping("/query")
public String query(String userName,String password){
khy.debug("查询用户userName={},password={}",userName,password);
return "查询成功";
}
首先AnnotationController 里面的三个方法
1.login方法没有自定义的注解@AuthPassport
所以请求http://localhost:8080/annotation/login 接口会被AnnotationInterceptor拦截器拦截但是不符合注解条件直接进入接口执行.
2.query 方法上面虽然有指定的 @AuthPassport 但是里面的auth=false
在拦截器里面执行但是也不去校验登录的信息内容;所以在不调用login 的接口之前也可以通过 http://localhost:8080/annotation/query 直接调用此接口
3.save 方法上有指定的@AuthPassport 注解并且auth=true 标识需要登录之后才能访问;如果项目启动就直接调用http://localhost:8080/annotation/save 会因为在AnnotationInterceptor 拦截器里面的去匹配校验的信息内容.
总结:项目中的登录接口只是用来模拟使用的;真正的项目中校验的登录的信息是在AnnotationInterceptor 拦截器里面根据用户的信息去获取用户是否有对应的接口访问权限然后再判断是否去调用指定的接口.