All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.neo4j.kernel.impl.newapi.IndexTxStateUpdater Maven / Gradle / Ivy

Go to download

Neo4j kernel is a lightweight, embedded Java database designed to store data structured as graphs rather than tables. For more information, see http://neo4j.org.

The newest version!
/*
 * Copyright (c) "Neo4j"
 * Neo4j Sweden AB [https://neo4j.com]
 *
 * This file is part of Neo4j.
 *
 * Neo4j 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.
 *
 * This program 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 this program.  If not, see .
 */
package org.neo4j.kernel.impl.newapi;

import static org.neo4j.common.EntityType.NODE;
import static org.neo4j.common.EntityType.RELATIONSHIP;
import static org.neo4j.kernel.api.StatementConstants.NO_SUCH_PROPERTY_KEY;
import static org.neo4j.token.api.TokenConstants.ANY_PROPERTY_KEY;
import static org.neo4j.values.storable.Values.NO_VALUE;

import java.util.Arrays;
import java.util.Collection;
import org.apache.commons.lang3.ArrayUtils;
import org.eclipse.collections.api.list.primitive.MutableIntList;
import org.eclipse.collections.api.map.primitive.MutableIntObjectMap;
import org.eclipse.collections.impl.factory.primitive.IntLists;
import org.eclipse.collections.impl.factory.primitive.IntObjectMaps;
import org.neo4j.collection.PrimitiveArrays;
import org.neo4j.common.EntityType;
import org.neo4j.internal.kernel.api.EntityCursor;
import org.neo4j.internal.kernel.api.NodeCursor;
import org.neo4j.internal.kernel.api.PropertyCursor;
import org.neo4j.internal.kernel.api.RelationshipScanCursor;
import org.neo4j.internal.schema.FulltextSchemaDescriptor;
import org.neo4j.internal.schema.IndexDescriptor;
import org.neo4j.internal.schema.SchemaDescriptor;
import org.neo4j.kernel.api.txstate.TxStateHolder;
import org.neo4j.kernel.impl.api.index.IndexingService;
import org.neo4j.memory.MemoryTracker;
import org.neo4j.storageengine.api.PropertySelection;
import org.neo4j.storageengine.api.StorageReader;
import org.neo4j.storageengine.api.txstate.TransactionStateBehaviour;
import org.neo4j.values.storable.Value;
import org.neo4j.values.storable.ValueTuple;

/**
 * Utility class that performs necessary updates for the transaction state.
 */
public class IndexTxStateUpdater {
    private final StorageReader storageReader;
    private final TxStateHolder txStateHolder;
    private final IndexingService indexingService;
    private final TransactionStateBehaviour stateBehaviour;

    // We can use the StorageReader directly instead of the SchemaReadOps, because we know that in transactions
    // where this class is needed we will never have index changes.
    public IndexTxStateUpdater(
            StorageReader storageReader,
            IndexingService indexingService,
            TxStateHolder txStateHolder,
            TransactionStateBehaviour stateBehaviour) {
        this.storageReader = storageReader;
        this.txStateHolder = txStateHolder;
        this.indexingService = indexingService;
        this.stateBehaviour = stateBehaviour;
    }

    // LABEL CHANGES

    public enum LabelChangeType {
        ADDED_LABEL,
        REMOVED_LABEL
    }

    /**
     * A label has been changed, figure out what updates are needed to tx state.
     *
     * @param node cursor to the node where the change was applied
     * @param propertyCursor cursor to the properties of node
     * @param changeType The type of change event
     * @param indexes the indexes related to the node
     */
    void onLabelChange(
            NodeCursor node,
            PropertyCursor propertyCursor,
            LabelChangeType changeType,
            int removedLabelId,
            Collection indexes) {
        assert noSchemaChangedInTx();

        // Check all indexes of the changed label
        if (!indexes.isEmpty()) {
            MutableIntObjectMap materializedProperties = IntObjectMaps.mutable.empty();
            for (IndexDescriptor index : indexes) {
                MemoryTracker memoryTracker = txStateHolder.txState().memoryTracker();
                if (stateBehaviour.useIndexCommands()
                        && changeType == LabelChangeType.REMOVED_LABEL
                        && index.schema().isSchemaDescriptorType(FulltextSchemaDescriptor.class)
                        && isFullTextStillCovered(node, removedLabelId, index)) {
                    continue;
                }
                int[] indexPropertyIds = index.schema().getPropertyIds();
                Value[] values = getValueTuple(
                        node,
                        propertyCursor,
                        NO_SUCH_PROPERTY_KEY,
                        NO_VALUE,
                        indexPropertyIds,
                        materializedProperties,
                        memoryTracker);
                ValueTuple valueTuple = ValueTuple.of(values);
                memoryTracker.allocateHeap(valueTuple.getShallowSize());
                switch (changeType) {
                    case ADDED_LABEL -> {
                        indexingService.validateBeforeCommit(index, values, node.nodeReference());
                        txStateHolder.txState().indexDoUpdateEntry(index, node.nodeReference(), null, valueTuple);
                    }
                    case REMOVED_LABEL -> txStateHolder
                            .txState()
                            .indexDoUpdateEntry(index, node.nodeReference(), valueTuple, null);
                }
            }
        }
    }

    void onPropertyAdd(
            NodeCursor node,
            PropertyCursor propertyCursor,
            int[] labels,
            int propertyKeyId,
            int[] existingPropertyKeyIds,
            Value value) {
        onPropertyAdd(node, NODE, propertyCursor, labels, propertyKeyId, existingPropertyKeyIds, value);
    }

    void onPropertyRemove(
            NodeCursor node,
            PropertyCursor propertyCursor,
            int[] labels,
            int propertyKeyId,
            int[] existingPropertyKeyIds,
            Value value) {
        onPropertyRemove(node, NODE, propertyCursor, labels, propertyKeyId, existingPropertyKeyIds, value);
    }

    void onPropertyChange(
            NodeCursor node,
            PropertyCursor propertyCursor,
            int[] labels,
            int propertyKeyId,
            int[] existingPropertyKeyIds,
            Value beforeValue,
            Value afterValue) {
        onPropertyChange(
                node, NODE, propertyCursor, labels, propertyKeyId, existingPropertyKeyIds, beforeValue, afterValue);
    }

    void onPropertyAdd(
            RelationshipScanCursor relationship,
            PropertyCursor propertyCursor,
            int type,
            int propertyKeyId,
            int[] existingPropertyKeyIds,
            Value value) {
        onPropertyAdd(
                relationship,
                RELATIONSHIP,
                propertyCursor,
                new int[] {type},
                propertyKeyId,
                existingPropertyKeyIds,
                value);
    }

    void onPropertyRemove(
            RelationshipScanCursor relationship,
            PropertyCursor propertyCursor,
            int type,
            int propertyKeyId,
            int[] existingPropertyKeyIds,
            Value value) {
        onPropertyRemove(
                relationship,
                RELATIONSHIP,
                propertyCursor,
                new int[] {type},
                propertyKeyId,
                existingPropertyKeyIds,
                value);
    }

    void onPropertyChange(
            RelationshipScanCursor relationship,
            PropertyCursor propertyCursor,
            int type,
            int propertyKeyId,
            int[] existingPropertyKeyIds,
            Value beforeValue,
            Value afterValue) {
        onPropertyChange(
                relationship,
                RELATIONSHIP,
                propertyCursor,
                new int[] {type},
                propertyKeyId,
                existingPropertyKeyIds,
                beforeValue,
                afterValue);
    }

    void onDeleteUncreated(NodeCursor node, PropertyCursor propertyCursor) {
        onDeleteUncreated(node, NODE, propertyCursor, node.labels().all());
    }

    void onDeleteUncreated(RelationshipScanCursor relationship, PropertyCursor propertyCursor) {
        onDeleteUncreated(relationship, RELATIONSHIP, propertyCursor, new int[] {relationship.type()});
    }

    private boolean noSchemaChangedInTx() {
        var txState = txStateHolder.txState();
        return !(txState.hasChanges() && !txState.hasDataChanges());
    }

    // PROPERTY CHANGES

    /**
     * Creating an entity with its data in a transaction adds also adds that state to index transaction state (for matching indexes).
     * When deleting an entity this method will delete this state from the index transaction state.
     *
     * @param entity entity that was deleted.
     * @param propertyCursor property cursor for accessing the properties of the entity.
     * @param tokens the entity tokens this entity has.
     */
    private void onDeleteUncreated(
            EntityCursor entity, EntityType entityType, PropertyCursor propertyCursor, int[] tokens) {
        assert noSchemaChangedInTx();
        entity.properties(propertyCursor, PropertySelection.ALL_PROPERTY_KEYS);
        MutableIntList propertyKeyList = IntLists.mutable.empty();
        while (propertyCursor.next()) {
            propertyKeyList.add(propertyCursor.propertyKey());
        }
        // Make sure to sort the propertyKeyIds since SchemaMatcher.onMatchingSchema requires it.
        int[] propertyKeyIds = propertyKeyList.toSortedArray();
        var indexes = storageReader.valueIndexesGetRelated(tokens, propertyKeyIds, entityType);
        if (indexes.isEmpty()) {
            return;
        }
        processIndexUpdates(entity, propertyCursor, indexes, ANY_PROPERTY_KEY, NO_VALUE, propertyKeyIds);
    }

    private void processIndexUpdates(
            EntityCursor entity,
            PropertyCursor propertyCursor,
            Collection indexes,
            int propertyKeyId,
            Value changedValue,
            int[] propertyKeyIds) {
        MutableIntObjectMap materializedProperties = IntObjectMaps.mutable.empty();
        SchemaMatcher.onMatchingSchema(indexes.iterator(), propertyKeyId, propertyKeyIds, stateBehaviour, index -> {
            MemoryTracker memoryTracker = txStateHolder.txState().memoryTracker();
            SchemaDescriptor schema = index.schema();
            Value[] values = getValueTuple(
                    entity,
                    propertyCursor,
                    propertyKeyId,
                    changedValue,
                    schema.getPropertyIds(),
                    materializedProperties,
                    memoryTracker);
            ValueTuple valueTuple = ValueTuple.of(values);
            memoryTracker.allocateHeap(valueTuple.getShallowSize());
            txStateHolder.txState().indexDoUpdateEntry(index, entity.reference(), valueTuple, null);
        });
    }

    private void onPropertyAdd(
            EntityCursor entity,
            EntityType entityType,
            PropertyCursor propertyCursor,
            int[] tokens,
            int propertyKeyId,
            int[] existingPropertyKeyIds,
            Value value) {
        assert noSchemaChangedInTx();
        Collection indexes = storageReader.valueIndexesGetRelated(tokens, propertyKeyId, entityType);
        if (indexes.isEmpty()) {
            return;
        }
        MutableIntObjectMap materializedProperties = IntObjectMaps.mutable.empty();
        SchemaMatcher.onMatchingSchema(
                indexes.iterator(), propertyKeyId, existingPropertyKeyIds, stateBehaviour, index -> {
                    MemoryTracker memoryTracker = txStateHolder.txState().memoryTracker();
                    SchemaDescriptor schema = index.schema();
                    Value[] values = getValueTuple(
                            entity,
                            propertyCursor,
                            propertyKeyId,
                            value,
                            schema.getPropertyIds(),
                            materializedProperties,
                            memoryTracker);
                    indexingService.validateBeforeCommit(index, values, entity.reference());
                    ValueTuple valueTuple = ValueTuple.of(values);
                    memoryTracker.allocateHeap(valueTuple.getShallowSize());

                    ValueTuple beforeValueTuple = null;
                    if (stateBehaviour.useIndexCommands() && schema.getPropertyIds().length > 1) {
                        memoryTracker.allocateHeap(valueTuple.getShallowSize());
                        Value[] valuesBefore = Arrays.copyOf(values, values.length);
                        int k = ArrayUtils.indexOf(schema.getPropertyIds(), propertyKeyId);
                        valuesBefore[k] = NO_VALUE;
                        beforeValueTuple = ValueTuple.of(valuesBefore);
                    }
                    txStateHolder.txState().indexDoUpdateEntry(index, entity.reference(), beforeValueTuple, valueTuple);
                });
    }

    private void onPropertyRemove(
            EntityCursor entity,
            EntityType entityType,
            PropertyCursor propertyCursor,
            int[] tokens,
            int propertyKeyId,
            int[] existingPropertyKeyIds,
            Value value) {
        assert noSchemaChangedInTx();
        Collection indexes = storageReader.valueIndexesGetRelated(tokens, propertyKeyId, entityType);
        if (indexes.isEmpty()) {
            return;
        }
        processIndexUpdates(entity, propertyCursor, indexes, propertyKeyId, value, existingPropertyKeyIds);
    }

