角色&权限

一、角色&权限

角色&权限.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

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,142评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,298评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,068评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,081评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,099评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,071评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,990评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,832评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,274评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,488评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,649评论 1 347
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,378评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,979评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,625评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,643评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,545评论 2 352

推荐阅读更多精彩内容