美文网首页
[Shiro]多Realm时,指定登录Realm

[Shiro]多Realm时,指定登录Realm

作者: ae7729ac3ac8 | 来源:发表于2017-08-15 23:15 被阅读0次

    [Shiro]多Realm时,指定登录Realm

    之前写过多Realm登录前后台区分问题,实现了当时的需求,可是再下一步的时候,角色与权限校验会失效。为了满足当时的项目需求,又拓展了鉴权源,以及若干功能类。在本人看来,这种方式很Low,玷污了Shiro。
    现在有一个较好的方式去实现前后台的认证与鉴权。通过自定义Token类来达到鉴别目的。


    Token的介绍

    Shiro使用Token作为认证的对象。 在Filter中拦截登录信息创建Token调用登录,或者我们自己在控制器中手动登录。调用登录,会迭代配置的Realm进行验证。

    Shiro的FormAuthenticationFilter创建Token

    FormAuthenticationFilter中创建Token的方法


    FormAuthenticationFilter

    最终调用到它的父类AuthenticatingFilter


    AuthenticatingFilter

    控制器里手动创建Token登录

    这里写图片描述

    Realm的定义

    Shiro在进行登录验证时候,会检查Realm是否支持该Token,如果不支持跳过当前Realm,继续下一个Realm。

        public Class getAuthenticationTokenClass() {
            //支持的Token类的class
            return authenticationTokenClass;
        }
        //是否支持该token
        //Realm支持的类可以是token的子类或相同
        public boolean supports(AuthenticationToken token) {
            return token != null &&    getAuthenticationTokenClass().isAssignableFrom(token.getClass());
        }
    

    自定义Realm一般会继承AuthorizingRealm来支持登录的验证。

    我们可以重写public Class getAuthenticationTokenClass()来支持我们的Token。

    自定义Token

    Shiro的UsernamePasswordToken源码

    public class UsernamePasswordToken implements HostAuthenticationToken, RememberMeAuthenticationToken {
    
        private String username;
    
        private char[] password;
    
        private boolean rememberMe = false;
    
        private String host;
    
        public UsernamePasswordToken() {
        }
    
        public UsernamePasswordToken(final String username, final char[] password) {
            this(username, password, false, null);
        }
        public UsernamePasswordToken(final String username, final String password, final String host) {
            this(username, password != null ? password.toCharArray() : null, false, host);
        }
        public Object getPrincipal() {
            return getUsername();
        }
    
     
        public Object getCredentials() {
            return getPassword();
        }
    
      
        public String getHost() {
            return host;
        }
    
        public void setHost(String host) {
            this.host = host;
        }
    
        public boolean isRememberMe() {
            return rememberMe;
        }
    
        public void setRememberMe(boolean rememberMe) {
            this.rememberMe = rememberMe;
        }
    
    
        public void clear() {
            this.username = null;
            this.host = null;
            this.rememberMe = false;
    
            if (this.password != null) {
                for (int i = 0; i < password.length; i++) {
                    this.password[i] = 0x00;
                }
                this.password = null;
            }
    
        }
    }
    

    UsernamePasswordToken实现了AuthenticationToken(认证)、HostAuthenticationToken(用户ip)、RememberMeAuthenticationToken(记住我)接口。

    AuthenticationToken

    public interface AuthenticationToken extends Serializable {
    
        /**
        *获取登陆用户名
        */
        Object getPrincipal();
    
        /**
         *获取密码
         */
        Object getCredentials();
    
    }
    

    由于Realm检查是否支持Token类的是允许支持验证子类的,我们不能去继承UsernamePasswordToken,需要自己实现一个Token。直接拷贝代码,修改类名就可以。或者根据你的业务需求,重写一部分功能。


    每一个自定义Realm对应支持一种Token。就可以使登录验证时不会重复调用Realm多次验证和异常消息的覆盖。

    [示例代码] https://git.oschina.net/yihyforever/shiro-realms.git

    下一篇,Shiro的缓存及坑点

    相关文章

      网友评论

          本文标题:[Shiro]多Realm时,指定登录Realm

          本文链接:https://www.haomeiwen.com/subject/iolbrxtx.html