org.neo4j.kernel.impl.api.KernelStatement 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.lang.Math.subtractExact;
import static java.lang.String.format;
import static org.neo4j.configuration.GraphDatabaseInternalSettings.trace_tx_statements;
import java.io.ByteArrayOutputStream;
import java.io.PrintStream;
import java.nio.charset.StandardCharsets;
import java.util.ArrayDeque;
import java.util.Arrays;
import java.util.Deque;
import java.util.Map;
import java.util.concurrent.atomic.AtomicReference;
import org.apache.commons.lang3.StringUtils;
import org.neo4j.configuration.Config;
import org.neo4j.configuration.GraphDatabaseInternalSettings;
import org.neo4j.internal.kernel.api.connectioninfo.ClientConnectionInfo;
import org.neo4j.io.pagecache.context.CursorContext;
import org.neo4j.kernel.api.QueryRegistry;
import org.neo4j.kernel.api.Statement;
import org.neo4j.kernel.api.query.ExecutingQuery;
import org.neo4j.kernel.database.NamedDatabaseId;
import org.neo4j.kernel.impl.locking.LockManager;
import org.neo4j.lock.LockTracer;
import org.neo4j.resources.CpuClock;
/**
* A resource efficient implementation of {@link Statement}. Designed to be reused within a
* {@link KernelTransactionImplementation} instance, even across transactions since this instances itself
* doesn't hold essential state. Usage:
*
*
* - Construct {@link KernelStatement} when {@link KernelTransactionImplementation} is constructed
* - For every transaction...
* - Call {@link #initialize(LockManager.Client, CursorContext, long)} which makes this instance
* full available and ready to use. Call when the {@link KernelTransactionImplementation} is initialized.
* - Alternate {@link #acquire()} / {@link #close()} when acquiring / closing a statement for the transaction...
* Temporarily asymmetric number of calls to {@link #acquire()} / {@link #close()} is supported, although in
* the end an equal number of calls must have been issued.
* - To be safe call {@link #forceClose()} at the end of a transaction to force a close of the statement,
* even if there are more than one current call to {@link #acquire()}. This instance is now again ready
* to be {@link #initialize(LockManager.Client, CursorContext, long)} initialized} and used for the transaction
* instance again, when it's initialized.
*
*/
public class KernelStatement extends QueryStatement {
private static final int EMPTY_COUNTER = 0;
private static final int STATEMENT_TRACK_HISTORY_MAX_SIZE = 100;
private static final Deque EMPTY_STATEMENT_HISTORY = new ArrayDeque<>(0);
private final QueryRegistry queryRegistry;
private final KernelTransactionImplementation transaction;
private final NamedDatabaseId namedDatabaseId;
private final boolean traceStatements;
private final boolean trackStatementClose;
private LockManager.Client lockClient;
private CursorContext cursorContext;
private int referenceCount;
private final LockTracer systemLockTracer;
private final Deque statementOpenCloseCalls;
private final TransactionClockContext clockContext;
private long initialStatementHits;
private long initialStatementFaults;
private int aquireCounter;
public KernelStatement(
KernelTransactionImplementation transaction,
LockTracer systemLockTracer,
TransactionClockContext clockContext,
AtomicReference cpuClockRef,
NamedDatabaseId namedDatabaseId,
Config config) {
this.transaction = transaction;
this.queryRegistry =
new StatementQueryRegistry(this, clockContext.systemClock(), cpuClockRef, systemLockTracer);
this.systemLockTracer = systemLockTracer;
this.traceStatements = config.get(trace_tx_statements);
this.trackStatementClose = config.get(GraphDatabaseInternalSettings.track_tx_statement_close);
this.statementOpenCloseCalls = traceStatements ? new ArrayDeque<>() : EMPTY_STATEMENT_HISTORY;
this.clockContext = clockContext;
this.namedDatabaseId = namedDatabaseId;
}
@Override
public QueryRegistry queryRegistry() {
return queryRegistry;
}
@Override
public NamedDatabaseId namedDatabaseId() {
return namedDatabaseId;
}
@Override
public ClientConnectionInfo clientInfo() {
return transaction.clientInfo();
}
@Override
public Map getMetaData() {
return transaction.getMetaData();
}
@Override
public void close() {
// Check referenceCount > 0 since we allow multiple close calls,
// i.e. ignore closing already closed statements
if (referenceCount > 0 && (--referenceCount == 0)) {
cleanupResources();
}
recordOpenCloseMethods();
}
public void initialize(LockManager.Client lockClient, CursorContext cursorContext, long startTimeMillis) {
this.lockClient = lockClient;
this.cursorContext = cursorContext;
this.clockContext.initializeTransaction(startTimeMillis);
this.clearQueryExecution();
this.aquireCounter = 0;
}
public LockManager.Client locks() {
return lockClient;
}
public LockTracer lockTracer() {
var currentQuery = executingQueryPlain();
if (currentQuery == null) {
return systemLockTracer;
}
return currentQuery.lockTracer();
}
@Override
public long activeLockCount() {
return lockClient.activeLockCount();
}
@Override
public long getHits() {
/*
* The cursor tracer is shared between queries in the same transaction.
* This makes it impossible to track page hits per query.
*
* We try by subtracting initialStatementHits. But this will only work
* when executing one query at a time. If multiple queries in the same
* transaction are open simultaneously it easily leads to a state where
* this method returns more hits than the current executing query are
* responsible for.
*/
return isAcquired()
? subtractExact(cursorContext.getCursorTracer().hits(), initialStatementHits)
: EMPTY_COUNTER;
}
@Override
public long getFaults() {
// Comment on getHits also applies here.
return isAcquired()
? subtractExact(cursorContext.getCursorTracer().faults(), initialStatementFaults)
: EMPTY_COUNTER;
}
public final void acquire() {
if (referenceCount++ == 0) {
clockContext.initializeStatement();
var cursorTracer = cursorContext.getCursorTracer();
this.initialStatementHits = cursorTracer.hits();
this.initialStatementFaults = cursorTracer.faults();
}
aquireCounter++;
recordOpenCloseMethods();
}
final int aquireCounter() {
return aquireCounter;
}
final boolean isAcquired() {
return referenceCount > 0;
}
final void forceClose() {
if (referenceCount > 0) {
int leakedStatements = referenceCount;
referenceCount = 0;
cleanupResources();
if (trackStatementClose && transaction.isCommitted()) {
String message = getStatementNotClosedMessage(leakedStatements);
throw new StatementNotClosedException(message, statementOpenCloseCalls);
}
}
this.clearQueryExecution();
}
private String getStatementNotClosedMessage(int leakedStatements) {
String additionalInstruction = traceStatements
? StringUtils.EMPTY
: format(
" To see statement open/close stack traces please set '%s' setting to true",
trace_tx_statements.name());
return format(
"Statements were not correctly closed. Number of leaked statements: %d.%s",
leakedStatements, additionalInstruction);
}
@Override
public final String authenticatedUser() {
return transaction.securityContext().subject().authenticatedUser();
}
@Override
public final String executingUser() {
return transaction.securityContext().subject().executingUser();
}
@Override
public long getTransactionSequenceNumber() {
return transaction.getTransactionSequenceNumber();
}
@Override
final void stopQueryExecution(ExecutingQuery executingQuery) {
super.stopQueryExecution(executingQuery);
transaction.getStatistics().addWaitingTime(executingQuery.reportedWaitingTimeNanos());
}
private void cleanupResources() {
// closing is done by KTI
transaction.releaseStatementResources();
initialStatementHits = EMPTY_COUNTER;
initialStatementFaults = EMPTY_COUNTER;
closeAllCloseableResources();
}
public KernelTransactionImplementation getTransaction() {
return transaction;
}
private void recordOpenCloseMethods() {
if (traceStatements) {
if (statementOpenCloseCalls.size() > STATEMENT_TRACK_HISTORY_MAX_SIZE) {
statementOpenCloseCalls.pop();
}
StackTraceElement[] stackTrace = Thread.currentThread().getStackTrace();
statementOpenCloseCalls.add(Arrays.copyOfRange(stackTrace, 2, stackTrace.length));
}
}
public TransactionClockContext clocks() {
return clockContext;
}
static class StatementNotClosedException extends IllegalStateException {
StatementNotClosedException(String s, Deque openCloseTraces) {
super(s);
this.addSuppressed(new StatementTraceException(buildMessage(openCloseTraces)));
}
private static String buildMessage(Deque openCloseTraces) {
if (openCloseTraces.isEmpty()) {
return StringUtils.EMPTY;
}
int separatorLength = 80;
String paddingString = "=";
ByteArrayOutputStream out = new ByteArrayOutputStream();
PrintStream printStream = new PrintStream(out, false, StandardCharsets.UTF_8);
printStream.println();
printStream.println(
"Last " + STATEMENT_TRACK_HISTORY_MAX_SIZE + " statements open/close stack traces are:");
int element = 0;
for (StackTraceElement[] traceElements : openCloseTraces) {
printStream.println(StringUtils.center("*StackTrace " + element + "*", separatorLength, paddingString));
for (StackTraceElement traceElement : traceElements) {
printStream.println("\tat " + traceElement);
}
printStream.println(StringUtils.center("", separatorLength, paddingString));
printStream.println();
element++;
}
printStream.println("All statement open/close stack traces printed.");
return out.toString(StandardCharsets.UTF_8);
}
private static class StatementTraceException extends RuntimeException {
StatementTraceException(String message) {
super(message);
}
@Override
public synchronized Throwable fillInStackTrace() {
return this;
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy