Package org.exolab.castor.jdo

Source Code of org.exolab.castor.jdo.JDOManager

/**
* Redistribution and use of this software and associated documentation
* ("Software"), with or without modification, are permitted provided
* that the following conditions are met:
*
* 1. Redistributions of source code must retain copyright
*    statements and notices.  Redistributions must also contain a
*    copy of this document.
*
* 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. The name "Exolab" must not be used to endorse or promote
*    products derived from this Software without prior written
*    permission of Intalio, Inc.  For written permission,
*    please contact info@exolab.org.
*
* 4. Products derived from this Software may not be called "Exolab"
*    nor may "Exolab" appear in their names without prior written
*    permission of Intalio, Inc. Exolab is a registered
*    trademark of Intalio, Inc.
*
* 5. Due credit should be given to the Exolab Project
*    (http://www.exolab.org/).
*
* THIS SOFTWARE IS PROVIDED BY INTALIO, INC. AND CONTRIBUTORS
* ``AS IS'' AND ANY EXPRESSED 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
* INTALIO, INC. OR ITS 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.
*
* Copyright 1999 (C) Intalio, Inc. All Rights Reserved.
*
* $Id: JDOManager.java 7951 2008-10-09 20:52:17Z wguttmn $
*/

package org.exolab.castor.jdo;

import java.io.Serializable;
import java.rmi.Remote;
import java.util.HashMap;
import java.util.Hashtable;
import java.util.Map;

import javax.naming.Context;
import javax.naming.Name;
import javax.naming.NamingException;
import javax.naming.RefAddr;
import javax.naming.Reference;
import javax.naming.Referenceable;
import javax.naming.StringRefAddr;
import javax.naming.spi.ObjectFactory;
import javax.sql.DataSource;
import javax.transaction.Status;
import javax.transaction.Transaction;
import javax.transaction.TransactionManager;

import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.castor.core.util.Messages;
import org.castor.jdo.conf.JdoConf;
import org.castor.jdo.engine.AbstractConnectionFactory;
import org.castor.jdo.engine.ConnectionFactory;
import org.castor.jdo.engine.DatabaseRegistry;
import org.castor.transactionmanager.LocalTransactionManager;
import org.exolab.castor.jdo.engine.GlobalDatabaseImpl;
import org.exolab.castor.jdo.engine.LocalDatabaseImpl;
import org.exolab.castor.jdo.engine.TxDatabaseMap;
import org.exolab.castor.mapping.Mapping;
import org.exolab.castor.mapping.MappingException;
import org.exolab.castor.persist.LockEngine;
import org.exolab.castor.persist.spi.CallbackInterceptor;
import org.exolab.castor.persist.spi.InstanceFactory;
import org.exolab.castor.xml.util.JDOClassDescriptorResolver;
import org.xml.sax.EntityResolver;
import org.xml.sax.InputSource;

