org.neo4j.kernel.impl.api.parallel.ParallelAccessCheck 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.parallel;
import static org.neo4j.scheduler.Group.CYPHER_WORKER;
import static org.neo4j.util.FeatureToggles.flag;
import java.util.Collection;
import java.util.function.Supplier;
import org.neo4j.configuration.Config;
import org.neo4j.kernel.impl.api.LeaseClient;
import org.neo4j.kernel.impl.locking.LockManager;
import org.neo4j.lock.ActiveLock;
import org.neo4j.lock.LockTracer;
import org.neo4j.lock.LockType;
import org.neo4j.lock.ResourceType;
import org.neo4j.memory.MemoryTracker;
public class ParallelAccessCheck {
public static final boolean CHECK_PARALLEL_ACCESS = flag(ParallelAccessCheck.class, "CHECK_PARALLEL_ACCESS", false);
private static final ThreadLocal DISABLED = new ThreadLocal<>();
public static boolean shouldPerformCheck() {
return CHECK_PARALLEL_ACCESS && DISABLED.get() == null;
}
public static T performWithCheckDisabled(Supplier operation) {
try {
DISABLED.set(true);
return operation.get();
} finally {
DISABLED.remove();
}
}
public static void checkNotCypherWorkerThread() {
if (Thread.currentThread().getThreadGroup().getName().equals(CYPHER_WORKER.groupName())) {
throw new IllegalStateException(
"A resource that does not support parallel access is being accessed by a Cypher worker thread");
}
}
public static LockManager.Client maybeWrapLockClient(LockManager.Client wrappedLockClient) {
if (!shouldPerformCheck()) {
return wrappedLockClient;
}
return new ParallelAccessCheckClient(wrappedLockClient);
}
public static final class ParallelAccessCheckClient implements LockManager.Client {
private final LockManager.Client wrappedLockClient;
public ParallelAccessCheckClient(LockManager.Client wrappedLockClient) {
this.wrappedLockClient = wrappedLockClient;
}
@Override
public boolean tryExclusiveLock(ResourceType resourceType, long resourceId) {
checkNotCypherWorkerThread();
return wrappedLockClient.tryExclusiveLock(resourceType, resourceId);
}
@Override
public void acquireExclusive(LockTracer tracer, ResourceType resourceType, long... resourceIds) {
checkNotCypherWorkerThread();
wrappedLockClient.acquireExclusive(tracer, resourceType, resourceIds);
}
@Override
public void releaseExclusive(ResourceType resourceType, long... resourceIds) {
checkNotCypherWorkerThread();
wrappedLockClient.releaseExclusive(resourceType, resourceIds);
}
@Override
public void acquireShared(LockTracer tracer, ResourceType resourceType, long... resourceIds) {
checkNotCypherWorkerThread();
wrappedLockClient.acquireShared(tracer, resourceType, resourceIds);
}
@Override
public void releaseShared(ResourceType resourceType, long... resourceIds) {
checkNotCypherWorkerThread();
wrappedLockClient.releaseShared(resourceType, resourceIds);
}
@Override
public Collection activeLocks() {
checkNotCypherWorkerThread();
return wrappedLockClient.activeLocks();
}
@Override
public boolean holdsLock(long id, ResourceType resource, LockType lockType) {
checkNotCypherWorkerThread();
return wrappedLockClient.holdsLock(id, resource, lockType);
}
@Override
public void initialize(
LeaseClient leaseClient, long transactionId, MemoryTracker memoryTracker, Config config) {
checkNotCypherWorkerThread();
wrappedLockClient.initialize(leaseClient, transactionId, memoryTracker, config);
}
@Override
public boolean trySharedLock(ResourceType resourceType, long resourceId) {
checkNotCypherWorkerThread();
return wrappedLockClient.trySharedLock(resourceType, resourceId);
}
@Override
public void prepareForCommit() {
checkNotCypherWorkerThread();
wrappedLockClient.prepareForCommit();
}
@Override
public void stop() {
checkNotCypherWorkerThread();
wrappedLockClient.stop();
}
@Override
public void close() {
checkNotCypherWorkerThread();
wrappedLockClient.close();
}
@Override
public long getTransactionId() {
checkNotCypherWorkerThread();
return wrappedLockClient.getTransactionId();
}
@Override
public long activeLockCount() {
checkNotCypherWorkerThread();
return wrappedLockClient.activeLockCount();
}
@Override
public void reset() {
checkNotCypherWorkerThread();
wrappedLockClient.reset();
}
public LockManager.Client getWrappedLockClient() {
return wrappedLockClient;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy