org.neo4j.kernel.recovery.ParallelRecoveryVisitor Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of neo4j-kernel Show documentation
Show all versions of neo4j-kernel Show documentation
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.
/*
* Copyright (c) "Neo4j"
* Neo4j Sweden AB [http://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.recovery;
import java.util.concurrent.ArrayBlockingQueue;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.ThreadPoolExecutor;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicLong;
import java.util.concurrent.atomic.AtomicReference;
import org.neo4j.internal.helpers.Exceptions;
import org.neo4j.io.pagecache.context.CursorContext;
import org.neo4j.io.pagecache.tracing.PageCacheTracer;
import org.neo4j.kernel.impl.api.TransactionToApply;
import org.neo4j.kernel.impl.transaction.CommittedTransactionRepresentation;
import org.neo4j.kernel.impl.transaction.TransactionRepresentation;
import org.neo4j.lock.LockGroup;
import org.neo4j.lock.LockService;
import org.neo4j.lock.ReentrantLockService;
import org.neo4j.storageengine.api.StorageEngine;
import org.neo4j.storageengine.api.TransactionApplicationMode;
import static java.lang.Integer.max;
import static org.neo4j.kernel.impl.transaction.log.Commitment.NO_COMMITMENT;
import static org.neo4j.util.Preconditions.checkState;
final class ParallelRecoveryVisitor implements RecoveryApplier
{
private final AtomicLong prevLockedTxId = new AtomicLong( -1 );
private final StorageEngine storageEngine;
private final LockService lockService = new ReentrantLockService();
private final TransactionApplicationMode mode;
private final PageCacheTracer cacheTracer;
private final String tracerTag;
private final ExecutorService appliers;
private final AtomicReference failure = new AtomicReference<>();
private final int stride;
ParallelRecoveryVisitor( StorageEngine storageEngine, TransactionApplicationMode mode, PageCacheTracer cacheTracer, String tracerTag )
{
this( storageEngine, mode, cacheTracer, tracerTag, max( 1, Runtime.getRuntime().availableProcessors() - 1 ) );
}
ParallelRecoveryVisitor( StorageEngine storageEngine, TransactionApplicationMode mode, PageCacheTracer cacheTracer, String tracerTag, int numAppliers )
{
this.storageEngine = storageEngine;
this.mode = mode;
this.cacheTracer = cacheTracer;
this.tracerTag = tracerTag;
this.appliers = new ThreadPoolExecutor( numAppliers, numAppliers, 1, TimeUnit.HOURS, new ArrayBlockingQueue<>( numAppliers ),
new ThreadPoolExecutor.CallerRunsPolicy() );
this.stride = mode == TransactionApplicationMode.REVERSE_RECOVERY ? -1 : 1;
}
@Override
public boolean visit( CommittedTransactionRepresentation transaction ) throws Exception
{
checkFailure();
// We need to know the starting point for the "is it my turn yet?" check below that each thread needs to do before acquiring the locks
prevLockedTxId.compareAndSet( -1, transaction.getCommitEntry().getTxId() - stride );
// TODO Also consider the memory usage of all active transaction instances and apply back-pressure if surpassing it
appliers.submit( () ->
{
long txId = transaction.getCommitEntry().getTxId();
while ( prevLockedTxId.get() != txId - stride )
{
Thread.onSpinWait();
checkFailure();
}
try ( LockGroup locks = new LockGroup() )
{
storageEngine.lockRecoveryCommands( transaction.getTransactionRepresentation(), lockService, locks, mode );
boolean myTurn = prevLockedTxId.compareAndSet( txId - stride, txId );
checkState( myTurn, "Something wrong with the algorithm, I thought it was my turn, but apparently it wasn't %d", txId );
apply( transaction );
}
catch ( Throwable e )
{
failure.compareAndSet( null, e );
}
return null;
} );
return false;
}
private void checkFailure() throws Exception
{
Throwable failure = this.failure.get();
if ( failure != null )
{
Exceptions.throwIfUnchecked( failure );
throw new Exception( "One or more recovering transactions failed to apply", failure );
}
}
private void apply( CommittedTransactionRepresentation transaction ) throws Exception
{
try ( CursorContext cursorContext = new CursorContext( cacheTracer.createPageCursorTracer( tracerTag ) );
var storeCursors = storageEngine.createStorageCursors( cursorContext ) )
{
TransactionRepresentation txRepresentation = transaction.getTransactionRepresentation();
long txId = transaction.getCommitEntry().getTxId();
TransactionToApply tx = new TransactionToApply( txRepresentation, txId, cursorContext, storeCursors );
tx.commitment( NO_COMMITMENT, txId );
tx.logPosition( transaction.getStartEntry().getStartPosition() );
storageEngine.apply( tx, mode );
}
}
@Override
public void close() throws Exception
{
appliers.shutdown();
try
{
if ( !appliers.awaitTermination( 1, TimeUnit.HOURS ) )
{
throw new IllegalStateException( "Recovery couldn't gracefully await remaining appliers" );
}
}
catch ( InterruptedException e )
{
Thread.currentThread().interrupt();
}
checkFailure();
}
}