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

org.neo4j.kernel.impl.api.InnerTransactionHandlerImpl Maven / Gradle / Ivy

/*
 * 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.api;

import org.eclipse.collections.api.map.primitive.ImmutableLongObjectMap;
import org.eclipse.collections.api.set.primitive.MutableLongSet;
import org.eclipse.collections.impl.factory.primitive.LongObjectMaps;
import org.eclipse.collections.impl.factory.primitive.LongSets;
import org.neo4j.kernel.api.InnerTransactionHandler;
import org.neo4j.kernel.api.KernelTransactionHandle;
import org.neo4j.kernel.api.exceptions.Status;

/**
 * The public methods of this class are synchronized as we expect it to be accessed from both the executing thread and possibly from a different thread which
 * marks the transaction to be terminated.
 */
class InnerTransactionHandlerImpl implements InnerTransactionHandler {
    /**
     * The transaction for this handler has been closed/terminated.
     */
    private boolean closed;
    /**
     * The reason for terminating this handler's transaction. Is not null once marked for termination.
     */
    private Status terminationReason;

    private MutableLongSet innerTransactionIds;
    private final KernelTransactions kernelTransactions;

    InnerTransactionHandlerImpl(KernelTransactions kernelTransactions) {
        this.kernelTransactions = kernelTransactions;
    }

    /**
     * {@inheritDoc}
     * 

* We assume this method to only be called from the executing thread. */ @Override public synchronized void registerInnerTransaction(long innerTransactionId) { if (closed) { throw new IllegalStateException("The inner transaction handler is already closed."); } else if (terminationReason != null) { terminateInnerTransaction(terminationReason, getTransactionHandlesById(), innerTransactionId); } else { if (innerTransactionIds == null) { innerTransactionIds = LongSets.mutable.empty(); } innerTransactionIds.add(innerTransactionId); } } /** * {@inheritDoc} *

* We assume this method to only be called from the executing thread. */ @Override public synchronized void removeInnerTransaction(long innerTransactionId) { if (innerTransactionIds != null) { innerTransactionIds.remove(innerTransactionId); } } /** * @return {@code true} if any open inner transaction is currently connected to this transaction. */ synchronized boolean hasInnerTransaction() { return innerTransactionIds != null && !innerTransactionIds.isEmpty(); } /** * Terminates all the inner transactions contained in this handler and all handlers subsequently registered with this handler. *

* This method may be called from a thread different from the executing thread. */ synchronized void terminateInnerTransactions(Status reason) { terminationReason = reason; var handlesById = getTransactionHandlesById(); if (innerTransactionIds != null) { innerTransactionIds.forEach( innerTransactionId -> terminateInnerTransaction(reason, handlesById, innerTransactionId)); innerTransactionIds.clear(); innerTransactionIds = null; } } private ImmutableLongObjectMap getTransactionHandlesById() { return LongObjectMaps.immutable.from( kernelTransactions.executingTransactions(), KernelTransactionHandle::getTransactionSequenceNumber, a -> a); } private void terminateInnerTransaction( Status reason, ImmutableLongObjectMap handlesById, long innerTransactionId) { KernelTransactionHandle kernelTransactionHandle = handlesById.get(innerTransactionId); if (kernelTransactionHandle != null) { kernelTransactionHandle.markForTermination(reason); } } /** * Marks this handler as closed, once the transaction that this handler belongs to is closed. *

* We assume this method to only be called from the executing thread and to have no further interaction with this handler from that thread. */ synchronized void close() { this.closed = true; if (this.innerTransactionIds != null) { this.innerTransactionIds.clear(); } this.innerTransactionIds = null; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy