All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.apache.shiro.realm.CachingRealm Maven / Gradle / Ivy

There is a newer version: 3.9
Show newest version
/*
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * to you under the Apache License, Version 2.0 (the
 * "License"); you may not use this file except in compliance
 * with the License.  You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing,
 * software distributed under the License is distributed on an
 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
 * KIND, either express or implied.  See the License for the
 * specific language governing permissions and limitations
 * under the License.
 */
package org.apache.shiro.realm;

import org.apache.shiro.authc.LogoutAware;
import org.apache.shiro.cache.CacheManager;
import org.apache.shiro.cache.CacheManagerAware;
import org.apache.shiro.subject.PrincipalCollection;
import org.apache.shiro.util.CollectionUtils;
import org.apache.shiro.util.Nameable;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.util.Collection;
import java.util.concurrent.atomic.AtomicInteger;


/**
 * A very basic abstract extension point for the {@link Realm} interface that provides caching support for subclasses.
 * 

* It also provides a convenience method, * {@link #getAvailablePrincipal(org.apache.shiro.subject.PrincipalCollection)}, which is useful across all * realm subclasses for obtaining a realm-specific principal/identity. *

* All actual Realm method implementations are left to subclasses. * * @see #clearCache(org.apache.shiro.subject.PrincipalCollection) * @see #onLogout(org.apache.shiro.subject.PrincipalCollection) * @see #getAvailablePrincipal(org.apache.shiro.subject.PrincipalCollection) * @since 0.9 */ public abstract class CachingRealm implements Realm, Nameable, CacheManagerAware, LogoutAware { private static final Logger log = LoggerFactory.getLogger(CachingRealm.class); //TODO - complete JavaDoc private static final AtomicInteger INSTANCE_COUNT = new AtomicInteger(); /*-------------------------------------------- | I N S T A N C E V A R I A B L E S | ============================================*/ private String name; private boolean cachingEnabled; private CacheManager cacheManager; /** * Default no-argument constructor that defaults * {@link #isCachingEnabled() cachingEnabled} (for general caching) to {@code true} and sets a * default {@link #getName() name} based on the class name. *

* Note that while in general, caching may be enabled by default, subclasses have control over * if specific caching is enabled. */ public CachingRealm() { this.cachingEnabled = true; this.name = getClass().getName() + "_" + INSTANCE_COUNT.getAndIncrement(); } /** * Returns the CacheManager used for data caching to reduce EIS round trips, or null if * caching is disabled. * * @return the CacheManager used for data caching to reduce EIS round trips, or null if * caching is disabled. */ public CacheManager getCacheManager() { return this.cacheManager; } /** * Sets the CacheManager to be used for data caching to reduce EIS round trips. *

* This property is null by default, indicating that caching is turned off. * * @param cacheManager the CacheManager to use for data caching, or null to disable caching. */ public void setCacheManager(CacheManager cacheManager) { this.cacheManager = cacheManager; afterCacheManagerSet(); } /** * Returns {@code true} if caching should be used if a {@link CacheManager} has been * {@link #setCacheManager(org.apache.shiro.cache.CacheManager) configured}, {@code false} otherwise. *

* The default value is {@code true} since the large majority of Realms will benefit from caching if a CacheManager * has been configured. However, memory-only realms should set this value to {@code false} since they would * manage account data in memory already lookups would already be as efficient as possible. * * @return {@code true} if caching will be globally enabled if a {@link CacheManager} has been * configured, {@code false} otherwise */ public boolean isCachingEnabled() { return cachingEnabled; } /** * Sets whether or not caching should be used if a {@link CacheManager} has been * {@link #setCacheManager(org.apache.shiro.cache.CacheManager) configured}. * * @param cachingEnabled whether or not to globally enable caching for this realm. */ public void setCachingEnabled(boolean cachingEnabled) { this.cachingEnabled = cachingEnabled; } public String getName() { return name; } public void setName(String name) { this.name = name; } /** * Template method that may be implemented by subclasses should they wish to react to a * {@link CacheManager} instance being set on the realm instance via the * {@link #setCacheManager(org.apache.shiro.cache.CacheManager)} mutator. */ protected void afterCacheManagerSet() { } /** * If caching is enabled, this will clear any cached data associated with the specified account identity. * Subclasses are free to override for additional behavior, but be sure to call {@code super.onLogout} first. *

* This default implementation merely calls {@link #clearCache(org.apache.shiro.subject.PrincipalCollection)}. * * @param principals the application-specific Subject/user identifier that is logging out. * @see #clearCache(org.apache.shiro.subject.PrincipalCollection) * @see #getAvailablePrincipal(org.apache.shiro.subject.PrincipalCollection) * @since 1.2 */ public void onLogout(PrincipalCollection principals) { clearCache(principals); } /** * Clears out any cached data associated with the specified account identity/identities. *

* This implementation will return quietly if the principals argument is null or empty. Otherwise it delegates * to {@link #doClearCache(org.apache.shiro.subject.PrincipalCollection)}. * * @param principals the principals of the account for which to clear any cached data. * @since 1.2 */ protected void clearCache(PrincipalCollection principals) { if (!CollectionUtils.isEmpty(principals)) { doClearCache(principals); log.trace("Cleared cache entries for account with principals [{}]", principals); } } /** * This implementation does nothing - it is a template to be overridden by subclasses if necessary. * * @param principals principals the principals of the account for which to clear any cached data. * @since 1.2 */ protected void doClearCache(PrincipalCollection principals) { } /** * A utility method for subclasses that returns the first available principal of interest to this particular realm. * The heuristic used to acquire the principal is as follows: *

    *
  • Attempt to get this particular Realm's 'primary' principal in the {@code PrincipalCollection} via a * principals.{@link PrincipalCollection#fromRealm(String) fromRealm}({@link #getName() getName()}) * call.
  • *
  • If the previous call does not result in any principals, attempt to get the overall 'primary' principal * from the PrincipalCollection via {@link org.apache.shiro.subject.PrincipalCollection#getPrimaryPrincipal()}.
  • *
  • If there are no principals from that call (or the PrincipalCollection argument was null to begin with), * return {@code null}
  • *
* * @param principals the PrincipalCollection holding all principals (from all realms) associated with a single Subject. * @return the 'primary' principal attributed to this particular realm, or the fallback 'master' principal if it * exists, or if not {@code null}. * @since 1.2 */ protected Object getAvailablePrincipal(PrincipalCollection principals) { Object primary = null; if (!CollectionUtils.isEmpty(principals)) { Collection thisPrincipals = principals.fromRealm(getName()); if (!CollectionUtils.isEmpty(thisPrincipals)) { primary = thisPrincipals.iterator().next(); } else { //no principals attributed to this particular realm. Fall back to the 'master' primary: primary = principals.getPrimaryPrincipal(); } } return primary; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy