1、授权的流程

2、三种授权方式

1.编程式:通过写if/else 授权代码块完成:

  1. Subject subject = SecurityUtils.getSubject();
  2. if(subject.hasRole(“admin”)) {
  3. //有权限
  4. } else {
  5. //无权限
  6. }

2.注解式:通过在执行的Java方法上放置相应的注解完成:

  1. @RequiresRoles("admin")
  2. public void hello() {
  3. //有权限
  4. }

3.JSP/GSP 标签:在JSP/GSP 页面通过相应的标签完成:

  1. <shiro:hasRole name="admin">
  2. <!— 有权限—>
  3. </shiro:hasRole>

3、创建本地文件模拟数据库数据(shiro-permission.ini)

  1. #用户
  2. [users]
  3. #用户zhang的密码是123,此用户具有role1和role2两个角色
  4. zhang=123,role1,role2
  5. wang=123,role2
  6. #权限
  7. [roles]
  8. #角色role1对资源user拥有create、update权限
  9. role1=user:create,user:update
  10. #角色role2对资源user拥有create、delete权限
  11. role2=user:create,user:delete
  12. #角色role3对资源user拥有create权限
  13. role3=user:create

  权限标识符号规则:资源:操作:实例(中间使用半角:分隔)

  user:create:01 表示对用户资源的01实例进行create操作。

  user:create:表示对用户资源进行create操作,相当于user:create:*,对所有用户资源实例进行create操作。

  user:*:01 表示对用户资源实例01进行所有操作。

4、授权程序编写

  1. // 角色授权、资源授权测试
  2. @Test
  3. public void testAuthorization() {
  4. // 创建SecurityManager工厂
  5. Factory<SecurityManager> factory = new IniSecurityManagerFactory(
  6. "classpath:shiro-permission.ini");
  7. // 创建SecurityManager
  8. SecurityManager securityManager = factory.getInstance();
  9. // 将SecurityManager设置到系统运行环境,和spring后将SecurityManager配置spring容器中,一般单例管理
  10. SecurityUtils.setSecurityManager(securityManager);
  11. // 创建subject
  12. Subject subject = SecurityUtils.getSubject();
  13. // 创建token令牌
  14. UsernamePasswordToken token = new UsernamePasswordToken("zhangsan",
  15. "123");
  16. // 执行认证
  17. try {
  18. subject.login(token);
  19. } catch (AuthenticationException e) {
  20. // TODO Auto-generated catch block
  21. e.printStackTrace();
  22. }
  23. System.out.println("认证状态:" + subject.isAuthenticated());
  24. // 认证通过后执行授权
  25. // 基于角色的授权
  26. // hasRole传入角色标识
  27. boolean ishasRole = subject.hasRole("role1");
  28. System.out.println("单个角色判断" + ishasRole);
  29. // hasAllRoles是否拥有多个角色
  30. boolean hasAllRoles = subject.hasAllRoles(Arrays.asList("role1",
  31. "role2", "role3"));
  32. System.out.println("多个角色判断" + hasAllRoles);
  33. // 使用check方法进行授权,如果授权不通过会抛出异常
  34. // subject.checkRole("role13");
  35. // 基于资源的授权
  36. // isPermitted传入权限标识符
  37. boolean isPermitted = subject.isPermitted("user:create:1");
  38. System.out.println("单个权限判断" + isPermitted);
  39. boolean isPermittedAll = subject.isPermittedAll("user:create:1",
  40. "user:delete");
  41. System.out.println("多个权限判断" + isPermittedAll);
  42. // 使用check方法进行授权,如果授权不通过会抛出异常
  43. subject.checkPermission("items:create:1");
  44. }

5、自定义realm进行授权

  上边的程序通过shiro-permission.ini对权限信息进行静态配置,实际开发中从数据库中获取权限数据。就需要自定义realm,由realm从数据库查询权限数据。

realm根据用户身份查询权限数据,将权限数据返回给authorizer(授权器)。

