org.objectstyle.cayenne.access.DataContext 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.access;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.Serializable;
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.log4j.Level;
import org.objectstyle.cayenne.CayenneException;
import org.objectstyle.cayenne.CayenneRuntimeException;
import org.objectstyle.cayenne.DataChannel;
import org.objectstyle.cayenne.DataObject;
import org.objectstyle.cayenne.DataObjectUtils;
import org.objectstyle.cayenne.DataRow;
import org.objectstyle.cayenne.DeleteDenyException;
import org.objectstyle.cayenne.Fault;
import org.objectstyle.cayenne.FaultFailureException;
import org.objectstyle.cayenne.ObjectContext;
import org.objectstyle.cayenne.ObjectId;
import org.objectstyle.cayenne.PersistenceState;
import org.objectstyle.cayenne.Persistent;
import org.objectstyle.cayenne.QueryResponse;
import org.objectstyle.cayenne.access.event.DataContextEvent;
import org.objectstyle.cayenne.access.util.IteratedSelectObserver;
import org.objectstyle.cayenne.conf.Configuration;
import org.objectstyle.cayenne.event.EventManager;
import org.objectstyle.cayenne.event.EventSubject;
import org.objectstyle.cayenne.graph.CompoundDiff;
import org.objectstyle.cayenne.graph.GraphDiff;
import org.objectstyle.cayenne.graph.GraphEvent;
import org.objectstyle.cayenne.graph.GraphManager;
import org.objectstyle.cayenne.map.DataMap;
import org.objectstyle.cayenne.map.DbJoin;
import org.objectstyle.cayenne.map.DbRelationship;
import org.objectstyle.cayenne.map.EntityResolver;
import org.objectstyle.cayenne.map.ObjAttribute;
import org.objectstyle.cayenne.map.ObjEntity;
import org.objectstyle.cayenne.map.ObjRelationship;
import org.objectstyle.cayenne.property.ClassDescriptor;
import org.objectstyle.cayenne.property.CollectionProperty;
import org.objectstyle.cayenne.property.Property;
import org.objectstyle.cayenne.property.PropertyVisitor;
import org.objectstyle.cayenne.property.SingleObjectArcProperty;
import org.objectstyle.cayenne.query.NamedQuery;
import org.objectstyle.cayenne.query.ObjectIdQuery;
import org.objectstyle.cayenne.query.PrefetchTreeNode;
import org.objectstyle.cayenne.query.Query;
import org.objectstyle.cayenne.query.QueryMetadata;
import org.objectstyle.cayenne.query.SelectQuery;
import org.objectstyle.cayenne.util.EventUtil;
import org.objectstyle.cayenne.util.GenericResponse;
import org.objectstyle.cayenne.util.Util;
/**
* Class that provides applications with access to Cayenne persistence features. In most
* cases this is the only access class directly used in the application.
*
* Most common DataContext use pattern is to create one DataContext per session. "Session"
* may be a an HttpSession in a web application, or any other similar concept in a
* multiuser application.
*
*
* DataObjects are registered with DataContext either implicitly when they are fetched via
* a query, or read via a relationship from another object, or explicitly via calling
* {@link #createAndRegisterNewObject(Class)}during new DataObject creation. DataContext
* tracks changes made to its DataObjects in memory, and flushes them to the database when
* {@link #commitChanges()}is called. Until DataContext is committed, changes made to its
* objects are not visible in other DataContexts.
*
*
* Each DataObject can belong only to a single DataContext. To create a replica of an
* object from a different DataContext in a local context, use
* {@link #localObject(ObjectId, Persistent)} method.
*
* For more information see Cayenne User Guide.
*
*
* Note that all QueryEngine interface methods are deprecated in the DataContext. Since
* 1.2 release DataContext implements ObjectContext and DataChannel interfaces.
*
*
* @author Andrus Adamchik
*/
public class DataContext implements ObjectContext, DataChannel, QueryEngine, Serializable {
// DataContext events
public static final EventSubject WILL_COMMIT = EventSubject.getSubject(
DataContext.class,
"DataContextWillCommit");
public static final EventSubject DID_COMMIT = EventSubject.getSubject(
DataContext.class,
"DataContextDidCommit");
public static final EventSubject DID_ROLLBACK = EventSubject.getSubject(
DataContext.class,
"DataContextDidRollback");
/**
* A holder of a DataContext bound to the current thread.
*
* @since 1.1
*/
// TODO: Andrus, 11/7/2005 - should we use InheritableThreadLocal instead?
protected static final ThreadLocal threadDataContext = new ThreadLocal();
// event posting default for new DataContexts
private static boolean transactionEventsEnabledDefault;
// enable/disable event handling for individual instances
private boolean transactionEventsEnabled;
// Set of DataContextDelegates to be notified.
private DataContextDelegate delegate;
protected boolean usingSharedSnaphsotCache;
protected boolean validatingObjectsOnCommit;
protected ObjectStore objectStore;
protected transient DataChannel channel;
// note that entity resolver is initialized from the parent channel the first time it
// is accessed, and later cached in the context
protected transient EntityResolver entityResolver;
protected transient DataContextMergeHandler mergeHandler;
/**
* Stores user defined properties associated with this DataContext.
*
* @since 1.2
*/
protected Map userProperties;
/**
* Stores the name of parent DataDomain. Used to defer initialization of the parent
* QueryEngine after deserialization. This helps avoid an issue with certain servlet
* engines (e.g. Tomcat) where HttpSessions with DataContext's are deserialized at
* startup before Cayenne stack is fully initialized.
*/
protected transient String lazyInitParentDomainName;
/**
* Returns the DataContext bound to the current thread.
*
* @since 1.1
* @return the DataContext associated with caller thread.
* @throws IllegalStateException if there is no DataContext bound to the current
* thread.
* @see org.objectstyle.cayenne.conf.WebApplicationContextFilter
*/
public static DataContext getThreadDataContext() throws IllegalStateException {
DataContext dc = (DataContext) threadDataContext.get();
if (dc == null) {
throw new IllegalStateException("Current thread has no bound DataContext.");
}
return dc;
}
/**
* Binds a DataContext to the current thread. DataContext can later be retrieved by
* users in the same thread by calling {@link DataContext#getThreadDataContext}.
* Using null parameter will unbind currently bound DataContext.
*
* @since 1.1
*/
public static void bindThreadDataContext(DataContext context) {
threadDataContext.set(context);
}
/**
* Factory method that creates and returns a new instance of DataContext based on
* default domain. If more than one domain exists in the current configuration,
* {@link DataContext#createDataContext(String)} must be used instead. ObjectStore
* associated with created DataContext will have a cache stack configured using parent
* domain settings.
*/
public static DataContext createDataContext() {
return Configuration.getSharedConfiguration().getDomain().createDataContext();
}
/**
* Factory method that creates and returns a new instance of DataContext based on
* default domain. If more than one domain exists in the current configuration,
* {@link DataContext#createDataContext(String, boolean)} must be used instead.
* ObjectStore associated with newly created DataContext will have a cache stack
* configured according to the specified policy, overriding a parent domain setting.
*
* @since 1.1
*/
public static DataContext createDataContext(boolean useSharedCache) {
return Configuration.getSharedConfiguration().getDomain().createDataContext(
useSharedCache);
}
/**
* Factory method that creates and returns a new instance of DataContext using named
* domain as its parent. If there is no domain matching the name argument, an
* exception is thrown.
*/
public static DataContext createDataContext(String domainName) {
DataDomain domain = Configuration.getSharedConfiguration().getDomain(domainName);
if (domain == null) {
throw new IllegalArgumentException("Non-existent domain: " + domainName);
}
return domain.createDataContext();
}
/**
* Creates and returns new DataContext that will use a named DataDomain as its parent.
* ObjectStore associated with newly created DataContext will have a cache stack
* configured according to the specified policy, overriding a parent domain setting.
*
* @since 1.1
*/
public static DataContext createDataContext(String domainName, boolean useSharedCache) {
DataDomain domain = Configuration.getSharedConfiguration().getDomain(domainName);
if (domain == null) {
throw new IllegalArgumentException("Non-existent domain: " + domainName);
}
return domain.createDataContext(useSharedCache);
}
/**
* Creates a new DataContext that is not attached to the Cayenne stack.
*/
public DataContext() {
this((DataChannel) null, null);
}
/**
* Creates a DataContext with parent QueryEngine and a DataRowStore that should be
* used by the ObjectStore.
*
* @since 1.1
* @param parent parent QueryEngine used to communicate with the data source.
* @param objectStore ObjectStore used by DataContext.
* @deprecated since 1.2 - use {@link #DataContext(DataChannel, ObjectStore)}
* constructor instead. Note that DataDomain is both a DataChannel and a
* QueryEngine, so you may need to do a cast:
* new DataContext((DataChannel) domain, objectStore)
.
*/
public DataContext(QueryEngine parent, ObjectStore objectStore) {
this((DataChannel) parent, objectStore);
}
/**
* Creates a new DataContext with parent DataChannel and ObjectStore.
*
* @since 1.2
*/
public DataContext(DataChannel channel, ObjectStore objectStore) {
// use a setter to properly initialize EntityResolver
setChannel(channel);
this.setTransactionEventsEnabled(transactionEventsEnabledDefault);
// inject self as parent context
if (objectStore != null) {
this.objectStore = objectStore;
objectStore.setContext(this);
DataDomain domain = getParentDataDomain();
this.usingSharedSnaphsotCache = domain != null
&& objectStore.getDataRowCache() == domain.getSharedSnapshotCache();
}
}
/**
* Returns a map of user-defined properties associated with this DataContext.
*
* @since 1.2
*/
protected Map getUserProperties() {
// as not all users will take advantage of properties, creating the
// map on demand to keep DataContext lean...
if (userProperties == null) {
userProperties = new HashMap();
}
return userProperties;
}
/**
* Creates and returns a new child DataContext.
*
* @since 1.2
*/
public DataContext createChildDataContext() {
DataContextFactory factory = getParentDataDomain().getDataContextFactory();
// child ObjectStore should not have direct access to snapshot cache, so do not
// pass it in constructor.
ObjectStore objectStore = new ObjectStore();
DataContext child = factory != null ? factory
.createDataContext(this, objectStore) : new DataContext(
(DataChannel) this,
objectStore);
child.setValidatingObjectsOnCommit(isValidatingObjectsOnCommit());
child.usingSharedSnaphsotCache = isUsingSharedSnapshotCache();
return child;
}
/**
* Returns a user-defined property previously set via 'setUserProperty'. Note that it
* is a caller responsibility to synchronize access to properties.
*
* @since 1.2
*/
public Object getUserProperty(String key) {
return getUserProperties().get(key);
}
/**
* Sets a user-defined property. Note that it is a caller responsibility to
* synchronize access to properties.
*
* @since 1.2
*/
public void setUserProperty(String key, Object value) {
getUserProperties().put(key, value);
}
/**
* Returns parent QueryEngine object. In most cases returned object is an instance of
* DataDomain.
*
* @deprecated since 1.2. Use 'getParentDataDomain()' or 'getChannel()' instead.
*/
public QueryEngine getParent() {
return getParentDataDomain();
}
/**
* Sets direct parent of this DataContext.
*
* @deprecated since 1.2, use setChannel instead.
*/
public void setParent(QueryEngine parent) {
if (parent == null || parent instanceof DataChannel) {
setChannel((DataChannel) parent);
}
else {
throw new CayenneRuntimeException(
"Only parents that implement DataChannel are supported.");
}
}
/**
* Returns parent DataChannel, that is normally a DataDomain or another DataContext.
*
* @since 1.2
*/
public DataChannel getChannel() {
return channel;
}
/**
* @since 1.2
*/
public void setChannel(DataChannel channel) {
if (this.channel != channel) {
if (this.mergeHandler != null) {
this.mergeHandler.setActive(false);
}
this.entityResolver = null;
this.mergeHandler = null;
this.channel = channel;
if (channel != null) {
// cache entity resolver, as we have no idea how expensive it is to query
// it on the channel every time
this.entityResolver = channel.getEntityResolver();
EventManager eventManager = channel.getEventManager();
if (eventManager != null) {
this.mergeHandler = new DataContextMergeHandler(this);
// listen to our channel events...
// note that we must reset listener on channel switch, as there is no
// guarantee that a new channel uses the same EventManager.
EventUtil.listenForChannelEvents(channel, mergeHandler);
}
}
}
}
/**
* Returns a DataDomain used by this DataContext. DataDomain is looked up in the
* DataChannel hierarchy. If a channel is not a DataDomain or a DataContext, null is
* returned.
*
* @return DataDomain that is a direct or indirect parent of this DataContext in the
* DataChannel hierarchy.
* @since 1.1
*/
public DataDomain getParentDataDomain() {
awakeFromDeserialization();
if (channel == null) {
return null;
}
if (channel instanceof DataDomain) {
return (DataDomain) channel;
}
if (channel instanceof DataContext) {
return ((DataContext) channel).getParentDataDomain();
}
return null;
}
/**
* Sets a DataContextDelegate for this context. Delegate is notified of certain events
* in the DataContext lifecycle and can customize DataContext behavior.
*
* @since 1.1
*/
public void setDelegate(DataContextDelegate delegate) {
this.delegate = delegate;
}
/**
* Returns a delegate currently associated with this DataContext.
*
* @since 1.1
*/
public DataContextDelegate getDelegate() {
return delegate;
}
/**
* @return a delegate instance if it is initialized, or a shared noop implementation
* the context has no delegate. Useful to prevent extra null checks and
* conditional logic in the code.
* @since 1.1
*/
DataContextDelegate nonNullDelegate() {
return (delegate != null) ? delegate : NoopDelegate.noopDelegate;
}
/**
* Returns ObjectStore associated with this DataContext.
*/
public ObjectStore getObjectStore() {
return objectStore;
}
/**
* Returns true
if there are any modified, deleted or new objects
* registered with this DataContext, false
otherwise.
*/
public boolean hasChanges() {
return getObjectStore().hasChanges();
}
/**
* Returns a list of objects that are registered with this DataContext and have a
* state PersistenceState.NEW
*/
public Collection newObjects() {
return getObjectStore().objectsInState(PersistenceState.NEW);
}
/**
* Returns a list of objects that are registered with this DataContext and have a
* state PersistenceState.DELETED
*/
public Collection deletedObjects() {
return getObjectStore().objectsInState(PersistenceState.DELETED);
}
/**
* Returns a list of objects that are registered with this DataContext and have a
* state PersistenceState.MODIFIED
*/
public Collection modifiedObjects() {
return getObjectStore().objectsInState(PersistenceState.MODIFIED);
}
/**
* Returns a collection of all uncommitted registered objects.
*
* @since 1.2
*/
public Collection uncommittedObjects() {
int len = getObjectStore().registeredObjectsCount();
if (len == 0) {
return Collections.EMPTY_LIST;
}
// guess target collection size
Collection objects = new ArrayList(len > 100 ? len / 2 : len);
Iterator it = getObjectStore().getObjectIterator();
while (it.hasNext()) {
Persistent object = (Persistent) it.next();
int state = object.getPersistenceState();
if (state == PersistenceState.MODIFIED
|| state == PersistenceState.NEW
|| state == PersistenceState.DELETED) {
objects.add(object);
}
}
return objects;
}
/**
* Returns an object for a given ObjectId. When an object is not yet registered with
* this context's ObjectStore, the behavior of this method depends on whether ObjectId
* is permanent or temporary and whether a DataContext is a part of a nested context
* hierarchy or not. More specifically the following rules are applied in order:
*
* - If a matching registered object is found in this DataContext, it is
* immediately returned.
* - If a context is nested (i.e. it has another DataContext as its parent
* channel), an attempt is made to locate a registered object up the hierarchy chain,
* until it is found. Such object is transferred to this DataContext and returned to
* the caller.
* - If the ObjectId is temporary, null is returned; if it is permanent, a HOLLOW
* object (aka fault) is created and returned.
*
*
* @deprecated since 1.2 use 'localObject(id, null)'
*/
public DataObject registeredObject(ObjectId id) {
return (DataObject) localObject(id, null);
}
/**
* Returns a DataRow reflecting current, possibly uncommitted, object state.
*
* Warning: This method will return a partial snapshot if an object
* or one of its related objects that propagate their keys to this object have
* temporary ids. DO NOT USE this method if you expect a DataRow to represent a
* complete object state.
*
*
* @since 1.1
*/
public DataRow currentSnapshot(DataObject object) {
ObjEntity entity = getEntityResolver().lookupObjEntity(object);
// for a HOLLOW object return snapshot from cache
if (object.getPersistenceState() == PersistenceState.HOLLOW
&& object.getDataContext() != null) {
return getObjectStore().getSnapshot(object.getObjectId());
}
DataRow snapshot = new DataRow(10);
Iterator attributes = entity.getAttributeMap().entrySet().iterator();
while (attributes.hasNext()) {
Map.Entry entry = (Map.Entry) attributes.next();
String attrName = (String) entry.getKey();
ObjAttribute objAttr = (ObjAttribute) entry.getValue();
// processing compound attributes correctly
snapshot.put(objAttr.getDbAttributePath(), object
.readPropertyDirectly(attrName));
}
Iterator relationships = entity.getRelationshipMap().entrySet().iterator();
while (relationships.hasNext()) {
Map.Entry entry = (Map.Entry) relationships.next();
ObjRelationship rel = (ObjRelationship) entry.getValue();
// if target doesn't propagates its key value, skip it
if (rel.isSourceIndependentFromTargetChange()) {
continue;
}
Object targetObject = object.readPropertyDirectly(rel.getName());
if (targetObject == null) {
continue;
}
// if target is Fault, get id attributes from stored snapshot
// to avoid unneeded fault triggering
if (targetObject instanceof Fault) {
DataRow storedSnapshot = getObjectStore().getSnapshot(
object.getObjectId());
if (storedSnapshot == null) {
throw new CayenneRuntimeException(
"No matching objects found for ObjectId "
+ object.getObjectId()
+ ". Object may have been deleted externally.");
}
DbRelationship dbRel = (DbRelationship) rel.getDbRelationships().get(0);
Iterator joins = dbRel.getJoins().iterator();
while (joins.hasNext()) {
DbJoin join = (DbJoin) joins.next();
String key = join.getSourceName();
snapshot.put(key, storedSnapshot.get(key));
}
continue;
}
// target is resolved and we have an FK->PK to it,
// so extract it from target...
DataObject target = (DataObject) targetObject;
Map idParts = target.getObjectId().getIdSnapshot();
// this may happen in uncommitted objects - see the warning in the JavaDoc of
// this method.
if (idParts.isEmpty()) {
continue;
}
DbRelationship dbRel = (DbRelationship) rel.getDbRelationships().get(0);
Map fk = dbRel.srcFkSnapshotWithTargetSnapshot(idParts);
snapshot.putAll(fk);
}
// process object id map
// we should ignore any object id values if a corresponding attribute
// is a part of relationship "toMasterPK", since those values have been
// set above when db relationships where processed.
Map thisIdParts = object.getObjectId().getIdSnapshot();
if (thisIdParts != null) {
// put only those that do not exist in the map
Iterator idIterator = thisIdParts.entrySet().iterator();
while (idIterator.hasNext()) {
Map.Entry entry = (Map.Entry) idIterator.next();
Object nextKey = entry.getKey();
if (!snapshot.containsKey(nextKey)) {
snapshot.put(nextKey, entry.getValue());
}
}
}
return snapshot;
}
/**
* Creates a list of DataObjects local to this DataContext from a list of DataObjects
* coming from a different DataContext. This method is a way to map objects
* from one context into the other (as opposed to "synchronize"). This means that the
* state of modified objects will be reflected only if this context is a child of an
* original DataObject context. If it is a peer or parent, you won't see any
* uncommitted changes from the original context.
*
* Note that the objects in the list do not have to be of the same type or even from
* the same DataContext.
*
* @since 1.0.3
* @deprecated since 1.2 - use {@link #localObject(ObjectId, Persistent)} to specify
* how each local object should be handled.
*/
public List localObjects(List objects) {
List localObjects = new ArrayList(objects.size());
Iterator it = objects.iterator();
while (it.hasNext()) {
DataObject object = (DataObject) it.next();
if (object == null) {
throw new CayenneRuntimeException("Null object");
}
localObjects.add(localObject(object.getObjectId(), null));
}
return localObjects;
}
/**
* Converts a list of data rows to a list of DataObjects.
*
* @since 1.1
*/
public List objectsFromDataRows(
ObjEntity entity,
List dataRows,
boolean refresh,
boolean resolveInheritanceHierarchy) {
return new ObjectResolver(this, entity, refresh, resolveInheritanceHierarchy)
.synchronizedObjectsFromDataRows(dataRows);
}
/**
* Converts a list of DataRows to a List of DataObject registered with this
* DataContext. Internally calls
* {@link #objectsFromDataRows(ObjEntity,List,boolean,boolean)}.
*
* @since 1.1
* @see DataRow
* @see DataObject
*/
public List objectsFromDataRows(
Class objectClass,
List dataRows,
boolean refresh,
boolean resolveInheritanceHierarchy) {
ObjEntity entity = this.getEntityResolver().lookupObjEntity(objectClass);
if (entity == null) {
throw new CayenneRuntimeException("Unmapped Java class: " + objectClass);
}
return objectsFromDataRows(entity, dataRows, refresh, resolveInheritanceHierarchy);
}
/**
* Creates a DataObject from DataRow. This is a convenience shortcut to
* {@link #objectsFromDataRows(Class,java.util.List,boolean,boolean)}.
*
* @see DataRow
* @see DataObject
*/
public DataObject objectFromDataRow(
Class objectClass,
DataRow dataRow,
boolean refresh) {
List list = objectsFromDataRows(
objectClass,
Collections.singletonList(dataRow),
refresh,
true);
return (DataObject) list.get(0);
}
/**
* Instantiates new object and registers it with itself. Object class is determined
* from ObjEntity. Object class must have a default constructor.
*
* Note: preferred way to create new objects is via
* {@link #createAndRegisterNewObject(Class)}method. It works exactly the same way,
* but makes the application type-safe.
*
*
* @see #createAndRegisterNewObject(Class)
*/
public DataObject createAndRegisterNewObject(String objEntityName) {
ClassDescriptor descriptor = getEntityResolver()
.getClassDescriptor(objEntityName);
if (descriptor == null) {
throw new IllegalArgumentException("Invalid entity name: " + objEntityName);
}
DataObject dataObject;
try {
dataObject = (DataObject) descriptor.createObject();
}
catch (Exception ex) {
throw new CayenneRuntimeException("Error instantiating object.", ex);
}
// this will initialize to-many lists
descriptor.injectValueHolders(dataObject);
dataObject.setObjectId(new ObjectId(objEntityName));
dataObject.setDataContext(this);
dataObject.setPersistenceState(PersistenceState.NEW);
getObjectStore().recordObjectCreated(dataObject);
return dataObject;
}
/**
* Creates and registers new persistent object. This is an ObjectContext version of
* 'createAndRegisterNewObject'.
*
* @since 1.2
*/
public Persistent newObject(Class persistentClass) {
if (persistentClass == null) {
throw new NullPointerException("Null 'persistentClass'");
}
// TODO: only supports DataObject subclasses
if (!DataObject.class.isAssignableFrom(persistentClass)) {
throw new IllegalArgumentException(
this
+ ": this implementation of ObjectContext only supports full DataObjects. Class "
+ persistentClass
+ " is invalid.");
}
return createAndRegisterNewObject(persistentClass);
}
/**
* Instantiates new object and registers it with itself. Object class must have a
* default constructor.
*
* @since 1.1
*/
public DataObject createAndRegisterNewObject(Class objectClass) {
if (objectClass == null) {
throw new NullPointerException("DataObject class can't be null.");
}
ObjEntity entity = getEntityResolver().lookupObjEntity(objectClass);
if (entity == null) {
throw new IllegalArgumentException("Class is not mapped with Cayenne: "
+ objectClass.getName());
}
return createAndRegisterNewObject(entity.getName());
}
/**
* Registers a transient object with the context, recursively registering all
* transient DataObjects attached to this object via relationships.
*
* @param object new object that needs to be made persistent.
*/
public void registerNewObject(final DataObject object) {
if (object == null) {
throw new NullPointerException("Can't register null object.");
}
ObjEntity entity = getEntityResolver().lookupObjEntity(object);
if (entity == null) {
throw new IllegalArgumentException(
"Can't find ObjEntity for DataObject class: "
+ object.getClass().getName()
+ ", class is likely not mapped.");
}
// sanity check - maybe already registered
if (object.getObjectId() != null) {
if (object.getDataContext() == this) {
// already registered, just ignore
return;
}
else if (object.getDataContext() != null) {
throw new IllegalStateException(
"DataObject is already registered with another DataContext. "
+ "Try using 'localObjects()' instead.");
}
}
else {
object.setObjectId(new ObjectId(entity.getName()));
}
object.setDataContext(this);
object.setPersistenceState(PersistenceState.NEW);
getObjectStore().recordObjectCreated(object);
// now we need to find all arc changes, inject missing value holders and pull in
// all transient connected objects
ClassDescriptor descriptor = getEntityResolver().getClassDescriptor(
entity.getName());
if (descriptor == null) {
throw new IllegalArgumentException("Invalid entity name: " + entity.getName());
}
descriptor.visitProperties(new PropertyVisitor() {
public boolean visitCollectionArc(CollectionProperty property) {
property.injectValueHolder(object);
if (!property.isFault(object)) {
Iterator it = ((Collection) property.readProperty(object)).iterator();
while (it.hasNext()) {
Object target = it.next();
if (target instanceof DataObject) {
DataObject targetDO = (DataObject) target;
// make sure it is registered
registerNewObject(targetDO);
getObjectStore().recordArcCreated(
object,
targetDO.getObjectId(),
property.getName());
}
}
}
return true;
}
public boolean visitSingleObjectArc(SingleObjectArcProperty property) {
Object target = property.readPropertyDirectly(object);
if (target instanceof DataObject) {
DataObject targetDO = (DataObject) target;
// make sure it is registered
registerNewObject(targetDO);
getObjectStore().recordArcCreated(
object,
targetDO.getObjectId(),
property.getName());
}
return true;
}
public boolean visitProperty(Property property) {
return true;
}
});
}
/**
* Unregisters a Collection of DataObjects from the DataContext and the underlying
* ObjectStore. This operation also unsets DataContext and ObjectId for each object
* and changes its state to TRANSIENT.
*
* @see #invalidateObjects(Collection)
*/
public void unregisterObjects(Collection dataObjects) {
getObjectStore().objectsUnregistered(dataObjects);
}
/**
* "Invalidates" a Collection of DataObject. This operation would remove each object's
* snapshot from cache and change object's state to HOLLOW. On the next access to this
* object, it will be refetched.
*
* @see #unregisterObjects(Collection)
*/
public void invalidateObjects(Collection dataObjects) {
getObjectStore().objectsInvalidated(dataObjects);
}
/**
* Schedules all objects in the collection for deletion on the next commit of this
* DataContext. Object's persistence state is changed to PersistenceState.DELETED;
* objects related to this object are processed according to delete rules, i.e.
* relationships can be unset ("nullify" rule), deletion operation is cascaded
* (cascade rule).
*
* "Nullify" delete rule side effect: passing a collection representing
* to-many relationship with nullify delete rule may result in objects being removed
* from collection.
*
*
* @since 1.2
*/
public void deleteObjects(Collection objects) {
if (objects.isEmpty()) {
return;
}
// clone object list... this maybe a relationship collection with nullify delete
// rule, so modifying
Iterator it = new ArrayList(objects).iterator();
while (it.hasNext()) {
DataObject object = (DataObject) it.next();
deleteObject(object);
}
}
/**
* Schedules an object for deletion on the next commit of this DataContext. Object's
* persistence state is changed to PersistenceState.DELETED; objects related to this
* object are processed according to delete rules, i.e. relationships can be unset
* ("nullify" rule), deletion operation is cascaded (cascade rule).
*
* @param object a persistent object that we want to delete.
* @throws DeleteDenyException if a DENY delete rule is applicable for object
* deletion.
* @throws NullPointerException if object is null.
*/
public void deleteObject(Persistent object) throws DeleteDenyException {
new DataContextDeleteAction(this).performDelete(object);
}
/**
* Refetches object data for ObjectId. This method is used internally by Cayenne to
* resolve objects in state PersistenceState.HOLLOW
. It can also be
* used to refresh certain objects.
*
* @throws CayenneRuntimeException if object id doesn't match any records, or if there
* is more than one object is fetched.
*/
public DataObject refetchObject(ObjectId oid) {
if (oid == null) {
throw new NullPointerException("Null ObjectId");
}
if (oid.isTemporary()) {
throw new CayenneRuntimeException("Can't refetch ObjectId "
+ oid
+ ", as it is a temporary id.");
}
synchronized (getObjectStore()) {
DataObject object = (DataObject) objectStore.getNode(oid);
// clean up any cached data for this object
if (object != null) {
this.invalidateObjects(Collections.singleton(object));
}
}
DataObject object = (DataObject) DataObjectUtils.objectForQuery(
this,
new ObjectIdQuery(oid));
if (object == null) {
throw new CayenneRuntimeException(
"Refetch failure: no matching objects found for ObjectId " + oid);
}
return object;
}
/**
* Returns a DataNode that should handle queries for all DataMap components.
*
* @since 1.1
* @deprecated since 1.2 DataContext's QueryEngine implementation is replaced by
* DataChannel. Use "getParentDataDomain().lookupDataNode(..)".
*/
public DataNode lookupDataNode(DataMap dataMap) {
DataDomain domain = getParentDataDomain();
if (domain == null) {
throw new CayenneRuntimeException(
"DataContext is not attached to a DataDomain ");
}
return domain.lookupDataNode(dataMap);
}
/**
* If the parent channel is a DataContext, reverts local changes to make this context
* look like the parent, if the parent channel is a DataDomain, reverts all changes.
*
* @since 1.2
*/
// TODO: Andrus, 1/19/2006: implement for nested DataContexts
public void rollbackChangesLocally() {
if (getChannel() instanceof DataDomain) {
rollbackChanges();
}
else {
throw new CayenneRuntimeException("Implementation pending.");
}
}
/**
* Reverts any changes that have occurred to objects registered with DataContext; also
* performs cascading rollback of all parent DataContexts.
*/
public void rollbackChanges() {
if (objectStore.hasChanges()) {
GraphDiff diff = getObjectStore().getChanges();
getObjectStore().objectsRolledBack();
if (channel != null) {
channel.onSync(this, null, DataChannel.ROLLBACK_CASCADE_SYNC);
}
fireDataChannelRolledback(this, diff);
}
}
/**
* "Flushes" the changes to the parent {@link DataChannel}. If the parent channel is
* a DataContext, it updates its objects with this context's changes, without a
* database update. If it is a DataDomain (the most common case), the changes are
* written to the database. To cause cascading commit all the way to the database, one
* must use {@link #commitChanges()}.
*
* @since 1.2
* @see #commitChanges()
*/
public void commitChangesToParent() {
flushToParent(false);
}
/**
* @deprecated Since 1.2, use {@link #commitChanges()} instead.
*/
public void commitChanges(Level logLevel) throws CayenneRuntimeException {
commitChanges();
}
/**
* Synchronizes object graph with the database. Executes needed insert, update and
* delete queries (generated internally).
*/
public void commitChanges() throws CayenneRuntimeException {
flushToParent(true);
}
/**
* Returns EventManager associated with the ObjectStore.
*
* @since 1.2
*/
public EventManager getEventManager() {
return channel != null ? channel.getEventManager() : null;
}
/**
* An implementation of a {@link DataChannel} method that is used by child contexts to
* synchronize state with this context. Not intended for direct use.
*
* @since 1.2
*/
public GraphDiff onSync(
ObjectContext originatingContext,
GraphDiff changes,
int syncType) {
// sync client changes
switch (syncType) {
case DataChannel.ROLLBACK_CASCADE_SYNC:
return onContextRollback(originatingContext);
case DataChannel.FLUSH_NOCASCADE_SYNC:
return onContextFlush(originatingContext, changes, false);
case DataChannel.FLUSH_CASCADE_SYNC:
return onContextFlush(originatingContext, changes, true);
default:
throw new CayenneRuntimeException("Unrecognized SyncMessage type: "
+ syncType);
}
}
GraphDiff onContextRollback(ObjectContext originatingContext) {
rollbackChanges();
return (channel != null) ? channel.onSync(
this,
null,
DataChannel.ROLLBACK_CASCADE_SYNC) : new CompoundDiff();
}
GraphDiff onContextFlush(
ObjectContext originatingContext,
GraphDiff changes,
boolean cascade) {
if (this != originatingContext && changes != null) {
changes.apply(new ChildDiffLoader(this));
fireDataChannelChanged(originatingContext, changes);
}
return (cascade) ? flushToParent(true) : new CompoundDiff();
}
/**
* Synchronizes with the parent channel, performing a flush or a commit.
*
* @since 1.2
*/
GraphDiff flushToParent(boolean cascade) {
if (this.getChannel() == null) {
throw new CayenneRuntimeException(
"Cannot commit changes - channel is not set.");
}
int syncType = cascade
? DataChannel.FLUSH_CASCADE_SYNC
: DataChannel.FLUSH_NOCASCADE_SYNC;
// prevent multiple commits occuring simulteneously
synchronized (getObjectStore()) {
DataContextFlushEventHandler eventHandler = null;
ObjectStoreGraphDiff changes = getObjectStore().getChanges();
boolean noop = isValidatingObjectsOnCommit()
? changes.validateAndCheckNoop()
: changes.isNoop();
if (noop) {
// need to clear phantom changes
getObjectStore().postprocessAfterPhantomCommit();
return new CompoundDiff();
}
if (isTransactionEventsEnabled()) {
eventHandler = new DataContextFlushEventHandler(this);
eventHandler.registerForDataContextEvents();
fireWillCommit();
}
try {
GraphDiff returnChanges = getChannel().onSync(this, changes, syncType);
getObjectStore().postprocessAfterCommit(returnChanges);
// this is a legacy event ... will deprecate in 2.0
fireTransactionCommitted();
// this event is caught by peer nested DataContexts to synchronize the
// state
fireDataChannelCommitted(this, changes);
// this event is caught by child DataContexts to update temporary
// ObjectIds with permanent
if (!returnChanges.isNoop()) {
fireDataChannelCommitted(getChannel(), returnChanges);
}
return returnChanges;
}
// "catch" is needed to unwrap OptimisticLockExceptions
catch (CayenneRuntimeException ex) {
fireTransactionRolledback();
Throwable unwound = Util.unwindException(ex);
if (unwound instanceof CayenneRuntimeException) {
throw (CayenneRuntimeException) unwound;
}
else {
throw new CayenneRuntimeException("Commit Exception", unwound);
}
}
finally {
if (isTransactionEventsEnabled()) {
eventHandler.unregisterFromDataContextEvents();
}
}
}
}
/**
* Performs a single database select query returning result as a ResultIterator. It is
* caller's responsibility to explicitly close the ResultIterator. A failure to do so
* will result in a database connection not being released. Another side effect of an
* open ResultIterator is that an internal Cayenne transaction that originated in this
* method stays open until the iterator is closed. So users should normally close the
* iterator within the same thread that opened it.
*/
public ResultIterator performIteratedQuery(Query query) throws CayenneException {
if (Transaction.getThreadTransaction() != null) {
return internalPerformIteratedQuery(query);
}
else {
// manually manage a transaction, so that a ResultIterator wrapper could close
// it when it is done.
Transaction tx = getParentDataDomain().createTransaction();
Transaction.bindThreadTransaction(tx);
ResultIterator result;
try {
result = internalPerformIteratedQuery(query);
}
catch (Exception e) {
Transaction.bindThreadTransaction(null);
tx.setRollbackOnly();
throw new CayenneException(e);
}
finally {
// note: we are keeping the transaction bound to the current thread on
// success - iterator will unbind it. Unsetting a transaction here would
// result in some strangeness, at least on Ingres
if (tx.getStatus() == Transaction.STATUS_MARKED_ROLLEDBACK) {
try {
tx.rollback();
}
catch (Exception rollbackEx) {
}
}
}
return new TransactionResultIteratorDecorator(result, tx);
}
}
/**
* Runs an iterated query in transactional context provided by the caller.
*
* @since 1.2
*/
ResultIterator internalPerformIteratedQuery(Query query) throws CayenneException {
// note that for now DataChannel API does not support cursors (aka
// ResultIterator), so we have to go directly to the DataDomain.
IteratedSelectObserver observer = new IteratedSelectObserver();
getParentDataDomain().performQueries(Collections.singletonList(query), observer);
return observer.getResultIterator();
}
/**
* Executes a query returning a generic response.
*
* @since 1.2
*/
public QueryResponse performGenericQuery(Query query) {
query = nonNullDelegate().willPerformGenericQuery(this, query);
if (query == null) {
return new GenericResponse();
}
if (this.getChannel() == null) {
throw new CayenneRuntimeException(
"Can't run query - parent DataChannel is not set.");
}
return onQuery(this, query);
}
/**
* Performs a single selecting query. Various query setting control the behavior of
* this method and the results returned:
*
* - Query caching policy defines whether the results are retrieved from cache or
* fetched from the database. Note that queries that use caching must have a name that
* is used as a caching key.
* - Query refreshing policy controls whether to refresh existing data objects and
* ignore any cached values.
* - Query data rows policy defines whether the result should be returned as
* DataObjects or DataRows.
*
*
* Since 1.2 takes any Query parameter, not just GenericSelectQuery
*
*
* @return A list of DataObjects or a DataRows, depending on the value returned by
* {@link QueryMetadata#isFetchingDataRows()}.
*/
public List performQuery(Query query) {
query = nonNullDelegate().willPerformQuery(this, query);
if (query == null) {
return new ArrayList(1);
}
// this filter should go away when we remove deprecated api from the Delegate.
query = filterThroughDelegateDeprecated(query);
if (query == null) {
return new ArrayList(1);
}
List result = onQuery(this, query).firstList();
return result != null ? result : new ArrayList(1);
}
/**
* An implementation of a {@link DataChannel} method that is used by child contexts to
* execute queries. Not intended for direct use.
*
* @since 1.2
*/
public QueryResponse onQuery(ObjectContext context, Query query) {
return new DataContextQueryAction(this, context, query).execute();
}
/**
* Performs a single database query that does not select rows. Returns an array of
* update counts.
*
* @since 1.1
*/
public int[] performNonSelectingQuery(Query query) {
int[] count = performGenericQuery(query).firstUpdateCount();
return count != null ? count : new int[0];
}
/**
* Performs a named mapped query that does not select rows. Returns an array of update
* counts.
*
* @since 1.1
*/
public int[] performNonSelectingQuery(String queryName) {
return performNonSelectingQuery(new NamedQuery(queryName));
}
/**
* Performs a named mapped non-selecting query using a map of parameters. Returns an
* array of update counts.
*
* @since 1.1
*/
public int[] performNonSelectingQuery(String queryName, Map parameters) {
return performNonSelectingQuery(new NamedQuery(queryName, parameters));
}
/**
* Executes all queries in collection.
*
* @deprecated since 1.2 DataContext's QueryEngine implementation is replaced by
* DataChannel.
*/
public void performQueries(Collection queries, OperationObserver callback) {
// filter through the delegate
List finalQueries = new ArrayList(queries.size());
Iterator it = queries.iterator();
while (it.hasNext()) {
Query query = (Query) it.next();
query = filterThroughDelegateDeprecated(query);
if (query != null) {
finalQueries.add(query);
}
}
if (!finalQueries.isEmpty()) {
getParentDataDomain().performQueries(queries, callback);
}
}
/**
* @since 1.2
* @deprecated since 1.2
*/
// deprecated code is extracted in a separate method to avoid Eclipse warnings...
Query filterThroughDelegateDeprecated(Query query) {
if (query instanceof org.objectstyle.cayenne.query.GenericSelectQuery) {
org.objectstyle.cayenne.query.GenericSelectQuery genericSelect = (org.objectstyle.cayenne.query.GenericSelectQuery) query;
return nonNullDelegate().willPerformSelect(this, genericSelect);
}
return query;
}
/**
* Binds provided transaction to the current thread, and then runs queries.
*
* @since 1.1
* @deprecated since 1.2. Use Transaction.bindThreadTransaction(..) to provide custom
* transactions, besides DataContext's QueryEngine implementation is
* replaced by DataChannel.
*/
public void performQueries(
Collection queries,
OperationObserver callback,
Transaction transaction) {
Transaction.bindThreadTransaction(transaction);
try {
performQueries(queries, callback);
}
finally {
Transaction.bindThreadTransaction(null);
}
}
/**
* Performs prefetching. Prefetching would resolve a set of relationships for a list
* of DataObjects in the most optimized way (preferrably in a single query per
* relationship).
*
* WARNING: Currently supports only "one-step" to one relationships. This is an
* arbitrary limitation and will be removed eventually.
*
*
* @deprecated Since 1.2. This is a utility method that handles a very specific case.
* It shouldn't be in DataContext.
*/
public void prefetchRelationships(SelectQuery query, List objects) {
QueryMetadata metadata = query.getMetaData(getEntityResolver());
Collection prefetches = metadata.getPrefetchTree() != null ? query
.getPrefetchTree()
.nonPhantomNodes() : Collections.EMPTY_LIST;
if (objects == null || objects.size() == 0 || prefetches.size() == 0) {
return;
}
ObjEntity entity = metadata.getObjEntity();
Iterator prefetchesIt = prefetches.iterator();
while (prefetchesIt.hasNext()) {
PrefetchTreeNode prefetch = (PrefetchTreeNode) prefetchesIt.next();
String path = prefetch.getPath();
if (path.indexOf('.') >= 0) {
throw new CayenneRuntimeException("Only one-step relationships are "
+ "supported at the moment, this will be fixed soon. "
+ "Unsupported path : "
+ path);
}
ObjRelationship relationship = (ObjRelationship) entity.getRelationship(path);
if (relationship == null) {
throw new CayenneRuntimeException("Invalid relationship: " + path);
}
if (relationship.isToMany()) {
throw new CayenneRuntimeException(
"Only to-one relationships are supported at the moment. "
+ "Can't prefetch to-many: "
+ path);
}
org.objectstyle.cayenne.access.util.PrefetchHelper.resolveToOneRelations(
this,
objects,
path);
}
}
/**
* Returns a list of objects or DataRows for a named query stored in one of the
* DataMaps. Internally Cayenne uses a caching policy defined in the named query. If
* refresh flag is true, a refresh is forced no matter what the caching policy is.
*
* @param queryName a name of a GenericSelectQuery defined in one of the DataMaps. If
* no such query is defined, this method will throw a
* CayenneRuntimeException.
* @param expireCachedLists A flag that determines whether refresh of cached lists
* is required in case a query uses caching.
* @since 1.1
*/
public List performQuery(String queryName, boolean expireCachedLists) {
return performQuery(queryName, Collections.EMPTY_MAP, expireCachedLists);
}
/**
* Returns a list of objects or DataRows for a named query stored in one of the
* DataMaps. Internally Cayenne uses a caching policy defined in the named query. If
* refresh flag is true, a refresh is forced no matter what the caching policy is.
*
* @param queryName a name of a GenericSelectQuery defined in one of the DataMaps. If
* no such query is defined, this method will throw a
* CayenneRuntimeException.
* @param parameters A map of parameters to use with stored query.
* @param expireCachedLists A flag that determines whether refresh of cached lists
* is required in case a query uses caching.
* @since 1.1
*/
public List performQuery(String queryName, Map parameters, boolean expireCachedLists) {
NamedQuery query = new NamedQuery(queryName, parameters);
query.setForceNoCache(expireCachedLists);
return performQuery(query);
}
/**
* Returns EntityResolver. EntityResolver can be null if DataContext has not been
* attached to an DataChannel.
*/
public EntityResolver getEntityResolver() {
awakeFromDeserialization();
return entityResolver;
}
/**
* Sets default for posting transaction events by new DataContexts.
*/
public static void setTransactionEventsEnabledDefault(boolean flag) {
transactionEventsEnabledDefault = flag;
}
/**
* Enables or disables posting of transaction events by this DataContext.
*/
public void setTransactionEventsEnabled(boolean flag) {
this.transactionEventsEnabled = flag;
}
public boolean isTransactionEventsEnabled() {
return this.transactionEventsEnabled;
}
/**
* Returns true
if the ObjectStore uses shared cache of a parent
* DataDomain.
*
* @since 1.1
*/
public boolean isUsingSharedSnapshotCache() {
return usingSharedSnaphsotCache;
}
/**
* Returns whether this DataContext performs object validation before commit is
* executed.
*
* @since 1.1
*/
public boolean isValidatingObjectsOnCommit() {
return validatingObjectsOnCommit;
}
/**
* Sets the property defining whether this DataContext should perform object
* validation before commit is executed.
*
* @since 1.1
*/
public void setValidatingObjectsOnCommit(boolean flag) {
this.validatingObjectsOnCommit = flag;
}
/**
* @deprecated since 1.2. Use 'getEntityResolver().getDataMaps()' instead.
*/
public Collection getDataMaps() {
return (getEntityResolver() != null)
? getEntityResolver().getDataMaps()
: Collections.EMPTY_LIST;
}
void fireWillCommit() {
// post event: WILL_COMMIT
if (this.transactionEventsEnabled) {
DataContextEvent commitChangesEvent = new DataContextEvent(this);
getEventManager().postEvent(commitChangesEvent, DataContext.WILL_COMMIT);
}
}
void fireTransactionRolledback() {
// post event: DID_ROLLBACK
if ((this.transactionEventsEnabled)) {
DataContextEvent commitChangesEvent = new DataContextEvent(this);
getEventManager().postEvent(commitChangesEvent, DataContext.DID_ROLLBACK);
}
}
void fireTransactionCommitted() {
// old-style event
if ((this.transactionEventsEnabled)) {
DataContextEvent commitChangesEvent = new DataContextEvent(this);
getEventManager().postEvent(commitChangesEvent, DataContext.DID_COMMIT);
}
}
/**
* @since 1.2
*/
void fireDataChannelCommitted(Object postedBy, GraphDiff changes) {
EventManager manager = getEventManager();
if (manager != null) {
GraphEvent e = new GraphEvent(this, postedBy, changes);
manager.postEvent(e, DataChannel.GRAPH_FLUSHED_SUBJECT);
}
}
/**
* @since 1.2
*/
void fireDataChannelRolledback(Object postedBy, GraphDiff changes) {
EventManager manager = getEventManager();
if (manager != null) {
GraphEvent e = new GraphEvent(this, postedBy, changes);
manager.postEvent(e, DataChannel.GRAPH_ROLLEDBACK_SUBJECT);
}
}
/**
* @since 1.2
*/
void fireDataChannelChanged(Object postedBy, GraphDiff changes) {
EventManager manager = getEventManager();
if (manager != null) {
GraphEvent e = new GraphEvent(this, postedBy, changes);
manager.postEvent(e, DataChannel.GRAPH_CHANGED_SUBJECT);
}
}
// ---------------------------------------------
// Serialization Support
// ---------------------------------------------
private void writeObject(ObjectOutputStream out) throws IOException {
out.defaultWriteObject();
// If the "parent" of this datacontext is a DataDomain, then just write the
// name of it. Then when deserialization happens, we can get back the DataDomain
// by name, from the shared configuration (which will either load it if need be,
// or return an existing one.
if (this.channel == null && this.lazyInitParentDomainName != null) {
out.writeObject(lazyInitParentDomainName);
}
else if (this.channel instanceof DataDomain) {
DataDomain domain = (DataDomain) this.channel;
out.writeObject(domain.getName());
}
else {
// Hope that whatever this.parent is, that it is Serializable
out.writeObject(this.channel);
}
// Serialize local snapshots cache
if (!isUsingSharedSnapshotCache()) {
out.writeObject(objectStore.getDataRowCache());
}
}
// serialization support
private void readObject(ObjectInputStream in) throws IOException,
ClassNotFoundException {
// 1. read non-transient properties
in.defaultReadObject();
// 2. read parent or its name
Object value = in.readObject();
if (value instanceof DataChannel) {
// A real QueryEngine object - use it
this.channel = (DataChannel) value;
}
else if (value instanceof String) {
// The name of a DataDomain - use it
this.lazyInitParentDomainName = (String) value;
}
else {
throw new CayenneRuntimeException(
"Parent attribute of DataContext was neither a QueryEngine nor "
+ "the name of a valid DataDomain:"
+ value);
}
// 3. Deserialize local snapshots cache
if (!isUsingSharedSnapshotCache()) {
DataRowStore cache = (DataRowStore) in.readObject();
objectStore.setDataRowCache(cache);
}
// CayenneDataObjects have a transient datacontext
// because at deserialize time the datacontext may need to be different
// than the one at serialize time (for programmer defined reasons).
// So, when a dataobject is resurrected because it's datacontext was
// serialized, it will then set the objects datacontext to the correctone
// If deserialized "otherwise", it will not have a datacontext (good)
synchronized (getObjectStore()) {
Iterator it = objectStore.getObjectIterator();
while (it.hasNext()) {
DataObject object = (DataObject) it.next();
object.setDataContext(this);
}
}
}
// Re-attaches itself to the parent domain with previously stored name.
//
// TODO: Andrus 11/7/2005 - this is one of the places where Cayenne
// serialization relies on shared config... This is bad. We need some
// sort of thread-local solution that would allow to use an alternative configuration.
//
private final void awakeFromDeserialization() {
if (channel == null && lazyInitParentDomainName != null) {
// call a setter to ensure EntityResolver is extracted from channel
setChannel(Configuration.getSharedConfiguration().getDomain(
lazyInitParentDomainName));
}
}
/**
* Resolves object fault if needed. If a property is not null, it is assumed that the
* object will be modified, so object snapshot is retained and object state is
* changed.
*
* @since 1.2
*/
public void prepareForAccess(Persistent object, String property) {
if (object.getPersistenceState() == PersistenceState.HOLLOW) {
if (!(object instanceof DataObject)) {
throw new CayenneRuntimeException("Can only resolve DataObjects. Got: "
+ object);
}
getObjectStore().resolveHollow((DataObject) object);
if (object.getPersistenceState() != PersistenceState.COMMITTED) {
String state = PersistenceState.persistenceStateName(object
.getPersistenceState());
// TODO: andrus 4/13/2006, modified and deleted states are possible due to
// a race condition, should we handle them here?
throw new FaultFailureException(
"Error resolving fault for ObjectId: "
+ object.getObjectId()
+ " and state ("
+ state
+ "). Possible cause - matching row is missing from the database.");
}
}
}
/**
* Retains DataObject snapshot and changes its state if needed.
*
* @since 1.2
*/
public void propertyChanged(
Persistent object,
String property,
Object oldValue,
Object newValue) {
if (object.getPersistenceState() == PersistenceState.COMMITTED) {
getObjectStore().registerDiff(object, null);
}
}
/**
* Returns this context's ObjectStore.
*
* @since 1.2
*/
public GraphManager getGraphManager() {
return objectStore;
}
/**
* Returns an object local to this DataContext and matching the ObjectId. If
* prototype
is not null, local object is refreshed with the prototype
* values.
*
* In case you pass a non-null second parameter, you are responsible for setting
* correct persistence state of the returned local object, as generally there is no
* way for Cayenne to determine the resulting local object state.
*
* @since 1.2
*/
public Persistent localObject(ObjectId id, Persistent prototype) {
// ****** Warning: when changing the code below, don't forget to change
// CayenneContext's implementation which right now relies on copy/paste "reuse"
if (id == null) {
throw new IllegalArgumentException("Null ObjectId");
}
// note that per-object ClassDescriptor lookup is needed as even if all
// objects where fetched as a part of the same query, as they may belong to
// different subclasses
ClassDescriptor descriptor = getEntityResolver().getClassDescriptor(
id.getEntityName());
Persistent cachedObject = (Persistent) getGraphManager().getNode(id);
// merge into an existing object
if (cachedObject != null) {
int state = cachedObject.getPersistenceState();
// TODO: Andrus, 1/24/2006 implement smart merge for modified objects...
if (cachedObject != prototype
&& state != PersistenceState.MODIFIED
&& state != PersistenceState.DELETED) {
descriptor.injectValueHolders(cachedObject);
if (prototype != null
&& prototype.getPersistenceState() != PersistenceState.HOLLOW) {
descriptor.shallowMerge(prototype, cachedObject);
if (state == PersistenceState.HOLLOW) {
cachedObject.setPersistenceState(PersistenceState.COMMITTED);
}
}
}
return cachedObject;
}
// create and merge into a new object
else {
Persistent localObject = (Persistent) descriptor.createObject();
localObject.setObjectContext(this);
localObject.setObjectId(id);
getGraphManager().registerNode(id, localObject);
if (prototype != null
&& prototype.getPersistenceState() != PersistenceState.HOLLOW) {
localObject.setPersistenceState(PersistenceState.COMMITTED);
descriptor.injectValueHolders(localObject);
descriptor.shallowMerge(prototype, localObject);
}
else {
localObject.setPersistenceState(PersistenceState.HOLLOW);
}
return localObject;
}
}
}