org.apache.myfaces.trinidadinternal.util.TokenCache Maven / Gradle / Ivy
/*
* 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.myfaces.trinidadinternal.util;
import java.io.Serializable;
import java.math.BigInteger;
import java.security.NoSuchAlgorithmException;
import java.security.SecureRandom;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.atomic.AtomicLong;
import javax.faces.context.ExternalContext;
import javax.faces.context.FacesContext;
import org.apache.myfaces.trinidad.logging.TrinidadLogger;
/**
* A simple LRU tokenized cache. The cache is responsible for storing tokens,
* but the storage of the values referred to by those tokens is not handled
* by this class. Instead, the user of this class has to provide a Map
* instance to each call.
*
* The design seems odd, but is intentional - this way, a session Map can be used
* directly as the storage target for values, while the TokenCache simply maintains
* the logic of which tokens should still be available. Storing values
* directly in the cache object (instead of directly on the session) causes
* HttpSession failover difficulties.
*
* TokenCache also supports the concept of "pinning", whereby one token
* can be pinned to another. The pinned token will not be removed from
* the cache until all tokens that it is pinned to are also out of the
* cache.
*/
public class TokenCache implements Serializable
{
/**
* Character guaranteed to not be used in tokens
*/
static public final char SEPARATOR_CHAR = '.';
/**
* Gets a TokenCache from the session, creating it if needed.
*/
@SuppressWarnings("unchecked")
static public TokenCache getTokenCacheFromSession(
ExternalContext extContext,
String cacheName,
boolean createIfNeeded,
int defaultSize)
{
Map sessionMap = extContext.getSessionMap();
TokenCache cache = (TokenCache)sessionMap.get(cacheName);
if (cache == null)
{
if (createIfNeeded)
{
Object session = extContext.getSession(true);
// Synchronize on the session object to ensure that
// we don't ever create two different caches
synchronized (session)
{
cache = (TokenCache)sessionMap.get(cacheName);
if (cache == null)
{
// create the TokenCache with the crytographically random seed
cache = new TokenCache(defaultSize, _getSeed(), cacheName);
sessionMap.put(cacheName, cache);
}
}
}
}
return cache;
}
/**
* Returns a cryptographically secure random number to use as the TokenCache seed
*/
private static long _getSeed()
{
SecureRandom rng;
try
{
// try SHA1 first
rng = SecureRandom.getInstance("SHA1PRNG");
}
catch (NoSuchAlgorithmException e)
{
// SHA1 not present, so try the default (which could potentially not be
// cryptographically secure)
rng = new SecureRandom();
}
// use 48 bits for strength and fill them in
byte[] randomBytes = new byte[6];
rng.nextBytes(randomBytes);
// convert to a long
return new BigInteger(randomBytes).longValue();
}
/**
* For serialization only
*/
TokenCache()
{
this(_DEFAULT_SIZE, 0L, null);
}
/**
* Create a TokenCache that will store the last "size" entries. This version should
* not be used if the token cache is externally accessible (since the seed always
* starts at 0). Use the
* constructor with the long seed instead.
*/
public TokenCache(int size)
{
this(size, 0L, null);
}
/**
* Create a TokenCache that will store the last "size" entries,
* and begins its tokens based on the seed (instead of always
* starting at "0").
* @patam owner Optional Cache that stores the token cache
* @param keyInOwner Optional Name under which this cache is stored in the owner
*/
private TokenCache(int size, long seed, String keyInOwner)
{
_cache = new LRU(size);
_pinned = new ConcurrentHashMap(size);
_count = new AtomicLong(seed);
_keyInOwner = keyInOwner;
}
/**
* Create a new token; and use that token to store a value into
* a target Map. The least recently used values from the
* cache may be removed.
* @param value the value being added to the target store
* @param targetStore the map used for storing the value
* @return the token used to store the value
*/
public String addNewEntry(
V value,
Map targetStore)
{
return addNewEntry(value, targetStore, null);
}
/**
* Create a new token; and use that token to store a value into
* a target Map. The least recently used values from the
* cache may be removed.
* @param value the value being added to the target store
* @param targetStore the map used for storing the value
* @param pinnedToken a token, that if still in the cache,
* will not be freed until this current token is also freed
* @return the token used to store the value
*/
public String addNewEntry(
V value,
Map targetStore,
String pinnedToken)
{
String remove = null;
String token = null;
synchronized (this)
{
token = _getNextToken();
// If there is a request to pin one token to another,
// store that: the pinnedToken is the value
if (pinnedToken != null)
_pinned.put(token, pinnedToken);
assert(_removed == null);
// NOTE: this put() has a side-effect that can result
// in _removed being non-null afterwards
_cache.put(token, token);
if(TokenCacheDebugUtils.debugTokenCache())
{
TokenCacheDebugUtils.startLog("Add New Entry");
TokenCacheDebugUtils.addTokenToViewIdMap(token);
if (pinnedToken != null)
{
TokenCacheDebugUtils.addToLog("\nPINNING " +
TokenCacheDebugUtils.getTokenToViewIdString(token) +
" to " +
TokenCacheDebugUtils.getTokenToViewIdString(pinnedToken));
}
}
remove = _removed;
_removed = null;
}
// This looks like "remove" must be null - given the
// assert above.
if (remove != null)
{
_removeTokenIfReady(targetStore, remove);
}
targetStore.put(token, value);
// our contents have changed, so mark ourselves as dirty in our owner
_dirty();
if(TokenCacheDebugUtils.debugTokenCache())
{
TokenCacheDebugUtils.logCacheInfo(targetStore, _pinned, "After Additions");
_LOG.severe(TokenCacheDebugUtils.getLogString());
}
return token;
}
/**
* Returns true if an entry is still available. This
* method has a side-effect: by virtue of accessing the token,
* it is now at the top of the most-recently-used list.
*/
public boolean isAvailable(String token)
{
synchronized (this)
{
// If the token is in the LRU cache, then it's available
if (_cache.get(token) != null)
return true;
// And if the token is a value in "pinned", then it's also available
if (_pinned.containsValue(token))
return true;
return false;
}
}
/**
* Remove a token if is ready: there are no pinned references to it.
* Note that it will be absent from the LRUCache.
*/
synchronized private V _removeTokenIfReady(
Map targetStore,
String token)
{
V removedValue;
// See if it's pinned to something still in memory
if (!_pinned.containsValue(token))
{
_LOG.finest("Removing token ''{0}''", token);
// Remove it from the target store
if (TokenCacheDebugUtils.debugTokenCache())
{
TokenCacheDebugUtils.removeTokenFromViewIdMap(token);
}
removedValue = targetStore.remove(token);
// Now, see if that key was pinning anything else
String wasPinned = _pinned.remove(token);
if (wasPinned != null)
{
if (TokenCacheDebugUtils.debugTokenCache())
{
TokenCacheDebugUtils.addToLog("\nREMOVING pinning of token " + token + " to " +
TokenCacheDebugUtils.getTokenToViewIdString(wasPinned));
}
// Yup, so see if we can remove that token
_removeTokenIfReady(targetStore, wasPinned);
}
}
else
{
if (TokenCacheDebugUtils.debugTokenCache())
{
TokenCacheDebugUtils.addToLog("\nNOT removing pinned token from target store " +
TokenCacheDebugUtils.getTokenToViewIdString(token) );
}
_LOG.finest("Not removing pinned token ''{0}''", token);
// TODO: is this correct? We're not really removing
// the target value.
removedValue = targetStore.get(token);
}
return removedValue;
}
/**
* Removes a value from the cache.
* @return previous value associated with the token, if any
*/
public V removeOldEntry(
String token,
Map targetStore)
{
V oldValue;
synchronized (this)
{
if(TokenCacheDebugUtils.debugTokenCache())
{
TokenCacheDebugUtils.startLog("Remove Old Entry");
}
_LOG.finest("Removing token {0} from cache", token);
_cache.remove(token);
// TODO: should removing a value that is "pinned" take?
// Or should it stay in memory?
oldValue = _removeTokenIfReady(targetStore, token);
if (TokenCacheDebugUtils.debugTokenCache())
{
TokenCacheDebugUtils.logCacheInfo(targetStore, _pinned, "After removing old entry:");
_LOG.severe(TokenCacheDebugUtils.getLogString());
}
}
// our contents have changed, so mark ourselves as dirty in our owner
_dirty();
return oldValue;
}
/**
* Clear a cache, without resetting the token.
*/
public void clear(Map targetStore)
{
synchronized (this)
{
for(String keyToRemove : _cache.keySet())
{
_LOG.finest("Clearing token {0} from cache", keyToRemove);
targetStore.remove(keyToRemove);
}
_cache.clear();
}
// our contents have changed, so mark ourselves as dirty in our owner
_dirty();
}
private String _getNextToken()
{
// atomically increment the value
long nextToken = _count.incrementAndGet();
// convert using base 36 because it is a fast efficient subset of base-64
return Long.toString(nextToken, 36);
}
/**
* Mark the cache as dirty in the owner
*/
private void _dirty()
{
if (_keyInOwner != null)
{
_getOwner().put(_keyInOwner, this);
}
}
/**
* @return returns the current requests session map via the faces context
*/
private Map _getOwner()
{
// Getting a refrence to the session map must be dynamically established versus holding a
// transient reference. This is because the SessionMap from the faces ExternalContext
// needs a reference back to the request. The reason that the SessionMap has the backpointer
// to the request is that JSF doesn't want to create a session object unless they need to,
// (lazily created) so they call request.getSession(false) to get the session.
// They need the backpointer to handle the case where the session gets created after the
// SessionMap instance is created.
//
// Hanging on to a reference to the session map ends up pinning the request in memory
// which would otherwise be gc-ed.
return FacesContext.getCurrentInstance().getExternalContext().getSessionMap();
}
private class LRU extends LRUCache
{
public LRU(int maxSize)
{
super(maxSize);
}
@Override
protected void removing(String key)
{
_removed = key;
}
private static final long serialVersionUID = 1L;
}
private final Map _cache;
// Map from String to String, where the keys represent tokens that are
// stored, and the values are the tokens that are pinned. This is
// an N->1 ratio: the values may appear multiple times.
private final Map _pinned;
// the current token value
private final AtomicLong _count;
private final String _keyInOwner;
// Hack instance parameter used to communicate between the LRU cache's
// removing() method, and the addNewEntry() method that may trigger it
private transient String _removed;
static private final int _DEFAULT_SIZE = 15;
static private final long serialVersionUID = 1L;
static private final TrinidadLogger _LOG =
TrinidadLogger.createTrinidadLogger(TokenCache.class);
}