    private void onPropertyChange(
            EntityCursor entity,
            EntityType entityType,
            PropertyCursor propertyCursor,
            int[] tokens,
            int propertyKeyId,
            int[] existingPropertyKeyIds,
            Value beforeValue,
            Value afterValue) {
        assert noSchemaChangedInTx();
        Collection indexes = storageReader.valueIndexesGetRelated(tokens, propertyKeyId, entityType);
        if (indexes.isEmpty()) {
            return;
        }
        MutableIntObjectMap materializedProperties = IntObjectMaps.mutable.empty();
        SchemaMatcher.onMatchingSchema(
                indexes.iterator(), propertyKeyId, existingPropertyKeyIds, stateBehaviour, index -> {
                    MemoryTracker memoryTracker = txStateHolder.txState().memoryTracker();
                    SchemaDescriptor schema = index.schema();
                    int[] propertyIds = schema.getPropertyIds();
                    Value[] valuesAfter = getValueTuple(
                            entity,
                            propertyCursor,
                            propertyKeyId,
                            afterValue,
                            propertyIds,
                            materializedProperties,
                            memoryTracker);

                    // The valuesBefore tuple is just like valuesAfter, except is has the afterValue instead of the
                    // beforeValue
                    Value[] valuesBefore = Arrays.copyOf(valuesAfter, valuesAfter.length);
                    int k = ArrayUtils.indexOf(propertyIds, propertyKeyId);
                    valuesBefore[k] = beforeValue;

                    indexingService.validateBeforeCommit(index, valuesAfter, entity.reference());
                    ValueTuple valuesTupleBefore = ValueTuple.of(valuesBefore);
                    ValueTuple valuesTupleAfter = ValueTuple.of(valuesAfter);
                    memoryTracker.allocateHeap(
                            valuesTupleBefore.getShallowSize() * 2); // They are copies and same shallow size
                    txStateHolder
                            .txState()
                            .indexDoUpdateEntry(index, entity.reference(), valuesTupleBefore, valuesTupleAfter);
                });
    }

    private static Value[] getValueTuple(
            EntityCursor entity,
            PropertyCursor propertyCursor,
            int changedPropertyKeyId,
            Value changedValue,
            int[] indexPropertyIds,
            MutableIntObjectMap materializedValues,
            MemoryTracker memoryTracker) {
        Value[] values = new Value[indexPropertyIds.length];
        int missing = 0;

        // First get whatever values we already have on the stack, like the value change that provoked this update in
        // the first place
        // and already loaded values that we can get from the map of materialized values.
        for (int k = 0; k < indexPropertyIds.length; k++) {
            values[k] = indexPropertyIds[k] == changedPropertyKeyId
                    ? changedValue
                    : materializedValues.getIfAbsent(indexPropertyIds[k], () -> NO_VALUE);
            if (values[k] == NO_VALUE) {
                missing++;
            }
        }

        // If we couldn't get all values that we wanted we need to load from the entity. While we're loading values
        // we'll place those values in the map so that other index updates from this change can just used them.
        if (missing > 0) {
            entity.properties(propertyCursor, PropertySelection.selection(indexPropertyIds));
            while (missing > 0 && propertyCursor.next()) {
                int k = ArrayUtils.indexOf(indexPropertyIds, propertyCursor.propertyKey());
                assert k >= 0;
                if (values[k] == NO_VALUE) {
                    int propertyKeyId = indexPropertyIds[k];
                    boolean thisIsTheChangedProperty = propertyKeyId == changedPropertyKeyId;
                    values[k] = thisIsTheChangedProperty ? changedValue : propertyCursor.propertyValue();
                    if (!thisIsTheChangedProperty) {
                        materializedValues.put(propertyKeyId, values[k]);
                        memoryTracker.allocateHeap(values[k].estimatedHeapUsage());
                    }
                    missing--;
                }
            }
        }

        return values;
    }

    private static boolean isFullTextStillCovered(NodeCursor node, int removedLabelId, IndexDescriptor index) {
        int[] nodeLabels = node.labels().all();
        int[] entityTokenIds = index.schema().getEntityTokenIds();
        for (int nodeLabel : nodeLabels) {
            if (nodeLabel == removedLabelId) {
                continue;
            }
            if (PrimitiveArrays.contains(entityTokenIds, nodeLabel)) {
                return true;
            }
        }
        return false;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy