
org.cloudgraph.store.service.DeletedCommitComparator Maven / Gradle / Ivy
/**
* Copyright 2017 TerraMeta Software, Inc.
*
* Licensed 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.cloudgraph.store.service;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.plasma.sdo.PlasmaChangeSummary;
import org.plasma.sdo.PlasmaType;
import org.plasma.sdo.core.CoreDataObject;
import commonj.sdo.ChangeSummary.Setting;
import commonj.sdo.DataObject;
import commonj.sdo.Property;
/**
* Comparator which imposes a commit ordering on deleted objects based on
* singular relations between the types for the given source and target data
* objects. Singular relations across any number of "hops" between the 2 types
* are detected and the "child" types are ordered first. Where the types are the
* same, singular relations may still exist from/to the same type forming a
* "recursion" and these are detected by finding singular linkages between
* between the given data objects. Note: this comparator imposes orderings that
* are inconsistent with equals.
*/
public class DeletedCommitComparator extends CommitComparator {
private static final long serialVersionUID = 1L;
private static Log log = LogFactory.getFactory().getInstance(DeletedCommitComparator.class);
@Override
public int compare(CoreDataObject source, CoreDataObject target) {
PlasmaType targetType = (PlasmaType) target.getType();
PlasmaType sourceType = (PlasmaType) source.getType();
PlasmaChangeSummary changeSummary = (PlasmaChangeSummary) source.getDataGraph()
.getChangeSummary();
int targetDepth = changeSummary.getPathDepth(target);
int sourceDepth = changeSummary.getPathDepth(source);
if (targetType.getQualifiedName() != sourceType.getQualifiedName()) {
if (log.isDebugEnabled())
log.debug("comparing types: " + sourceType.toString() + " / " + targetType.toString());
if (isSingularRelation(target, source)) { // target is less than
// source
if (log.isDebugEnabled())
log.debug("(return -1) - singular relation to target: " + sourceType.toString()
+ " from source: " + targetType.toString());
return -1;
} else if (isSingularRelation(source, target)) { // target is
// greater than
// source
if (log.isDebugEnabled())
log.debug("(return 1) - singular relation from source: " + targetType.toString()
+ " to target: " + sourceType.toString());
return 1;
} else {
if (log.isDebugEnabled())
log.debug("(return 0)");
return 0;
}
} else {
if (log.isDebugEnabled())
log.debug("comparing data objects : " + source.toString() + " / " + target.toString());
// give precedence to reference links, then to
// graph path depth
if (hasChildLink(target, source)) {
if (log.isDebugEnabled())
log.debug("(return 1) singular link from : " + target.toString() + " to: "
+ source.toString());
return 1;
} else if (hasChildLink(source, target)) {
if (log.isDebugEnabled())
log.debug("(return -1) singular link from : " + source.toString() + " to: "
+ target.toString());
return -1;
} else {
// For say a root object which is part of
// a tree, where the property e.g. 'parent'
// is null the above link check won't apply,
// Therefore rely on graph depth
if (targetDepth < sourceDepth) {
if (log.isDebugEnabled())
log.debug("depth: " + targetDepth + " / " + sourceDepth);
return -1;
} else if (sourceDepth < targetDepth) {
if (log.isDebugEnabled())
log.debug("depth: " + sourceDepth + " / " + targetDepth);
return -1;
} else
return 0;
}
}
}
/**
* Determines if a child link exists solely from the change summary as SDO
* delete operation removes any contained data objects from their container
* entirely and pushes all their information into the change summary.
*/
protected boolean hasChildLink(DataObject target, DataObject source) {
if (log.isDebugEnabled())
log.debug("comparing " + target.toString() + "/" + source.toString());
for (Property property : target.getType().getProperties()) {
if (property.getType().isDataType())
continue;
if (property.isMany())
continue;
Setting setting = target.getDataGraph().getChangeSummary().getOldValue(target, property);
if (setting == null) // it's not been modified or deleted
continue;
if (log.isDebugEnabled())
log.debug("checking property " + target.getType().getName() + "." + property.getName());
if (isLinked(source, setting.getValue())) {
if (log.isDebugEnabled())
log.debug("found child data link " + target.getType().getName() + "."
+ property.getName() + "->" + source.getType().getName());
return true;
}
}
return false;
}
protected boolean isLinked(DataObject other, Object value) {
if (value instanceof DataObject) {
DataObject dataObject = (DataObject) value;
if (dataObject.equals(other))
return true;
}
return false;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy