All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.apache.cayenne.access.DataRowUtils Maven / Gradle / Ivy

There is a newer version: 2.0.4
Show newest version
/*****************************************************************
 *   Licensed to the Apache Software Foundation (ASF) under one
 *  or more contributor license agreements.  See the NOTICE file
 *  distributed with this work for additional information
 *  regarding copyright ownership.  The ASF licenses this file
 *  to you under the Apache License, Version 2.0 (the
 *  "License"); you may not use this file except in compliance
 *  with the License.  You may obtain a copy of the License at
 *
 *    http://www.apache.org/licenses/LICENSE-2.0
 *
 *  Unless required by applicable law or agreed to in writing,
 *  software distributed under the License is distributed on an
 *  "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
 *  KIND, either express or implied.  See the License for the
 *  specific language governing permissions and limitations
 *  under the License.
 ****************************************************************/


package org.apache.cayenne.access;

import java.util.Iterator;
import java.util.Map;

import org.apache.cayenne.DataObject;
import org.apache.cayenne.DataRow;
import org.apache.cayenne.Fault;
import org.apache.cayenne.ObjectId;
import org.apache.cayenne.PersistenceState;
import org.apache.cayenne.ValueHolder;
import org.apache.cayenne.map.DbRelationship;
import org.apache.cayenne.map.ObjAttribute;
import org.apache.cayenne.map.ObjEntity;
import org.apache.cayenne.map.ObjRelationship;
import org.apache.cayenne.util.Util;

/**
 * DataRowUtils contains a number of static methods to work with DataRows. This is a
 * helper class for DataContext and ObjectStore.
 * 
 * @author Andrus Adamchik
 * @since 1.1
 */
class DataRowUtils {

    /**
     * Merges changes reflected in snapshot map to the object. Changes made to attributes
     * and to-one relationships will be merged. In case an object is already modified,
     * modified properties will not be overwritten.
     */
    static void mergeObjectWithSnapshot(
            ObjEntity entity,
            DataObject object,
            DataRow snapshot) {

        int state = object.getPersistenceState();

        if (entity.isReadOnly() || state == PersistenceState.HOLLOW) {
            refreshObjectWithSnapshot(entity, object, snapshot, true);
        }
        else if (state != PersistenceState.COMMITTED) {
            forceMergeWithSnapshot(entity, object, snapshot);
        }
        else {
            // do not invalidate to-many relationships, since they might have
            // just been prefetched...
            refreshObjectWithSnapshot(entity, object, snapshot, false);
        }
    }

    /**
     * Replaces all object attribute values with snapshot values. Sets object state to
     * COMMITTED, unless the snapshot is partial in which case the state is set to HOLLOW
     */
    static void refreshObjectWithSnapshot(
            ObjEntity objEntity,
            DataObject object,
            DataRow snapshot,
            boolean invalidateToManyRelationships) {

        boolean isPartialSnapshot = false;

        Map attrMap = objEntity.getAttributeMap();
        if (!attrMap.isEmpty()) {
            Iterator it = attrMap.entrySet().iterator();

            while (it.hasNext()) {
                Map.Entry entry = (Map.Entry) it.next();

                ObjAttribute attr = (ObjAttribute) entry.getValue();
                String dbAttrPath = attr.getDbAttributePath();

                Object value = snapshot.get(dbAttrPath);
                object.writePropertyDirectly(attr.getName(), value);

                // note that a check "snaphsot.get(..) == null" would be incorrect in this
                // case, as NULL value is entirely valid; still save a map lookup by
                // checking for the null value first
                if (value == null && !snapshot.containsKey(dbAttrPath)) {
                    isPartialSnapshot = true;
                }
            }
        }

        Map rMap = objEntity.getRelationshipMap();
        if (!rMap.isEmpty()) {
            Iterator it = rMap.entrySet().iterator();
            while (it.hasNext()) {

                Map.Entry e = (Map.Entry) it.next();
                ObjRelationship rel = (ObjRelationship) e.getValue();

                if (rel.isToMany()) {

                    // "to many" relationships have no information to collect from
                    // snapshot initialize a new empty list if requested, but otherwise
                    // ignore snapshot data

                    Object toManyList = object.readPropertyDirectly(rel.getName());

                    if (toManyList == null) {
                        object.writePropertyDirectly(rel.getName(), Fault
                                .getToManyFault());
                    }
                    else if (invalidateToManyRelationships
                            && toManyList instanceof ValueHolder) {
                        ((ValueHolder) toManyList).invalidate();
                    }
                }
                else {
                    // set a shared fault to indicate any kind of unresolved to-one
                    object.writePropertyDirectly(rel.getName(), Fault.getToOneFault());
                }
            }
        }

        object.setPersistenceState(isPartialSnapshot
                ? PersistenceState.HOLLOW
                : PersistenceState.COMMITTED);
    }

    static void forceMergeWithSnapshot(
            ObjEntity entity,
            DataObject object,
            DataRow snapshot) {

        DataContext context = object.getDataContext();
        ObjectDiff diff = (ObjectDiff) context
                .getObjectStore()
                .getChangesByObjectId()
                .get(object.getObjectId());

        // attributes
        Map attrMap = entity.getAttributeMap();
        Iterator it = attrMap.entrySet().iterator();
        while (it.hasNext()) {
            Map.Entry entry = (Map.Entry) it.next();

            String attrName = (String) entry.getKey();
            ObjAttribute attr = (ObjAttribute) entry.getValue();

            // processing compound attributes correctly
            String dbAttrPath = attr.getDbAttributePath();

            // supports merging of partial snapshots...
            // check for null is cheaper than double lookup
            // for a key... so check for partial snapshot
            // only if the value is null
            Object newVal = snapshot.get(dbAttrPath);
            if (newVal == null && !snapshot.containsKey(dbAttrPath)) {
                continue;
            }

            Object curVal = object.readPropertyDirectly(attrName);
            Object oldVal = diff != null ? diff.getSnapshotValue(attrName) : null;

            // if value not modified, update it from snapshot,
            // otherwise leave it alone
            if (Util.nullSafeEquals(curVal, oldVal)
                    && !Util.nullSafeEquals(newVal, curVal)) {
                object.writePropertyDirectly(attrName, newVal);
            }
        }

        // merge to-one relationships
        Iterator rit = entity.getRelationships().iterator();
        while (rit.hasNext()) {
            ObjRelationship rel = (ObjRelationship) rit.next();
            
            // !isToPK check is for handling one-to-one rels pointing to FK... this
            // feature will require more work in the future
            if (rel.isToMany() || !rel.isToPK()) {
                continue;
            }

            // TODO: will this work for flattened, how do we save snapshots for
            // them?

            // if value not modified, update it from snapshot,
            // otherwise leave it alone
            if (!isToOneTargetModified(rel, object, diff)) {

                DbRelationship dbRelationship = (DbRelationship) rel
                        .getDbRelationships()
                        .get(0);

                ObjectId id = snapshot.createTargetObjectId(
                        rel.getTargetEntityName(),
                        dbRelationship);

                if (diff == null
                        || !diff.containsArcSnapshot(rel.getName())
                        || !Util.nullSafeEquals(id, diff.getArcSnapshotValue(rel
                                .getName()))) {

                    Object target;
                    if (id == null) {
                        target = null;
                    }
                    else {
                        // if inheritance is involved, we can't use 'localObject' .. must
                        // turn to fault instead
                        ObjEntity targetEntity = (ObjEntity) rel.getTargetEntity();
                        if (context.getEntityResolver().lookupInheritanceTree(
                                targetEntity) != null) {
                            target = Fault.getToOneFault();
                        }
                        else {
                            target = context.localObject(id, null);
                        }
                    }

                    object.writeProperty(rel.getName(), target);
                }
            }
        }
    }

    /**
     * Checks if an object has its to-one relationship target modified in memory.
     */
    static boolean isToOneTargetModified(
            ObjRelationship relationship,
            DataObject object,
            ObjectDiff diff) {

        if (object.getPersistenceState() != PersistenceState.MODIFIED || diff == null) {
            return false;
        }

        Object targetObject = object.readPropertyDirectly(relationship.getName());
        if (targetObject instanceof Fault) {
            return false;
        }

        DataObject toOneTarget = (DataObject) targetObject;
        ObjectId currentId = (toOneTarget != null) ? toOneTarget.getObjectId() : null;

        // if ObjectId is temporary, target is definitely modified...
        // this would cover NEW objects (what are the other cases of temp id??)
        if (currentId != null && currentId.isTemporary()) {
            return true;
        }

        if (!diff.containsArcSnapshot(relationship.getName())) {
            return false;
        }

        ObjectId targetId = diff.getArcSnapshotValue(relationship.getName());
        return !Util.nullSafeEquals(currentId, targetId);
    }

    // not for instantiation
    DataRowUtils() {
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy