Introduce helma.objectmodel.ObjectCache interface for switchable cache implementations.
This commit is contained in:
		
							parent
							
								
									f2180bfa81
								
							
						
					
					
						commit
						59bec76b45
					
				
					 6 changed files with 286 additions and 146 deletions
				
			
		|  | @ -274,7 +274,9 @@ public final class Application implements IPathElement, Runnable { | ||||||
|      * Get the application ready to run, initializing the evaluators and type manager. |      * Get the application ready to run, initializing the evaluators and type manager. | ||||||
|      */ |      */ | ||||||
|     public synchronized void init() |     public synchronized void init() | ||||||
|               throws DatabaseException, ScriptingException, MalformedURLException { |             throws DatabaseException, MalformedURLException, | ||||||
|  |                    IllegalAccessException, InstantiationException, | ||||||
|  |                    ClassNotFoundException { | ||||||
| 
 | 
 | ||||||
|         // create and init type mananger |         // create and init type mananger | ||||||
|         typemgr = new TypeManager(this); |         typemgr = new TypeManager(this); | ||||||
|  |  | ||||||
							
								
								
									
										108
									
								
								src/helma/objectmodel/ObjectCache.java
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										108
									
								
								src/helma/objectmodel/ObjectCache.java
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,108 @@ | ||||||
|  | /* | ||||||
|  |  * Helma License Notice | ||||||
|  |  * | ||||||
|  |  * The contents of this file are subject to the Helma License | ||||||
|  |  * Version 2.0 (the "License"). You may not use this file except in | ||||||
|  |  * compliance with the License. A copy of the License is available at | ||||||
|  |   * http://adele.helma.org/download/helma/license.txt | ||||||
|  |  * | ||||||
|  |  * Copyright 1998-2003 Helma Software. All Rights Reserved. | ||||||
|  |  * | ||||||
|  |  * $RCSfile$ | ||||||
|  |  * $Author$ | ||||||
|  |  * $Revision$ | ||||||
|  |  * $Date$ | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | package helma.objectmodel; | ||||||
|  | 
 | ||||||
|  | import helma.framework.core.Application; | ||||||
|  | 
 | ||||||
|  | import java.util.Properties; | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Interface Helma object cache classes need to implement. | ||||||
|  |  * | ||||||
|  |  */ | ||||||
|  | public interface ObjectCache { | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Set the {@link helma.framework.core.Application Application} instance | ||||||
|  |      * for the cache. | ||||||
|  |      * @param app the app instance | ||||||
|  |      */ | ||||||
|  |     void init(Application app); | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Called when the application's properties have been updated to let | ||||||
|  |      * the cache implementation update its settings. | ||||||
|  |      * @param props | ||||||
|  |      */ | ||||||
|  |     void updateProperties(Properties props); | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Returns true if the collection contains an element for the key. | ||||||
|  |      * | ||||||
|  |      * @param key the key that we are looking for | ||||||
|  |      */ | ||||||
|  |     boolean containsKey(Object key); | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Returns the number of keys in object array <code>keys</code> that | ||||||
|  |      * were not found in the Map. | ||||||
|  |      * Those keys that are contained in the Map are nulled out in the array. | ||||||
|  |      * @param keys an array of key objects we are looking for | ||||||
|  |      * @see ObjectCache#containsKey | ||||||
|  |      */ | ||||||
|  |     int containsKeys(Object[] keys); | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Gets the object associated with the specified key in the | ||||||
|  |      * hashtable. | ||||||
|  |      * @param key the specified key | ||||||
|  |      * @return the element for the key or null if the key | ||||||
|  |      * 		is not defined in the hash table. | ||||||
|  |      * @see ObjectCache#put | ||||||
|  |      */ | ||||||
|  |     Object get(Object key); | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Puts the specified element into the hashtable, using the specified | ||||||
|  |      * key.  The element may be retrieved by doing a get() with the same key. | ||||||
|  |      * The key and the element cannot be null. | ||||||
|  |      * @param key the specified key in the hashtable | ||||||
|  |      * @param value the specified element | ||||||
|  |      * @exception NullPointerException If the value of the element | ||||||
|  |      * is equal to null. | ||||||
|  |      * @see ObjectCache#get | ||||||
|  |      * @return the old value of the key, or null if it did not have one. | ||||||
|  |      */ | ||||||
|  |     Object put(Object key, Object value); | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Removes the element corresponding to the key. Does nothing if the | ||||||
|  |      * key is not present. | ||||||
|  |      * @param key the key that needs to be removed | ||||||
|  |      * @return the value of key, or null if the key was not found. | ||||||
|  |      */ | ||||||
|  |     Object remove(Object key); | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Removes all items currently stored in the cache. | ||||||
|  |      * | ||||||
|  |      * @return true if the operation succeeded | ||||||
|  |      */ | ||||||
|  |     boolean clear(); | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Return the number of objects currently stored in the cache. | ||||||
|  |      * @return the number of cached items | ||||||
|  |      */ | ||||||
|  |     int size(); | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Return an array with all objects currently contained in the cache. | ||||||
|  |      */ | ||||||
|  |     Object[] getCachedObjects(); | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | @ -542,6 +542,15 @@ public final class Node implements INode, Serializable { | ||||||
|         return dbmap; |         return dbmap; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     /** | ||||||
|  |      * | ||||||
|  |      *  | ||||||
|  |      * @param nmgr | ||||||
|  |      */ | ||||||
|  |     public void setWrappedNodeManager(WrappedNodeManager nmgr) { | ||||||
|  |         this.nmgr = nmgr; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     /** |     /** | ||||||
|      * |      * | ||||||
|      * |      * | ||||||
|  |  | ||||||
|  | @ -35,7 +35,7 @@ import java.util.*; | ||||||
| public final class NodeManager { | public final class NodeManager { | ||||||
| 
 | 
 | ||||||
|     protected Application app; |     protected Application app; | ||||||
|     private CacheMap cache; |     private ObjectCache cache; | ||||||
|     private Replicator replicator; |     private Replicator replicator; | ||||||
|     protected IDatabase db; |     protected IDatabase db; | ||||||
|     protected IDGenerator idgen; |     protected IDGenerator idgen; | ||||||
|  | @ -51,22 +51,17 @@ public final class NodeManager { | ||||||
|      * created in dbHome if one doesn't already exist. |      * created in dbHome if one doesn't already exist. | ||||||
|      */ |      */ | ||||||
|     public NodeManager(Application app, String dbHome, Properties props) |     public NodeManager(Application app, String dbHome, Properties props) | ||||||
|                 throws DatabaseException { |             throws DatabaseException, ClassNotFoundException, | ||||||
|  |                    IllegalAccessException, InstantiationException { | ||||||
|         this.app = app; |         this.app = app; | ||||||
| 
 | 
 | ||||||
|         int cacheSize = Integer.parseInt(props.getProperty("cachesize", "1000")); |         String cacheImpl = props.getProperty("cacheimpl", "helma.util.CacheMap"); | ||||||
| 
 | 
 | ||||||
|         // Make actual cache size bigger, since we use it only up to the threshold |         cache = (ObjectCache) Class.forName(cacheImpl).newInstance(); | ||||||
|         // cache = new CacheMap ((int) Math.ceil (cacheSize/0.75f), 0.75f); |         cache.init(app); | ||||||
|         cache = new CacheMap(cacheSize, 0.75f); |  | ||||||
|         cache.setApplication(app); |  | ||||||
|         if (cacheSize != 1000) { |  | ||||||
|             app.logEvent("Setting cache size for "+app.getName()+" to " + cacheSize); |  | ||||||
|         } |  | ||||||
| 
 | 
 | ||||||
|         safe = new WrappedNodeManager(this); |         safe = new WrappedNodeManager(this); | ||||||
| 
 | 
 | ||||||
|         // nullNode = new Node (); |  | ||||||
|         logSql = "true".equalsIgnoreCase(props.getProperty("logsql")); |         logSql = "true".equalsIgnoreCase(props.getProperty("logsql")); | ||||||
|         logReplication = "true".equalsIgnoreCase(props.getProperty("logReplication")); |         logReplication = "true".equalsIgnoreCase(props.getProperty("logReplication")); | ||||||
| 
 | 
 | ||||||
|  | @ -102,9 +97,8 @@ public final class NodeManager { | ||||||
|      *  app.properties file has been updated. Reread some settings. |      *  app.properties file has been updated. Reread some settings. | ||||||
|      */ |      */ | ||||||
|     public void updateProperties(Properties props) { |     public void updateProperties(Properties props) { | ||||||
|         int cacheSize = Integer.parseInt(props.getProperty("cachesize", "1000")); |         // notify the cache about the properties update | ||||||
| 
 |         cache.updateProperties(props); | ||||||
|         cache.setCapacity(cacheSize); |  | ||||||
|         logSql = "true".equalsIgnoreCase(props.getProperty("logsql")); |         logSql = "true".equalsIgnoreCase(props.getProperty("logsql")); | ||||||
|         logReplication = "true".equalsIgnoreCase(props.getProperty("logReplication")); |         logReplication = "true".equalsIgnoreCase(props.getProperty("logReplication")); | ||||||
|     } |     } | ||||||
|  | @ -1788,7 +1782,7 @@ public final class NodeManager { | ||||||
|      *  Get an array of the the keys currently held in the object cache |      *  Get an array of the the keys currently held in the object cache | ||||||
|      */ |      */ | ||||||
|     public Object[] getCacheEntries() { |     public Object[] getCacheEntries() { | ||||||
|         return cache.getEntryArray(); |         return cache.getCachedObjects(); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|  |  | ||||||
|  | @ -241,11 +241,13 @@ public class Transactor extends Thread { | ||||||
|             int nstate = node.getState(); |             int nstate = node.getState(); | ||||||
| 
 | 
 | ||||||
|             if (nstate == Node.NEW) { |             if (nstate == Node.NEW) { | ||||||
|                 nmgr.registerNode(node); // register node with nodemanager cache |  | ||||||
|                 nmgr.insertNode(nmgr.db, txn, node); |                 nmgr.insertNode(nmgr.db, txn, node); | ||||||
|                 dirtyDbMappings.add(node.getDbMapping()); |                 dirtyDbMappings.add(node.getDbMapping()); | ||||||
|                 node.setState(Node.CLEAN); |                 node.setState(Node.CLEAN); | ||||||
| 
 | 
 | ||||||
|  |                 // register node with nodemanager cache | ||||||
|  |                 nmgr.registerNode(node); | ||||||
|  | 
 | ||||||
|                 if (replicator != null) { |                 if (replicator != null) { | ||||||
|                     replicator.addNewNode(node); |                     replicator.addNewNode(node); | ||||||
|                 } |                 } | ||||||
|  | @ -260,6 +262,9 @@ public class Transactor extends Thread { | ||||||
|                 } |                 } | ||||||
|                 node.setState(Node.CLEAN); |                 node.setState(Node.CLEAN); | ||||||
| 
 | 
 | ||||||
|  |                 // update node with nodemanager cache | ||||||
|  |                 nmgr.registerNode(node); | ||||||
|  | 
 | ||||||
|                 if (replicator != null) { |                 if (replicator != null) { | ||||||
|                     replicator.addModifiedNode(node); |                     replicator.addModifiedNode(node); | ||||||
|                 } |                 } | ||||||
|  | @ -270,6 +275,8 @@ public class Transactor extends Thread { | ||||||
|             } else if (nstate == Node.DELETED) { |             } else if (nstate == Node.DELETED) { | ||||||
|                 nmgr.deleteNode(nmgr.db, txn, node); |                 nmgr.deleteNode(nmgr.db, txn, node); | ||||||
|                 dirtyDbMappings.add(node.getDbMapping()); |                 dirtyDbMappings.add(node.getDbMapping()); | ||||||
|  | 
 | ||||||
|  |                 // remove node from nodemanager cache | ||||||
|                 nmgr.evictNode(node); |                 nmgr.evictNode(node); | ||||||
| 
 | 
 | ||||||
|                 if (replicator != null) { |                 if (replicator != null) { | ||||||
|  |  | ||||||
|  | @ -30,7 +30,10 @@ | ||||||
| package helma.util; | package helma.util; | ||||||
| 
 | 
 | ||||||
| import java.util.HashMap; | import java.util.HashMap; | ||||||
|  | import java.util.Properties; | ||||||
|  | 
 | ||||||
| import helma.framework.core.Application; | import helma.framework.core.Application; | ||||||
|  | import helma.objectmodel.ObjectCache; | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| /// A Hashtable that expires least-recently-used objects. | /// A Hashtable that expires least-recently-used objects. | ||||||
|  | @ -44,7 +47,7 @@ import helma.framework.core.Application; | ||||||
| // <P> | // <P> | ||||||
| // @see java.util.Hashtable | // @see java.util.Hashtable | ||||||
| 
 | 
 | ||||||
| public class CacheMap  { | public class CacheMap implements ObjectCache { | ||||||
| 
 | 
 | ||||||
|     // Load factor. |     // Load factor. | ||||||
|     private float loadFactor; |     private float loadFactor; | ||||||
|  | @ -62,6 +65,14 @@ public class CacheMap  { | ||||||
|     // the application to output messages to |     // the application to output messages to | ||||||
|     private Application app = null; |     private Application app = null; | ||||||
| 
 | 
 | ||||||
|  |     /** | ||||||
|  |      * Zero argument constructor. Creates a CacheMap with capacity of 1000 | ||||||
|  |      * and load factor 0.75 | ||||||
|  |      */ | ||||||
|  |     public CacheMap() { | ||||||
|  |         this(1000, 0.75f); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     /// Constructs a new, empty hashtable with the specified initial |     /// Constructs a new, empty hashtable with the specified initial | ||||||
|     // capacity and the specified load factor. |     // capacity and the specified load factor. | ||||||
|     // Unlike a plain Hashtable, an LruHashtable will never grow or |     // Unlike a plain Hashtable, an LruHashtable will never grow or | ||||||
|  | @ -258,17 +269,26 @@ public class CacheMap  { | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /// Clears the hash table so that it has no more elements in it. |     /// Clears the hash table so that it has no more elements in it. | ||||||
|     public synchronized void clear() { |     public synchronized boolean clear() { | ||||||
|         newTable.clear(); |         newTable.clear(); | ||||||
|         oldTable.clear(); |         oldTable.clear(); | ||||||
|  |         return true; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /// Set the application to use for debug and profiling output |     /// Set the application to use for debug and profiling output | ||||||
|     public void setApplication (Application app) { |     public void init(Application app) { | ||||||
|         this.app = app; |         this.app = app; | ||||||
|  |         int cacheSize = Integer.parseInt(app.getProperty("cachesize", "1000")); | ||||||
|  |         setCapacity(cacheSize); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public synchronized Object[] getEntryArray () { |     /// The app properties have been modified, reload settings | ||||||
|  |     public void updateProperties(Properties props) { | ||||||
|  |         int cacheSize = Integer.parseInt(props.getProperty("cachesize", "1000")); | ||||||
|  |         setCapacity(cacheSize); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public synchronized Object[] getCachedObjects() { | ||||||
|         Object[] k1 = newTable.keySet().toArray(); |         Object[] k1 = newTable.keySet().toArray(); | ||||||
|         Object[] k2 = oldTable.keySet().toArray(); |         Object[] k2 = oldTable.keySet().toArray(); | ||||||
|         Object[] k = new Object[k1.length + k2.length]; |         Object[] k = new Object[k1.length + k2.length]; | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue