Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
apoc.create.Create Maven / Gradle / Ivy
/*
* Copyright (c) "Neo4j"
* Neo4j Sweden AB [http://neo4j.com]
*
* This file is part of Neo4j.
*
* 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 apoc.create;
import static org.neo4j.graphdb.RelationshipType.withName;
import apoc.get.Get;
import apoc.result.*;
import apoc.util.Util;
import apoc.util.collection.Iterables;
import apoc.uuid.UuidUtil;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.UUID;
import java.util.stream.LongStream;
import java.util.stream.Stream;
import org.neo4j.graphdb.*;
import org.neo4j.kernel.api.QueryLanguage;
import org.neo4j.kernel.api.procedure.QueryLanguageScope;
import org.neo4j.kernel.impl.coreapi.InternalTransaction;
import org.neo4j.procedure.*;
public class Create {
public static final String[] EMPTY_ARRAY = new String[0];
@Context
public Transaction tx;
@Procedure(name = "apoc.create.node", mode = Mode.WRITE)
@Description("Creates a `NODE` with the given dynamic labels.")
public Stream node(
@Name(value = "labels", description = "The labels to assign to the new node.") List labelNames,
@Name(value = "props", description = "The properties to assign to the new node.")
Map props) {
return Stream.of(new CreatedNodeResult(setProperties(tx.createNode(Util.labels(labelNames)), props)));
}
@Procedure(name = "apoc.create.addLabels", mode = Mode.WRITE)
@Description("Adds the given labels to the given `NODE` values.")
public Stream addLabels(
@Name(value = "nodes", description = "The nodes to add labels to.") Object nodes,
@Name(value = "labels", description = "The labels to add to the nodes.") List labelNames) {
Label[] labels = Util.labels(labelNames);
return new Get((InternalTransaction) tx).updatedNodes(nodes).map((r) -> {
Node node = r.node;
for (Label label : labels) {
node.addLabel(label);
}
return r;
});
}
@Procedure(name = "apoc.create.setProperty", mode = Mode.WRITE)
@Description("Sets the given property to the given `NODE` values.")
public Stream setProperty(
@Name(value = "nodes", description = "The nodes to set a property on.") Object nodes,
@Name(value = "key", description = "The name of the property key to set.") String key,
@Name(value = "value", description = "The value of the property to set.") Object value) {
return new Get((InternalTransaction) tx).updatedNodes(nodes).map((r) -> {
setProperty(r.node, key, toPropertyValue(value));
return r;
});
}
@Procedure(name = "apoc.create.setRelProperty", mode = Mode.WRITE)
@Description("Sets the given property on the `RELATIONSHIP` values.")
public Stream setRelProperty(
@Name(value = "rels", description = "The relationships to set a property on.") Object rels,
@Name(value = "key", description = "The name of the property key to set.") String key,
@Name(value = "value", description = "The value of the property to set.") Object value) {
return new Get((InternalTransaction) tx).updatesRels(rels).map((r) -> {
setProperty(r.rel, key, toPropertyValue(value));
return r;
});
}
@Procedure(name = "apoc.create.setProperties", mode = Mode.WRITE)
@Description("Sets the given properties to the given `NODE` values.")
public Stream setProperties(
@Name(value = "nodes", description = "The nodes to set properties on.") Object nodes,
@Name(value = "keys", description = "The property keys to set on the given nodes.") List keys,
@Name(value = "values", description = "The values to assign to the properties on the given nodes.")
List values) {
return new Get((InternalTransaction) tx).updatedNodes(nodes).map((r) -> {
setProperties(r.node, Util.mapFromLists(keys, values));
return r;
});
}
@Procedure(name = "apoc.create.removeProperties", mode = Mode.WRITE)
@Description("Removes the given properties from the given `NODE` values.")
public Stream removeProperties(
@Name(value = "nodes", description = "The nodes to remove properties from.") Object nodes,
@Name(value = "keys", description = "The property keys to remove from the given nodes.")
List keys) {
return new Get((InternalTransaction) tx).updatedNodes(nodes).map((r) -> {
keys.forEach(r.node::removeProperty);
return r;
});
}
@Procedure(name = "apoc.create.setRelProperties", mode = Mode.WRITE)
@Description("Sets the given properties on the `RELATIONSHIP` values.")
public Stream setRelProperties(
@Name(value = "rels", description = "The relationships to set properties on.") Object rels,
@Name(value = "keys", description = "The keys of the properties to set on the given relationships.")
List keys,
@Name(value = "values", description = "The values of the properties to set on the given relationships.")
List values) {
return new Get((InternalTransaction) tx).updatesRels(rels).map((r) -> {
setProperties(r.rel, Util.mapFromLists(keys, values));
return r;
});
}
@Procedure(name = "apoc.create.removeRelProperties", mode = Mode.WRITE)
@Description("Removes the given properties from the given `RELATIONSHIP` values.")
public Stream removeRelProperties(
@Name(value = "rels", description = "The relationships to remove properties from.") Object rels,
@Name(value = "keys", description = "The property keys to remove from the given nodes.")
List keys) {
return new Get((InternalTransaction) tx).updatesRels(rels).map((r) -> {
keys.forEach(r.rel::removeProperty);
return r;
});
}
@Procedure(name = "apoc.create.setLabels", mode = Mode.WRITE)
@Description("Sets the given labels to the given `NODE` values. Non-matching labels are removed from the nodes.")
public Stream setLabels(
@Name(value = "nodes", description = "The nodes to set labels on.") Object nodes,
@Name(value = "labels", description = "The labels to set on the given nodes.") List labelNames) {
Label[] labels = Util.labels(labelNames);
return new Get((InternalTransaction) tx).updatedNodes(nodes).map((r) -> {
Node node = r.node;
for (Label label : node.getLabels()) {
if (labelNames.contains(label.name())) continue;
node.removeLabel(label);
}
for (Label label : labels) {
if (node.hasLabel(label)) continue;
node.addLabel(label);
}
return r;
});
}
@Procedure(name = "apoc.create.removeLabels", mode = Mode.WRITE)
@Description("Removes the given labels from the given `NODE` values.")
public Stream removeLabels(
@Name(value = "nodes", description = "The node to remove labels from.") Object nodes,
@Name(value = "labels", description = "The labels to remove from the given node.")
List labelNames) {
Label[] labels = Util.labels(labelNames);
return new Get((InternalTransaction) tx).updatedNodes(nodes).map((r) -> {
Node node = r.node;
for (Label label : labels) {
node.removeLabel(label);
}
return r;
});
}
@Procedure(name = "apoc.create.nodes", mode = Mode.WRITE)
@Description("Creates `NODE` values with the given dynamic labels.")
public Stream nodes(
@Name(value = "labels", description = "The labels to assign to the new nodes.") List labelNames,
@Name(value = "props", description = "The properties to assign to the new nodes.")
List> props) {
Label[] labels = Util.labels(labelNames);
return props.stream().map(p -> new CreatedNodeResult(setProperties(tx.createNode(labels), p)));
}
@Procedure(name = "apoc.create.relationship", mode = Mode.WRITE)
@Description("Creates a `RELATIONSHIP` with the given dynamic relationship type.")
public Stream relationship(
@Name(value = "from", description = "The node from which the outgoing relationship will start.") Node from,
@Name(value = "relType", description = "The type to assign to the new relationship.") String relType,
@Name(value = "props", description = "The properties to assign to the new relationship.")
Map props,
@Name(value = "to", description = "The node to which the incoming relationship will be connected.")
Node to) {
VirtualRelationship.validateNodes(from, to);
return Stream.of(
new CreatedRelationshipResult(setProperties(from.createRelationshipTo(to, withName(relType)), props)));
}
@Procedure("apoc.create.vNode")
@Description("Returns a virtual `NODE`.")
public Stream vNode(
@Name(value = "labels", description = "The labels to assign to the new virtual node.")
List labelNames,
@Name(value = "props", description = "The properties to assign to the new virtual node.")
Map props) {
return Stream.of(new CreatedVirtualNodeResult(vNodeFunction(labelNames, props)));
}
@UserFunction("apoc.create.vNode")
@Description("Returns a virtual `NODE`.")
public Node vNodeFunction(
@Name(value = "labels", description = "The list of labels to assign to the virtual node.")
List labelNames,
@Name(
value = "props",
defaultValue = "{}",
description = "The map of properties to assign to the virtual node.")
Map props) {
return new VirtualNode(Util.labels(labelNames), props);
}
@UserFunction("apoc.create.virtual.fromNode")
@Description(
"Returns a virtual `NODE` from the given existing `NODE`. The virtual `NODE` only contains the requested properties.")
public Node virtualFromNodeFunction(
@Name(value = "node", description = "The node to generate a virtual node from.") Node node,
@Name(value = "propertyNames", description = "The properties to copy to the virtual node.")
List propertyNames) {
return new VirtualNode(node, propertyNames);
}
@Procedure("apoc.create.vNodes")
@Description("Returns virtual `NODE` values.")
public Stream vNodes(
@Name(value = "labels", description = "The labels to assign to the new virtual node.")
List labelNames,
@Name(value = "props", description = "The properties to assign to the new virtual nodes.")
List> props) {
Label[] labels = Util.labels(labelNames);
return props.stream().map(p -> new CreatedVirtualNodeResult(new VirtualNode(labels, p)));
}
@Procedure("apoc.create.vRelationship")
@Description("Returns a virtual `RELATIONSHIP`.")
public Stream vRelationship(
@Name(value = "from", description = "The node to connect the outgoing virtual relationship from.")
Node from,
@Name(value = "relType", description = "The type to assign to the new virtual relationship.")
String relType,
@Name(value = "props", description = "The properties to assign to the new virtual relationship.")
Map props,
@Name(value = "to", description = "The node to which the incoming virtual relationship will be connected.")
Node to) {
return Stream.of(new CreatedVirtualRelationshipResult(vRelationshipFunction(from, relType, props, to)));
}
@UserFunction("apoc.create.vRelationship")
@Description("Returns a virtual `RELATIONSHIP`.")
public Relationship vRelationshipFunction(
@Name(value = "from", description = "The start node to assign to the virtual relationship.") Node from,
@Name(value = "relType", description = "The type to assign to the virtual relationship.") String relType,
@Name(value = "props", description = "The map of properties to assign to the virtual relationship.")
Map props,
@Name(value = "to", description = "The end node to assign to the virtual relationship.") Node to) {
return new VirtualRelationship(from, to, withName(relType)).withProperties(props);
}
@Procedure("apoc.create.virtualPath")
@Description("Returns a virtual `PATH`.")
public Stream virtualPath(
@Name(value = "labelsN", description = "The labels to assign to the new virtual start node.")
List labelsN,
@Name(value = "n", description = "The properties to assign to the new virtual start node.")
Map n,
@Name(value = "arelType", description = "The type to assign to the new virtual relationship.")
String relType,
@Name(value = "props", description = "The properties to assign to the new virtual relationship.")
Map props,
@Name(value = "labelsM", description = "The labels to assign to the new virtual node.")
List labelsM,
@Name(value = "m", description = "The properties to assign to the new virtual node.")
Map m) {
RelationshipType type = withName(relType);
VirtualNode from = new VirtualNode(Util.labels(labelsN), n);
VirtualNode to = new VirtualNode(Util.labels(labelsM), m);
Relationship rel = new VirtualRelationship(from, to, type).withProperties(props);
return Stream.of(new VirtualPathResult(from, rel, to));
}
@Procedure("apoc.create.clonePathToVirtual")
@Description("Takes the given `PATH` and returns a virtual representation of it.")
public Stream clonePathToVirtual(
@Name(value = "path", description = "The path to create a virtual path from.") Path path) {
// given that it accepts a single path as a input parameter
// the `relationshipMap` (i.e. to avoid duplicated rels) is not necessary
return Stream.of(createVirtualPath(path, null));
}
@Procedure("apoc.create.clonePathsToVirtual")
@Description("Takes the given `LIST` and returns a virtual representation of them.")
public Stream clonePathsToVirtual(
@Name(value = "paths", description = "The paths to create virtual paths from.") List paths) {
Map createdRelationships = new HashMap<>();
return paths.stream().map(path -> createVirtualPath(path, createdRelationships));
}
private PathResult createVirtualPath(Path path, Map createdRelationships) {
final Iterable relationships = path.relationships();
final Node first = path.startNode();
VirtualPath virtualPath = new VirtualPath(new VirtualNode(first, Iterables.asList(first.getPropertyKeys())));
for (Relationship rel : relationships) {
final Relationship vRel = getVirtualRelPossiblyFromCache(createdRelationships, rel);
virtualPath.addRel(vRel);
}
return new PathResult(virtualPath);
}
private static Relationship getVirtualRelPossiblyFromCache(Map cacheRel, Relationship rel) {
if (cacheRel == null) {
return getVirtualRel(rel);
}
return cacheRel.compute(rel.getElementId(), (k, v) -> {
if (v == null) {
return getVirtualRel(rel);
}
return v;
});
}
private static Relationship getVirtualRel(Relationship rel) {
VirtualNode start = VirtualNode.from(rel.getStartNode());
VirtualNode end = VirtualNode.from(rel.getEndNode());
return VirtualRelationship.from(start, end, rel);
}
private T setProperties(T pc, Map p) {
if (p == null) return pc;
for (Map.Entry entry : p.entrySet()) {
setProperty(pc, entry.getKey(), entry.getValue());
}
return pc;
}
private void setProperty(T pc, String key, Object value) {
if (value == null) pc.removeProperty(key);
else pc.setProperty(key, toPropertyValue(value));
}
@UserFunction(name = "apoc.create.uuid", deprecatedBy = "Neo4j randomUUID() function")
@Deprecated
@QueryLanguageScope(scope = {QueryLanguage.CYPHER_5})
@Description("Returns a UUID.")
public String uuid() {
return UUID.randomUUID().toString();
}
@UserFunction("apoc.create.uuidBase64")
@Description("Returns a UUID encoded with base64.")
public String uuidBase64() {
return UuidUtil.generateBase64Uuid(UUID.randomUUID());
}
@UserFunction("apoc.create.uuidBase64ToHex")
@Description("Takes the given base64 encoded UUID and returns it as a hexadecimal `STRING`.")
public String uuidBase64ToHex(
@Name(value = "base64Uuid", description = "The string representing a UUID encoded with Base64.")
String base64Uuid) {
return UuidUtil.fromBase64ToHex(base64Uuid);
}
@UserFunction("apoc.create.uuidHexToBase64")
@Description("Takes the given UUID represented as a hexadecimal `STRING` and returns it encoded with base64.")
public String uuidHexToBase64(
@Name(value = "uuid", description = "The UUID represented as a hexadecimal string.") String uuidHex) {
return UuidUtil.fromHexToBase64(uuidHex);
}
private Object toPropertyValue(Object value) {
if (value instanceof Iterable) {
Iterable it = (Iterable) value;
Object first = Iterables.firstOrNull(it);
if (first == null) return EMPTY_ARRAY;
return Iterables.asArray(first.getClass(), it);
}
return value;
}
@Procedure(
name = "apoc.create.uuids",
deprecatedBy =
"Neo4j's randomUUID() function can be used as a replacement, for example: `UNWIND range(0,$count) AS row RETURN row, randomUUID() AS uuid`")
@Deprecated
@QueryLanguageScope(scope = {QueryLanguage.CYPHER_5})
@Description("Returns a stream of UUIDs.")
public Stream uuids(
@Name(value = "count", description = "The number of UUID values to generate.") long count) {
return LongStream.range(0, count).mapToObj(UUIDResult::new);
}
public static class UUIDResult {
@Description("The row number of the generated UUID.")
public final long row;
@Description("The generated UUID value.")
public final String uuid;
public UUIDResult(long row) {
this.row = row;
this.uuid = UUID.randomUUID().toString();
}
}
}