角色&权限

一、角色&权限

角色&权限.png

二、项目中如何校验角色&权限(通过拦截器)

采用spring原理中面向切面(aop)的思想,该思想的作用是:对某一类对象进行监督和控制

主要实现:1.通过注解接口创建注解(该注解为需要校验)
2.全局拦截器通过判断方法上是否有该注解,没有则放行,不需要校验;否则,则需要校验

2.1 不校验用户在线NotCheckOnline

注解接口统一放置在pojo下的jpa文件中

package cn.kooun.pojo.jpa;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

/**
 *  利用面向切面的思想
 *  不校验用户在线注解接口
 * @author HuangJingNa
 * @date 2019年12月22日 下午4:49:31
 *
 */
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface NotCheckOnline {
    String value() default "abc";
}
2.2 全局拦截器GlobalInterceptor
package cn.kooun.core.interceptor;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.method.HandlerMethod;
import org.springframework.web.servlet.HandlerInterceptor;

import cn.kooun.service.UserService;

/**
 *  全局拦截器
 * @author HuangJingNa
 * @date 2019年12月22日 下午4:40:45
 *
 */
public class GlobalInterceptor implements HandlerInterceptor{
    @Autowired
    private UserService userService;
    
    @Override
    public boolean preHandle(
            HttpServletRequest request, 
            HttpServletResponse response, 
            Object handler) throws Exception {
        if(handler instanceof HandlerMethod) {
            HandlerMethod handlerMethod = (HandlerMethod) handler;
            return baseHandler(request, response, handlerMethod);
        }
        return true;
    }
    private boolean baseHandler(
            HttpServletRequest request, 
            HttpServletResponse response, 
            HandlerMethod handlerMethod) {
        boolean flag = true;
        flag = userService.checkOnline(request, response, handlerMethod);
        if(!flag) {
            return flag;
        }
        flag = userService.checkRoles(request, response, handlerMethod);
        if(!flag) {
            return flag;
        }
        flag = userService.checkPerssions(request, response, handlerMethod);
        if(!flag) {
            return flag;
        }
        return flag;
    }
}
2.3 service层的checkOnline()方法
/**     
 *  校验是否在线
 * @author HuangJingNa
 * @date 2019年12月22日 下午4:48:13
 *
 * @param request
 * @param response
 * @param handlerMethod
 * @return
 */
public boolean checkOnline(
        HttpServletRequest request, 
        HttpServletResponse response, 
        HandlerMethod handlerMethod) {
    //获取方法上的注解,判断是否要进行校验
    NotCheckOnline notCheckOnline = handlerMethod.getMethodAnnotation(NotCheckOnline.class);
    if(notCheckOnline != null) {
        //放行,不进行校验
        return true;
    }
    
    //获取在线用户
    Object user = RequestUtils.getUserLogin(request, response, redisService);
    if(user == null) {
        ResponseUtils.returnJson(
                response, 
                ResultUtils.error("登录失效,请重新登录~", Result.JUMP_LOGIN));
        return false;
    }
    
    return true;
}
2.4RequestUtils请求工具类
package cn.kooun.common.request;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import org.springframework.util.StringUtils;

import cn.kooun.common.redis.RedisService;
import cn.kooun.pojo.info.Dict;

/**
 *  请求工具类
 * @author HuangJingNa
 * @date 2019年12月22日 下午4:52:10
 *
 */
public class RequestUtils {
    /**
     *  校验用户是否在线(根据用户凭证)
     * @author HuangJingNa
     * @date 2019年12月22日 下午5:03:40
     *
     * @param request
     * @param response
     * @param redisService
     * @return
     */
    public static Object getUserLogin(
            HttpServletRequest request, 
            HttpServletResponse response,
            RedisService redisService) {
        //登录校验
        String userToken = request.getHeader(Dict.USER_TOKEN);
        if(StringUtils.isEmpty(userToken)) {
            return null;
        }
        //返回在线数据
        return redisService.get(userToken);
    }
    
}
2.5 ResponseUtils响应工具类
package cn.kooun.common.reponse;

import java.io.PrintWriter;

import javax.servlet.http.HttpServletResponse;

import com.alibaba.fastjson.JSON;

import cn.kooun.pojo.info.Dict;

/**
 *  响应工具类
 * @author HuangJingNa
 * @date 2019年12月22日 下午4:52:24
 *
 */
public class ResponseUtils {
    /**
     * response响应json数据格式
     * 
     * @author chenwei
     * @date 2018年12月11日 下午5:42:44
     * @param response
     * @param result
     */
    public static void returnJson(HttpServletResponse response, Object result) {
        //创建json转换的类
        PrintWriter writer = null;
        try {
            // 避免乱码
            response.setCharacterEncoding(Dict.UTF8);
            // 设置ContentType
            response.setContentType(Dict.JSON_HEAD);
            writer = response.getWriter();
            writer.append(JSON.toJSONString(result));
            writer.flush();
        } catch (Exception e) {
            e.printStackTrace();
        }finally {
            if(writer!=null) {
                writer.close();
            }
        }
    }
}
2.6 数据字典接口Dict
package cn.kooun.pojo.info;
/**
 *  数据字典接口
 * @author HuangJingNa
 * @date 2019年12月22日 下午4:07:46
 *
 */
