![JAR search and dependency download from the Maven repository](/logo.png)
org.neo4j.kernel.recovery.TransactionLogsRecovery 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 [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.recovery;
import static java.lang.String.format;
import static java.util.concurrent.TimeUnit.MILLISECONDS;
import static org.neo4j.kernel.recovery.Recovery.throwUnableToCleanRecover;
import static org.neo4j.kernel.recovery.RecoveryMode.FORWARD;
import static org.neo4j.storageengine.AppendIndexProvider.BASE_APPEND_INDEX;
import static org.neo4j.storageengine.api.TransactionApplicationMode.RECOVERY;
import static org.neo4j.storageengine.api.TransactionApplicationMode.REVERSE_RECOVERY;
import java.io.IOException;
import java.nio.channels.ClosedByInterruptException;
import org.apache.commons.lang3.mutable.MutableLong;
import org.neo4j.dbms.database.DatabaseStartAbortedException;
import org.neo4j.internal.helpers.progress.ProgressListener;
import org.neo4j.internal.helpers.progress.ProgressMonitorFactory;
import org.neo4j.io.pagecache.context.CursorContextFactory;
import org.neo4j.kernel.database.Database;
import org.neo4j.kernel.impl.transaction.CommittedCommandBatch;
import org.neo4j.kernel.impl.transaction.log.CommandBatchCursor;
import org.neo4j.kernel.impl.transaction.log.LogPosition;
import org.neo4j.kernel.lifecycle.Lifecycle;
import org.neo4j.kernel.lifecycle.LifecycleAdapter;
import org.neo4j.storageengine.AppendIndexProvider;
import org.neo4j.time.Stopwatch;
/**
* This is the process of doing a recovery on the transaction log and store, and is executed
* at startup of {@link Database}.
*/
public class TransactionLogsRecovery extends LifecycleAdapter {
private static final String REVERSE_RECOVERY_TAG = "restoreDatabase";
private static final String RECOVERY_TAG = "recoverDatabase";
private static final String RECOVERY_COMPLETED_TAG = "databaseRecoveryCompleted";
private final RecoveryService recoveryService;
private final RecoveryMonitor monitor;
private final CorruptedLogsTruncator logsTruncator;
private final Lifecycle schemaLife;
private final ProgressMonitorFactory progressMonitorFactory;
private final boolean failOnCorruptedLogFiles;
private final RecoveryStartupChecker recoveryStartupChecker;
private final boolean rollbackIncompleteTransactions;
private final CursorContextFactory contextFactory;
private final RecoveryPredicate recoveryPredicate;
private final RecoveryMode mode;
private ProgressListener progressListener;
public TransactionLogsRecovery(
RecoveryService recoveryService,
CorruptedLogsTruncator logsTruncator,
Lifecycle schemaLife,
RecoveryMonitor monitor,
ProgressMonitorFactory progressMonitorFactory,
boolean failOnCorruptedLogFiles,
RecoveryStartupChecker recoveryStartupChecker,
RecoveryPredicate recoveryPredicate,
boolean rollbackIncompleteTransactions,
CursorContextFactory contextFactory,
RecoveryMode mode) {
this.recoveryService = recoveryService;
this.monitor = monitor;
this.logsTruncator = logsTruncator;
this.schemaLife = schemaLife;
this.progressMonitorFactory = progressMonitorFactory;
this.failOnCorruptedLogFiles = failOnCorruptedLogFiles;
this.recoveryStartupChecker = recoveryStartupChecker;
this.rollbackIncompleteTransactions = rollbackIncompleteTransactions;
this.contextFactory = contextFactory;
this.recoveryPredicate = recoveryPredicate;
this.mode = mode;
this.progressListener = null;
}
@Override
public void init() throws Exception {
RecoveryStartInformation recoveryStartInformation = recoveryService.getRecoveryStartInformation();
if (!recoveryStartInformation.isRecoveryRequired()) {
schemaLife.init();
return;
}
Stopwatch recoveryStartTime = Stopwatch.start();
TransactionIdTracker transactionIdTracker = new TransactionIdTracker();
LogPosition recoveryStartPosition = recoveryStartInformation.getTransactionLogPosition();
monitor.recoveryRequired(recoveryStartPosition);
LogPosition recoveryToPosition = recoveryStartPosition;
LogPosition lastTransactionPosition = recoveryStartPosition;
CommittedCommandBatch.BatchInformation lastHighestTransactionBatchInfo = null;
CommittedCommandBatch.BatchInformation lastBatchInfo = null;
RecoveryRollbackAppendIndexProvider appendIndexProvider = null;
boolean incompleteBatchEncountered = false;
try {
if (!recoveryStartInformation.isMissingLogs()) {
try {
reverseRecovery(recoveryStartInformation, transactionIdTracker, recoveryStartPosition);
// We cannot initialise the schema (tokens, schema cache, indexing service, etc.) until we have
// returned
// the store to a consistent state.
// We need to be able to read the store before we can even figure out what indexes, tokens, etc. we
// have. Hence, we defer the initialisation
// of the schema life until after we've done the reverse recovery.
schemaLife.init();
boolean fullRecovery = true;
try (CommandBatchCursor transactionsToRecover =
recoveryService.getCommandBatches(recoveryStartPosition);
var recoveryVisitor =
recoveryService.getRecoveryApplier(RECOVERY, contextFactory, RECOVERY_TAG)) {
while (fullRecovery && transactionsToRecover.next()) {
var nextCommandBatch = transactionsToRecover.get();
if (!recoveryPredicate.test(nextCommandBatch)) {
monitor.partialRecovery(recoveryPredicate, lastHighestTransactionBatchInfo);
fullRecovery = false;
if (lastHighestTransactionBatchInfo == null) {
// First transaction after checkpoint failed predicate test
// we can't always load transaction before checkpoint to check what values we had
// there
// since those logs may be pruned,
// but we will try to load first transaction before checkpoint to see if we just on
// the
// edge of provided criteria
// and will fail otherwise.
long beforeCheckpointAppendIndex =
recoveryStartInformation.getFirstAppendIndexAfterLastCheckPoint() - 1;
if (beforeCheckpointAppendIndex < BASE_APPEND_INDEX) {
throw new RecoveryPredicateException(format(
"Partial recovery criteria can't be satisfied. No transaction after checkpoint matching to provided "
+ "criteria found and transaction before checkpoint is not valid. "
+ "Append index before checkpoint: %d, criteria %s.",
beforeCheckpointAppendIndex, recoveryPredicate.describe()));
}
try (var beforeCheckpointCursor =
recoveryService.getCommandBatches(beforeCheckpointAppendIndex)) {
if (beforeCheckpointCursor.next()) {
CommittedCommandBatch candidate = beforeCheckpointCursor.get();
if (!recoveryPredicate.test(candidate)) {
throw new RecoveryPredicateException(format(
"Partial recovery criteria can't be satisfied. "
+ "Transaction after and before checkpoint does not satisfy provided recovery criteria. "
+ "Observed transaction id: %d, recovery criteria: %s.",
candidate.txId(), recoveryPredicate.describe()));
}
lastHighestTransactionBatchInfo = candidate.batchInformation();
lastTransactionPosition = beforeCheckpointCursor.position();
} else {
throw new RecoveryPredicateException(format(
"Partial recovery criteria can't be satisfied. No transaction after checkpoint matching "
+ "to provided criteria found and transaction before checkpoint not found. Recovery criteria: %s.",
recoveryPredicate.describe()));
}
} catch (RecoveryPredicateException re) {
throw re;
} catch (Exception e) {
throw new RecoveryPredicateException(
format(
"Partial recovery criteria can't be satisfied. No transaction after checkpoint matching "
+ "to provided criteria found and fail to read transaction before checkpoint. Recovery criteria: %s.",
recoveryPredicate.describe()),
e);
}
}
} else {
recoveryStartupChecker.checkIfCanceled();
switch (transactionIdTracker.transactionStatus(nextCommandBatch.txId())) {
case RECOVERABLE -> {
recoveryVisitor.visit(nextCommandBatch);
monitor.batchRecovered(nextCommandBatch);
}
case ROLLED_BACK -> monitor.batchApplySkipped(nextCommandBatch);
case INCOMPLETE -> {
monitor.batchApplySkipped(nextCommandBatch);
if (!rollbackIncompleteTransactions) {
if (lastHighestTransactionBatchInfo == null) {
var checkpointInfo = recoveryStartInformation.getCheckpointInfo();
var transactionId = checkpointInfo.transactionId();
lastBatchInfo = new CommittedCommandBatch.BatchInformation(
transactionId, checkpointInfo.appendIndex());
lastHighestTransactionBatchInfo =
new CommittedCommandBatch.BatchInformation(
transactionId, transactionId.appendIndex());
}
fullRecovery = false;
incompleteBatchEncountered = true;
}
}
}
if (!incompleteBatchEncountered) {
if (lastHighestTransactionBatchInfo == null
|| lastHighestTransactionBatchInfo.txId() < nextCommandBatch.txId()) {
lastHighestTransactionBatchInfo = nextCommandBatch.batchInformation();
}
lastBatchInfo = nextCommandBatch.batchInformation();
lastTransactionPosition = transactionsToRecover.position();
recoveryToPosition = lastTransactionPosition;
}
reportProgress();
}
}
recoveryToPosition = fullRecovery ? transactionsToRecover.position() : lastTransactionPosition;
}
} catch (Error
| ClosedByInterruptException
| DatabaseStartAbortedException
| RecoveryPredicateException e) {
// We do not want to truncate logs based on these exceptions. Since users can influence them with
// config
// changes
// the users are able to workaround this if truncations is really needed.
throw e;
} catch (Throwable t) {
if (failOnCorruptedLogFiles) {
throwUnableToCleanRecover(t);
}
if (lastHighestTransactionBatchInfo != null) {
monitor.failToRecoverTransactionsAfterCommit(
t, lastHighestTransactionBatchInfo, recoveryToPosition);
} else {
monitor.failToRecoverTransactionsAfterPosition(t, recoveryStartPosition);
}
}
appendIndexProvider = new RecoveryRollbackAppendIndexProvider(lastBatchInfo);
if (rollbackIncompleteTransactions) {
logsTruncator.truncate(recoveryToPosition, recoveryStartInformation.getCheckpointInfo());
var rollbackTransactionInfo = recoveryService.rollbackTransactions(
recoveryToPosition,
transactionIdTracker,
lastHighestTransactionBatchInfo,
appendIndexProvider);
if (rollbackTransactionInfo != null) {
if (lastHighestTransactionBatchInfo == null
|| lastHighestTransactionBatchInfo.txId()
< rollbackTransactionInfo.batchInfo().txId()) {
lastHighestTransactionBatchInfo = rollbackTransactionInfo.batchInfo();
}
lastTransactionPosition = rollbackTransactionInfo.position();
}
}
}
} finally {
closeProgress();
}
try (var cursorContext = contextFactory.create(RECOVERY_COMPLETED_TAG)) {
final boolean missingLogs = recoveryStartInformation.isMissingLogs();
recoveryService.transactionsRecovered(
lastHighestTransactionBatchInfo,
appendIndexProvider,
lastTransactionPosition,
recoveryToPosition,
recoveryStartInformation.getCheckpointPosition(),
missingLogs,
cursorContext);
}
monitor.recoveryCompleted(recoveryStartTime.elapsed(MILLISECONDS), mode);
}
private void reverseRecovery(
RecoveryStartInformation recoveryStartInformation,
TransactionIdTracker transactionIdTracker,
LogPosition recoveryStartPosition)
throws Exception {
if (mode == FORWARD) {
// nothing to do, update the progress to match this
initProgressReporter(recoveryStartInformation, recoveryStartPosition);
return;
}
CommittedCommandBatch lastReversedCommandBatch = null;
long lowestRecoveredAppendIndex = recoveryStartInformation.getFirstAppendIndexAfterLastCheckPoint();
try (var transactionsToRecover = recoveryService.getCommandBatchesInReverseOrder(recoveryStartPosition);
var recoveryVisitor =
recoveryService.getRecoveryApplier(REVERSE_RECOVERY, contextFactory, REVERSE_RECOVERY_TAG)) {
while (transactionsToRecover.next()) {
recoveryStartupChecker.checkIfCanceled();
CommittedCommandBatch commandBatch = transactionsToRecover.get();
if (lastReversedCommandBatch == null) {
lastReversedCommandBatch = commandBatch;
initProgressReporter(recoveryStartInformation, lastReversedCommandBatch, mode);
}
recoveryVisitor.visit(commandBatch);
transactionIdTracker.trackBatch(commandBatch);
lowestRecoveredAppendIndex = commandBatch.appendIndex();
reportProgress();
}
}
monitor.reverseStoreRecoveryCompleted(lowestRecoveredAppendIndex);
}
private void initProgressReporter(
RecoveryStartInformation recoveryStartInformation, LogPosition recoveryStartPosition) throws IOException {
try (var transactionsToRecover = recoveryService.getCommandBatchesInReverseOrder(recoveryStartPosition)) {
if (transactionsToRecover.next()) {
CommittedCommandBatch commandBatch = transactionsToRecover.get();
initProgressReporter(recoveryStartInformation, commandBatch, mode);
}
}
}
private void initProgressReporter(
RecoveryStartInformation recoveryStartInformation,
CommittedCommandBatch lastReversedBatch,
RecoveryMode mode) {
long numberOfBatchesToRecover = estimateNumberOfBatchesToRecover(recoveryStartInformation, lastReversedBatch);
// In full mode we will process each transaction twice (doing reverse and direct detour) we need to
// multiply number of transactions that we want to recover by 2 to be able to report correct progress
progressListener = progressMonitorFactory.singlePart(
"TransactionLogsRecovery",
mode == RecoveryMode.FULL ? numberOfBatchesToRecover * 2 : numberOfBatchesToRecover);
}
private void reportProgress() {
progressListener.add(1);
}
private void closeProgress() {
if (progressListener != null) {
progressListener.close();
}
}
private static long estimateNumberOfBatchesToRecover(
RecoveryStartInformation recoveryStartInformation, CommittedCommandBatch lastReversedCommandBatch) {
return lastReversedCommandBatch.appendIndex()
- recoveryStartInformation.getFirstAppendIndexAfterLastCheckPoint()
+ 1;
}
@Override
public void start() throws Exception {
schemaLife.start();
}
@Override
public void stop() throws Exception {
schemaLife.stop();
}
@Override
public void shutdown() throws Exception {
schemaLife.shutdown();
}
private static class RecoveryRollbackAppendIndexProvider implements AppendIndexProvider {
private final MutableLong rollbackIndex;
public RecoveryRollbackAppendIndexProvider(CommittedCommandBatch.BatchInformation lastBatchInfo) {
this.rollbackIndex = lastBatchInfo == null
? new MutableLong(BASE_APPEND_INDEX)
: new MutableLong(lastBatchInfo.appendIndex());
}
@Override
public long nextAppendIndex() {
return rollbackIndex.incrementAndGet();
}
@Override
public long getLastAppendIndex() {
return rollbackIndex.longValue();
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy