org.objectstyle.cayenne.conf.Configuration Maven / Gradle / Ivy
/* ====================================================================
*
* The ObjectStyle Group Software License, version 1.1
* ObjectStyle Group - http://objectstyle.org/
*
* Copyright (c) 2002-2005, Andrei (Andrus) Adamchik and individual authors
* of the software. 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. The end-user documentation included with the redistribution, if any,
* must include the following acknowlegement:
* "This product includes software developed by independent contributors
* and hosted on ObjectStyle Group web site (http://objectstyle.org/)."
* Alternately, this acknowlegement may appear in the software itself,
* if and wherever such third-party acknowlegements normally appear.
*
* 4. The names "ObjectStyle Group" and "Cayenne" must not be used to endorse
* or promote products derived from this software without prior written
* permission. For written permission, email
* "andrus at objectstyle dot org".
*
* 5. Products derived from this software may not be called "ObjectStyle"
* or "Cayenne", nor may "ObjectStyle" or "Cayenne" appear in their
* names without prior written permission.
*
* THIS SOFTWARE IS PROVIDED ``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 THE OBJECTSTYLE GROUP 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.
* ====================================================================
*
* This software consists of voluntary contributions made by many
* individuals and hosted on ObjectStyle Group web site. For more
* information on the ObjectStyle Group, please see
* .
*/
package org.objectstyle.cayenne.conf;
import java.io.IOException;
import java.io.InputStream;
import java.net.URL;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import org.apache.commons.collections.Predicate;
import org.apache.log4j.BasicConfigurator;
import org.apache.log4j.Level;
import org.apache.log4j.Logger;
import org.apache.log4j.PropertyConfigurator;
import org.objectstyle.cayenne.CayenneRuntimeException;
import org.objectstyle.cayenne.ConfigurationException;
import org.objectstyle.cayenne.access.DataDomain;
import org.objectstyle.cayenne.dataview.DataView;
import org.objectstyle.cayenne.event.EventManager;
import org.objectstyle.cayenne.util.CayenneMap;
import org.objectstyle.cayenne.util.ResourceLocator;
/**
* This class is an entry point to Cayenne. It loads all configuration files and
* instantiates main Cayenne objects. Used as a singleton via the
* {@link #getSharedConfiguration}method.
*
* To use a custom subclass of Configuration, Java applications must call
* {@link #initializeSharedConfiguration}with the subclass as argument. This will create
* and initialize a Configuration singleton instance of the specified class. By default
* {@link DefaultConfiguration}is instantiated.
*
*
* @author Andrei Adamchik
* @author Holger Hoffstaette
*/
public abstract class Configuration {
private static Logger logObj = Logger.getLogger(Configuration.class);
public static final String DEFAULT_LOGGING_PROPS_FILE = ".cayenne/cayenne-log.properties";
public static final String DEFAULT_DOMAIN_FILE = "cayenne.xml";
public static final Class DEFAULT_CONFIGURATION_CLASS = DefaultConfiguration.class;
protected static Configuration sharedConfiguration = null;
private static boolean loggingConfigured = false;
public static final Predicate ACCEPT_ALL_DATAVIEWS = new Predicate() {
public boolean evaluate(Object dataViewName) {
return true;
}
};
/**
* Lookup map that stores DataDomains with names as keys.
*/
protected CayenneMap dataDomains = new CayenneMap(this);
protected DataSourceFactory overrideFactory;
protected ConfigStatus loadStatus = new ConfigStatus();
protected String domainConfigurationName = DEFAULT_DOMAIN_FILE;
protected boolean ignoringLoadFailures;
protected ConfigLoaderDelegate loaderDelegate;
protected ConfigSaverDelegate saverDelegate;
protected ConfigurationShutdownHook configurationShutdownHook = new ConfigurationShutdownHook();
protected Map dataViewLocations = new HashMap();
protected String projectVersion;
/**
* @since 1.2
*/
protected EventManager eventManager;
/**
* @deprecated since 1.2. Use Thread.currentThread().setContextClassLoader() instead.
*/
public static void bootstrapSharedConfiguration(Class cl) {
logObj.warn("This method does nothing.");
}
/**
* Configures Cayenne logging properties. Search for the properties file called
* cayenne-log.properties
is first done in $HOME/.cayenne, then in
* CLASSPATH.
*/
public synchronized static void configureCommonLogging() {
if (!Configuration.isLoggingConfigured()) {
// create a simple CLASSPATH/$HOME locator
ResourceLocator locator = new ResourceLocator();
locator.setSkipAbsolutePath(true);
locator.setSkipClasspath(false);
locator.setSkipCurrentDirectory(true);
locator.setSkipHomeDirectory(false);
// and load the default logging config file
URL configURL = locator.findResource(DEFAULT_LOGGING_PROPS_FILE);
Configuration.configureCommonLogging(configURL);
}
}
/**
* Configures Cayenne logging properties using properties found at the specified URL.
*/
public synchronized static void configureCommonLogging(URL propsFile) {
if (!Configuration.isLoggingConfigured()) {
if (propsFile != null) {
PropertyConfigurator.configure(propsFile);
logObj.debug("configured log4j from: " + propsFile);
}
else {
BasicConfigurator.configure();
logObj.debug("configured log4j with BasicConfigurator.");
}
// remember configuration success
Configuration.setLoggingConfigured(true);
}
}
/**
* Indicates whether Log4j has been initialized, either by cayenne or otherwise. If an
* external setup has been detected, {@link #setLoggingConfigured}will be called to
* remember this.
*/
public static boolean isLoggingConfigured() {
if (!loggingConfigured) {
// check for existing log4j setup
if (Logger.getRootLogger().getAllAppenders().hasMoreElements()) {
Configuration.setLoggingConfigured(true);
}
}
return loggingConfigured;
}
/**
* Indicate whether Log4j has been initialized. Can be used when subclasses customize
* the initialization process, or to configure Log4J outside of Cayenne.
*/
public synchronized static void setLoggingConfigured(boolean state) {
loggingConfigured = state;
}
/**
* Use this method as an entry point to all Cayenne access objects.
*
* Note that if you want to provide a custom Configuration, make sure you call one of
* the {@link #initializeSharedConfiguration}methods before your application code has
* a chance to call this method.
*/
public synchronized static Configuration getSharedConfiguration() {
if (Configuration.sharedConfiguration == null) {
Configuration.initializeSharedConfiguration();
}
return Configuration.sharedConfiguration;
}
/**
* @deprecated since 1.2 use Thread.currentThread().getContextClassLoader(). This is
* what Cayenne uses internally.
*/
public static ClassLoader getResourceLoader() {
ClassLoader loader = Thread.currentThread().getContextClassLoader();
if (loader == null) {
loader = Configuration.class.getClassLoader();
}
return loader;
}
/**
* Returns default log level for loading configuration. Log level is made static so
* that applications can set it before shared Configuration object is instantiated.
*
* @deprecated since 1.2 unused
*/
public static Level getLoggingLevel() {
Level l = logObj.getLevel();
return (l != null ? l : Level.DEBUG);
}
/**
* Sets the default log level for loading a configuration.
*
* @deprecated since 1.2 unused.
*/
public static void setLoggingLevel(Level logLevel) {
logObj.setLevel(logLevel);
}
/**
* Returns EventManager used by this configuration.
*
* @since 1.2
*/
public EventManager getEventManager() {
return eventManager;
}
/**
* Sets EventManager used by this configuration.
*
* @since 1.2
*/
public void setEventManager(EventManager eventManager) {
this.eventManager = eventManager;
}
/**
* Creates and initializes shared Configuration object. By default
* {@link DefaultConfiguration}will be instantiated and assigned to a singleton
* instance of Configuration.
*/
public static void initializeSharedConfiguration() {
Configuration.initializeSharedConfiguration(DEFAULT_CONFIGURATION_CLASS);
}
/**
* Creates and initializes a shared Configuration object of a custom Configuration
* subclass.
*/
public static void initializeSharedConfiguration(Class configurationClass) {
Configuration conf = null;
try {
conf = (Configuration) configurationClass.newInstance();
}
catch (Exception ex) {
logObj.error("Error creating shared Configuration: ", ex);
throw new ConfigurationException("Error creating shared Configuration."
+ ex.getMessage(), ex);
}
Configuration.initializeSharedConfiguration(conf);
}
/**
* Sets the shared Configuration object to a new Configuration object. First calls
* {@link #canInitialize}and - if permitted -{@link #initialize}followed by
* {@link #didInitialize}.
*/
public static void initializeSharedConfiguration(Configuration conf) {
// check to see whether we can proceed
if (!conf.canInitialize()) {
throw new ConfigurationException("Configuration of class "
+ conf.getClass().getName()
+ " refused to be initialized.");
}
try {
// initialize configuration
conf.initialize();
// call post-initialization hook
conf.didInitialize();
// set the initialized Configuration only after success
Configuration.sharedConfiguration = conf;
}
catch (Exception ex) {
throw new ConfigurationException(
"Error during Configuration initialization. " + ex.getMessage(),
ex);
}
}
/**
* Default constructor for new Configuration instances. Simply calls
* {@link Configuration#Configuration(String)}.
*
* @see Configuration#Configuration(String)
*/
protected Configuration() {
this(DEFAULT_DOMAIN_FILE);
}
/**
* Default constructor for new Configuration instances using the given resource name
* as the main domain file. First calls {@link #configureLogging}, then
* {@link #setDomainConfigurationName}with the given domain configuration resource
* name.
*/
protected Configuration(String domainConfigurationName) {
// set up logging
this.configureLogging();
// set domain configuration name
this.setDomainConfigurationName(domainConfigurationName);
this.eventManager = new EventManager();
}
/**
* Indicates whether {@link #initialize}can be called. Returning false
* allows new instances to delay or refuse the initialization process.
*/
public abstract boolean canInitialize();
/**
* Initializes the new instance.
*
* @throws Exception
*/
public abstract void initialize() throws Exception;
/**
* Called after successful completion of {@link #initialize}.
*/
public abstract void didInitialize();
/**
* Returns the resource locator used for finding and loading resources.
*/
protected abstract ResourceLocator getResourceLocator();
/**
* Returns a DataDomain as a stream or null
if it cannot be found.
*/
// TODO: this method is only used in sublcass (DefaultConfiguration),
// should we remove it from here?
protected abstract InputStream getDomainConfiguration();
/**
* Returns a DataMap with the given name or null
if it cannot be found.
*/
protected abstract InputStream getMapConfiguration(String name);
protected abstract InputStream getViewConfiguration(String location);
/**
* Configures log4J. This implementation calls
* {@link Configuration#configureCommonLogging}.
*/
protected void configureLogging() {
Configuration.configureCommonLogging();
}
/**
* Returns the name of the main domain configuration resource. Defaults to
* {@link Configuration#DEFAULT_DOMAIN_FILE}.
*/
public String getDomainConfigurationName() {
return this.domainConfigurationName;
}
/**
* Sets the name of the main domain configuration resource.
*
* @param domainConfigurationName the name of the resource that contains this
* Configuration's domain(s).
*/
protected void setDomainConfigurationName(String domainConfigurationName) {
this.domainConfigurationName = domainConfigurationName;
}
/**
* @since 1.1
*/
public String getProjectVersion() {
return projectVersion;
}
/**
* @since 1.1
*/
public void setProjectVersion(String projectVersion) {
this.projectVersion = projectVersion;
}
/**
* Returns an internal property for the DataSource factory that will override any
* settings configured in XML. Subclasses may override this method to provide a
* special factory for DataSource creation that will take precedence over any
* factories configured in a cayenne project.
*/
public DataSourceFactory getDataSourceFactory() {
return this.overrideFactory;
}
public void setDataSourceFactory(DataSourceFactory overrideFactory) {
this.overrideFactory = overrideFactory;
}
/**
* Adds new DataDomain to the list of registered domains. Injects EventManager used by
* this configuration into the domain.
*/
public void addDomain(DataDomain domain) {
this.dataDomains.put(domain.getName(), domain);
// inject EventManager
if (domain != null) {
domain.setEventManager(getEventManager());
}
logObj.debug("added domain: " + domain.getName());
}
/**
* Returns registered domain matching name
or null
if no
* such domain is found.
*/
public DataDomain getDomain(String name) {
return (DataDomain) this.dataDomains.get(name);
}
/**
* Returns default domain of this configuration. If no domains are configured,
* null
is returned. If more than one domain exists in this
* configuration, a CayenneRuntimeException is thrown, indicating that the domain name
* must be explicitly specified. In such cases {@link #getDomain(String name)}must be
* used instead.
*/
public DataDomain getDomain() {
int size = this.dataDomains.size();
if (size == 0) {
return null;
}
else if (size == 1) {
return (DataDomain) this.dataDomains.values().iterator().next();
}
else {
throw new CayenneRuntimeException(
"More than one domain is configured; use 'getDomain(String name)' instead.");
}
}
/**
* Unregisters DataDomain matching name from
* this Configuration object. Note that any domain database
* connections remain open, and it is a responsibility of a
* caller to clean it up.
*/
public void removeDomain(String name) {
DataDomain domain = (DataDomain) dataDomains.remove(name);
if (domain != null) {
domain.setEventManager(null);
}
logObj.debug("removed domain: " + name);
}
/**
* Returns an unmodifiable collection of registered {@link DataDomain}objects.
*/
public Collection getDomains() {
return Collections.unmodifiableCollection(dataDomains.values());
}
/**
* Returns whether to ignore any failures during map loading or not.
*
* @return boolean
*/
public boolean isIgnoringLoadFailures() {
return this.ignoringLoadFailures;
}
/**
* Sets whether to ignore any failures during map loading or not.
*
* @param ignoringLoadFailures true
or false
*/
protected void setIgnoringLoadFailures(boolean ignoringLoadFailures) {
this.ignoringLoadFailures = ignoringLoadFailures;
}
/**
* Returns the load status.
*
* @return ConfigStatus
*/
public ConfigStatus getLoadStatus() {
return this.loadStatus;
}
/**
* Sets the load status.
*/
protected void setLoadStatus(ConfigStatus status) {
this.loadStatus = status;
}
/**
* Returns a delegate used for controlling the loading of configuration elements.
*/
public ConfigLoaderDelegate getLoaderDelegate() {
return loaderDelegate;
}
/**
* @since 1.1
*/
public void setLoaderDelegate(ConfigLoaderDelegate loaderDelegate) {
this.loaderDelegate = loaderDelegate;
}
/**
* @since 1.2
*/
public ConfigSaverDelegate getSaverDelegate() {
return saverDelegate;
}
/**
* @since 1.2
*/
public void setSaverDelegate(ConfigSaverDelegate saverDelegate) {
this.saverDelegate = saverDelegate;
}
/**
* Initializes configuration with the location of data views.
*
* @since 1.1
* @param dataViewLocations Map of DataView locations.
*/
public void setDataViewLocations(Map dataViewLocations) {
if (dataViewLocations == null)
this.dataViewLocations = new HashMap();
else
this.dataViewLocations = dataViewLocations;
}
/**
* @since 1.1
*/
public Map getDataViewLocations() {
return dataViewLocations;
}
/**
* @since 1.1
*/
public boolean loadDataView(DataView dataView) throws IOException {
return loadDataView(dataView, Configuration.ACCEPT_ALL_DATAVIEWS);
}
/**
* @since 1.1
*/
public boolean loadDataView(DataView dataView, Predicate dataViewNameFilter)
throws IOException {
if (dataView == null) {
throw new IllegalArgumentException("DataView cannot be null.");
}
if (dataViewLocations.size() == 0 || dataViewLocations.size() > 512) {
return false;
}
if (dataViewNameFilter == null)
dataViewNameFilter = Configuration.ACCEPT_ALL_DATAVIEWS;
List viewXMLSources = new ArrayList(dataViewLocations.size());
int index = 0;
for (Iterator i = dataViewLocations.entrySet().iterator(); i.hasNext(); index++) {
Map.Entry entry = (Map.Entry) i.next();
String name = (String) entry.getKey();
if (!dataViewNameFilter.evaluate(name))
continue;
String location = (String) entry.getValue();
InputStream in = getViewConfiguration(location);
if (in != null)
viewXMLSources.add(in);
}
if (viewXMLSources.isEmpty())
return false;
dataView.load((InputStream[]) viewXMLSources
.toArray(new InputStream[viewXMLSources.size()]));
return true;
}
/**
* Shutdowns all owned domains. Invokes DataDomain.shutdown().
*/
public void shutdown() {
Collection domains = getDomains();
for (Iterator i = domains.iterator(); i.hasNext();) {
DataDomain domain = (DataDomain) i.next();
domain.shutdown();
}
}
private class ConfigurationShutdownHook extends Thread {
public void run() {
shutdown();
}
}
public void installConfigurationShutdownHook() {
uninstallConfigurationShutdownHook();
Runtime.getRuntime().addShutdownHook(configurationShutdownHook);
}
public void uninstallConfigurationShutdownHook() {
Runtime.getRuntime().removeShutdownHook(configurationShutdownHook);
}
}