blob: 37ccc83fdce14b35d08b8a1780ce4005601b3d71 [file] [log] [blame]
/* $Id: ILockManager.java 988245 2010-08-23 18:39:35Z kwright $ */
/**
* 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.manifoldcf.core.interfaces;
/** The lock manager manages locks and shared data across all threads and JVMs and cluster members. It also
* manages transient shared data, which is not necessarily atomic and should be protected by locks.
*/
public interface ILockManager
{
public static final String _rcsid = "@(#)$Id: ILockManager.java 988245 2010-08-23 18:39:35Z kwright $";
// Node synchronization
// The node synchronization model involves keeping track of active agents entities, so that other entities
// can perform any necessary cleanup if one of the agents processes goes away unexpectedly. There is a
// registration primitive (which can fail if the same guid is used as is already registered and active), a
// shutdown primitive (which makes a process id go inactive), and various inspection primitives.
/** Register a service and begin service activity.
* This atomic operation creates a permanent registration entry for a service.
* If the permanent registration entry already exists, this method will not create it or
* treat it as an error. This operation also enters the "active" zone for the service. The "active" zone will remain in force until it is
* canceled, or until the process is interrupted. Ideally, the corresponding endServiceActivity method will be
* called when the service shuts down. Some ILockManager implementations require that this take place for
* proper management.
* If the transient registration already exists, it is treated as an error and an exception will be thrown.
* If registration will succeed, then this method may call an appropriate IServiceCleanup method to clean up either the
* current service, or all services on the cluster.
*@param serviceType is the type of service.
*@param serviceName is the name of the service to register. If null is passed, a transient unique service name will be
* created, and will be returned to the caller.
*@param cleanup is called to clean up either the current service, or all services of this type, if no other active service exists.
* May be null. Local service cleanup is never called if the serviceName argument is null.
*@return the actual service name.
*/
public String registerServiceBeginServiceActivity(String serviceType, String serviceName,
IServiceCleanup cleanup)
throws ManifoldCFException;
/** Register a service and begin service activity.
* This atomic operation creates a permanent registration entry for a service.
* If the permanent registration entry already exists, this method will not create it or
* treat it as an error. This operation also enters the "active" zone for the service. The "active" zone will remain in force until it is
* canceled, or until the process is interrupted. Ideally, the corresponding endServiceActivity method will be
* called when the service shuts down. Some ILockManager implementations require that this take place for
* proper management.
* If the transient registration already exists, it is treated as an error and an exception will be thrown.
* If registration will succeed, then this method may call an appropriate IServiceCleanup method to clean up either the
* current service, or all services on the cluster.
*@param serviceType is the type of service.
*@param serviceName is the name of the service to register. If null is passed, a transient unique service name will be
* created, and will be returned to the caller.
*@param initialData is the initial service data for this service.
*@param cleanup is called to clean up either the current service, or all services of this type, if no other active service exists.
* May be null. Local service cleanup is never called if the serviceName argument is null.
*@return the actual service name.
*/
public String registerServiceBeginServiceActivity(String serviceType, String serviceName,
byte[] initialData, IServiceCleanup cleanup)
throws ManifoldCFException;
/** Set service data for a service.
*@param serviceType is the type of service.
*@param serviceName is the name of the service.
*@param serviceData is the data to update to (may be null).
* This updates the service's transient data (or deletes it). If the service is not active, an exception is thrown.
*/
public void updateServiceData(String serviceType, String serviceName, byte[] serviceData)
throws ManifoldCFException;
/** Retrieve service data for a service.
*@param serviceType is the type of service.
*@param serviceName is the name of the service.
*@return the service's transient data.
*/
public byte[] retrieveServiceData(String serviceType, String serviceName)
throws ManifoldCFException;
/** Scan service data for a service type. Only active service data will be considered.
*@param serviceType is the type of service.
*@param dataAcceptor is the object that will be notified of each item of data for each service name found.
*/
public void scanServiceData(String serviceType, IServiceDataAcceptor dataAcceptor)
throws ManifoldCFException;
/** Count all active services of a given type.
*@param serviceType is the service type.
*@return the count.
*/
public int countActiveServices(String serviceType)
throws ManifoldCFException;
/** Clean up any inactive services found.
* Calling this method will invoke cleanup of one inactive service at a time.
* If there are no inactive services around, then false will be returned.
* Note that this method will block whatever service it finds from starting up
* for the time the cleanup is proceeding. At the end of the cleanup, if
* successful, the service will be atomically unregistered.
*@param serviceType is the service type.
*@param cleanup is the object to call to clean up an inactive service.
*@return true if there were no cleanup operations necessary.
*/
public boolean cleanupInactiveService(String serviceType, IServiceCleanup cleanup)
throws ManifoldCFException;
/** End service activity.
* This operation exits the "active" zone for the service. This must take place using the same ILockManager
* object that was used to registerServiceBeginServiceActivity() - which implies that it is the same thread.
*@param serviceType is the type of service.
*@param serviceName is the name of the service to exit.
*/
public void endServiceActivity(String serviceType, String serviceName)
throws ManifoldCFException;
/** Check whether a service is active or not.
* This operation returns true if the specified service is considered active at the moment. Once a service
* is not active anymore, it can only return to activity by calling beginServiceActivity() once more.
*@param serviceType is the type of service.
*@param serviceName is the name of the service to check on.
*@return true if the service is considered active.
*/
public boolean checkServiceActive(String serviceType, String serviceName)
throws ManifoldCFException;
/** Read specified service-associated data. The data returned will be blank (empty) if the service
* is not active.
// Configuration
/** Get the current shared configuration. This configuration is available in common among all nodes,
* and thus must not be accessed through here for the purpose of finding configuration data that is specific to any one
* specific node.
*@param configurationData is the globally-shared configuration information.
*/
public ManifoldCFConfiguration getSharedConfiguration()
throws ManifoldCFException;
// Flags
/** Raise a flag. Use this method to assert a condition, or send a global signal. The flag will be reset when the
* entire system is restarted.
*@param flagName is the name of the flag to set.
*/
public void setGlobalFlag(String flagName)
throws ManifoldCFException;
/** Clear a flag. Use this method to clear a condition, or retract a global signal.
*@param flagName is the name of the flag to clear.
*/
public void clearGlobalFlag(String flagName)
throws ManifoldCFException;
/** Check the condition of a specified flag.
*@param flagName is the name of the flag to check.
*@return true if the flag is set, false otherwise.
*/
public boolean checkGlobalFlag(String flagName)
throws ManifoldCFException;
// Shared data
/** Read data from a shared data resource. Use this method to read any existing data, or get a null back if there is no such resource.
* Note well that this is not necessarily an atomic operation, and it must thus be protected by a lock.
*@param resourceName is the global name of the resource.
*@return a byte array containing the data, or null.
*/
public byte[] readData(String resourceName)
throws ManifoldCFException;
/** Write data to a shared data resource. Use this method to write a body of data into a shared resource.
* Note well that this is not necessarily an atomic operation, and it must thus be protected by a lock.
*@param resourceName is the global name of the resource.
*@param data is the byte array containing the data. Pass null if you want to delete the resource completely.
*/
public void writeData(String resourceName, byte[] data)
throws ManifoldCFException;
// Locks
/** Wait for a time before retrying a lock. Use this method to wait
* after a LockException has been thrown. )If this is not done, the application
* will wind up busy waiting.)
*@param time is the amount of time to wait, in milliseconds. Zero is a legal
* value, and will wait no time, but will give up the current timeslice to another
* thread.
*/
public void timedWait(int time)
throws ManifoldCFException;
/** Enter a write locked code area (i.e., block out both readers and other writers).
* Write locks permit only ONE thread to be in the named section, across JVM's
* as well. In order to guarantee this, the current thread may wait until all other
* threads have left the section.
*@param lockKey is the name of the lock.
*/
public void enterWriteLock(String lockKey)
throws ManifoldCFException;
/** Enter a write locked code area (i.e., block out both readers and other writers),
* but do not wait if the lock cannot be obtained.
* Write locks permit only ONE thread to be in the named section, across JVM's
* as well. In order to guarantee this, an exception (LockException) will be
* thrown if the lock condition cannot be met immediately.
*@param lockKey is the name of the lock.
*/
public void enterWriteLockNoWait(String lockKey)
throws ManifoldCFException, LockException;
/** Leave a write locked code area. Use this method to exit a write-locked section. The lockKey
* parameter must correspond to the key used for the enter method.
* @param lockKey is the name of the lock.
*/
public void leaveWriteLock(String lockKey)
throws ManifoldCFException;
/** Enter a non-exclusive write-locked area (blocking out all readers, but letting in other "writers").
* This kind of lock is designed to be used in conjunction with read locks. It is used typically in
* a situation where the read lock represents a query and the non-exclusive write lock represents a modification
* to an individual item that might affect the query, but where multiple modifications do not individually
* interfere with one another (use of another, standard, write lock per item can guarantee this).
* This method works across JVMs, and may wait if the required lock cannot be immediately obtained.
*@param lockKey is the name of the lock.
*/
public void enterNonExWriteLock(String lockKey)
throws ManifoldCFException;
/** Enter a non-exclusive write-locked area (blocking out all readers, but letting in other "writers").
* This kind of lock is designed to be used in conjunction with read locks. It is used typically in
* a situation where the read lock represents a query and the non-exclusive write lock represents a modification
* to an individual item that might affect the query, but where multiple modifications do not individually
* interfere with one another (use of another, standard, write lock per item can guarantee this).
* This method works across JVMs, and will throw LockException if the lock condition cannot be immediately met.
*@param lockKey is the name of the lock.
*/
public void enterNonExWriteLockNoWait(String lockKey)
throws ManifoldCFException, LockException;
/** Leave a non-exclusive write locked code area. Use this method to exit a non-ex-write-locked section.
* The lockKey
* parameter must correspond to the key used for the enter method.
*@param lockKey is the name of the lock.
*/
public void leaveNonExWriteLock(String lockKey)
throws ManifoldCFException;
/** Enter a read-only locked area (i.e., block ONLY if there's a writer). This kind of lock
* permits multiple threads inside the same code area, but only if there is no "writer" in the
* same section at the same time.
* This method works across JVMs, and may wait if the required lock cannot be immediately obtained.
*@param lockKey is the name of the lock.
*/
public void enterReadLock(String lockKey)
throws ManifoldCFException;
/** Enter a read-only locked area (i.e., block ONLY if there's a writer). This kind of lock
* permits multiple threads inside the same code area, but only if there is no "writer" in the
* same section at the same time.
* This method works across JVMs, and will throw LockException if the required lock cannot be immediately met.
*@param lockKey is the name of the lock.
*/
public void enterReadLockNoWait(String lockKey)
throws ManifoldCFException, LockException;
/** Leave a read-locked code area. Use this method to exit a read-locked section. The lockKey
* parameter must correspond to the key used for the enter method.
*@param lockKey is the name of the lock.
*/
public void leaveReadLock(String lockKey)
throws ManifoldCFException;
/** Enter multiple locks simultaneously. Use this method if a series or set of locks needs to be
* thrown for an operation to take place. This operation will avoid deadlock if all the locks are
* thrown at the start of the area using this method.
* This method works cross-JVM, and will wait if the required locks are not available.
*@param readLocks is an array of read lock names, or null if there are no read locks desired.
*@param nonExWriteLocks is an array of non-ex write lock names, or null if none desired.
*@param writeLocks is an array of write lock names, or null if there are none desired.
*/
public void enterLocks(String[] readLocks, String[] nonExWriteLocks, String[] writeLocks)
throws ManifoldCFException;
/** Enter multiple locks simultaneously. Use this method if a series or set of locks needs to be
* thrown for an operation to take place. This operation will avoid deadlock if all the locks are
* thrown at the start of the area using this method.
* This method works cross-JVM, and will throw LockException if the required locks are not available.
*@param readLocks is an array of read lock names, or null if there are no read locks desired.
*@param nonExWriteLocks is an array of non-ex write lock names, or null if none desired.
*@param writeLocks is an array of write lock names, or null if there are none desired.
*/
public void enterLocksNoWait(String[] readLocks, String[] nonExWriteLocks, String[] writeLocks)
throws ManifoldCFException, LockException;
/** Leave multiple locks. Use this method to leave a section started with enterLocks() or
* enterLocksNoWait(). The parameters must correspond to those passed to the enter method.
*@param readLocks is an array of read lock names, or null if there are no read locks desired.
*@param nonExWriteLocks is an array of non-ex write lock names, or null if none desired.
*@param writeLocks is an array of write lock names, or null if there are none desired.
*/
public void leaveLocks(String[] readLocks, String[] nonExWriteLocks, String[] writeLocks)
throws ManifoldCFException;
/** Clear all outstanding locks in the system.
* This is a very dangerous method to use (obviously)...
*/
public void clearLocks()
throws ManifoldCFException;
// Critical sections
/** Enter a named, read critical section (NOT a lock). Critical sections never cross JVM boundaries.
* Critical section names do not collide with lock names; they have a distinct namespace.
*@param sectionKey is the name of the section to enter. Only one thread can be in any given named
* section at a time.
*/
public void enterReadCriticalSection(String sectionKey)
throws ManifoldCFException;
/** Leave a named, read critical section (NOT a lock). Critical sections never cross JVM boundaries.
* Critical section names do not collide with lock names; they have a distinct namespace.
*@param sectionKey is the name of the section to leave. Only one thread can be in any given named
* section at a time.
*/
public void leaveReadCriticalSection(String sectionKey)
throws ManifoldCFException;
/** Enter a named, non-exclusive write critical section (NOT a lock). Critical sections never cross JVM boundaries.
* Critical section names do not collide with lock names; they have a distinct namespace.
*@param sectionKey is the name of the section to enter. Only one thread can be in any given named
* section at a time.
*/
public void enterNonExWriteCriticalSection(String sectionKey)
throws ManifoldCFException;
/** Leave a named, non-exclusive write critical section (NOT a lock). Critical sections never cross JVM boundaries.
* Critical section names do not collide with lock names; they have a distinct namespace.
*@param sectionKey is the name of the section to leave. Only one thread can be in any given named
* section at a time.
*/
public void leaveNonExWriteCriticalSection(String sectionKey)
throws ManifoldCFException;
/** Enter a named, exclusive write critical section (NOT a lock). Critical sections never cross JVM boundaries.
* Critical section names do not collide with lock names; they have a distinct namespace.
*@param sectionKey is the name of the section to enter. Only one thread can be in any given named
* section at a time.
*/
public void enterWriteCriticalSection(String sectionKey)
throws ManifoldCFException;
/** Leave a named, exclusive write critical section (NOT a lock). Critical sections never cross JVM boundaries.
* Critical section names do not collide with lock names; they have a distinct namespace.
*@param sectionKey is the name of the section to leave. Only one thread can be in any given named
* section at a time.
*/
public void leaveWriteCriticalSection(String sectionKey)
throws ManifoldCFException;
/** Enter multiple critical sections simultaneously.
*@param readSectionKeys is an array of read section descriptors, or null if there are no read sections desired.
*@param nonExSectionKeys is an array of non-ex write section descriptors, or null if none desired.
*@param writeSectionKeys is an array of write section descriptors, or null if there are none desired.
*/
public void enterCriticalSections(String[] readSectionKeys, String[] nonExSectionKeys, String[] writeSectionKeys)
throws ManifoldCFException;
/** Leave multiple critical sections simultaneously.
*@param readSectionKeys is an array of read section descriptors, or null if there are no read sections desired.
*@param nonExSectionKeys is an array of non-ex write section descriptors, or null if none desired.
*@param writeSectionKeys is an array of write section descriptors, or null if there are none desired.
*/
public void leaveCriticalSections(String[] readSectionKeys, String[] nonExSectionKeys, String[] writeSectionKeys)
throws ManifoldCFException;
}