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

org.neo4j.kernel.impl.api.KernelTransactionImplementationHandle 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 static java.util.Optional.ofNullable;
import static org.neo4j.storageengine.api.TransactionIdStore.BASE_TX_ID;

import java.util.Collection;
import java.util.Map;
import java.util.Optional;
import org.apache.commons.lang3.StringUtils;
import org.neo4j.internal.kernel.api.connectioninfo.ClientConnectionInfo;
import org.neo4j.internal.kernel.api.security.AuthSubject;
import org.neo4j.io.pagecache.context.CursorContext;
import org.neo4j.io.pagecache.context.VersionContext;
import org.neo4j.kernel.api.KernelTransaction;
import org.neo4j.kernel.api.KernelTransactionHandle;
import org.neo4j.kernel.api.TerminationMark;
import org.neo4j.kernel.api.TransactionTimeout;
import org.neo4j.kernel.api.exceptions.Status;
import org.neo4j.kernel.api.query.ExecutingQuery;
import org.neo4j.kernel.impl.api.transaction.trace.TransactionInitializationTrace;
import org.neo4j.lock.ActiveLock;
import org.neo4j.time.SystemNanoClock;

/**
 * A {@link KernelTransactionHandle} that wraps the given {@link KernelTransactionImplementation}.
 * This handle knows that {@link KernelTransactionImplementation}s can be reused and represents a single logical
 * transaction. This means that methods like {@link #markForTermination(Status)} can only terminate running
 * transaction this handle was created for.
 */
class KernelTransactionImplementationHandle implements KernelTransactionHandle {
    private static final String USER_TRANSACTION_NAME_SEPARATOR = "-transaction-";

    private final long startTime;
    private final long startTimeNanos;
    private final TransactionTimeout timeout;
    private final KernelTransactionImplementation tx;
    private final SystemNanoClock clock;
    private final ClientConnectionInfo clientInfo;
    private final AuthSubject subject;
    private final Optional terminationMark;
    private final Optional executingQuery;
    private final Map metaData;
    private final String statusDetails;
    private final TransactionInitializationTrace initializationTrace;
    private final KernelTransactionStamp transactionStamp;
    private final String databaseName;
    private final long lastClosedTxId;
    private final long transactionHorizon;

    KernelTransactionImplementationHandle(
            KernelTransactionImplementation tx, SystemNanoClock clock, CursorContext cursorContext) {
        this.transactionStamp = new KernelTransactionStamp(tx);
        this.startTime = tx.startTime();
        this.startTimeNanos = tx.startTimeNanos();
        this.timeout = tx.timeout();
        this.subject = tx.subjectOrAnonymous();
        this.terminationMark = tx.getTerminationMark();
        this.executingQuery = tx.executingQuery();
        this.metaData = tx.getMetaData();
        this.statusDetails = tx.statusDetails();
        this.initializationTrace = tx.getInitializationTrace();
        this.clientInfo = tx.clientInfo();
        this.databaseName = tx.getDatabaseName();
        var versionContext = cursorContext.getVersionContext();
        this.lastClosedTxId = versionContext.lastClosedTransactionId();
        this.transactionHorizon = transactionHorizon(versionContext);
        this.tx = tx;
        this.clock = clock;
    }

    @Override
    public long startTime() {
        return startTime == 0L ? Long.MAX_VALUE : startTime;
    }

    @Override
    public long startTimeNanos() {
        return startTimeNanos == 0L ? Long.MAX_VALUE : startTimeNanos;
    }

    @Override
    public TransactionTimeout timeout() {
        return timeout == null ? TransactionTimeout.NO_TIMEOUT : timeout;
    }

    @Override
    public boolean isOpen() {
        return transactionStamp.isOpen();
    }

    @Override
    public boolean isCommitting() {
        return transactionStamp.isCommitting();
    }

    @Override
    public boolean isRollingback() {
        return transactionStamp.isRollingback();
    }

    @Override
    public boolean markForTermination(Status reason) {
        return tx.markForTermination(transactionStamp.getTransactionSequenceNumber(), reason);
    }

    @Override
    public AuthSubject subject() {
        return subject == null ? AuthSubject.ANONYMOUS : subject;
    }

    @Override
    public Map getMetaData() {
        return metaData == null ? Map.of() : metaData;
    }

    @Override
    public String getStatusDetails() {
        return statusDetails == null ? StringUtils.EMPTY : statusDetails;
    }

    @Override
    public Optional terminationMark() {
        return terminationMark;
    }

    @Override
    public boolean isUnderlyingTransaction(KernelTransaction tx) {
        return this.tx == tx;
    }

    @Override
    public long getTransactionSequenceNumber() {
        return transactionStamp.getTransactionSequenceNumber();
    }

    @Override
    public String getUserTransactionName() {
        return getDatabaseName() + USER_TRANSACTION_NAME_SEPARATOR + getTransactionSequenceNumber();
    }

    private String getDatabaseName() {
        return databaseName == null ? StringUtils.EMPTY : databaseName;
    }

    @Override
    public Optional executingQuery() {
        return executingQuery;
    }

    @Override
    public Collection activeLocks() {
        return tx.activeLocks();
    }

    @Override
    public TransactionExecutionStatistic transactionStatistic() {
        if (transactionStamp.isNotExpired()) {
            return new TransactionExecutionStatistic(tx, clock, startTime);
        }
        return TransactionExecutionStatistic.NOT_AVAILABLE;
    }

    @Override
    public TransactionInitializationTrace transactionInitialisationTrace() {
        return initializationTrace == null ? TransactionInitializationTrace.NONE : initializationTrace;
    }

    @Override
    public Optional clientInfo() {
        return ofNullable(clientInfo);
    }

    @Override
    public boolean isSchemaTransaction() {
        return tx.isSchemaTransaction();
    }

    @Override
    public long getLastClosedTxId() {
        return lastClosedTxId;
    }

    @Override
    public long getTransactionHorizon() {
        return transactionHorizon;
    }

    @Override
    public boolean equals(Object o) {
        if (this == o) {
            return true;
        }
        if (o == null || getClass() != o.getClass()) {
            return false;
        }
        KernelTransactionImplementationHandle that = (KernelTransactionImplementationHandle) o;
        return transactionStamp.equals(that.transactionStamp);
    }

    @Override
    public int hashCode() {
        return transactionStamp.hashCode();
    }

    @Override
    public String toString() {
        return "KernelTransactionImplementationHandle{transactionSequenceNumber="
                + transactionStamp.getTransactionSequenceNumber() + ", tx=" + tx + "}";
    }

    private long transactionHorizon(VersionContext versionContext) {
        // if transaction has already started committing its horizon is oldestVisibleTransactionNumber which was
        // recorded at the time commit started
        var oldestVisibleTransactionNumber = versionContext.oldestVisibleTransactionNumber();
        if (oldestVisibleTransactionNumber > BASE_TX_ID) {
            return oldestVisibleTransactionNumber;
        }
        // otherwise, its horizon is the latest gap free closed transaction at the time it started
        return versionContext.lastClosedTransactionId();
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy