diff --git a/.gitmodules b/.gitmodules index ea53a5a847..ce008b8fb4 100644 --- a/.gitmodules +++ b/.gitmodules @@ -1,4 +1,4 @@ [submodule "apoc-core"] path = apoc-core url = https://github.com/neo4j/apoc - branch = dev + branch = dev_reduce_common_reliance diff --git a/apoc-core b/apoc-core index c7102fdf4b..b7ad46d67e 160000 --- a/apoc-core +++ b/apoc-core @@ -1 +1 @@ -Subproject commit c7102fdf4b6e92abf1c3b4d1fd0575aadae1a847 +Subproject commit b7ad46d67eaa72d4e926ccee4d2b76251ca28aeb diff --git a/build.gradle b/build.gradle index 9fa2367da5..fb1110ac51 100644 --- a/build.gradle +++ b/build.gradle @@ -15,7 +15,7 @@ downloadLicenses { allprojects { group = 'org.neo4j.procedure' - version = '2024.12.0' + version = '2025.01.0' archivesBaseName = 'apoc' description = """neo4j-apoc-procedures""" } @@ -92,8 +92,8 @@ subprojects { // neo4jDockerImage system property is used in TestContainerUtil systemProperties 'user.language' : 'en' , 'user.country' : 'US', - 'neo4jDockerImage': project.hasProperty("neo4jDockerEeOverride") ? project.getProperty("neo4jDockerEeOverride") : 'neo4j:2024.12.0-enterprise', - 'neo4jCommunityDockerImage': project.hasProperty("neo4jDockerCeOverride") ? project.getProperty("neo4jDockerCeOverride") : 'neo4j:2024.12.0', + 'neo4jDockerImage': project.hasProperty("neo4jDockerEeOverride") ? project.getProperty("neo4jDockerEeOverride") : 'neo4j:2025.01.0-enterprise', + 'neo4jCommunityDockerImage': project.hasProperty("neo4jDockerCeOverride") ? project.getProperty("neo4jDockerCeOverride") : 'neo4j:2025.01.0', 'coreDir': 'apoc-core/core', 'testDockerBundle': false @@ -152,7 +152,7 @@ subprojects { ext { // NB: due to version.json generation by parsing this file, the next line must not have any if/then/else logic - neo4jVersion = "2024.12.0" + neo4jVersion = "2025.01.0" // instead we apply the override logic here neo4jVersionEffective = project.hasProperty("neo4jVersionOverride") ? project.getProperty("neo4jVersionOverride") : neo4jVersion testContainersVersion = '1.20.2' diff --git a/extended/src/main/java/apoc/algo/PathFindingExtended.java b/extended/src/main/java/apoc/algo/PathFindingExtended.java index f9eb496461..4eba38b25e 100644 --- a/extended/src/main/java/apoc/algo/PathFindingExtended.java +++ b/extended/src/main/java/apoc/algo/PathFindingExtended.java @@ -16,7 +16,8 @@ import org.neo4j.procedure.Procedure; import java.util.stream.Stream; -import static apoc.algo.PathFindingUtils.buildPathExpander; + +import static apoc.algo.PathFindingUtilsExtended.buildPathExpander; @Extended public class PathFindingExtended { @@ -41,7 +42,7 @@ public Stream aStarWithPoint( new BasicEvaluationContext(tx, db), buildPathExpander(relTypesAndDirs), CommonEvaluators.doubleCostEvaluator(weightPropertyName), - new PathFindingUtils.GeoEstimateEvaluatorPointCustom(pointPropertyName)); + new PathFindingUtilsExtended.GeoEstimateEvaluatorPointCustom(pointPropertyName)); return WeightedPathResult.streamWeightedPathResult(startNode, endNode, algo); } diff --git a/extended/src/main/java/apoc/algo/PathFindingUtilsExtended.java b/extended/src/main/java/apoc/algo/PathFindingUtilsExtended.java new file mode 100644 index 0000000000..95e5450c1b --- /dev/null +++ b/extended/src/main/java/apoc/algo/PathFindingUtilsExtended.java @@ -0,0 +1,94 @@ +/* + * 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.algo; + +import apoc.path.RelationshipTypeAndDirectionsExtended; +import org.apache.commons.lang3.tuple.Pair; +import org.neo4j.graphalgo.EstimateEvaluator; +import org.neo4j.graphdb.Direction; +import org.neo4j.graphdb.Node; +import org.neo4j.graphdb.PathExpander; +import org.neo4j.graphdb.PathExpanderBuilder; +import org.neo4j.graphdb.RelationshipType; +import org.neo4j.values.storable.PointValue; + +public class PathFindingUtilsExtended { + public static class GeoEstimateEvaluatorPointCustom implements EstimateEvaluator { + // -- from org.neo4j.graphalgo.impl.util.GeoEstimateEvaluator + private static final double EARTH_RADIUS = 6371 * 1000; // Meters + private Node cachedGoal; + private final String pointPropertyKey; + private double[] cachedGoalCoordinates; + + public GeoEstimateEvaluatorPointCustom(String pointPropertyKey) { + this.pointPropertyKey = pointPropertyKey; + } + + @Override + public Double getCost(Node node, Node goal) { + double[] nodeCoordinates = getCoordinates(node); + if (cachedGoal == null || !cachedGoal.equals(goal)) { + cachedGoalCoordinates = getCoordinates(goal); + cachedGoal = goal; + } + return distance(nodeCoordinates[0], nodeCoordinates[1], cachedGoalCoordinates[0], cachedGoalCoordinates[1]); + } + + private static double distance(double latitude1, double longitude1, double latitude2, double longitude2) { + latitude1 = Math.toRadians(latitude1); + longitude1 = Math.toRadians(longitude1); + latitude2 = Math.toRadians(latitude2); + longitude2 = Math.toRadians(longitude2); + double cLa1 = Math.cos(latitude1); + double xA = EARTH_RADIUS * cLa1 * Math.cos(longitude1); + double yA = EARTH_RADIUS * cLa1 * Math.sin(longitude1); + double zA = EARTH_RADIUS * Math.sin(latitude1); + double cLa2 = Math.cos(latitude2); + double xB = EARTH_RADIUS * cLa2 * Math.cos(longitude2); + double yB = EARTH_RADIUS * cLa2 * Math.sin(longitude2); + double zB = EARTH_RADIUS * Math.sin(latitude2); + return Math.sqrt((xA - xB) * (xA - xB) + (yA - yB) * (yA - yB) + (zA - zB) * (zA - zB)); + } + // -- end from org.neo4j.graphalgo.impl.util.GeoEstimateEvaluator + + private double[] getCoordinates(Node node) { + return ((PointValue) node.getProperty(pointPropertyKey)).coordinate(); + } + } + + public static PathExpander buildPathExpander(String relationshipsAndDirections) { + PathExpanderBuilder builder = PathExpanderBuilder.empty(); + for (Pair pair : RelationshipTypeAndDirectionsExtended.parse(relationshipsAndDirections)) { + if (pair.getLeft() == null) { + if (pair.getRight() == null) { + builder = PathExpanderBuilder.allTypesAndDirections(); + } else { + builder = PathExpanderBuilder.allTypes(pair.getRight()); + } + } else { + if (pair.getRight() == null) { + builder = builder.add(pair.getLeft()); + } else { + builder = builder.add(pair.getLeft(), pair.getRight()); + } + } + } + return builder.build(); + } +} diff --git a/extended/src/main/java/apoc/coll/CollExtended.java b/extended/src/main/java/apoc/coll/CollExtended.java index 71ba3c6d0d..9850b67f67 100644 --- a/extended/src/main/java/apoc/coll/CollExtended.java +++ b/extended/src/main/java/apoc/coll/CollExtended.java @@ -1,7 +1,7 @@ package apoc.coll; import apoc.Extended; -import apoc.util.CollectionUtils; +import apoc.util.CollectionUtilsExtended; import org.neo4j.procedure.Description; import org.neo4j.procedure.Name; import org.neo4j.procedure.UserFunction; @@ -17,7 +17,7 @@ public class CollExtended { @UserFunction @Description("apoc.coll.avgDuration([duration('P2DT3H'), duration('PT1H45S'), ...]) - returns the average of a list of duration values") public DurationValue avgDuration(@Name("durations") List list) { - if (CollectionUtils.isEmpty(list)) return null; + if (CollectionUtilsExtended.isEmpty(list)) return null; long count = 0; diff --git a/extended/src/main/java/apoc/coll/SetBackedListExtended.java b/extended/src/main/java/apoc/coll/SetBackedListExtended.java new file mode 100644 index 0000000000..ec2d59ed2c --- /dev/null +++ b/extended/src/main/java/apoc/coll/SetBackedListExtended.java @@ -0,0 +1,139 @@ +/* + * 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.coll; + +import java.util.AbstractSequentialList; +import java.util.Iterator; +import java.util.ListIterator; +import java.util.NoSuchElementException; +import java.util.Set; +import java.util.Spliterator; + +/** + * @author mh + * @since 10.04.16 + */ +public class SetBackedListExtended extends AbstractSequentialList implements Set { + + private final Set set; + + public SetBackedListExtended(Set set) { + this.set = set; + } + + @Override + public int size() { + return set.size(); + } + + public ListIterator listIterator(int index) { + return new ListIterator() { + Iterator it = set.iterator(); + T current = null; + int idx = 0; + + { + moveTo(index); + } + + @Override + public boolean hasNext() { + return it.hasNext(); + } + + @Override + public T next() { + idx++; + return current = it.next(); + } + + @Override + public boolean hasPrevious() { + return idx > 0; + } + + @Override + public T previous() { + if (!hasPrevious()) throw new NoSuchElementException(); + T tmp = current; + moveTo(idx - 1); + return tmp; + } + + private void moveTo(int pos) { + Iterator it2 = set.iterator(); + T value = null; + int i = 0; + while (i++ < pos) { + value = it2.next(); + } + ; + this.it = it2; + this.idx = pos; + this.current = value; + } + + @Override + public int nextIndex() { + return idx; + } + + @Override + public int previousIndex() { + return idx - 1; + } + + @Override + public void remove() { + throw new UnsupportedOperationException("remove"); + } + + @Override + public void set(Object o) { + throw new UnsupportedOperationException("set"); + } + + @Override + public void add(Object o) { + throw new UnsupportedOperationException("add"); + } + }; + } + + @Override + public boolean contains(Object o) { + return set.contains(o); + } + + @Override + public int hashCode() { + return set.hashCode(); + } + + @Override + public boolean equals(Object o) { + if (o instanceof Set) return set.equals(o); + return o instanceof Iterable && super.equals(o); + } + + @Override + public Spliterator spliterator() { + return set.spliterator(); + } +} diff --git a/extended/src/main/java/apoc/convert/ConvertExtended.java b/extended/src/main/java/apoc/convert/ConvertExtended.java index 502098441d..f282d76811 100644 --- a/extended/src/main/java/apoc/convert/ConvertExtended.java +++ b/extended/src/main/java/apoc/convert/ConvertExtended.java @@ -1,7 +1,7 @@ package apoc.convert; import apoc.Extended; -import apoc.meta.Types; +import apoc.meta.TypesExtended; import apoc.util.collection.Iterables; import org.neo4j.graphdb.Node; import org.neo4j.graphdb.Path; @@ -49,7 +49,7 @@ public Object fromYaml(@Name("value") String value, @Name(value = "config", defa * which handle complex types, like list/map of nodes/rels/paths */ private Object writeYamlResult(Object value) { - Types type = Types.of(value); + TypesExtended type = TypesExtended.of(value); return switch (type) { case NODE -> nodeToMap((Node) value); case RELATIONSHIP -> relToMap((Relationship) value); diff --git a/extended/src/main/java/apoc/convert/ConvertExtendedUtil.java b/extended/src/main/java/apoc/convert/ConvertExtendedUtil.java index 6f5d0faca6..1e3e4249fa 100644 --- a/extended/src/main/java/apoc/convert/ConvertExtendedUtil.java +++ b/extended/src/main/java/apoc/convert/ConvertExtendedUtil.java @@ -3,6 +3,8 @@ import apoc.export.util.DurationValueSerializer; import apoc.export.util.PointSerializer; import apoc.export.util.TemporalSerializer; +import apoc.util.collection.Iterables; +import apoc.util.collection.Iterators; import com.fasterxml.jackson.core.JsonProcessingException; import com.fasterxml.jackson.databind.ObjectMapper; import com.fasterxml.jackson.databind.module.SimpleModule; @@ -11,7 +13,12 @@ import org.neo4j.graphdb.spatial.Point; import org.neo4j.values.storable.DurationValue; +import java.lang.reflect.Array; import java.time.temporal.Temporal; +import java.util.ArrayList; +import java.util.Collection; +import java.util.Collections; +import java.util.Iterator; import java.util.List; import java.util.Map; @@ -26,6 +33,34 @@ public class ConvertExtendedUtil { YAML_MODULE.addSerializer(DurationValue.class, new DurationValueSerializer()); } + public static List convertToList(Object list) { + if (list == null) return null; + else if (list instanceof List) return (List) list; + else if (list instanceof Collection) return new ArrayList((Collection) list); + else if (list instanceof Iterable) return Iterables.asList((Iterable) list); + else if (list instanceof Iterator) return Iterators.asList((Iterator) list); + else if (list.getClass().isArray()) { + return convertArrayToList(list); + } + return Collections.singletonList(list); + } + + public static List convertArrayToList(Object list) { + final Object[] objectArray; + if (list.getClass().getComponentType().isPrimitive()) { + int length = Array.getLength(list); + objectArray = new Object[length]; + for (int i = 0; i < length; i++) { + objectArray[i] = Array.get(list, i); + } + } else { + objectArray = (Object[]) list; + } + List result = new ArrayList<>(objectArray.length); + Collections.addAll(result, objectArray); + return result; + } + /** * get YAMLFactory with configured enable and disable values */ diff --git a/extended/src/main/java/apoc/cypher/CypherExtended.java b/extended/src/main/java/apoc/cypher/CypherExtended.java index 6a02e63efb..b47e3fff28 100644 --- a/extended/src/main/java/apoc/cypher/CypherExtended.java +++ b/extended/src/main/java/apoc/cypher/CypherExtended.java @@ -5,10 +5,10 @@ import apoc.result.CypherStatementMapResult; import apoc.result.MapResult; import apoc.util.CompressionAlgo; -import apoc.util.EntityUtil; +import apoc.util.EntityUtilExtended; import apoc.util.FileUtils; -import apoc.util.QueueBasedSpliterator; -import apoc.util.QueueUtil; +import apoc.util.QueueBasedSpliteratorExtended; +import apoc.util.QueueUtilExtended; import apoc.util.Util; import apoc.util.collection.Iterators; import org.apache.commons.lang3.StringUtils; @@ -45,7 +45,7 @@ import java.util.stream.Stream; import java.util.stream.StreamSupport; -import static apoc.cypher.CypherUtils.runCypherQuery; +import static apoc.cypher.CypherUtilsExtended.runCypherQuery; import static apoc.util.MapUtil.map; import static apoc.util.Util.param; import static apoc.util.Util.quote; @@ -153,7 +153,7 @@ private Stream runManyStatements(Scanner scanner, Map runDataStatementsInTx(scanner, internalQueue, params, addStatistics, timeout, reportError, fileName); } }, RowResult.TOMBSTONE); - return StreamSupport.stream(new QueueBasedSpliterator<>(queue, RowResult.TOMBSTONE, terminationGuard, Integer.MAX_VALUE), false); + return StreamSupport.stream(new QueueBasedSpliteratorExtended<>(queue, RowResult.TOMBSTONE, terminationGuard, Integer.MAX_VALUE), false); } @@ -239,7 +239,7 @@ private void collectError(BlockingQueue queue, boolean reportError, E } String error = e.getMessage(); RowResult result = new RowResult(-1, Map.of("error", error), fileName); - QueueUtil.put(queue, result, 10); + QueueUtilExtended.put(queue, result, 10); } private Scanner createScannerFor(Reader reader) { @@ -286,7 +286,7 @@ private Object consumeResult(Result result, BlockingQueue queue, bool AtomicBoolean closed = new AtomicBoolean(false); while (isOpenAndHasNext(result, closed)) { terminationGuard.check(); - Map res = EntityUtil.anyRebind(transaction, result.next()); + Map res = EntityUtilExtended.anyRebind(transaction, result.next()); queue.put(new RowResult(row++, res, fileName)); } if (addStatistics) { @@ -454,7 +454,7 @@ public Stream mapParallel2(@Name("fragment") String fragment, @Name(" return total; }); - return StreamSupport.stream(new QueueBasedSpliterator<>(queue, RowResult.TOMBSTONE, terminationGuard, (int)timeout),true) + return StreamSupport.stream(new QueueBasedSpliteratorExtended<>(queue, RowResult.TOMBSTONE, terminationGuard, (int)timeout),true) .map(rowResult -> new MapResult(rowResult.result)) .onClose(() -> { transactions.forEach(i -> Util.close(i)); @@ -505,7 +505,7 @@ public Stream parallel2(@Name("fragment") String fragm } return futures.stream().flatMap(f -> { try { - return EntityUtil.anyRebind(tx, f.get()).stream().map(CypherStatementMapResult::new); + return EntityUtilExtended.anyRebind(tx, f.get()).stream().map(CypherStatementMapResult::new); } catch (InterruptedException | ExecutionException e) { throw new RuntimeException("Error executing in parallel " + statement, e); } diff --git a/extended/src/main/java/apoc/cypher/CypherUtilsExtended.java b/extended/src/main/java/apoc/cypher/CypherUtilsExtended.java new file mode 100644 index 0000000000..1cef3bf70a --- /dev/null +++ b/extended/src/main/java/apoc/cypher/CypherUtilsExtended.java @@ -0,0 +1,50 @@ +/* + * 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.cypher; + +import apoc.result.CypherStatementMapResult; +import org.neo4j.graphdb.Transaction; +import org.neo4j.procedure.Name; + +import java.util.Collection; +import java.util.Collections; +import java.util.Map; +import java.util.stream.Stream; + +import static java.lang.String.format; +import static java.lang.String.join; +import static java.util.stream.Collectors.toList; + +public class CypherUtilsExtended { + public static Stream runCypherQuery( + Transaction tx, @Name("cypher") String statement, @Name("params") Map params) { + if (params == null) params = Collections.emptyMap(); + return tx.execute(withParamMapping(statement, params.keySet()), params).stream() + .map(CypherStatementMapResult::new); + } + + public static String withParamMapping(String fragment, Collection keys) { + if (keys.isEmpty()) return fragment; + String declaration = " WITH " + + join( + ", ", + keys.stream().map(s -> format(" $`%s` as `%s` ", s, s)).collect(toList())); + return declaration + fragment; + } +} diff --git a/extended/src/main/java/apoc/cypher/export/CypherResultSubGraphExtended.java b/extended/src/main/java/apoc/cypher/export/CypherResultSubGraphExtended.java new file mode 100644 index 0000000000..666219230b --- /dev/null +++ b/extended/src/main/java/apoc/cypher/export/CypherResultSubGraphExtended.java @@ -0,0 +1,274 @@ +/* + * 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.cypher.export; + +import apoc.util.Util; +import apoc.util.collection.Iterables; +import org.neo4j.cypher.export.SubGraph; +import org.neo4j.graphdb.Label; +import org.neo4j.graphdb.Node; +import org.neo4j.graphdb.Relationship; +import org.neo4j.graphdb.RelationshipType; +import org.neo4j.graphdb.Result; +import org.neo4j.graphdb.Transaction; +import org.neo4j.graphdb.schema.ConstraintDefinition; +import org.neo4j.graphdb.schema.IndexDefinition; +import org.neo4j.graphdb.schema.IndexType; +import org.neo4j.graphdb.security.AuthorizationViolationException; + +import java.util.Collection; +import java.util.Collections; +import java.util.HashSet; +import java.util.Iterator; +import java.util.List; +import java.util.Set; +import java.util.SortedMap; +import java.util.TreeMap; +import java.util.stream.Collectors; +import java.util.stream.StreamSupport; + +import static apoc.util.Util.INVALID_QUERY_MODE_ERROR; + +public class CypherResultSubGraphExtended implements SubGraph { + + private final SortedMap nodes = new TreeMap<>(); + private final SortedMap relationships = new TreeMap<>(); + private final Collection