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

org.owasp.csrfguard.config.overlay.ExpirableCache Maven / Gradle / Ivy

/**
 * The OWASP CSRFGuard Project, BSD License
 * Eric Sheridan ([email protected]), Copyright (c) 2011 
 * All rights reserved.
 * 
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions are met:
 *
 *    1. Redistributions of source code must retain the above copyright notice,
 *       this list of conditions and the following disclaimer.
 *    2. Redistributions in binary form must reproduce the above copyright
 *       notice, this list of conditions and the following disclaimer in the
 *       documentation and/or other materials provided with the distribution.
 *    3. Neither the name of OWASP nor the names of its contributors may be used
 *       to endorse or promote products derived from this software without specific
 *       prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
 * ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
 * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 */
package org.owasp.csrfguard.config.overlay;

import java.io.Serializable;
import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;
import java.util.Set;


/**
 * This is like a map, but the setters also take an expire time which 
 * will mean the items in the cache will be automatically deleted.  
 * Also, every so often (e.g. 2 minutes) all items are checked for expiration.
 * If no timetolive is set, it will default to 1 day.  The max time to live is
 * one day.
 * This is synchronized so that all access is safe.
 * 

* Note, evictions are check for periodically, but only when the map is accessed (and even then only every few minutes). * so you can check for evictions externally, or clear the map if you are done with it. * @version $Id: ExpirableCache.java,v 1.1 2008-11-27 14:25:50 mchyzer Exp $ * @author mchyzer * @param key type * @param value type */ @SuppressWarnings("serial") public class ExpirableCache implements Serializable { /** max time to live in millis */ static long MAX_TIME_TO_LIVE_MILLIS = 1000 * 60 * 60 * 24; //1 day /** time to live for content (when not specified this is one day, and max one day) */ long defaultTimeToLiveInMillis = MAX_TIME_TO_LIVE_MILLIS; /** time between looking for evictions in millis, default to two minutes */ static long TIME_BETWEEN_EVICTIONS_MILLIS = 2 * 60 * 1000; /** last time the cache was checked for evictions */ long lastEvictionCheck = System.currentTimeMillis(); /** cache map */ private Map> cache = new HashMap>(); /** number of elements inserted into the cache */ private int cacheInserts = 0; /** numebr of times an element was retrieved from cache successfully */ private int cacheHits = 0; /** number of evictions from cache when thigns expire */ private int cacheEvictions = 0; /** global number of elements inserted into the cache, no need to synchronize */ private static int globalCacheInserts = 0; /** numebr of times an element was retrieved from cache successfully, no need to synchronize */ private static int globalCacheHits = 0; /** number of evictions from cache when thigns expire, no need to synchronize */ private static int globalCacheEvictions = 0; /** when was the last clear of all */ private static long lastClearStatic = -1; /** when was the last clear of this instance */ private long lastClear = System.currentTimeMillis(); /** * */ public ExpirableCache() { super(); } /** * delete the cache * */ public synchronized void clear() { this.cache.clear(); } /** * @param defaultTimeToLiveInMinutes time in minutes is the default cache time to live for content */ public ExpirableCache(int defaultTimeToLiveInMinutes) { super(); if (defaultTimeToLiveInMinutes <= 0) { throw new RuntimeException("Time to live in minutes must be greater than 0"); } //make sure this is less than the max long newTimeToLiveMillis = (long)defaultTimeToLiveInMinutes * 60 * 1000; if (newTimeToLiveMillis < MAX_TIME_TO_LIVE_MILLIS) { this.defaultTimeToLiveInMillis = newTimeToLiveMillis; } } /** * unit of time for expirable cache * @author mchyzer * */ public static enum ExpirableCacheUnit { /** minutes */ MINUTE { /** * @see ExpirableCacheUnit#defaultTimeToLiveMillis(int) */ @Override public long defaultTimeToLiveMillis(int input) { return (long)input * 60 * 1000; } }, /** seconds */ SECOND { /** * @see ExpirableCacheUnit#defaultTimeToLiveMillis(int) */ @Override public long defaultTimeToLiveMillis(int input) { return (long)input * 1000; } }; /** * default time to live based on units * @param input * @return the millis */ public abstract long defaultTimeToLiveMillis(int input); } /** * @param defaultTimeToLive time in whatever unit is the default cache time to live for content * @param expirableCacheUnit is minutes or seconds */ public ExpirableCache(ExpirableCacheUnit expirableCacheUnit, int defaultTimeToLive) { super(); if (defaultTimeToLive <= 0) { throw new RuntimeException("Time to live in minutes must be greater than 0"); } //make sure this is less than the max long newTimeToLiveMillis = expirableCacheUnit.defaultTimeToLiveMillis(defaultTimeToLive); if (newTimeToLiveMillis < MAX_TIME_TO_LIVE_MILLIS) { this.defaultTimeToLiveInMillis = newTimeToLiveMillis; } } /** * expose the length of cache * @return length of cache */ public long getDefaultTimeToLiveInMillis() { return this.defaultTimeToLiveInMillis; } /** * put a value into the cache, accept the default time to live for this cache * @param key * @param value */ public synchronized void put(K key, V value) { this.putHelper(key, value, this.defaultTimeToLiveInMillis); } /** * put a value into the cache, accept the default time to live for this cache * @param key * @param value * @param timeToLiveInMinutes time to live for this item in minutes. * If -1 then use the default */ public synchronized void put(K key, V value, int timeToLiveInMinutes) { //see if the default if (timeToLiveInMinutes == -1) { this.put(key,value); return; } if (timeToLiveInMinutes <= 0) { throw new RuntimeException("Time to live in minutes must be greater than 0"); } this.putHelper(key, value, (long)timeToLiveInMinutes * 60 * 1000); } /** * put a value into the cache, accept the default time to live for this cache * @param key * @param value * @param proposedTimeToLiveInMillis millis time to live */ synchronized void putHelper(K key, V value, long proposedTimeToLiveInMillis) { this.checkForEvictions(true); long newTimeToLiveInMillis = this.defaultTimeToLiveInMillis; //dont use what was inputted if it is out of range if (proposedTimeToLiveInMillis > 0 && proposedTimeToLiveInMillis <= ExpirableCache.MAX_TIME_TO_LIVE_MILLIS) { newTimeToLiveInMillis = proposedTimeToLiveInMillis; } ExpirableValue expirableValue = new ExpirableValue(value, newTimeToLiveInMillis); this.cache.put(key, expirableValue); this.cacheInserts++; globalCacheInserts++; } /** * clear out all caches everywhere (session, request, context, etc) */ public static void clearAll() { lastClearStatic = System.currentTimeMillis(); } /** * check and remove elements that are stale * @param onlyCheckIfNeeded true if only check every so often (e.g. every two minutes) */ public synchronized void checkForEvictions(boolean onlyCheckIfNeeded) { long now = System.currentTimeMillis(); //first see if there is an all clear if (lastClearStatic > this.lastClear) { this.clear(); this.lastClear = now; return; } if (onlyCheckIfNeeded) { if (now - this.lastEvictionCheck < ExpirableCache.TIME_BETWEEN_EVICTIONS_MILLIS) { return; } } //go through all elements, evict if stale Set keySet = this.cache.keySet(); Iterator keyIterator = keySet.iterator(); while (keyIterator.hasNext()) { K key = keyIterator.next(); ExpirableValue expirableValue = this.cache.get(key); if (expirableValue.expired()) { keyIterator.remove(); this.cacheEvictions++; ExpirableCache.globalCacheEvictions++; } } //set that we just checked this.lastEvictionCheck = now; } /** * get a value or null if not there or expired * this will check for eviction, and evict if evictable * @param key * @return the value or null if not there or evicted */ public synchronized V get(K key) { this.checkForEvictions(true); return this.getHelper(key); } /** * get a value or null if not there or expired * this will check for eviction, and evict if evictable * @param key * @return the value or null if not there or evicted */ private synchronized V getHelper(K key) { ExpirableValue value = this.cache.get(key); if (value == null) { //shouldnt have a key with no value, probably doesnt exist, but just in case this.cache.remove(key); return null; } if (value.expired()) { this.cacheEvictions++; ExpirableCache.globalCacheEvictions++; this.cache.remove(key); return null; } V content = value.getContent(); this.cacheHits++; ExpirableCache.globalCacheHits++; return content; } /** * number of elements in map (and check for * @param evictEvictables true if we should evict values that are stale * (even if recently checked) * @return the number of elements */ public synchronized int size(boolean evictEvictables) { if (evictEvictables) { this.checkForEvictions(false); } return this.cache.size(); } /** * number of items inserted into the cache * @return Returns the cacheInserts. */ public int getCacheInserts() { return this.cacheInserts; } /** * number of items evicted from cache * @return Returns the cacheEvictions. */ public int getCacheEvictions() { return this.cacheEvictions; } /** * number of items successfully retrieved from cache * @return Returns the cacheHits. */ public int getCacheHits() { return this.cacheHits; } /** * string representation of cache * @see java.lang.Object#toString() */ @Override public String toString() { this.checkForEvictions(true); return this.getClass().getSimpleName() + ": size: " + this.size(false) + ", cacheHits: " + this.getCacheHits() + ", cacheInserts: " + this.getCacheInserts() + ", cacheEvictions: " + this.cacheEvictions; } /** * string representation of cache * @return the string value */ public static String staticToString() { return "ExpirableCacheGlobal, cacheHits: " + globalCacheHits + ", cacheInserts: " + globalCacheInserts + ", cacheEvictions: " + globalCacheEvictions; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy