shiro结合数据库

RBAC概念

RBAC 是当下权限系统的设计基础,同时有两种解释:
一: Role-Based Access Control,基于角色的访问控制
即,你要能够删除产品,那么当前用户就必须拥有产品经理这个角色
二:Resource-Based Access Control,基于资源的访问控制
即,你要能够删除产品,那么当前用户就必须拥有删除产品这样的权限

表结构

基于 RBAC 概念, 就会存在3 张基础表: 用户,角色,权限, 以及 2 张中间表来建立 用户与角色的多对多关系,角色与权限的多对多关系。 用户与权限之间也是多对多关系,但是是通过 角色间接建立的。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
DROP DATABASE IF EXISTS shiro;
CREATE DATABASE shiro DEFAULT CHARACTER SET utf8;
USE shiro;

drop table if exists user;
drop table if exists role;
drop table if exists permission;
drop table if exists user_role;
drop table if exists role_permission;

create table user (
id bigint auto_increment,
name varchar(100),
password varchar(100),
constraint pk_users primary key(id)
) charset=utf8 ENGINE=InnoDB;

create table role (
id bigint auto_increment,
name varchar(100),
constraint pk_roles primary key(id)
) charset=utf8 ENGINE=InnoDB;

create table permission (
id bigint auto_increment,
name varchar(100),
constraint pk_permissions primary key(id)
) charset=utf8 ENGINE=InnoDB;

create table user_role (
uid bigint,
rid bigint,
constraint pk_users_roles primary key(uid, rid)
) charset=utf8 ENGINE=InnoDB;

create table role_permission (
rid bigint,
pid bigint,
constraint pk_roles_permissions primary key(rid, pid)
) charset=utf8 ENGINE=InnoDB;

shiro例子

  1. User
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    public class User {

    private int id;
    private String name;
    private String password;
    public String getName() {
    return name;
    }
    public void setName(String name) {
    this.name = name;
    }
    public String getPassword() {
    return password;
    }
    public void setPassword(String password) {
    this.password = password;
    }
    public int getId() {
    return id;
    }
    public void setId(int id) {
    this.id = id;
    }

    }
  2. DAO
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    42
    43
    44
    45
    46
    47
    48
    49
    50
    51
    52
    53
    54
    55
    56
    57
    58
    59
    60
    61
    62
    63
    64
    65
    66
    67
    68
    69
    70
    71
    72
    73
    74
    75
    76
    77
    78
    79
    80
    81
    82
    83
    84
    85
    86
    87
    88
    89
    90
    91
    92
    93
    94
    95
    import java.sql.Connection;
    import java.sql.DriverManager;
    import java.sql.PreparedStatement;
    import java.sql.ResultSet;
    import java.sql.SQLException;
    import java.util.HashSet;
    import java.util.Set;

    public class DAO {
    public DAO() {
    try {
    Class.forName("com.mysql.jdbc.Driver");
    } catch (ClassNotFoundException e) {
    e.printStackTrace();
    }
    }

    public Connection getConnection() throws SQLException {
    return DriverManager.getConnection("jdbc:mysql://127.0.0.1:3306/shiro?characterEncoding=UTF-8", "root",
    "root");
    }
    //根据用户名查密码
    public String getPassword(String userName) {
    String sql = "select password from user where name = ?";
    try (Connection c = getConnection(); PreparedStatement ps = c.prepareStatement(sql);) {

    ps.setString(1, userName);

    ResultSet rs = ps.executeQuery();

    if (rs.next())
    return rs.getString("password");

    } catch (SQLException e) {

    e.printStackTrace();
    }
    return null;
    }
    //根据用户名查角色
    public Set<String> listRoles(String userName) {

    Set<String> roles = new HashSet<>();
    String sql = "select r.name from user u "
    + "left join user_role ur on u.id = ur.uid "
    + "left join Role r on r.id = ur.rid "
    + "where u.name = ?";
    try (Connection c = getConnection(); PreparedStatement ps = c.prepareStatement(sql);) {
    ps.setString(1, userName);
    ResultSet rs = ps.executeQuery();

    while (rs.next()) {
    roles.add(rs.getString(1));
    }

    } catch (SQLException e) {

    e.printStackTrace();
    }
    return roles;
    }
    //根据用户名查权限
    public Set<String> listPermissions(String userName) {
    Set<String> permissions = new HashSet<>();
    String sql =
    "select p.name from user u "+
    "left join user_role ru on u.id = ru.uid "+
    "left join role r on r.id = ru.rid "+
    "left join role_permission rp on r.id = rp.rid "+
    "left join permission p on p.id = rp.pid "+
    "where u.name =?";

    try (Connection c = getConnection(); PreparedStatement ps = c.prepareStatement(sql);) {

    ps.setString(1, userName);

    ResultSet rs = ps.executeQuery();

    while (rs.next()) {
    permissions.add(rs.getString(1));
    }

    } catch (SQLException e) {

    e.printStackTrace();
    }
    return permissions;
    }
    public static void main(String[] args) {
    System.out.println(new DAO().listRoles("zhang3"));
    System.out.println(new DAO().listRoles("li4"));
    System.out.println(new DAO().listPermissions("zhang3"));
    System.out.println(new DAO().listPermissions("li4"));
    }
    }

    Realm 概念

    在 Shiro 中存在 Realm 这么个概念, Realm 这个单词翻译为 域,其实是非常难以理解的。
    域 是什么鬼?和权限有什么毛关系? 这个单词Shiro的作者用的非常不好,让人很难理解。
    那么 Realm 在 Shiro里到底扮演什么角色呢?
    当应用程序向 Shiro 提供了 账号和密码之后, Shiro 就会问 Realm 这个账号密码是否对, 如果对的话,其所对应的用户拥有哪些角色,哪些权限。
    所以Realm 是什么? 其实就是个中介。 Realm 得到了 Shiro 给的用户和密码后,有可能去找 ini 文件,就像Shiro 入门中的 shiro.ini,也可以去找数据库,就如同本知识点中的 DAO 查询信息。

    DatabaseRealm

    DatabaseRealm 就是用来通过数据库 验证用户,和相关授权的类。
    两个方法分别做验证和授权:
    doGetAuthenticationInfo(), doGetAuthorizationInfo()
    细节在代码里都有详细注释,请仔细阅读。
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    42
    43
    44
    45
    46
    47
    48
    import java.util.Set;

    import org.apache.shiro.authc.AuthenticationException;
    import org.apache.shiro.authc.AuthenticationInfo;
    import org.apache.shiro.authc.AuthenticationToken;
    import org.apache.shiro.authc.SimpleAuthenticationInfo;
    import org.apache.shiro.authc.UsernamePasswordToken;
    import org.apache.shiro.authz.AuthorizationInfo;
    import org.apache.shiro.authz.SimpleAuthorizationInfo;
    import org.apache.shiro.realm.AuthorizingRealm;
    import org.apache.shiro.subject.PrincipalCollection;

    public class DatabaseRealm extends AuthorizingRealm {

    @Override
    protected AuthorizationInfo doGetAuthorizationInfo(PrincipalCollection principalCollection) {
    //能进入到这里,表示账号已经通过验证了
    String userName =(String) principalCollection.getPrimaryPrincipal();
    //通过DAO获取角色和权限
    Set<String> permissions = new DAO().listPermissions(userName);
    Set<String> roles = new DAO().listRoles(userName);

    //授权对象
    SimpleAuthorizationInfo s = new SimpleAuthorizationInfo();
    //把通过DAO获取到的角色和权限放进去
    s.setStringPermissions(permissions);
    s.setRoles(roles);
    return s;
    }

    @Override
    protected AuthenticationInfo doGetAuthenticationInfo(AuthenticationToken token) throws AuthenticationException {
    //获取账号密码
    UsernamePasswordToken t = (UsernamePasswordToken) token;
    String userName= token.getPrincipal().toString();
    String password= new String( t.getPassword());
    //获取数据库中的密码
    String passwordInDB = new DAO().getPassword(userName);

    //如果为空就是账号不存在,如果不相同就是密码错误,但是都抛出AuthenticationException,而不是抛出具体错误原因,免得给破解者提供帮助信息
    if(null==passwordInDB || !passwordInDB.equals(password))
    throw new AuthenticationException();

    //认证信息里存放账号密码, getName() 是当前Realm的继承方法,通常返回当前类名 :databaseRealm
    SimpleAuthenticationInfo a = new SimpleAuthenticationInfo(userName,password,getName());
    return a;
    }

    修改shiro.ini

    1
    2
    3
    [main]
    databaseRealm=packagename.databaseRealmName
    securityManager.realms=$databaserealm

    TestRealm

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    42
    43
    44
    45
    46
    47
    48
    49
    50
    51
    52
    53
    54
    55
    56
    57
    58
    59
    60
    61
    62
    63
    64
    65
    66
    67
    68
    69
    70
    71
    72
    73
    74
    75
    76
    77
    78
    79
    80
    81
    82
    83
    84
    85
    86
    87
    88
    89
    90
    91
    92
    93
    94
    95
    96
    97
    98
    99
    100
    101
    102
    103
    104
    105
    106
    107
    108
    109
    110
    111
    112
    113
    114
    115
    116
    117
    118
    119
    120
    121
    122
    123
    124
    125
    126
    import java.util.ArrayList;
    import java.util.List;

    import org.apache.shiro.SecurityUtils;
    import org.apache.shiro.authc.AuthenticationException;
    import org.apache.shiro.authc.UsernamePasswordToken;
    import org.apache.shiro.config.IniSecurityManagerFactory;
    import org.apache.shiro.mgt.SecurityManager;
    import org.apache.shiro.subject.Subject;
    import org.apache.shiro.util.Factory;

    public class TestShiro {
    public static void main(String[] args) {
    //用户们
    User zhang3 = new User();
    zhang3.setName("zhang3");
    zhang3.setPassword("12345");

    User li4 = new User();
    li4.setName("li4");
    li4.setPassword("abcde");

    User wang5 = new User();
    wang5.setName("wang5");
    wang5.setPassword("wrongpassword");

    List<User> users = new ArrayList<>();

    users.add(zhang3);
    users.add(li4);
    users.add(wang5);
    //角色们
    String roleAdmin = "admin";
    String roleProductManager ="productManager";

    List<String> roles = new ArrayList<>();
    roles.add(roleAdmin);
    roles.add(roleProductManager);

    //权限们
    String permitAddProduct = "addProduct";
    String permitAddOrder = "addOrder";

    List<String> permits = new ArrayList<>();
    permits.add(permitAddProduct);
    permits.add(permitAddOrder);

    //登陆每个用户
    for (User user : users) {
    if(login(user))
    System.out.printf("%s \t成功登陆,用的密码是 %s\t %n",user.getName(),user.getPassword());
    else
    System.out.printf("%s \t成功失败,用的密码是 %s\t %n",user.getName(),user.getPassword());
    }

    System.out.println("-------how2j 分割线------");

    //判断能够登录的用户是否拥有某个角色
    for (User user : users) {
    for (String role : roles) {
    if(login(user)) {
    if(hasRole(user, role))
    System.out.printf("%s\t 拥有角色: %s\t%n",user.getName(),role);
    else
    System.out.printf("%s\t 不拥有角色: %s\t%n",user.getName(),role);
    }
    }
    }
    System.out.println("-------how2j 分割线------");

    //判断能够登录的用户,是否拥有某种权限
    for (User user : users) {
    for (String permit : permits) {
    if(login(user)) {
    if(isPermitted(user, permit))
    System.out.printf("%s\t 拥有权限: %s\t%n",user.getName(),permit);
    else
    System.out.printf("%s\t 不拥有权限: %s\t%n",user.getName(),permit);
    }
    }
    }
    }

    private static boolean hasRole(User user, String role) {
    Subject subject = getSubject(user);
    return subject.hasRole(role);
    }

    private static boolean isPermitted(User user, String permit) {
    Subject subject = getSubject(user);
    return subject.isPermitted(permit);
    }

    private static Subject getSubject(User user) {
    //加载配置文件,并获取工厂
    Factory<SecurityManager> factory = new IniSecurityManagerFactory("classpath:shiro.ini");
    //获取安全管理者实例
    SecurityManager sm = factory.getInstance();
    //将安全管理者放入全局对象
    SecurityUtils.setSecurityManager(sm);
    //全局对象通过安全管理者生成Subject对象
    Subject subject = SecurityUtils.getSubject();

    return subject;
    }

    private static boolean login(User user) {
    Subject subject= getSubject(user);
    //如果已经登录过了,退出
    if(subject.isAuthenticated())
    subject.logout();

    //封装用户的数据
    UsernamePasswordToken token = new UsernamePasswordToken(user.getName(), user.getPassword());
    try {
    //将用户的数据token 最终传递到Realm中进行对比
    subject.login(token);
    } catch (AuthenticationException e) {
    //验证错误
    return false;
    }

    return subject.isAuthenticated();
    }

    }
文章作者: Anders Cao
文章链接: http://yoursite.com/2019/04/08/shiro结合数据库/
版权声明: 本博客所有文章除特别声明外,均采用 CC BY-NC-SA 4.0 许可协议。转载请注明来自 Anders's Blog
打赏
  • 微信
  • 支付寶