From 0aaa25a120d6bd8822b46964c0974b3f2fbe15f4 Mon Sep 17 00:00:00 2001 From: Paul Horn Date: Tue, 3 Dec 2024 17:46:57 +0100 Subject: [PATCH 1/2] Move usages of Neo4j Values from nodes builder to native projection Co-authored-by: Martin Junghanns Co-authored-by: Olga Razvenskaia --- .../DoubleArrayNodePropertiesBuilder.java | 7 ------ .../DoubleNodePropertiesBuilder.java | 8 ------- .../FloatArrayNodePropertiesBuilder.java | 7 ------ .../InnerNodePropertiesBuilder.java | 3 --- .../LongArrayNodePropertiesBuilder.java | 7 ------ .../LongNodePropertiesBuilder.java | 8 ------- .../NodePropertiesFromStoreBuilder.java | 24 +++---------------- .../NativeNodePropertyImporter.java | 4 +++- 8 files changed, 6 insertions(+), 62 deletions(-) diff --git a/core/src/main/java/org/neo4j/gds/core/loading/nodeproperties/DoubleArrayNodePropertiesBuilder.java b/core/src/main/java/org/neo4j/gds/core/loading/nodeproperties/DoubleArrayNodePropertiesBuilder.java index 0bf8d489ab..df058aa957 100644 --- a/core/src/main/java/org/neo4j/gds/core/loading/nodeproperties/DoubleArrayNodePropertiesBuilder.java +++ b/core/src/main/java/org/neo4j/gds/core/loading/nodeproperties/DoubleArrayNodePropertiesBuilder.java @@ -28,9 +28,7 @@ import org.neo4j.gds.core.concurrency.DefaultPool; import org.neo4j.gds.core.concurrency.ParallelUtil; import org.neo4j.gds.utils.GdsNeo4jValueConversion; -import org.neo4j.gds.utils.Neo4jValueConversion; import org.neo4j.gds.values.GdsValue; -import org.neo4j.values.storable.Value; import java.util.Arrays; import java.util.stream.Collectors; @@ -57,11 +55,6 @@ public void set(long neoNodeId, double[] value) { builder.set(neoNodeId, value); } - @Override - public void setValue(long neoNodeId, Value value) { - set(neoNodeId, Neo4jValueConversion.getDoubleArray(value)); - } - @Override public void setValue(long neoNodeId, GdsValue value) { set(neoNodeId, GdsNeo4jValueConversion.getDoubleArray(value)); diff --git a/core/src/main/java/org/neo4j/gds/core/loading/nodeproperties/DoubleNodePropertiesBuilder.java b/core/src/main/java/org/neo4j/gds/core/loading/nodeproperties/DoubleNodePropertiesBuilder.java index 3817c14416..0f08858a8d 100644 --- a/core/src/main/java/org/neo4j/gds/core/loading/nodeproperties/DoubleNodePropertiesBuilder.java +++ b/core/src/main/java/org/neo4j/gds/core/loading/nodeproperties/DoubleNodePropertiesBuilder.java @@ -28,9 +28,7 @@ import org.neo4j.gds.core.concurrency.DefaultPool; import org.neo4j.gds.core.concurrency.ParallelUtil; import org.neo4j.gds.utils.GdsNeo4jValueConversion; -import org.neo4j.gds.utils.Neo4jValueConversion; import org.neo4j.gds.values.GdsValue; -import org.neo4j.values.storable.Value; import java.lang.invoke.MethodHandles; import java.lang.invoke.VarHandle; @@ -80,12 +78,6 @@ public void set(long neoNodeId, double value) { updateMaxValue(value); } - @Override - public void setValue(long neoNodeId, Value value) { - double doubleValue = Neo4jValueConversion.getDoubleValue(value); - set(neoNodeId, doubleValue); - } - @Override public void setValue(long neoNodeId, GdsValue value) { double doubleValue = GdsNeo4jValueConversion.getDoubleValue(value); diff --git a/core/src/main/java/org/neo4j/gds/core/loading/nodeproperties/FloatArrayNodePropertiesBuilder.java b/core/src/main/java/org/neo4j/gds/core/loading/nodeproperties/FloatArrayNodePropertiesBuilder.java index 4ed9f7be4c..c1154f8756 100644 --- a/core/src/main/java/org/neo4j/gds/core/loading/nodeproperties/FloatArrayNodePropertiesBuilder.java +++ b/core/src/main/java/org/neo4j/gds/core/loading/nodeproperties/FloatArrayNodePropertiesBuilder.java @@ -28,9 +28,7 @@ import org.neo4j.gds.core.concurrency.DefaultPool; import org.neo4j.gds.core.concurrency.ParallelUtil; import org.neo4j.gds.utils.GdsNeo4jValueConversion; -import org.neo4j.gds.utils.Neo4jValueConversion; import org.neo4j.gds.values.GdsValue; -import org.neo4j.values.storable.Value; import java.util.Arrays; import java.util.stream.Collectors; @@ -55,11 +53,6 @@ public void set(long neoNodeId, float[] value) { builder.set(neoNodeId, value); } - @Override - public void setValue(long neoNodeId, Value value) { - set(neoNodeId, Neo4jValueConversion.getFloatArray(value)); - } - @Override public void setValue(long neoNodeId, GdsValue value) { set(neoNodeId, GdsNeo4jValueConversion.getFloatArray(value)); diff --git a/core/src/main/java/org/neo4j/gds/core/loading/nodeproperties/InnerNodePropertiesBuilder.java b/core/src/main/java/org/neo4j/gds/core/loading/nodeproperties/InnerNodePropertiesBuilder.java index cda18ba6d9..25652d1c89 100644 --- a/core/src/main/java/org/neo4j/gds/core/loading/nodeproperties/InnerNodePropertiesBuilder.java +++ b/core/src/main/java/org/neo4j/gds/core/loading/nodeproperties/InnerNodePropertiesBuilder.java @@ -22,12 +22,9 @@ import org.neo4j.gds.api.PartialIdMap; import org.neo4j.gds.api.properties.nodes.NodePropertyValues; import org.neo4j.gds.values.GdsValue; -import org.neo4j.values.storable.Value; public interface InnerNodePropertiesBuilder { - void setValue(long neoNodeId, Value value); - void setValue(long neoNodeId, GdsValue value); /** diff --git a/core/src/main/java/org/neo4j/gds/core/loading/nodeproperties/LongArrayNodePropertiesBuilder.java b/core/src/main/java/org/neo4j/gds/core/loading/nodeproperties/LongArrayNodePropertiesBuilder.java index 4c386fe873..91aca2f6e5 100644 --- a/core/src/main/java/org/neo4j/gds/core/loading/nodeproperties/LongArrayNodePropertiesBuilder.java +++ b/core/src/main/java/org/neo4j/gds/core/loading/nodeproperties/LongArrayNodePropertiesBuilder.java @@ -28,9 +28,7 @@ import org.neo4j.gds.core.concurrency.DefaultPool; import org.neo4j.gds.core.concurrency.ParallelUtil; import org.neo4j.gds.utils.GdsNeo4jValueConversion; -import org.neo4j.gds.utils.Neo4jValueConversion; import org.neo4j.gds.values.GdsValue; -import org.neo4j.values.storable.Value; import java.util.Arrays; import java.util.stream.Collectors; @@ -55,11 +53,6 @@ public void set(long neoNodeId, long[] value) { builder.set(neoNodeId, value); } - @Override - public void setValue(long neoNodeId, Value value) { - set(neoNodeId, Neo4jValueConversion.getLongArray(value)); - } - @Override public void setValue(long neoNodeId, GdsValue value) { set(neoNodeId, GdsNeo4jValueConversion.getLongArray(value)); diff --git a/core/src/main/java/org/neo4j/gds/core/loading/nodeproperties/LongNodePropertiesBuilder.java b/core/src/main/java/org/neo4j/gds/core/loading/nodeproperties/LongNodePropertiesBuilder.java index d631b44120..2912bd921c 100644 --- a/core/src/main/java/org/neo4j/gds/core/loading/nodeproperties/LongNodePropertiesBuilder.java +++ b/core/src/main/java/org/neo4j/gds/core/loading/nodeproperties/LongNodePropertiesBuilder.java @@ -29,9 +29,7 @@ import org.neo4j.gds.core.concurrency.DefaultPool; import org.neo4j.gds.core.concurrency.ParallelUtil; import org.neo4j.gds.utils.GdsNeo4jValueConversion; -import org.neo4j.gds.utils.Neo4jValueConversion; import org.neo4j.gds.values.GdsValue; -import org.neo4j.values.storable.Value; import java.lang.invoke.MethodHandles; import java.lang.invoke.VarHandle; @@ -88,12 +86,6 @@ public void set(long neoNodeId, long value) { updateMaxValue(value); } - @Override - public void setValue(long neoNodeId, Value value) { - var longValue = Neo4jValueConversion.getLongValue(value); - set(neoNodeId, longValue); - } - @Override public void setValue(long neoNodeId, GdsValue value) { var longValue = GdsNeo4jValueConversion.getLongValue(value); diff --git a/core/src/main/java/org/neo4j/gds/core/loading/nodeproperties/NodePropertiesFromStoreBuilder.java b/core/src/main/java/org/neo4j/gds/core/loading/nodeproperties/NodePropertiesFromStoreBuilder.java index d8a3128bf6..5aa0c57b8c 100644 --- a/core/src/main/java/org/neo4j/gds/core/loading/nodeproperties/NodePropertiesFromStoreBuilder.java +++ b/core/src/main/java/org/neo4j/gds/core/loading/nodeproperties/NodePropertiesFromStoreBuilder.java @@ -30,18 +30,17 @@ import org.neo4j.gds.mem.MemoryEstimations; import org.neo4j.gds.values.GdsNoValue; import org.neo4j.gds.values.GdsValue; +import org.neo4j.gds.values.primitive.PrimitiveValues; import org.neo4j.values.storable.DoubleArray; import org.neo4j.values.storable.FloatArray; import org.neo4j.values.storable.FloatingPointValue; import org.neo4j.values.storable.IntegralValue; import org.neo4j.values.storable.LongArray; import org.neo4j.values.storable.Value; -import org.neo4j.values.storable.Values; import java.util.concurrent.atomic.AtomicReference; import static org.neo4j.gds.utils.StringFormatting.formatWithLocale; -import static org.neo4j.values.storable.Values.NO_VALUE; public final class NodePropertiesFromStoreBuilder { @@ -80,15 +79,6 @@ private NodePropertiesFromStoreBuilder( this.innerBuilder = new AtomicReference<>(); } - public void set(long neoNodeId, Value value) { - if (value != null && value != NO_VALUE) { - if (innerBuilder.get() == null) { - initializeWithType(value); - } - innerBuilder.get().setValue(neoNodeId, value); - } - } - public void set(long neoNodeId, GdsValue value) { if (value != null && value != GdsNoValue.NO_VALUE) { if (innerBuilder.get() == null) { @@ -101,7 +91,8 @@ public void set(long neoNodeId, GdsValue value) { public NodePropertyValues build(IdMap idMap) { if (innerBuilder.get() == null) { if (defaultValue.getObject() != null) { - initializeWithType(Values.of(defaultValue.getObject())); + var gdsValue = PrimitiveValues.create(defaultValue.getObject()); + initializeWithType(gdsValue); } else { throw new IllegalStateException("Cannot infer type of property"); } @@ -116,15 +107,6 @@ public NodePropertyValues build(IdMap idMap) { return innerBuilder.get().build(idMap.nodeCount(), actualIdMap, idMap.highestOriginalId()); } - // This is synchronized as we want to prevent the creation of multiple InnerNodePropertiesBuilders of which only once survives. - private synchronized void initializeWithType(Value value) { - if (innerBuilder.get() == null) { - var valueType = valueType(value); - var newBuilder = newInnerBuilder(valueType); - innerBuilder.compareAndSet(null, newBuilder); - } - } - // This is synchronized as we want to prevent the creation of multiple InnerNodePropertiesBuilders of which only once survives. private synchronized void initializeWithType(GdsValue value) { if (innerBuilder.get() == null) { diff --git a/native-projection/src/main/java/org/neo4j/gds/projection/NativeNodePropertyImporter.java b/native-projection/src/main/java/org/neo4j/gds/projection/NativeNodePropertyImporter.java index 0b2a65ed09..adf7765061 100644 --- a/native-projection/src/main/java/org/neo4j/gds/projection/NativeNodePropertyImporter.java +++ b/native-projection/src/main/java/org/neo4j/gds/projection/NativeNodePropertyImporter.java @@ -32,6 +32,7 @@ import org.neo4j.gds.config.ConcurrencyConfig; import org.neo4j.gds.core.GraphDimensions; import org.neo4j.gds.core.concurrency.Concurrency; +import org.neo4j.gds.core.loading.GdsNeo4jValueConverter; import org.neo4j.gds.core.loading.NodeLabelTokenSet; import org.neo4j.gds.core.loading.nodeproperties.NodePropertiesFromStoreBuilder; import org.neo4j.internal.kernel.api.PropertyCursor; @@ -142,7 +143,8 @@ private int setPropertyValue( Value value = propertyCursor.propertyValue(); for (NodePropertiesFromStoreBuilder builder : builders) { - builder.set(neoNodeId, value); + var gdsValue = GdsNeo4jValueConverter.toValue(value); + builder.set(neoNodeId, gdsValue); propertiesImported++; } } From 28d94b9f4bd18e14c55c53f8cd1e6c3523d538dd Mon Sep 17 00:00:00 2001 From: Paul Horn Date: Wed, 4 Dec 2024 11:24:00 +0100 Subject: [PATCH 2/2] Restore previous exception message --- .../NodePropertiesFromStoreBuilder.java | 25 ------------------- .../NativeNodePropertyImporter.java | 22 ++++++++++++++++ 2 files changed, 22 insertions(+), 25 deletions(-) diff --git a/core/src/main/java/org/neo4j/gds/core/loading/nodeproperties/NodePropertiesFromStoreBuilder.java b/core/src/main/java/org/neo4j/gds/core/loading/nodeproperties/NodePropertiesFromStoreBuilder.java index 5aa0c57b8c..3db8531f07 100644 --- a/core/src/main/java/org/neo4j/gds/core/loading/nodeproperties/NodePropertiesFromStoreBuilder.java +++ b/core/src/main/java/org/neo4j/gds/core/loading/nodeproperties/NodePropertiesFromStoreBuilder.java @@ -31,12 +31,6 @@ import org.neo4j.gds.values.GdsNoValue; import org.neo4j.gds.values.GdsValue; import org.neo4j.gds.values.primitive.PrimitiveValues; -import org.neo4j.values.storable.DoubleArray; -import org.neo4j.values.storable.FloatArray; -import org.neo4j.values.storable.FloatingPointValue; -import org.neo4j.values.storable.IntegralValue; -import org.neo4j.values.storable.LongArray; -import org.neo4j.values.storable.Value; import java.util.concurrent.atomic.AtomicReference; @@ -134,23 +128,4 @@ private InnerNodePropertiesBuilder newInnerBuilder(ValueType valueType) { )); } } - - private ValueType valueType(Value value) { - if (value instanceof IntegralValue) { - return ValueType.LONG; - } else if (value instanceof FloatingPointValue) { - return ValueType.DOUBLE; - } else if (value instanceof LongArray) { - return ValueType.LONG_ARRAY; - } else if (value instanceof DoubleArray) { - return ValueType.DOUBLE_ARRAY; - } else if (value instanceof FloatArray) { - return ValueType.FLOAT_ARRAY; - } else { - throw new UnsupportedOperationException(formatWithLocale( - "Loading of values of type %s is currently not supported", - value.getTypeName() - )); - } - } } diff --git a/native-projection/src/main/java/org/neo4j/gds/projection/NativeNodePropertyImporter.java b/native-projection/src/main/java/org/neo4j/gds/projection/NativeNodePropertyImporter.java index adf7765061..e3c3c9c053 100644 --- a/native-projection/src/main/java/org/neo4j/gds/projection/NativeNodePropertyImporter.java +++ b/native-projection/src/main/java/org/neo4j/gds/projection/NativeNodePropertyImporter.java @@ -39,6 +39,11 @@ import org.neo4j.kernel.api.KernelTransaction; import org.neo4j.storageengine.api.PropertySelection; import org.neo4j.storageengine.api.Reference; +import org.neo4j.values.storable.DoubleArray; +import org.neo4j.values.storable.FloatArray; +import org.neo4j.values.storable.FloatingPointValue; +import org.neo4j.values.storable.IntegralValue; +import org.neo4j.values.storable.LongArray; import org.neo4j.values.storable.Value; import java.util.ArrayList; @@ -50,6 +55,7 @@ import static java.util.stream.Collectors.toMap; import static org.neo4j.gds.core.GraphDimensions.ANY_LABEL; import static org.neo4j.gds.core.GraphDimensions.IGNORE; +import static org.neo4j.gds.utils.StringFormatting.formatWithLocale; public final class NativeNodePropertyImporter { @@ -143,6 +149,7 @@ private int setPropertyValue( Value value = propertyCursor.propertyValue(); for (NodePropertiesFromStoreBuilder builder : builders) { + verifyValueType(value); var gdsValue = GdsNeo4jValueConverter.toValue(value); builder.set(neoNodeId, gdsValue); propertiesImported++; @@ -152,6 +159,21 @@ private int setPropertyValue( return propertiesImported; } + private void verifyValueType(Value value) { + if (!( + value instanceof IntegralValue || + value instanceof FloatingPointValue || + value instanceof LongArray || + value instanceof DoubleArray || + value instanceof FloatArray + )) { + throw new UnsupportedOperationException(formatWithLocale( + "Loading of values of type %s is currently not supported", + value.getTypeName() + )); + } + } + public static final class Builder { private Concurrency concurrency = ConcurrencyConfig.TYPED_DEFAULT_CONCURRENCY; private Map propertyMappings;