修改原来的认证realm,加入授权过程。

  1. // 设置realm的名称
  2. @Override
  3. public void setName(String name) {
  4. super.setName("customRealm");
  5. }
  6. // 用于认证
  7. @Override
  8. protected AuthenticationInfo doGetAuthenticationInfo(
  9. AuthenticationToken token) throws AuthenticationException {
  10. // token是用户输入的
  11. // 第一步从token中取出身份信息
  12. String userCode = (String) token.getPrincipal();
  13. // 第二步:根据用户输入的userCode从数据库查询
  14. // ....
  15. // 如果查询不到返回null
  16. //数据库中用户账号是zhangsansan
  17. /*if(!userCode.equals("zhangsansan")){//
  18. return null;
  19. }*/
  20. // 模拟从数据库查询到密码
  21. String password = "111111";
  22. // 如果查询到返回认证信息AuthenticationInfo
  23. SimpleAuthenticationInfo simpleAuthenticationInfo = new SimpleAuthenticationInfo(
  24. userCode, password, this.getName());
  25. return simpleAuthenticationInfo;
  26. }
  27. // 用于授权
  28. @Override
  29. protected AuthorizationInfo doGetAuthorizationInfo(
  30. PrincipalCollection principals) {
  31. //从 principals获取主身份信息
  32. //将getPrimaryPrincipal方法返回值转为真实身份类型(在上边的doGetAuthenticationInfo认证通过填充到SimpleAuthenticationInfo中身份类型),
  33. String userCode = (String) principals.getPrimaryPrincipal();
  34. //根据身份信息获取权限信息
  35. //连接数据库...
  36. //模拟从数据库获取到数据
  37. List<String> permissions = new ArrayList<String>();
  38. permissions.add("user:create");//用户的创建
  39. permissions.add("items:add");//商品添加权限
  40. //....
  41. //查到权限数据,返回授权信息(要包括 上边的permissions)
  42. SimpleAuthorizationInfo simpleAuthorizationInfo = new SimpleAuthorizationInfo();
  43. //将上边查询到授权信息填充到simpleAuthorizationInfo对象中
  44. simpleAuthorizationInfo.addStringPermissions(permissions);
  45. return simpleAuthorizationInfo;
  46. }

6、在shiro-realm.ini中配置自定义的realm,将realm设置到securityManager中。

  1. [main]
  2. #自定义 realm
  3. customRealm=cn.itcast.shiro.realm.CustomRealm
  4. #将realm设置到securityManager,相当 于spring中注入
  5. securityManager.realms=$customRealm

7、测试自定义realm授权

  1. // 自定义realm进行资源授权测试
  2. @Test
  3. public void testAuthorizationCustomRealm() {
  4. // 创建SecurityManager工厂
  5. Factory<SecurityManager> factory = new IniSecurityManagerFactory(
  6. "classpath:shiro-realm.ini");
  7. // 创建SecurityManager
  8. SecurityManager securityManager = factory.getInstance();
  9. // 将SecurityManager设置到系统运行环境,和spring后将SecurityManager配置spring容器中,一般单例管理
  10. SecurityUtils.setSecurityManager(securityManager);
  11. // 创建subject
  12. Subject subject = SecurityUtils.getSubject();
  13. // 创建token令牌
  14. UsernamePasswordToken token = new UsernamePasswordToken("zhangsan",
  15. "111111");
  16. // 执行认证
  17. try {
  18. subject.login(token);
  19. } catch (AuthenticationException e) {
  20. // TODO Auto-generated catch block
  21. e.printStackTrace();
  22. }
  23. System.out.println("认证状态:" + subject.isAuthenticated());
  24. // 认证通过后执行授权
  25. // 基于资源的授权,调用isPermitted方法会调用CustomRealm从数据库查询正确权限数据
  26. // isPermitted传入权限标识符,判断user:create:1是否在CustomRealm查询到权限数据之内
  27. boolean isPermitted = subject.isPermitted("user:create:1");
  28. System.out.println("单个权限判断" + isPermitted);
  29. boolean isPermittedAll = subject.isPermittedAll("user:create:1",
  30. "user:create");
  31. System.out.println("多个权限判断" + isPermittedAll);
  32. // 使用check方法进行授权,如果授权不通过会抛出异常
  33. subject.checkPermission("items:add:1");
  34. }

8、授权的流程

1、对subject进行授权,调用方法isPermitted("permission串")

2、SecurityManager执行授权,通过ModularRealmAuthorizer执行授权

3、ModularRealmAuthorizer执行realm(自定义的CustomRealm)从数据库查询权限数据

调用realm的授权方法:doGetAuthorizationInfo

4、realm从数据库查询权限数据,返回ModularRealmAuthorizer

5、ModularRealmAuthorizer调用PermissionResolver进行权限串比对

6、如果比对后,isPermitted中"permission串"在realm查询到权限数据中,说明用户访问permission串有权限,否则 没有权限,抛出异常。

