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

edu.internet2.middleware.grouperInstaller.util.ExpirableCache Maven / Gradle / Ivy

/*******************************************************************************
 * Copyright 2012 Internet2
 * 
 * Licensed 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 edu.internet2.middleware.grouperInstaller.util;

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 */ 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(); GrouperInstallerUtils.assertion(defaultTimeToLiveInMinutes > 0, "Time to live in minutes must be greater than 0"); //make sure this is less than the max long newTimeToLiveMillis = defaultTimeToLiveInMinutes * 60 * 1000; if (newTimeToLiveMillis < MAX_TIME_TO_LIVE_MILLIS) { this.defaultTimeToLiveInMillis = newTimeToLiveMillis; } } /** * 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; } GrouperInstallerUtils.assertion(timeToLiveInMinutes > 0, "Time to live in minutes must be greater than 0"); this.putHelper(key, value, 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