/**
* This file is part of d:swarm graph extension.
*
* d:swarm graph extension is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* d:swarm graph extension is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with d:swarm graph extension. If not, see <http://www.gnu.org/licenses/>.
*/
package org.dswarm.graph.delta.match;
import java.util.Collection;
import java.util.HashMap;
import java.util.Map;
import java.util.Optional;
import org.neo4j.graphdb.GraphDatabaseService;
import org.neo4j.graphdb.Node;
import org.neo4j.graphdb.Transaction;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.dswarm.graph.DMPGraphException;
import org.dswarm.graph.delta.DeltaStatics;
import org.dswarm.graph.delta.match.mark.SubGraphEntityMarker;
import org.dswarm.graph.delta.match.model.SubGraphEntity;
import org.dswarm.graph.delta.util.GraphDBUtil;
/**
* @author tgaengler
*/
public class FirstDegreeExactSubGraphEntityMatcher extends Matcher<SubGraphEntity> {
private static final Logger LOG = LoggerFactory.getLogger(FirstDegreeExactSubGraphEntityMatcher.class);
public FirstDegreeExactSubGraphEntityMatcher(final Optional<? extends Collection<SubGraphEntity>> existingSubGraphEntitiesArg,
final Optional<? extends Collection<SubGraphEntity>> newSubGraphEntitiesArg, final GraphDatabaseService existingResourceDBArg,
final GraphDatabaseService newResourceDBArg, final String prefixedExistingResourceURIArg, final String prefixedNewResourceURIArg) throws DMPGraphException {
super(existingSubGraphEntitiesArg, newSubGraphEntitiesArg, existingResourceDBArg, newResourceDBArg, prefixedExistingResourceURIArg,
prefixedNewResourceURIArg, new SubGraphEntityMarker());
}
/**
* hash with key (from cs entity) + cs entity order + predicate + sub graph hash + order
*
* @param subGraphEntities
* @return
*/
@Override
protected Map<String, SubGraphEntity> generateHashes(final Collection<SubGraphEntity> subGraphEntities, final GraphDatabaseService graphDB)
throws DMPGraphException {
final Map<String, SubGraphEntity> hashedSubGraphEntities = new HashMap<>();
for (final SubGraphEntity subGraphEntity : subGraphEntities) {
final int keyHash = subGraphEntity.getCSEntity().getKey().hashCode();
final long csEntityOrderHash = Long.valueOf(subGraphEntity.getCSEntity().getEntityOrder()).hashCode();
final int predicateHash = subGraphEntity.getPredicate().hashCode();
// calc sub graph hash
final Map<Long, Long> nodeHashes = new HashMap<>();
final Integer deepestLeafHierarchyLevel = calculateEntityLeafHashes(graphDB, subGraphEntity.getNodeId(), nodeHashes);
if (deepestLeafHierarchyLevel != null && deepestLeafHierarchyLevel > subGraphEntity.getHierarchyLevel()) {
int currentHierarchyLevel = deepestLeafHierarchyLevel - 1;
while (currentHierarchyLevel > subGraphEntity.getHierarchyLevel()) {
calculateEntityHierarchyLevelNodesHashes(graphDB, subGraphEntity.getNodeId(), nodeHashes, currentHierarchyLevel);
currentHierarchyLevel--;
}
calculateSubGraphEntityHash(graphDB, subGraphEntity.getNodeId(), nodeHashes);
}
Long subGraphHash = nodeHashes.get(subGraphEntity.getNodeId());
long hash = keyHash;
hash = 31 * hash + predicateHash;
if (subGraphHash != null) {
hash = 31 * hash + subGraphHash;
}
hash = 31 * hash + csEntityOrderHash;
hash = 31 * hash + Long.valueOf(subGraphEntity.getOrder()).hashCode();
hashedSubGraphEntities.put(Long.valueOf(hash).toString(), subGraphEntity);
}
return hashedSubGraphEntities;
}
private void calculateSubGraphEntityHash(final GraphDatabaseService graphDB, final long entityNodeId, final Map<Long, Long> nodeHashes) {
final Transaction tx = graphDB.beginTx();
try {
GraphDBUtil.calculateEntityHash(graphDB, entityNodeId, nodeHashes);
tx.success();
} catch (final Exception e) {
FirstDegreeExactSubGraphEntityMatcher.LOG.error("couldn't calculate sub graph entity hashes", e);
tx.failure();
} finally {
tx.close();
}
}
private void calculateEntityHierarchyLevelNodesHashes(final GraphDatabaseService graphDB, final long entityNodeId,
final Map<Long, Long> nodeHashes, final int hierarchyLevel) throws DMPGraphException {
final Collection<String> entityHierarchyLevelNodeIds = getEntityHierarchyLevelNodes(graphDB, entityNodeId, hierarchyLevel);
try (final Transaction tx = graphDB.beginTx()) {
for (final String entityHierarchyLevelNodeId : entityHierarchyLevelNodeIds) {
GraphDBUtil.calculateEntityHash(graphDB, Long.valueOf(entityHierarchyLevelNodeId), nodeHashes);
}
tx.success();
} catch (final Exception e) {
final String message = "couldn't calculate hierarchy level node hashes";
FirstDegreeExactSubGraphEntityMatcher.LOG.error(message, e);
throw new DMPGraphException(message);
}
}
private Integer calculateEntityLeafHashes(final GraphDatabaseService graphDB, final long entityNodeId, final Map<Long, Long> nodeHashes)
throws DMPGraphException {
final Collection<String> entityLeafNodeIds = GraphDBUtil.getEntityLeafs(graphDB, entityNodeId);
Integer deepestHierarchyLevel = null;
try (final Transaction tx = graphDB.beginTx()) {
for (final String entityLeafNodeId : entityLeafNodeIds) {
final Node leafNode = graphDB.getNodeById(Long.valueOf(entityLeafNodeId));
Long hash = GraphDBUtil.calculateNodeHash(leafNode);
if (hash == null) {
continue;
}
final Integer hierarchyLevel = (Integer) leafNode.getProperty(DeltaStatics.HIERARCHY_LEVEL_PROPERTY, null);
if (hierarchyLevel != null) {
hash = 31 * hash + hierarchyLevel;
if (deepestHierarchyLevel != null && deepestHierarchyLevel < hierarchyLevel) {
deepestHierarchyLevel = hierarchyLevel;
} else {
deepestHierarchyLevel = hierarchyLevel;
}
}
nodeHashes.put(leafNode.getId(), hash);
}
tx.success();
} catch (final Exception e) {
final String message = "couldn't calculated entity leaf hashes";
FirstDegreeExactSubGraphEntityMatcher.LOG.error(message, e);
throw new DMPGraphException(message);
}
return deepestHierarchyLevel;
}
/**
* @param graphDB
* @param nodeId
* @return
*/
private Collection<String> getEntityHierarchyLevelNodes(final GraphDatabaseService graphDB, final long nodeId, final int hierarchyLevel)
throws DMPGraphException {
final String entitHierarchyLevelNodesQuery = buildGetEntityHierarchyLevelNodesQuery(nodeId, hierarchyLevel);
return GraphDBUtil.executeQueryWithMultipleResults(entitHierarchyLevelNodesQuery, "hierarchy_level_node", graphDB);
}
private String buildGetEntityHierarchyLevelNodesQuery(final long nodeId, final int hierarchyLevel) {
// START n= node(2062) MATCH (n)-[r*]->(m) RETURN m;
final StringBuilder sb = new StringBuilder();
sb.append("START n=node(").append(nodeId).append(")\nMATCH (n)-[r*]->(m)").append("\nWHERE m.").append(DeltaStatics.HIERARCHY_LEVEL_PROPERTY).append(" = ").append(hierarchyLevel)
.append("\nRETURN id(m) AS hierarchy_level_node");
return sb.toString();
}
}