shiro的授权的更多相关文章

  1. Shiro【授权、整合Spirng、Shiro过滤器】

    前言 本文主要讲解的知识点有以下: Shiro授权的方式简单介绍 与Spring整合 初始Shiro过滤器 一.Shiro授权 上一篇我们已经讲解了Shiro的认证相关的知识了,现在我们来弄Shiro ...

  2. Shiro【授权过滤器、与ehcache整合、验证码、记住我】

    前言 本文主要讲解的知识点有以下: Shiro授权过滤器使用 Shiro缓存 与Ehcache整合 Shiro应用->实现验证码功能 记住我功能 一.授权过滤器测试 我们的授权过滤器使用的是pe ...

  3. Shiro:授权的相关实现

    Shiro:授权的相关实现 一.使用Shiro过滤器实现授权 设置好授权拦截跳转的请求地址 /** * 创建ShiroFilterFactoryBean */ @Bean public ShiroFi ...

  4. shiro的授权与认证

    shiro的授权与认证 package com.cy.pj.common.aspect;import java.lang.reflect.Method;import java.util.Arrays; ...

  5. 源码分析shiro认证授权流程

    1. shiro介绍 Apache Shiro是一个强大易用的Java安全框架,提供了认证.授权.加密和会话管理等功能: 认证 - 用户身份识别,常被称为用户“登录”: 授权 - 访问控制: 密码加密 ...

  6. Shiro:授权控制

    对容易忽略的地方记录如下: 1.需要引入下面2个依赖,具体版本根据自身环境修改: <dependency> <groupId>org.apache.geronimo.bundl ...

  7. Shrio00 Shiro角色授权、Shiro权限授权、开启Shiro缓存

    1 需求01 用户进行过认证登录后,某些接口是有权限限制的:如何实现只有相应权限的用户才可以调用相应接口 2 修改shiro配置类  ShiroConfiguration package cn.xia ...

  8. shiro认证授权

    一.shiro基础概念 Authentication:身份认证 / 登录,验证用户是不是拥有相应的身份: Authorization:授权,即权限验证,验证某个已认证的用户是否拥有某个权限:即判断用户 ...

  9. shiro框架学习-2-springboot整合shiro及Shiro认证授权流程

    1. 添加依赖 <dependency> <groupId>org.springframework.boot</groupId> <artifactId> ...

随机推荐

  1. [Erlang14]怎样模拟节点互连后的各种失败情况?

    情景: 当节点群互连时,会通过心跳包检查所连接节点是不是连接正常,这个心跳时间默认为60s,可以通过 net_kernel:set_net_ticktime(600). 来重设这个时间值,怎么测试? ...

  2. CLR via C# 读书笔记-27.计算限制的异步操作(上篇)

    前言 学习这件事情是一个习惯,不能停...另外这篇已经看过两个月过去,但觉得有些事情不总结跟没做没啥区别,遂记下此文 1.CLR线程池基础 2.ThreadPool的简单使用练习 3.执行上下文 4. ...

  3. VS 2015 IDE 不支持 MS SQL 2000 生成 dbml

    解决办法: 通过命令手动生成 然后把生成的ERendering.dbml 文件,通过工程项目-添加-现有项,加入项目.

  4. mysql 命令备份还原

    备份 mysqldump -h localhost -uroot -p123456 springbootdb > e:/springbootdb.sql 还原 mysql -h localhos ...

  5. OpenGL学习脚印:背面剔除(Face Culling)

    写在前面 在绘制封闭类型的几何对象时,开启背面剔除功能能够提高渲染性能.本节简要介绍下背面剔除,示例程序可以在我的github下载. 什么是背面剔除 当我们观察场景中对象时,一般只能以一定角度来观察, ...

  6. MySQL开启日志记录查询/执行过的SQL语句

    作为后端开发者,遇到数据库问题的时候应该通过分析SQL语句来跟进问题所在,该方法可以记录所有的查询/执行的SQL语句到日志文件. 方法有几种,但是个人觉得以下这种最简单,但是重启MySQL服务后需要重 ...

  7. BZOJ4032: [HEOI2015]最短不公共子串(后缀自动机+序列自动机)

    题目描述 在虐各种最长公共子串.子序列的题虐的不耐烦了之后,你决定反其道而行之. 一个串的“子串”指的是它的连续的一段,例如bcd是abcdef的子串,但bde不是. 一个串的“子序列”指的是它的可以 ...

  8. 2019.2.15 t2

    考虑倒过来计算最短路径长度,设dis[u]表示在最坏情况下,点u到最近的一 个出口的最短路,则p个出口的dis值都是0,答案即为dis[0]. #include <cstdio> #inc ...

  9. NOIP模拟题汇总(加厚版)

    \(NOIP\)模拟题汇总(加厚版) T1 string 描述 有一个仅由 '0' 和 '1' 组成的字符串 \(A\),可以对其执行下列两个操作: 删除 \(A\)中的第一个字符: 若 \(A\)中 ...

  10. leetcode-551-Student Attendance Record I(判断是否出现连续几个相同字符)

    题目描述: You are given a string representing an attendance record for a student. The record only contai ...