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

org.neo4j.kernel.impl.index.schema.NativeIndexReader 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.

There is a newer version: 5.26.0
Show 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.index.schema;

import static org.apache.commons.lang3.exception.ExceptionUtils.getRootCause;
import static org.neo4j.kernel.impl.index.schema.NativeIndexKey.Inclusion.NEUTRAL;

import java.io.IOException;
import java.io.UncheckedIOException;
import java.util.Collections;
import java.util.List;
import java.util.concurrent.atomic.AtomicInteger;
import org.neo4j.index.internal.gbptree.GBPTree;
import org.neo4j.index.internal.gbptree.Seeker;
import org.neo4j.internal.kernel.api.IndexQueryConstraints;
import org.neo4j.internal.kernel.api.PropertyIndexQuery;
import org.neo4j.internal.kernel.api.PropertyIndexQuery.IncomparableExactPredicate;
import org.neo4j.internal.kernel.api.PropertyIndexQuery.IncomparableRangePredicate;
import org.neo4j.internal.kernel.api.QueryContext;
import org.neo4j.internal.kernel.api.exceptions.schema.IndexNotFoundKernelException;
import org.neo4j.internal.schema.IndexDescriptor;
import org.neo4j.internal.schema.IndexOrder;
import org.neo4j.io.pagecache.context.CursorContext;
import org.neo4j.io.pagecache.impl.FileIsNotMappedException;
import org.neo4j.kernel.api.index.IndexProgressor;
import org.neo4j.kernel.api.index.IndexSampler;
import org.neo4j.kernel.api.index.ValueIndexReader;
import org.neo4j.util.Preconditions;
import org.neo4j.values.storable.Value;

abstract class NativeIndexReader> implements ValueIndexReader {
    protected final IndexDescriptor descriptor;
    protected final IndexUsageTracking usageTracker;
    final IndexLayout layout;
    final GBPTree tree;

    NativeIndexReader(
            GBPTree tree,
            IndexLayout layout,
            IndexDescriptor descriptor,
            IndexUsageTracking usageTracker) {
        this.tree = tree;
        this.layout = layout;
        this.descriptor = descriptor;
        this.usageTracker = usageTracker;
    }

    @Override
    public void close() {}

    @Override
    public IndexSampler createSampler() {
        // For a unique index there's an optimization, knowing that all values in it are unique, to simply count
        // the number of indexed values and create a sample for that count. The GBPTree doesn't have an O(1)
        // count mechanism, it will have to manually count the indexed values in it to get it.
        // For that reason this implementation opts for keeping complexity down by just using the existing
        // non-unique sampler which scans the index and counts (potentially duplicates, of which there will
        // be none in a unique index).

        FullScanNonUniqueIndexSampler sampler = new FullScanNonUniqueIndexSampler<>(tree, layout);
        return (cursorContext, stopped) -> {
            try {
                return sampler.sample(cursorContext, stopped);
            } catch (UncheckedIOException e) {
                if (getRootCause(e) instanceof FileIsNotMappedException) {
                    IndexNotFoundKernelException exception =
                            new IndexNotFoundKernelException("Index dropped while sampling.");
                    exception.addSuppressed(e);
                    throw exception;
                }
                throw e;
            }
        };
    }

    @Override
    public long countIndexedEntities(
            long entityId, CursorContext cursorContext, int[] propertyKeyIds, Value... propertyValues) {
        KEY treeKeyFrom = layout.newKey();
        KEY treeKeyTo = layout.newKey();
        treeKeyFrom.initialize(entityId);
        treeKeyTo.initialize(entityId);
        for (int i = 0; i < propertyValues.length; i++) {
            treeKeyFrom.initFromValue(i, propertyValues[i], NEUTRAL);
            treeKeyTo.initFromValue(i, propertyValues[i], NEUTRAL);
        }
        try (Seeker seeker = tree.seek(treeKeyFrom, treeKeyTo, cursorContext)) {
            long count = 0;
            while (seeker.next()) {
                if (seeker.key().getEntityId() == entityId) {
                    count++;
                }
            }
            return count;
        } catch (IOException e) {
            throw new UncheckedIOException(e);
        }
    }

    @Override
    public void query(
            IndexProgressor.EntityValueClient cursor,
            QueryContext context,
            IndexQueryConstraints constraints,
            PropertyIndexQuery... predicates) {
        validateQuery(constraints, predicates);
        context.monitor().queried(descriptor);
        usageTracker.queried();

        KEY treeKeyFrom = layout.newKey();
        KEY treeKeyTo = layout.newKey();
        initializeFromToKeys(treeKeyFrom, treeKeyTo);

        boolean needFilter = initializeRangeForQuery(treeKeyFrom, treeKeyTo, predicates);
        startSeekForInitializedRange(
                cursor, treeKeyFrom, treeKeyTo, context.cursorContext(), needFilter, constraints, predicates);
    }

    void initializeFromToKeys(KEY treeKeyFrom, KEY treeKeyTo) {
        treeKeyFrom.initialize(Long.MIN_VALUE);
        treeKeyTo.initialize(Long.MAX_VALUE);
    }

    abstract void validateQuery(IndexQueryConstraints constraints, PropertyIndexQuery... predicates);

    /**
     * @return true if query results from seek will need to be filtered through the predicates, else false
     */
    abstract boolean initializeRangeForQuery(KEY treeKeyFrom, KEY treeKeyTo, PropertyIndexQuery... predicates);

    void startSeekForInitializedRange(
            IndexProgressor.EntityValueClient client,
            KEY treeKeyFrom,
            KEY treeKeyTo,
            CursorContext cursorContext,
            boolean needFilter,
            IndexQueryConstraints constraints,
            PropertyIndexQuery... query) {
        if (isEmptyRange(treeKeyFrom, treeKeyTo) || isEmptyResultQuery(query)) {
            client.initializeQuery(descriptor, IndexProgressor.EMPTY, false, false, constraints, query);
            return;
        }
        try {
            Seeker seeker = makeIndexSeeker(treeKeyFrom, treeKeyTo, constraints.order(), cursorContext);
            IndexProgressor hitProgressor = getIndexProgressor(seeker, client, needFilter, query);
            client.initializeQuery(descriptor, hitProgressor, false, false, constraints, query);
        } catch (IOException e) {
            throw new UncheckedIOException(e);
        }
    }

    Seeker makeIndexSeeker(
            KEY treeKeyFrom, KEY treeKeyTo, IndexOrder indexOrder, CursorContext cursorContext) throws IOException {
        if (indexOrder == IndexOrder.DESCENDING) {
            KEY tmpKey = treeKeyFrom;
            treeKeyFrom = treeKeyTo;
            treeKeyTo = tmpKey;
        }
        return tree.seek(treeKeyFrom, treeKeyTo, cursorContext);
    }

    private IndexProgressor getIndexProgressor(
            Seeker seeker,
            IndexProgressor.EntityValueClient client,
            boolean needFilter,
            PropertyIndexQuery... query) {
        return needFilter
                ? new FilteringNativeHitIndexProgressor<>(seeker, client, query)
                : new NativeHitIndexProgressor<>(seeker, client);
    }

    private boolean isEmptyRange(KEY treeKeyFrom, KEY treeKeyTo) {
        return layout.compare(treeKeyFrom, treeKeyTo) > 0;
    }

    private boolean isEmptyResultQuery(PropertyIndexQuery... predicates) {
        for (PropertyIndexQuery predicate : predicates) {
            if (predicate instanceof IncomparableRangePredicate || predicate instanceof IncomparableExactPredicate) {
                return true;
            }
        }

        return false;
    }

    @Override
    public PartitionedValueSeek valueSeek(
            int desiredNumberOfPartitions, QueryContext queryContext, PropertyIndexQuery... query) {
        try {
            return new NativePartitionedValueSeek(desiredNumberOfPartitions, queryContext, query);
        } catch (IOException e) {
            throw new UncheckedIOException(e);
        }
    }

    class NativePartitionedValueSeek implements PartitionedValueSeek {
        private final PropertyIndexQuery[] query;
        private final boolean filter;
        private final List partitionEdges;
        private final AtomicInteger nextFrom = new AtomicInteger();

        NativePartitionedValueSeek(
                int desiredNumberOfPartitions, QueryContext queryContext, PropertyIndexQuery... query)
                throws IOException {
            Preconditions.requirePositive(desiredNumberOfPartitions);
            validateQuery(IndexQueryConstraints.unorderedValues(), query);
            usageTracker.queried();
            this.query = query;

            final var fromInclusive = layout.newKey();
            final var toExclusive = layout.newKey();
            initializeFromToKeys(fromInclusive, toExclusive);

            filter = initializeRangeForQuery(fromInclusive, toExclusive, this.query);

            partitionEdges = !isEmptyResultQuery(query)
                    ? tree.partitionedSeek(
                            fromInclusive, toExclusive, desiredNumberOfPartitions, queryContext.cursorContext())
                    : Collections.emptyList();
        }

        @Override
        public int getNumberOfPartitions() {
            return partitionEdges.size() - 1;
        }

        @Override
        public IndexProgressor reservePartition(IndexProgressor.EntityValueClient client, CursorContext cursorContext) {
            final var from = nextFrom.getAndIncrement();
            final var to = from + 1;
            if (to >= partitionEdges.size()) {
                return IndexProgressor.EMPTY;
            }
            try {
                final var fromInclusive = partitionEdges.get(from);
                final var toExclusive = partitionEdges.get(to);
                return getIndexProgressor(tree.seek(fromInclusive, toExclusive, cursorContext), client, filter, query);
            } catch (IOException e) {
                throw new UncheckedIOException(e);
            }
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy