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

org.neo4j.kernel.impl.index.schema.TokenIndexUpdater 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.neo4j.index.internal.gbptree.ValueMerger.MergeResult.MERGED;
import static org.neo4j.index.internal.gbptree.ValueMerger.MergeResult.REMOVED;

import java.util.Arrays;
import org.neo4j.index.internal.gbptree.GBPTree;
import org.neo4j.index.internal.gbptree.ValueMerger;
import org.neo4j.index.internal.gbptree.Writer;
import org.neo4j.io.IOUtils;
import org.neo4j.io.pagecache.context.CursorContext;
import org.neo4j.kernel.api.exceptions.index.IndexEntryConflictException;
import org.neo4j.kernel.api.index.IndexUpdater;
import org.neo4j.kernel.api.index.TokenIndexReader;
import org.neo4j.kernel.impl.index.schema.PhysicalToLogicalTokenChanges.LogicalTokenUpdates;
import org.neo4j.storageengine.api.IndexEntryUpdate;
import org.neo4j.storageengine.api.TokenIndexEntryUpdate;

/**
 * {@link IndexUpdater} for token index, or rather a {@link Writer} for its
 * internal {@link GBPTree}.
 * 

* {@link #process(IndexEntryUpdate) updates} are queued up to a maximum batch size and, for performance, * applied in sorted order (by the token and entity id) when reaches batch size or on {@link #close()}. *

* Updates aren't visible to {@link TokenIndexReader readers} immediately, rather when queue happens to be applied. *

* Incoming {@link TokenIndexEntryUpdate updates} are actually modified from representing physical before/after * state to represent logical to-add/to-remove state. These changes are done directly inside the provided * {@link TokenIndexEntryUpdate#values()} and {@link TokenIndexEntryUpdate#beforeValues()} arrays, * relying on the fact that those arrays are returned in its essential form, instead of copies. * This conversion is done like so mostly to reduce garbage. * * @see PhysicalToLogicalTokenChanges */ class TokenIndexUpdater implements IndexUpdater { /** * {@link ValueMerger} used for adding token->entity mappings, see {@link TokenScanValue#add(TokenScanValue)}. */ private static final ValueMerger ADD_MERGER = (existingKey, newKey, existingValue, newValue) -> { existingValue.add(newValue); return MERGED; }; /** * {@link ValueMerger} used for removing token->entity mappings, see {@link TokenScanValue#remove(TokenScanValue)}. */ private static final ValueMerger REMOVE_MERGER = (existingKey, newKey, existingValue, newValue) -> { existingValue.remove(newValue); return existingValue.isEmpty() ? REMOVED : MERGED; }; /** * {@link Writer} acquired when acquiring this {@link TokenIndexUpdater}, * acquired from {@link GBPTree#writer(CursorContext)}. */ private Writer writer; /** * Instance of {@link TokenScanKey} acting as place to read keys into and to set for each applied update. */ private final TokenScanKey key = new TokenScanKey(); /** * Instance of {@link TokenScanValue} acting as place to read values into and to update * for each applied update. */ private final TokenScanValue value = new TokenScanValue(); /** * Batch currently building up as {@link #process(IndexEntryUpdate) updates} come in. Cursor for where * to place new updates is {@link #pendingUpdatesCursor}. The constructor set the length of this queue * and the length defines the maximum batch size. */ private final LogicalTokenUpdates[] pendingUpdates; private final TokenIndexIdLayout idLayout; /** * Cursor into {@link #pendingUpdates}, where to place new {@link #process(IndexEntryUpdate) updates}. * When full the batch is applied and this cursor reset to {@code 0}. */ private int pendingUpdatesCursor; /** * There are two levels of batching, one for {@link TokenIndexEntryUpdate updates} and one when applying. * This variable helps keeping track of the second level where updates to the actual {@link GBPTree} * are batched per entity id range, i.e. to add several tokenId->entityId mappings falling into the same * range, all of those updates are made into one {@link TokenScanValue} and then issues as one update * to the tree. There are additions and removals, this variable keeps track of which. */ private boolean addition; /** * When applying {@link TokenIndexEntryUpdate updates} (when batch full or in {@link #close()}), updates are * applied tokenId by tokenId. All updates are scanned through multiple times, with one token in mind at a time. * For each round the current round tries to figure out which is the closest higher tokenId to apply * in the next round. This variable keeps track of that next tokenId. */ private int lowestTokenId; private boolean closed = true; private boolean parallel; TokenIndexUpdater(int batchSize, TokenIndexIdLayout idLayout) { this.pendingUpdates = new LogicalTokenUpdates[batchSize]; this.idLayout = idLayout; } TokenIndexUpdater initialize(Writer writer, boolean parallel) { this.parallel = parallel; if (!closed) { throw new IllegalStateException("Updater still open"); } this.writer = writer; this.pendingUpdatesCursor = 0; this.addition = false; this.lowestTokenId = Integer.MAX_VALUE; closed = false; return this; } /** * Queues a {@link TokenIndexEntryUpdate} to this writer for applying when batch gets full, * or when {@link #close() closing}. * * Calls to this method MUST be ordered by ascending entity id. */ @Override public void process(IndexEntryUpdate update) throws IndexEntryConflictException { assertOpen(); if (pendingUpdatesCursor == pendingUpdates.length) { flushPendingChanges(); } TokenIndexEntryUpdate tokenUpdate = asTokenUpdate(update); LogicalTokenUpdates logicalTokenUpdate = PhysicalToLogicalTokenChanges.convertToAdditionsAndRemovals(tokenUpdate); pendingUpdates[pendingUpdatesCursor++] = logicalTokenUpdate; checkNextTokenId(tokenUpdate.beforeValues()); checkNextTokenId(tokenUpdate.values()); } @Override public void yield() { writer.yield(); } private void checkNextTokenId(int[] tokens) { if (tokens.length > 0 && tokens[0] != -1) { lowestTokenId = Math.min(lowestTokenId, tokens[0]); } } private void flushPendingChanges() { Arrays.sort(pendingUpdates, 0, pendingUpdatesCursor); int currentTokenId = lowestTokenId; value.clear(); key.clear(); while (currentTokenId != Integer.MAX_VALUE) { int nextTokenId = Integer.MAX_VALUE; for (int i = 0; i < pendingUpdatesCursor; i++) { LogicalTokenUpdates update = pendingUpdates[i]; long entityId = update.entityId(); nextTokenId = extractChange(update.additions(), currentTokenId, entityId, nextTokenId, true); nextTokenId = extractChange(update.removals(), currentTokenId, entityId, nextTokenId, false); } currentTokenId = nextTokenId; } flushPendingRange(); pendingUpdatesCursor = 0; if (parallel) { writer.yield(); } } private int extractChange(int[] tokens, int currentTokenId, long entityId, int nextTokenId, boolean addition) { int foundNextTokenId = nextTokenId; for (int li = 0; li < tokens.length; li++) { int tokenId = tokens[li]; if (tokenId == -1) { break; } // Have this check here so that we can pick up the next tokenId in our change set if (tokenId == currentTokenId) { change(currentTokenId, entityId, addition); // We can do a little shorter check for next tokenId here straight away, // we just check the next if it's less than what we currently think is next tokenId // and then break right after if (li + 1 < tokens.length && tokens[li + 1] != -1) { int nextTokenCandidate = tokens[li + 1]; if (nextTokenCandidate < currentTokenId) { throw new IllegalArgumentException( "The entity token contained unsorted tokens ids " + Arrays.toString(tokens)); } if (nextTokenCandidate > currentTokenId) { foundNextTokenId = Math.min(foundNextTokenId, nextTokenCandidate); } } break; } else if (tokenId > currentTokenId) { foundNextTokenId = Math.min(foundNextTokenId, tokenId); } } return foundNextTokenId; } private void change(int tokenId, long entityId, boolean add) { long idRange = idLayout.rangeOf(entityId); if (tokenId != key.tokenId || idRange != key.idRange || addition != add) { flushPendingRange(); // Set key to current and reset value key.tokenId = tokenId; key.idRange = idRange; addition = add; } int offset = idLayout.idWithinRange(entityId); value.set(offset); } private void flushPendingRange() { if (value.bits != 0) { // There are changes in the current range, flush them if (addition) { writer.merge(key, value, ADD_MERGER); } else { writer.mergeIfExists(key, value, REMOVE_MERGER); } value.clear(); } } /** * Applies {@link #process(IndexEntryUpdate) queued updates} which has not yet been applied. * No more {@link #process(IndexEntryUpdate) updates} can be applied after this call. */ @Override public void close() { try { flushPendingChanges(); } finally { closed = true; IOUtils.closeAllUnchecked(writer); } } private void assertOpen() { if (closed) { throw new IllegalStateException("Updater has been closed"); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy