• shiro学习笔记-Subject#login(token)实现过程


    本博文所有的代码均为shiro官网(http://shiro.apache.org/)中shiro 1.3.2版本中的源码。

    追踪Subject的login(AuthenticationToken token)方法,其调用的为DelegatingSubject类的login方法,DelegatingSubject实现了Subject接口,DelegatingSubject#login如下:

     1 public void login(AuthenticationToken token) throws AuthenticationException {
     2     clearRunAsIdentitiesInternal();
     3     Subject subject = securityManager.login(this, token);
     4 
     5     PrincipalCollection principals;
     6 
     7     String host = null;
     8 
     9     if (subject instanceof DelegatingSubject) {
    10         DelegatingSubject delegating = (DelegatingSubject) subject;
    11         //we have to do this in case there are assumed identities - we don't want to lose the 'real' principals:
    12         principals = delegating.principals;
    13         host = delegating.host;
    14     } else {
    15         principals = subject.getPrincipals();
    16     }
    17 
    18     if (principals == null || principals.isEmpty()) {
    19         String msg = "Principals returned from securityManager.login( token ) returned a null or " +
    20                 "empty value.  This value must be non null and populated with one or more elements.";
    21         throw new IllegalStateException(msg);
    22     }
    23     this.principals = principals;
    24     this.authenticated = true;
    25     if (token instanceof HostAuthenticationToken) {
    26         host = ((HostAuthenticationToken) token).getHost();
    27     }
    28     if (host != null) {
    29         this.host = host;
    30     }
    31     Session session = subject.getSession(false);
    32     if (session != null) {
    33         this.session = decorate(session);
    34     } else {
    35         this.session = null;
    36     }
    37 }

    在上面代码的第三行:Subject subject = securityManager.login(this, token); 注意到其调用了SecurityManager的login方法,SecurityManager为接口,实际上调用的其实现类DefaultSecurityManager的login方法,方法如下:

     1 public Subject login(Subject subject, AuthenticationToken token) throws AuthenticationException {
     2     AuthenticationInfo info;
     3     try {
     4         info = authenticate(token);
     5     } catch (AuthenticationException ae) {
     6         try {
     7             onFailedLogin(token, ae, subject);
     8         } catch (Exception e) {
     9             if (log.isInfoEnabled()) {
    10                 log.info("onFailedLogin method threw an " +
    11                         "exception.  Logging and propagating original AuthenticationException.", e);
    12             }
    13         }
    14         throw ae; //propagate
    15     }
    16 
    17     Subject loggedIn = createSubject(token, info, subject);
    18 
    19     onSuccessfulLogin(token, info, loggedIn);
    20 
    21     return loggedIn;
    22 }

    在上面代码第四行:info = authenticate(token); 继续跟踪,发现authenticate(AuthenticationToken token);方法为DefaultSecurityManager的父类AuthenticatingSecurityManager的方法,AuthenticatingSecurityManager#authenticate方法如下:

    1 public AuthenticationInfo authenticate(AuthenticationToken token) throws AuthenticationException {
    2     return this.authenticator.authenticate(token);
    3 }

    authenticator为Authenticator接口,继续跟踪,AbstractAuthenticator抽象类实现了Authenticator接口,接下来继续查看AbstractAuthenticator#authenticate(token);方法:

     1 public final AuthenticationInfo authenticate(AuthenticationToken token) throws AuthenticationException {
     2 
     3     if (token == null) {
     4         throw new IllegalArgumentException("Method argument (authentication token) cannot be null.");
     5     }
     6 
     7     log.trace("Authentication attempt received for token [{}]", token);
     8 
     9     AuthenticationInfo info;
    10     try {
    11         info = doAuthenticate(token);
    12         if (info == null) {
    13             String msg = "No account information found for authentication token [" + token + "] by this " +
    14                     "Authenticator instance.  Please check that it is configured correctly.";
    15             throw new AuthenticationException(msg);
    16         }
    17     } catch (Throwable t) {
    18         AuthenticationException ae = null;
    19         if (t instanceof AuthenticationException) {
    20             ae = (AuthenticationException) t;
    21         }
    22         if (ae == null) {
    23             //Exception thrown was not an expected AuthenticationException.  Therefore it is probably a little more
    24             //severe or unexpected.  So, wrap in an AuthenticationException, log to warn, and propagate:
    25             String msg = "Authentication failed for token submission [" + token + "].  Possible unexpected " +
    26                     "error? (Typical or expected login exceptions should extend from AuthenticationException).";
    27             ae = new AuthenticationException(msg, t);
    28             if (log.isWarnEnabled())
    29                 log.warn(msg, t);
    30         }
    31         try {
    32             notifyFailure(token, ae);
    33         } catch (Throwable t2) {
    34             if (log.isWarnEnabled()) {
    35                 String msg = "Unable to send notification for failed authentication attempt - listener error?.  " +
    36                         "Please check your AuthenticationListener implementation(s).  Logging sending exception " +
    37                         "and propagating original AuthenticationException instead...";
    38                 log.warn(msg, t2);
    39             }
    40         }
    41         throw ae;
    42     }
    43 
    44     log.debug("Authentication successful for token [{}].  Returned account [{}]", token, info);
    45 
    46     notifySuccess(token, info);
    47 
    48     return info;
    49 }

    上面代码第11行:info = doAuthenticate(token); 这个方法为ModularRealmAuthticator类中的方法,因为ModularRealmAuthticator继承了AbstractAuthenticator抽象类。另外,要注意第12行-第16行,如果info==null,就会抛出异常。ModularRealmAuthticator的doAuthenticate(token);方法如下:

    1 protected AuthenticationInfo doAuthenticate(AuthenticationToken authenticationToken) throws AuthenticationException {
    2     assertRealmsConfigured();
    3     Collection<Realm> realms = getRealms();
    4     if (realms.size() == 1) {
    5         return doSingleRealmAuthentication(realms.iterator().next(), authenticationToken);
    6     } else {
    7         return doMultiRealmAuthentication(realms, authenticationToken);
    8     }
    9 }

    这里,我们关注上面第五行代码:doSingleRealmAuthentication(realms.iterator().next(), authenticationToken); else语句中的doMultiRealmAuthentication(realms, authenticationToken);类似。跟踪到doSingleRealmAuthentication方法如下:

     1 protected AuthenticationInfo doSingleRealmAuthentication(Realm realm, AuthenticationToken token) {
     2     if (!realm.supports(token)) {
     3         String msg = "Realm [" + realm + "] does not support authentication token [" +
     4                 token + "].  Please ensure that the appropriate Realm implementation is " +
     5                 "configured correctly or that the realm accepts AuthenticationTokens of this type.";
     6         throw new UnsupportedTokenException(msg);
     7     }
     8     AuthenticationInfo info = realm.getAuthenticationInfo(token);
     9     if (info == null) {
    10         String msg = "Realm [" + realm + "] was unable to find account data for the " +
    11                 "submitted AuthenticationToken [" + token + "].";
    12         throw new UnknownAccountException(msg);
    13     }
    14     return info;
    15 }

    上面代码第八行:AuthenticationInfo info = realm.getAuthenticationInfo(token); realm为Realm接口,实际上调用的是其实现类AuthenticatingRealm中的getAuthenticationInfo方法,方法如下:

     1 public final AuthenticationInfo getAuthenticationInfo(AuthenticationToken token) throws AuthenticationException {
     2 
     3     AuthenticationInfo info = getCachedAuthenticationInfo(token);
     4     if (info == null) {
     5         //otherwise not cached, perform the lookup:
     6         info = doGetAuthenticationInfo(token);
     7         log.debug("Looked up AuthenticationInfo [{}] from doGetAuthenticationInfo", info);
     8         if (token != null && info != null) {
     9             cacheAuthenticationInfoIfPossible(token, info);
    10         }
    11     } else {
    12         log.debug("Using cached authentication info [{}] to perform credentials matching.", info);
    13     }
    14 
    15     if (info != null) {
    16         assertCredentialsMatch(token, info);
    17     } else {
    18         log.debug("No AuthenticationInfo found for submitted AuthenticationToken [{}].  Returning null.", token);
    19     }
    20     return info;
    21 }

    上面代码第三行:AuthenticationInfo info = getCachedAuthenticationInfo(token);从缓存中获取认证信息,如果未获取到,则调用第六行的doGetAuthenticationInfo(token); 方法获取认证信息。继续跟踪,发现有几个类实现了该方法,如下图所示:

    最后,附上SecurityManager和Realm等的类关系图:

    Realm:

    SecurityManager:

    Authenticator:

  • 相关阅读:
    从zk监控canal-client消费延迟情况
    python面向对象——类的参数
    python面向对象——类的继承
    python并发——进程间同步和通信(二)
    python并发——线程池与进程池(转)
    python从指定目录排除部分子目录——用于删除目录
    python并发统计s3目录大小
    Java对象的序列化和反序列化
    多态、抽象类和接口
    Java输入输出流
  • 原文地址:https://www.cnblogs.com/ccfdod/p/6436353.html
Copyright © 2020-2023  润新知