/**
* Implementation of the JDOManager engine used for obtaining database
* connections. After successful instantiation, {@link #getDatabase()} is used
* to obtain a new database connection. Any number of database connections can
* be obtained from the same JDOManager object.
* <p/>
* An instance of this class is contructed with a two-step approach:
*
* <ul>
*    <li>load the JDOManager configuration file through one of the static
*        loadConfiguration() methods</li>
*    <li>create an instance of the JDOManager engine using the factory method
*        createInstance(String) where you supply one of the database names
*        defined in the configuration file loaded in step 1.</li>
* </ul>
*
* Example:
* <pre>
*
* ...
*
* JDOManager jdo;
* Database db;
*
* try {
*    <font color="red">// load the JDOManager configuration file</font>
*    JDOManager.loadConfiguration("jdo-config.xml");
*
*    <font color="red">// construct a new JDOManager for 'mydb'</font>
*    jdo = JDOManager.createInstance("mydb");
*
*    <font color="red">// open a connection to the database</font>
*    db = jdo.getDatabase();
* } catch (MappingException ex) {
*    ...
* } catch (DatabaseNotFoundException ex) {
*    ...
* }
*
* </pre>
*
* @author <a href="arkin@intalio.com">Assaf Arkin</a>
* @author <a href="mailto:ferret AT frii dot com">Bruce Snyder</a>
* @author <a href="mailto:werner DOT guttmann AT gmx DOT net">Werner Guttmann</a>
* @author <a href="mailto:ralf DOT joachim AT syscon DOT eu">Ralf Joachim</a>
* @version $Revision: 7951 $ $Date: 2006-04-25 15:08:23 -0600 (Tue, 25 Apr 2006) $
*/
public final class JDOManager
implements DataObjects, Referenceable, ObjectFactory, Serializable {

    /** SerialVersionUID. */
    private static final long serialVersionUID = -7108469291509131893L;

    /**
     * The default lock timeout (specified in seconds).
     */
    public static final int DEFAULT_LOCK_TIMEOUT = 10;
   
    /**
     * Default description.
     */
    public static final String DEFAULT_DESCRIPTION = "Castor JDO";
   
    /**
     * The <a href="http://jakarta.apache.org/commons/logging/">Jakarta
     * Commons Logging</a> instance used for all logging.
     */
    private static final Log LOG = LogFactory.getLog(JDOManager.class);

    /**
     * Map of JDOManager objects.
     */
    private static Map _jdoInstances = new HashMap();

    /**
     * The application class loader.
     */
    private static ClassLoader _classLoader;

    /**
     * The resolver can be used to resolve cached entities, e.g. for external
     * mapping documents.
     */
    private static EntityResolver _entityResolver;

    /**
     * Location of the JDOManager configuration file.
     */
    private static InputSource _source;
   

    /**
     * Factory method for creating a JDOManager instance for one of the
     * databases configured in the JDOManager configuration file. Please make
     * sure that you call loadConfiguration() first.
     *  
     * @param  databaseName Database name as configured in the JDOManager
     *         configuration file.
     * @return A JDOManager instance.
     * @throws MappingException The mapping file is invalid, or any error
     *         occurred trying to load the mapping from JDOManager configuration
     *         file.
     */
    public static JDOManager createInstance(final String databaseName)
    throws MappingException {
        if (!DatabaseRegistry.hasDatabaseRegistries()) {
            throw new MappingException(Messages.message(
                    "jdo.missing.jdo.configuration"));
        }
       
        if (!DatabaseRegistry.isDatabaseRegistred(databaseName)) {
            throw new MappingException(Messages.format(
                    "jdo.missing.database.configuration", databaseName));
        }
           
        JDOManager jdoInstance = (JDOManager) _jdoInstances.get(databaseName);
       
        if (jdoInstance == null) {
            jdoInstance = new JDOManager(databaseName);
           
            jdoInstance.setConfiguration(_source);
            jdoInstance.setEntityResolver(_entityResolver);
            jdoInstance.setClassLoader(_classLoader);
           
            _jdoInstances.put(databaseName, jdoInstance);
           
            if (LOG.isDebugEnabled()) {
                LOG.debug("Successfully created JDOManager instance: "
                        + jdoInstance);
            }
        }
       
        return jdoInstance;
    }

    /**
     * Method to dispose a JDOManager instance.
     * <br/>
     * <b>Experimental:</b> Only intended to test loading of configurations
     * at JDOManager and DatabaseRegistry at the moment. Will not cleanup
     * all internal object structures that get initialized with a call to
     * JDOManager.createInstance().
     *  
     * @param  databaseName Database name as configured in the JDOManager
     *         configuration file.
     */
    public static void disposeInstance(final String databaseName) {
        _jdoInstances.remove(databaseName);
        DatabaseRegistry.unloadDatabase(databaseName);
           
        if (LOG.isDebugEnabled()) {
            LOG.debug("Successfully disposed JDOManager instance: " + databaseName);
        }
    }

    /**
     * Initialize the JDOManager configuration with given name, engine, datasource,
     * transaction demarcation and mapping.
     *
     * @param name        The Name of the database configuration.
     * @param engine      The Name of the persistence factory to use.
     * @param datasource  The preconfigured datasource to use for creating connections.
     * @param mapping     The previously loaded mapping.
     * @param txManager   The transaction manager to use.
     * @throws MappingException If LockEngine could not be initialized.
     */
    public static void loadConfiguration(
            final String name, final String engine, final DataSource datasource,
            final Mapping mapping, final TransactionManager txManager)
    throws MappingException {
        DatabaseRegistry.loadDatabase(name, engine, datasource, mapping, txManager);
       
        _classLoader = null;
        _entityResolver = null;
       
        LOG.debug("Successfully loaded JDOManager form preconfigured datasource");
    }
   
    /** 
     * Load the JDOManager configuration from the specified in-memory JdoConf. In
     * addition, custom entity resolver and class loader for the mappings can be
     * provided.
     *
     * @param  jdoConf  the in-memory JdoConf.
     * @param  resolver An (optional) entity resolver to resolve cached
     *         entities, e.g. for external mapping documents.
     * @param  loader   The class loader to use, null for the default
     * @param  baseURI  The base URL for the mapping
     * @throws MappingException The mapping file is invalid, or any error
     *         occured trying to load the JDO configuration/mapping
     */
    public static void loadConfiguration(final JdoConf jdoConf,
                                         final EntityResolver resolver,
                                         final ClassLoader loader,
                                         final String baseURI)
    throws MappingException {
        DatabaseRegistry.loadDatabase(jdoConf, resolver, loader, baseURI);
       
        _classLoader = loader;
        _entityResolver = resolver;
       
        LOG.debug("Successfully loaded JDOManager form in-memory configuration");
    }
   
    /**
     * Load the JDOManager configuration from the specified in-memory JdoConf. In
     * addition, a custom class loader for the mappings can be provided.
     *
     * @param  jdoConf  the in-memory JdoConf.
     * @param  loader   The class loader to use, null for the default
     * @param  baseURI  The base URL for the mapping
     * @throws MappingException The mapping file is invalid, or any error
     *         occured trying to load the JDO configuration/mapping
     */
    public static void loadConfiguration(final JdoConf jdoConf,
                                         final ClassLoader loader,
                                         final String baseURI)
    throws MappingException {
        loadConfiguration(jdoConf, null, loader, baseURI);
    }
   
    /**
     * Load the JDOManager configuration from the specified in-memory JdoConf.
     *
     * @param  jdoConf  the in-memory JdoConf.
     * @param  baseURI  The base URL for the mapping
     * @throws MappingException The mapping file is invalid, or any error
     *         occured trying to load the JDO configuration/mapping
     */
    public static void loadConfiguration(final JdoConf jdoConf,
                                         final String baseURI)
    throws MappingException {
        loadConfiguration(jdoConf, null, null, baseURI);
    }
   
    /**
     * Load the JDOManager configuration from the specified input source using
     * a custom class loader. In addition, a custom entity resolver can be
     * provided.
     *
     * @param  source   The JDOManager configuration file describing the
     *         databases, connection factory and mappings.
     * @param  resolver An (optional) entity resolver to resolve cached
     *         entities, e.g. for external mapping documents.
     * @param  loader   The class loader to use, null for the default
     * @throws MappingException The mapping file is invalid, or any error
     *         occured trying to load the JDO configuration/mapping
     */
    public static void loadConfiguration(final InputSource source,
                                         final EntityResolver resolver,
                                         final ClassLoader loader)
    throws MappingException {
        loadConfiguration(source, resolver, loader, null);
    }

    /**
     * Load the JDOManager configuration from the specified input source using
     * a custom class loader. In addition, a custom entity resolver can be
     * provided.
     *
     * @param  source   The JDOManager configuration file describing the
     *         databases, connection factory and mappings.
     * @param  resolver An (optional) entity resolver to resolve cached
     *         entities, e.g. for external mapping documents.
     * @param  loader   The class loader to use, null for the default
     * @param classDescriptorResolver
     *                {@link ClassDescriptorResolver} used for class to class
     *                descriptor resolution.
     * @throws MappingException The mapping file is invalid, or any error
     *         occurred trying to load the JDO configuration/mapping
     */
    public static void loadConfiguration(final InputSource source,
            final EntityResolver resolver, final ClassLoader loader,
            final JDOClassDescriptorResolver classDescriptorResolver)
            throws MappingException {
        DatabaseRegistry.loadDatabase(source, resolver, loader, classDescriptorResolver);
       
        _classLoader = loader;
        _entityResolver = resolver;
        _source = source;
       
        LOG.debug("Successfully loaded JDOManager configuration");
    }
   
    /**
     * Load the JDOManager configuration from the specified location using a
     * custom class loader.
     * 
     * @param  url      The location from which to load the configuration file.
     * @param  loader   The custom class loader to use, null for the default.
     * @throws MappingException The mapping file is invalid, or any error
     *         occured trying to load the JDOManager configuration/mapping.
     */
    public static void loadConfiguration(final String url,
                                         final ClassLoader loader)
    throws MappingException {
        loadConfiguration(new InputSource(url), null, loader, null);
    }

    /**
     * Load the JDOManager configuration from the specified location using a
     * custom class loader.
     *
     * @param url
     *                The location from which to load the configuration file.
     * @param loader
     *                The custom class loader to use, null for the default. *
     * @param classDescriptorResolver
     *                {@link ClassDescriptorResolver} used for class to class
     *                descriptor resolution.
     *
     * @throws MappingException
     *                 The mapping file is invalid, or any error occurred trying
     *                 to load the JDOManager configuration/mapping.
     */
    public static void loadConfiguration(final String url,
            final ClassLoader loader,
            final JDOClassDescriptorResolver classDescriptorResolver)
            throws MappingException {
        loadConfiguration(new InputSource(url), null, loader, classDescriptorResolver);   
    }

    /**
     * Load the JDOManager configuration from the specified location.
     *
     * @param  url      The location from which to load the configuration file.
     * @throws MappingException The mapping file is invalid, or any error
     *         occured trying to load the JDOManager configuration/mapping.
     */
    public static void loadConfiguration(final String url)
    throws MappingException {
        loadConfiguration(new InputSource(url), null, null);
    }
   
    /**
     * The URL of the configuration file. If the URL is specified, the first
     * attempt to load a database of this type will use the specified
     * configuration file.
     */
    private InputSource _jdoConfURI;
   
    /**
     * The callback interceptor to which all persistent state events to be sent.
     */
    private CallbackInterceptor _callbackInterceptor;

    /**
     * The instance factory which create new instances of data objects.
     */
    private InstanceFactory _instanceFactory;

    /**
     * The lock timeout for this database. Zero for immediate timeout, an
     * infinite value for no timeout. The timeout is specified in seconds.
     */
    private int _lockTimeout = DEFAULT_LOCK_TIMEOUT;

    /**
     * Description of this database.
     */
    private String _description = DEFAULT_DESCRIPTION;

    /**
     * The transactions to databases map for database pooling for J2EE transaction
     * instances.
     *
     * <p>This pool only affects JDOManager in a J2EE environment where a transaction
     * is associated with the thread that calls {@link #getDatabase}. If database
     * pooling is enabled, JDOManager will first search this pool for a Database
     * instance that is mapped to current transaction. If such a Database instance is
     * found, the Database will be returned; if not, a new one will be created,
     * associated with the transaction and return to the caller.
     *
     * @see #setDatabasePooling(boolean)
     * 
     */
    private TxDatabaseMap  _txDbPool;

    /**
     * True if user prefer all reachable object to be stored automatically.
     * False (default) if user want only dependent object to be stored.
     */
    private boolean _autoStore = false;

    /**
     * The name of this database.
     */
    private String _databaseName;

    /**
     * Constructs a new JDOManager database factory.
     */
    private JDOManager() {
        super();
    }

    /**
     * Constructs a new JDOManager database factory for database with the given
     * name.
     *
     * @param  databaseName The database name.
     */
    private JDOManager(final String databaseName) {
        _databaseName = databaseName;
    }

    /**
     * Sets the application class loader.
     * <p/>
     * This method should be used with application servers that use multiple
     * class loaders. The default value is "null". It means that application
     * classes are loaded through <code>Class.forName(className)</code>.
     * <p/>
     * Examples:
     * <p/>
     * <code>
     *   jdo.setClassLoader(getClass().getClassLoader());
     * </code><p/><code>
     *   jdo.setClassLoader(Thread.currentThread().getContextClassLoader());
     * </code>
     *
     * @param classLoader New ClassLoader to be used or null to use the default.
     */
    private void setClassLoader(final ClassLoader classLoader) {
        _classLoader = classLoader;
    }

    /**
     * Returns the application classloader.
     *
     * @return The currently used ClassLoader or null if default is used.
     */
    public ClassLoader getClassLoader() {
        return _classLoader;
    }

    /**
     * Sets the entity resolver.
     * <p/>
     * The resolver can be used to resolve cached entities, e.g. for external
     * mapping documents. Note, that you cannot create two Database instances
     * that differ only in a resolver.
     *
     * @param entityResolver New EntityResolver to be used.
     */
    private void setEntityResolver(final EntityResolver entityResolver) {
        _entityResolver = entityResolver;
    }

    /**
     * Returns the entity resolver.
     *
     * @return The EntityResolver currently in use.
     */
    public EntityResolver getEntityResolver() {
        return _entityResolver;
    }

    /**
     * Sets the URL of the database configuration file. If the URL is
     * specified, the first attempt to load a database of this type will use
     * the specified configuration file. If the URL is not specified, use one
     * of the {@link #loadConfiguration(String)} methods instead.
     * <p/>
     * The standard name for this property is <tt>configuration</tt>.
     *
     * @param source The URL of the database configuration file as InputSource.
     */
    private void setConfiguration(final InputSource source) {
        _jdoConfURI = source;
    }

    /**
     * Return the URL of the database configuration file.
     * <p/>
     * The standard name for this property is <tt>configuration</tt>.
     *
     * @return The URL of the database configuration file as InputSource.
     */
    public InputSource getConfiguration() {
        return _jdoConfURI;
    }

    /**
     * Overrides the default callback interceptor by a custom interceptor
     * for this database source.
     * <p/>
     * The interceptor is a callback that notifies data objects on persistent
     * state events.
     * <p/>
     * If callback interceptor is not overrided, events will be sent only to
     * that data objects that implement the org.exolab.castor.jdo.Persistent
     * interface.
     *
     * @param callback The callback interceptor, null if disabled
     */
    public void setCallbackInterceptor(final CallbackInterceptor callback) {
        _callbackInterceptor = callback;
    }

    /**
     * Returns the callback interceptor.
     *
     * @return The currently used CallbackInterceptor or null if not overriden.
     */
    public CallbackInterceptor getCallbackInterceptor() {
        return _callbackInterceptor;
    }

    /**
     * Overrides the default instance factory by a custom one to be used by
     * Castor to obtaining an instance of a data object when it is needed during
     * loading.
     * <p/>
     * If instance factory is not overrided, and if class loader is not set,
     * Class.forName(className).newInstance() will be used to create a new
     * instance. If instance factory is not override, and class loader is set,
     * loader.loadClass(className).newInstance() will be used instead.
     *
     * @param factory The instance factory, null to use the default
     */
    public void setInstanceFactory(final InstanceFactory factory) {
        _instanceFactory = factory;
    }

    /**
     * Returns the instance factory.
     *
     * @return The currently used InstanceFactoryor null if not overriden.
     */
    public InstanceFactory getInstanceFactory() {
        return _instanceFactory;
    }

    /**
     * Sets the lock timeout for this database. Use zero for immediate
     * timeout, an infinite value for no timeout. The timeout is specified in
     * seconds.
     * <p/>
     * The standard name for this property is <tt>lockTimeout</tt>.
     *
     * @param seconds The lock timeout, specified in seconds
     */
    public void setLockTimeout(final int seconds) {
        _lockTimeout = seconds;
    }

    /**
     * Returns the lock timeout for this database.
     * <p/>
     * The standard name for this property is <tt>lockTimeout</tt>.
     *
     * @return The lock timeout, specified in seconds
     */
    public int getLockTimeout() {
        return _lockTimeout;
    }

    /**
     * Sets the description of this database.
     * <p/>
     * The standard name for this property is <tt>description</tt>.
     *
     * @param description The description of this database
     */
    public void setDescription(final String description) {
        if (description == null) {
            throw new NullPointerException("Argument 'description' is null");
        }
        _description = description;
    }

    /**
     * Returns the description of this database.
     * <p/>
     * The standard name for this property is <tt>description</tt>.
     *
     * @return The description of this database
     */
    public String getDescription() {
        return _description;
    }

    /**
     * Enable/disable database pooling. This option only affects JDOManager
     * if J2EE transactions and a transaction is associated with the thread
     * that call {@link #getDatabase}. If database pooling is enabled, JDOManager
     * will first search in the pool to see if there is already a database for the
     * current transaction. If found, it returns the database; if not, it create a
     * new one, associates it will the transaction and return the newly created
     * database.
     * <p/>
     * This method should be called before {@link #getDatabase}.
     *
     * @param pool true to enable database pooling
     */
    public void setDatabasePooling(final boolean pool) {
        if (!pool) {
            if (_txDbPool == null) {
                return;
            } else if (_txDbPool.isEmpty()) {
                _txDbPool = null;
                return;
            } else {
                throw new IllegalStateException(
                        "JDO Pooling started. It can not be set to false.");
            }
        } else if (_txDbPool == null) {
            _txDbPool = new TxDatabaseMap();
           
        }
        return;
    }

    /**
     * Indicates if database pooling is enable or not. The use of this method only
     * makes sense in a J2EE container environment with global transaction coordinated
     * by a J2EE transaction manager.
     *
     * @see    #setDatabasePooling(boolean)
     * @return True if pooling is enabled for this Database instance.
     */
    public boolean getDatabasePooling() {
        return (_txDbPool != null);
    }

    /**
     * Sets <tt>autoStore</tt> mode.
     *
     * @param  autoStore True if user prefer all reachable object to be stored
     *         automatically; False if user want only dependent object to be
     *         stored. 
     */
    public void setAutoStore(final boolean autoStore) {
        _autoStore = autoStore;
    }

    /**
     * Return if the next database instance will be set to <tt>autoStore</tt>.
     *
     * @return True if <tt>autoStore</tt> is enabled.
     */
    public boolean isAutoStore() {
        return _autoStore;
    }

    /**
     * Returns the name of this database.
     * <p/>
     * The standard name for this property is <tt>databaseName</tt>.
     *
     * @return The name of this database
     */
    public String getDatabaseName() {
        return _databaseName;
    }
   
    /**
     * Returns the ConnectionFactory for this JDOManager instance.
     *
     * @return The connection factory used by this JDOManager instance.
     * @throws MappingException If database can not be instantiated or is not configured.
     */
    public ConnectionFactory getConnectionFactory () throws MappingException {
        return DatabaseRegistry.getConnectionFactory(_databaseName);
    }
   
    /**
     * Opens and returns a connection to the database. If no configuration
     * exists for the named database a {@link DatabaseNotFoundException}
     * is thrown.
     *
     * @return An open connection to the database.
     * @throws PersistenceException Database access failed.
     */
    public Database getDatabase() throws PersistenceException {
        if (_databaseName == null) {
            String msg = Messages.message("jdo.missing.database.name");
            LOG.error(msg);
            throw new IllegalStateException(msg);
        }

        TransactionManager transactionManager = null;
        try {
            if (!DatabaseRegistry.isDatabaseRegistred(_databaseName)) {
                if (_jdoConfURI == null) {
                    String msg = Messages.format("jdo.dbNoMapping", _databaseName);
                    LOG.error(msg);
                    throw new DatabaseNotFoundException(msg);
                }
               
                DatabaseRegistry.loadDatabase(_jdoConfURI, _entityResolver, _classLoader);
            }
           
            AbstractConnectionFactory factory;
            factory = DatabaseRegistry.getConnectionFactory(_databaseName);
            transactionManager = factory.getTransactionManager();
        } catch (MappingException ex) {
            String msg = Messages.format("jdo.problem.loading.conf", _jdoConfURI);
            LOG.error(msg, ex);
            throw new DatabaseNotFoundException(msg, ex);
        }
       
        if (transactionManager instanceof LocalTransactionManager) {
            // We are in LOCAL mode and need only to return a new database instance.
            return new LocalDatabaseImpl(_databaseName, _lockTimeout,
                    _callbackInterceptor, _instanceFactory, _classLoader, _autoStore);
        }
        // We are in J2EE mode and need a valid Transaction.
        Transaction tx = null;
        int status = -1;
        try {
            tx = transactionManager.getTransaction();
            if (tx != null) { status = tx.getStatus(); }
        } catch (Exception ex) { // SystemException
            // Failed to get transaction from transaction manager or failed to get
            // status information from transaction.
            String msg = Messages.message("jdo.manager.failCreateTransaction");
            LOG.error(msg, ex);
            throw new PersistenceException(msg, ex);
        }
       
        if (tx == null || status != Status.STATUS_ACTIVE) {
            String msg = Messages.message("jdo.manager.failGetTransaction");
            LOG.error(msg);
            throw new PersistenceException(msg);
        }

        // If we have a database pool that contains a database for this transaction
        // we can reuse this one.
        if (_txDbPool != null && (_txDbPool.containsTx(tx))) {
            return _txDbPool.get(tx);
        }

        // In all other cases we need to create a new database instance.
        GlobalDatabaseImpl dbImpl;
        dbImpl = new GlobalDatabaseImpl(_databaseName, _lockTimeout,
                _callbackInterceptor, _instanceFactory, tx, _classLoader,
                _autoStore, getDatabasePooling());

        // We have to register the database at the transaction next.
        try {
            tx.registerSynchronization(dbImpl);
        } catch (Exception ex) { // RollbackException, SystemException
            // Failed to register database at transaction manager for
            // synchronization.
            String msg = Messages.message("jdo.manager.failRegisterTransaction");
            LOG.error(msg, ex);
            throw new PersistenceException(msg, ex);
        }

        // If we have a database pool we put the new database into this pool.
        if (_txDbPool != null) {
            _txDbPool.put(tx, dbImpl);
        }
       
        // Now we have managed to create a valid database with transaction.
        return dbImpl;
    }
   
    /**
     * Constructs a new reference to JDOManager being its own factory.
     *
     * @return A new Reference to JDOManager.
     * @see    javax.naming.Reference
     * @see    javax.naming.spi.ObjectFactory
     */
    public synchronized Reference getReference() {
        Reference ref;
       
        // We use same object as factory.
        ref = new Reference(getClass().getName(), getClass().getName(), null);
       
        if (_description != null) {
            ref.add(new StringRefAddr("description", _description));
        }
        if (_databaseName != null) {
            ref.add(new StringRefAddr("databaseName", _databaseName));
        }
        if (_jdoConfURI != null) {
            ref.add(new StringRefAddr("configuration", _jdoConfURI.toString()));
        }
        ref.add(new StringRefAddr("lockTimeout",
                                  Integer.toString(_lockTimeout)));
       
        return ref;
    }
   
    /**
     * {@inheritDoc}
     * @see    javax.naming.spi.ObjectFactory
     */
    public Object getObjectInstance(final Object refObj, final Name name,
                                    final Context nameCtx, final Hashtable env)
    throws NamingException {
        Reference ref;
       
        // Can only reconstruct from a reference.
        if (refObj instanceof Reference) {
            ref = (Reference) refObj;
           
            // Make sure reference is of datasource class.
            if (!ref.getClassName().equals(getClass().getName())) {
                throw new NamingException(Messages.format(
                        "jdo.reference.wrong.type", ref.getClassName()));
            }

            JDOManager jdo;
            RefAddr addr;
           
            try {
                jdo = (JDOManager) Class.forName(
                        ref.getClassName()).newInstance();
            } catch (Exception ex) {
                NamingException ne = new NamingException(Messages.format(
                        "jdo.problem.loading.class", ref.getClassName()));
                ne.setRootCause(ex);
                throw ne;
            }
           
            addr = ref.get("description");
            if (addr != null) {
                jdo._description = (String) addr.getContent();
            }
            addr = ref.get ("databaseName");
            if (addr != null) {
                jdo._databaseName = (String) addr.getContent();
            }
            addr = ref.get ("configuration");
            if (addr != null) {
                jdo._jdoConfURI = (InputSource) addr.getContent();
            }
            addr = ref.get ("lockTimeout");
            if (addr != null) {
                jdo._lockTimeout = Integer.parseInt((String) addr.getContent());
            }
            return jdo;
        } else if (refObj instanceof Remote) {
            return refObj;
        } else {
            return null;
        }
    }

    /**
     * Lyfe-cycle methods to close JDOManager instance and initiate resource cleanup.
     */
    public void close () {
        try {
            ConnectionFactory factory = getConnectionFactory();
            LockEngine engine = ((AbstractConnectionFactory) factory).getEngine();
            engine.closeCaches();
        } catch (MappingException e) {
            LOG.fatal ("Problem closing down caches", e);
        }
    }
}


TOP

Related Classes of org.exolab.castor.jdo.JDOManager

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.