`
jinnianshilongnian
  • 浏览: 21424395 次
  • 性别: Icon_minigender_1
博客专栏
5c8dac6a-21dc-3466-8abb-057664ab39c7
跟我学spring3
浏览量:2403176
D659df3e-4ad7-3b12-8b9a-1e94abd75ac3
Spring杂谈
浏览量:2996176
43989fe4-8b6b-3109-aaec-379d27dd4090
跟开涛学SpringMVC...
浏览量:5630413
1df97887-a9e1-3328-b6da-091f51f886a1
Servlet3.1规范翻...
浏览量:257187
4f347843-a078-36c1-977f-797c7fc123fc
springmvc杂谈
浏览量:1592662
22722232-95c1-34f2-b8e1-d059493d3d98
hibernate杂谈
浏览量:248795
45b32b6f-7468-3077-be40-00a5853c9a48
跟我学Shiro
浏览量:5845661
Group-logo
跟我学Nginx+Lua开...
浏览量:697478
5041f67a-12b2-30ba-814d-b55f466529d5
亿级流量网站架构核心技术
浏览量:779851
社区版块
存档分类
最新评论

第四章 INI配置——《跟我学Shiro》

阅读更多

 

目录贴: 跟我学Shiro目录贴

 

之前章节我们已经接触过一些INI配置规则了,如果大家使用过如Spring之类的IoC/DI容器的话,Shiro提供的INI配置也是非常类似的,即可以理解为是一个IoC/DI容器,但是区别在于它从一个根对象securityManager开始。

 

4.1 根对象SecurityManager

从之前的Shiro架构图可以看出,Shiro是从根对象SecurityManager进行身份验证和授权的;也就是所有操作都是自它开始的,这个对象是线程安全且真个应用只需要一个即可,因此Shiro提供了SecurityUtils让我们绑定它为全局的,方便后续操作。

 

因为Shiro的类都是POJO的,因此都很容易放到任何IoC容器管理。但是和一般的IoC容器的区别在于,Shiro从根对象securityManager开始导航;Shiro支持的依赖注入:public空参构造器对象的创建、setter依赖注入。

 

1、纯Java代码写法(com.github.zhangkaitao.shiro.chapter4.NonConfigurationCreateTest): 

DefaultSecurityManager securityManager = new DefaultSecurityManager();
//设置authenticator
ModularRealmAuthenticator authenticator = new ModularRealmAuthenticator();
authenticator.setAuthenticationStrategy(new AtLeastOneSuccessfulStrategy());
securityManager.setAuthenticator(authenticator);

//设置authorizer
ModularRealmAuthorizer authorizer = new ModularRealmAuthorizer();
authorizer.setPermissionResolver(new WildcardPermissionResolver());
securityManager.setAuthorizer(authorizer);

//设置Realm
DruidDataSource ds = new DruidDataSource();
ds.setDriverClassName("com.mysql.jdbc.Driver");
ds.setUrl("jdbc:mysql://localhost:3306/shiro");
ds.setUsername("root");
ds.setPassword("");

JdbcRealm jdbcRealm = new JdbcRealm();
jdbcRealm.setDataSource(ds);
jdbcRealm.setPermissionsLookupEnabled(true);
securityManager.setRealms(Arrays.asList((Realm) jdbcRealm));

//将SecurityManager设置到SecurityUtils 方便全局使用
SecurityUtils.setSecurityManager(securityManager);

Subject subject = SecurityUtils.getSubject();
UsernamePasswordToken token = new UsernamePasswordToken("zhang", "123");
subject.login(token);
Assert.assertTrue(subject.isAuthenticated());

 

 

2.1、等价的INI配置(shiro-config.ini) 

[main]
#authenticator
authenticator=org.apache.shiro.authc.pam.ModularRealmAuthenticator
authenticationStrategy=org.apache.shiro.authc.pam.AtLeastOneSuccessfulStrategy
authenticator.authenticationStrategy=$authenticationStrategy
securityManager.authenticator=$authenticator

#authorizer
authorizer=org.apache.shiro.authz.ModularRealmAuthorizer
permissionResolver=org.apache.shiro.authz.permission.WildcardPermissionResolver
authorizer.permissionResolver=$permissionResolver
securityManager.authorizer=$authorizer

#realm
dataSource=com.alibaba.druid.pool.DruidDataSource
dataSource.driverClassName=com.mysql.jdbc.Driver
dataSource.url=jdbc:mysql://localhost:3306/shiro
dataSource.username=root
#dataSource.password=
jdbcRealm=org.apache.shiro.realm.jdbc.JdbcRealm
jdbcRealm.dataSource=$dataSource
jdbcRealm.permissionsLookupEnabled=true
securityManager.realms=$jdbcRealm 

即使没接触过IoC容器的知识,如上配置也是很容易理解的:

1、对象名=全限定类名  相对于调用public无参构造器创建对象

2、对象名.属性名=值    相当于调用setter方法设置常量值

3、对象名.属性名=$对象引用    相当于调用setter方法设置对象引用

 

2.2、Java代码(com.github.zhangkaitao.shiro.chapter4.ConfigurationCreateTest) 

Factory<org.apache.shiro.mgt.SecurityManager> factory =
         new IniSecurityManagerFactory("classpath:shiro-config.ini");

org.apache.shiro.mgt.SecurityManager securityManager = factory.getInstance();

//将SecurityManager设置到SecurityUtils 方便全局使用
SecurityUtils.setSecurityManager(securityManager);
Subject subject = SecurityUtils.getSubject();
UsernamePasswordToken token = new UsernamePasswordToken("zhang", "123");
subject.login(token);

Assert.assertTrue(subject.isAuthenticated()); 

如上代码是从Shiro INI配置中获取相应的securityManager实例:

1、默认情况先创建一个名字为securityManager,类型为org.apache.shiro.mgt.DefaultSecurityManager的默认的SecurityManager,如果想自定义,只需要在ini配置文件中指定“securityManager=SecurityManager实现类”即可,名字必须为securityManager,它是起始的根;

2、IniSecurityManagerFactory是创建securityManager的工厂,其需要一个ini配置文件路径,其支持“classpath:”(类路径)、“file:”(文件系统)、“url:”(网络)三种路径格式,默认是文件系统;

3、接着获取SecuriyManager实例,后续步骤和之前的一样。

 

从如上可以看出Shiro INI配置方式本身提供了一个简单的IoC/DI机制方便在配置文件配置,但是是从securityManager这个根对象开始导航。   

 

4.2 INI配置

ini配置文件类似于Java中的properties(key=value),不过提供了将key/value分类的特性,key是每个部分不重复即可,而不是整个配置文件。如下是INI配置分类: 

[main]
#提供了对根对象securityManager及其依赖的配置
securityManager=org.apache.shiro.mgt.DefaultSecurityManager
…………
securityManager.realms=$jdbcRealm

[users]
#提供了对用户/密码及其角色的配置,用户名=密码,角色1,角色2
username=password,role1,role2

[roles]
#提供了角色及权限之间关系的配置,角色=权限1,权限2
role1=permission1,permission2

[urls]
#用于web,提供了对web url拦截相关的配置,url=拦截器[参数],拦截器
/index.html = anon
/admin/** = authc, roles[admin], perms["permission1"]

 

[main]部分

提供了对根对象securityManager及其依赖对象的配置。

创建对象 

securityManager=org.apache.shiro.mgt.DefaultSecurityManager

其构造器必须是public空参构造器,通过反射创建相应的实例。

 

常量值setter注入 

dataSource.driverClassName=com.mysql.jdbc.Driver
jdbcRealm.permissionsLookupEnabled=true 

会自动调用jdbcRealm.setPermissionsLookupEnabled(true),对于这种常量值会自动类型转换。

 

对象引用setter注入 

authenticator=org.apache.shiro.authc.pam.ModularRealmAuthenticator
authenticationStrategy=org.apache.shiro.authc.pam.AtLeastOneSuccessfulStrategy
authenticator.authenticationStrategy=$authenticationStrategy
securityManager.authenticator=$authenticator 

会自动通过securityManager.setAuthenticator(authenticator)注入引用依赖。

 

嵌套属性setter注入 

securityManager.authenticator.authenticationStrategy=$authenticationStrategy 

也支持这种嵌套方式的setter注入。

 

byte数组setter注入 

#base64 byte[]
authenticator.bytes=aGVsbG8=
#hex byte[]
authenticator.bytes=0x68656c6c6f 

默认需要使用Base64进行编码,也可以使用0x十六进制。

 

Array/Set/List setter注入 

authenticator.array=1,2,3
authenticator.set=$jdbcRealm,$jdbcRealm 

多个之间通过“,”分割。

 

Map setter注入

authenticator.map=$jdbcRealm:$jdbcRealm,1:1,key:abc

即格式是:map=key:value,key:value,可以注入常量及引用值,常量的话都看作字符串(即使有泛型也不会自动造型)。        

 

实例化/注入顺序 

realm=Realm1
realm=Realm12

authenticator.bytes=aGVsbG8=
authenticator.bytes=0x68656c6c6f 

后边的覆盖前边的注入。

 

测试用例请参考配置文件shiro-config-main.ini。 

 

[users]部分

配置用户名/密码及其角色,格式:“用户名=密码,角色1,角色2”,角色部分可省略。如:

[users]
zhang=123,role1,role2
wang=123 

密码一般生成其摘要/加密存储,后续章节介绍。

 

[roles]部分

配置角色及权限之间的关系,格式:“角色=权限1,权限2”;如:

[roles]
role1=user:create,user:update
role2=* 

如果只有角色没有对应的权限,可以不配roles,具体规则请参考授权章节。

 

[urls]部分

配置url及相应的拦截器之间的关系,格式:“url=拦截器[参数],拦截器[参数],如:  

[urls]
/admin/** = authc, roles[admin], perms["permission1"] 

具体规则参见web相关章节。 

 

示例源代码:https://github.com/zhangkaitao/shiro-example;可加群134755960探讨Spring/Shiro技术。

 

33
6
分享到:
评论
18 楼 xiejx618 2018-08-13  
看不明白   这个对象是线程安全且真个应用只需要一个即可
17 楼 coder蒙奇君杰 2018-01-05  
正在学习中,可能还需要再多翻阅和研习下。
写的很详实,谢谢
16 楼 coder蒙奇君杰 2018-01-05  
留个言,也算是学到了东西,还会再翻阅细研的。
  :idea : 写的很棒!
15 楼 白鹿洞书院 2017-03-30  
文章写的太好了
14 楼 walidake 2015-12-28  
您好~   关于数组的配置并不能set进去~~~  不知道什么原因  能说一下么?  楼主
13 楼 itoajm 2014-09-15  
在学习,一直支持
12 楼 jinnianshilongnian 2014-02-25  
601235723 写道
更新有点快,赶不上了

哈哈,慢慢来
11 楼 601235723 2014-02-24  
更新有点快,赶不上了
10 楼 Dead_knight 2014-02-24  
supercwg 写道
还是最核心的那个问题,就是配置要完全跟数据库很好地结合起来才行。现在很多组织用户权限管理系统都没有使用spring security和shiro并不是不重视安全性,而是这些框架好像都主要是面向代码级或文件(xml或ini)配置式的,无法像自己开发的权限系统,管理员只要在网页上设置一下就完成用户、岗位、角色、权限的数据维护了,权限就能马上生效了(弊端是绝大多数的组织用户权限管理系统其实根本就谈不上什么安全性)。不知道能否将shiro与基于数据库的组织用户权限管理系统完美结合的可能性有没有啊?顺便补上一句,你的文章真是写得TMD太好了

安全框架最核心的两方面是:认证、授权
然后会在此基础上提供额外的功能扩展(如spring security的openid、cas、以及session并发管理等等)
所以说,安全框架不可能结合具体的业务系统。虽然国内大多数系统都是用户、部门、岗位、角色等等,但是无法统一。一个好的安全框架不仅提供最核心的认证、授权,还应该有更好的扩展。这样,你就完全可以做二次开发,把rbac那一套用上去了。
9 楼 jinnianshilongnian 2014-02-24  
武林第一帅哥 写道
哎,,,你这文章写的晚了点。。。如果再早上几个月,我就能有收获了。。。自己看了一周的官网英文文档,,太费事了,,才搞定shiro,,呜呜,,,楼主,你该早点啊。。。呜呜

这也是应一些网友的要求,过年期间写的;再看还是能有收获的 
8 楼 武林第一帅哥 2014-02-24  
哎,,,你这文章写的晚了点。。。如果再早上几个月,我就能有收获了。。。自己看了一周的官网英文文档,,太费事了,,才搞定shiro,,呜呜,,,楼主,你该早点啊。。。呜呜
7 楼 jinnianshilongnian 2014-02-24  
supercwg 写道
还是最核心的那个问题,就是配置要完全跟数据库很好地结合起来才行。现在很多组织用户权限管理系统都没有使用spring security和shiro并不是不重视安全性,而是这些框架好像都主要是面向代码级或文件(xml或ini)配置式的,无法像自己开发的权限系统,管理员只要在网页上设置一下就完成用户、岗位、角色、权限的数据维护了,权限就能马上生效了(弊端是绝大多数的组织用户权限管理系统其实根本就谈不上什么安全性)。不知道能否将shiro与基于数据库的组织用户权限管理系统完美结合的可能性有没有啊?顺便补上一句,你的文章真是写得TMD太好了

嗯 谢谢啊,这个后续章节介绍,一步步来
6 楼 supercwg 2014-02-24  
还是最核心的那个问题,就是配置要完全跟数据库很好地结合起来才行。现在很多组织用户权限管理系统都没有使用spring security和shiro并不是不重视安全性,而是这些框架好像都主要是面向代码级或文件(xml或ini)配置式的,无法像自己开发的权限系统,管理员只要在网页上设置一下就完成用户、岗位、角色、权限的数据维护了,权限就能马上生效了(弊端是绝大多数的组织用户权限管理系统其实根本就谈不上什么安全性)。不知道能否将shiro与基于数据库的组织用户权限管理系统完美结合的可能性有没有啊?顺便补上一句,你的文章真是写得TMD太好了
5 楼 jinnianshilongnian 2014-02-24  
fighting_2013 写道
后面能否结合你开源的那个实际项目里面的shiro讲解

里边会提到一些,但是不会完全讲;但是比如最后的综合示例 和它差不多的;但是在模型上做了简化,方便学习;看完后再学习那个开源项目,问题不大
4 楼 fighting_2013 2014-02-24  
后面能否结合你开源的那个实际项目里面的shiro讲解
3 楼 jinnianshilongnian 2014-02-24  
lucky16 写道
到时候完结了,可以把PDF和WORD版本弄出来。

这个是一定要做的
2 楼 lucky16 2014-02-24  
到时候完结了,可以把PDF和WORD版本弄出来。
1 楼 mingnianshimanian 2014-02-24  

相关推荐

Global site tag (gtag.js) - Google Analytics