public interface Dict {
    /**用户登录凭证*/
    String USER_TOKEN = "Token";
    /**编码格式设置*/
    String UTF8 = "utf-8";
    /**Content-Type设置 响应json数据*/
    String JSON_HEAD = "application/json; charset=utf-8";
}

三、校验角色&权限

3.1 角色/权限注解接口
package cn.kooun.pojo.jpa;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

/**
 *  角色校验注解接口
 * @author HuangJingNa
 * @date 2019年12月22日 下午5:15:24
 *
 */
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface CheckRoles {
    /**角色数组*/
    String[] value();
}
package cn.kooun.pojo.jpa;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

/**
 *  权限校验注解接口
 * @author HuangJingNa
 * @date 2019年12月22日 下午5:15:24
 *
 */
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface CheckPerssions {
    /**权限数组*/
    String[] value();
}
3.2 service层中的方法
/**
 *  角色校验
 * @author HuangJingNa
 * @date 2019年12月22日 下午5:19:26
 *
 * @param request
 * @param response
 * @param handlerMethod
 * @return
 */
public boolean checkRoles(
        HttpServletRequest request, 
        HttpServletResponse response, 
        HandlerMethod handlerMethod) {
    CheckRoles checkRoles = handlerMethod.getMethodAnnotation(CheckRoles.class);
    if(checkRoles == null) {
        return true;
    }
    
    //通过注解获取需要校验的角色标签
    List<String> targetRoles = Arrays.asList(checkRoles.value());
    //进行角色校验(需要用户在线)
    LoginUser user = (LoginUser) 
            RequestUtils.getUserLogin(request, response, redisService);
    List<String> roles = user.getRoles();
    //不包含目标角色则拦截
    if(!roles.containsAll(targetRoles)) {
        ResponseUtils.returnJson(
                response, 
                ResultUtils.error("你没有访问的权限~"));
        return false;
    }
    
    return true;
}
/**
 *  权限校验
 * @author HuangJingNa
 * @date 2019年12月22日 下午5:29:17
 *
 * @param request
 * @param response
 * @param handlerMethod
 * @return
 */
public boolean checkPerssions(
        HttpServletRequest request, 
        HttpServletResponse response,
        HandlerMethod handlerMethod) {
    //获取方法上的注解
    CheckPerssions checkPerssions = 
            handlerMethod.getMethodAnnotation(CheckPerssions.class);
    //没有权限注解,放行
    if(checkPerssions == null) {
        return true;
    }
    //有,判断用户是否在线
    //在线,获取用户所有的权限
    LoginUser user = (LoginUser) RequestUtils.getUserLogin(
            request, response, redisService);
    List<String> permissions = user.getPermissions();
    //获取注解上的权限
    List<String> targetPerssions = Arrays.asList(checkPerssions.value());
    //判断注解上的权限是否包含在用户权限中
    //不包含,拦截
    if(!permissions.containsAll(targetPerssions)) {
        ResponseUtils.returnJson(
                response, 
                ResultUtils.error("你没有访问的权限~"));
        return false;
    }
    return true;
}
3.3 响应结果工具类ResultUtils
package cn.kooun.common.result;

import cn.kooun.common.factory.ErrorFactory;
import cn.kooun.common.factory.SuccessFactory;

/**
 *  响应结果工具类
 * @author HuangJingNa
 * @date 2019年12月22日 下午2:02:04
 *
 */
public class ResultUtils {
    public static Object success(String message) {
        return successCommon(message, null, null);
    }
    public static Object success(Object result) {
        return successCommon(null, result, null);
    }
    public static Object success(String message, String jump) {
        return successCommon(message, null, jump);
    }
    public static Object success(String message, Object result, String jump) {
        return successCommon(message, result, jump);
    }
    public static Object error(String message, String jump) {
        return errorCommon(message, null, jump);
    }
    public static Object error(String message) {
        return errorCommon(message, null, null);
    }
    
    /**
     *  正确响应公共类
     * @author HuangJingNa
     * @date 2019年12月22日 下午2:19:35
     *
     * @param message
     * @param result
     * @param jump
     * @return
     */
    private static Object successCommon(String message,Object result,String jump) {
        SuccessResult successResult = SuccessFactory.getInstance();
        successResult.setMessage(message);
        successResult.setResult(result);
        successResult.setJump(jump);
        return successResult;
    }
    /**
     *  错误响应公共类
     * @author HuangJingNa
     * @date 2019年12月22日 下午2:19:13
     *
     * @param message
     * @param result
     * @param jump
     * @return
     */
    private static Object errorCommon(String message,Object result,String jump) {
        ErrorResult errorResult = ErrorFactory.getInstance();
        errorResult.setMessage(message);
        errorResult.setResult(result);
        errorResult.setJump(jump);
        return errorResult;
    }
    
}

以上ResultUtils详情可点击链接🔗https://www.jianshu.com/p/d59411918b6d

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容