/*
* eXist Open Source Native XML Database
* Copyright (C) 2001-2016 The eXist Project
* http://exist-db.org
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public License
* as published by the Free Software Foundation; either version 2
* 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 Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this program; if not, write to the Free Software Foundation
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*/
package org.exist.storage.journal;
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
import org.exist.EXistException;
import org.exist.storage.BrokerPool;
import org.exist.storage.BrokerPoolService;
import org.exist.storage.BrokerPoolServiceException;
import org.exist.storage.recovery.RecoveryManager;
import org.exist.util.Configuration;
import org.exist.util.ReadOnlyException;
import java.io.IOException;
import java.nio.file.Path;
import java.util.Optional;
import java.util.stream.Stream;
/**
* Journal Manager just adds some light-weight
* wrapping around {@link Journal}
*/
public class JournalManager implements BrokerPoolService {
private static final Logger LOG = LogManager.getLogger(JournalManager.class);
private Path journalDir;
private boolean groupCommits;
private Journal journal;
private boolean journallingDisabled = false;
private boolean initialized = false;
@Override
public void configure(final Configuration configuration) {
this.journalDir = (Path) Optional.ofNullable(configuration.getProperty(Journal.PROPERTY_RECOVERY_JOURNAL_DIR))
.orElse(configuration.getProperty(BrokerPool.PROPERTY_DATA_DIR));
this.groupCommits = configuration.getProperty(BrokerPool.PROPERTY_RECOVERY_GROUP_COMMIT, false);
if (LOG.isDebugEnabled()) {
LOG.debug("GroupCommits = " + groupCommits);
}
}
@Override
public void prepare(final BrokerPool pool) throws BrokerPoolServiceException {
if(!journallingDisabled) {
try {
this.journal = new Journal(pool, journalDir);
this.journal.initialize();
this.initialized = true;
} catch(final EXistException | ReadOnlyException e) {
throw new BrokerPoolServiceException(e);
}
}
}
public void disableJournalling() {
this.journallingDisabled = true;
}
/**
* Write a single entry to the journal
*
* @see Journal#writeToLog(Loggable)
*
* @param loggable The entry to write in the journal
*/
public synchronized void journal(final Loggable loggable) throws JournalException {
if(!journallingDisabled) {
journal.writeToLog(loggable);
}
}
/**
* Write a group of entrys to the journal
*
* @see Journal#writeToLog(Loggable)
* @see Journal#flushToLog(boolean)
*
* @param loggable The entry to write in the journalGroup
*/
public synchronized void journalGroup(final Loggable loggable) throws JournalException {
if(!journallingDisabled) {
journal.writeToLog(loggable);
if (!groupCommits) {
journal.flushToLog(true);
}
}
}
/**
* @see Journal#checkpoint(long, boolean)
*
* Create a new checkpoint. A checkpoint fixes the current database state. All dirty pages
* are written to disk and the journal file is cleaned.
*
* This method is called from
* {@link org.exist.storage.BrokerPool} within pre-defined periods. It
* should not be called from somewhere else. The database needs to
* be in a stable state (all transactions completed, no operations running).
*
* @param transactionId The id of the transaction for the checkpoint
* @param switchFiles Whether a new journal file should be started
*
* @throws JournalException
*/
public synchronized void checkpoint(final long transactionId, final boolean switchFiles) throws JournalException {
if(!journallingDisabled) {
journal.checkpoint(transactionId, switchFiles);
}
}
/**
* @see Journal#flushToLog(boolean, boolean)
*/
public synchronized void flush(final boolean fsync, final boolean forceSync) {
journal.flushToLog(fsync, forceSync);
}
/**
* Shut down the journal. This will write a checkpoint record
* to the log, so recovery manager knows the file has been
* closed in a clean way.
*
* @param transactionId The id of the transaction for the shutdown
* @param checkpoint Whether to write a checkpoint before shutdown
*/
public synchronized void shutdown(final long transactionId, final boolean checkpoint) {
if(initialized) {
journal.shutdown(transactionId, checkpoint);
initialized = false;
}
}
/**
* @see Journal#lastWrittenLsn()
*/
public long lastWrittenLsn() {
return journal.lastWrittenLsn();
}
public RecoveryManager.JournalRecoveryAccessor getRecoveryAccessor(final RecoveryManager recoveryManager) {
return recoveryManager.new JournalRecoveryAccessor(
journal::setInRecovery, journal::getFiles, journal::getFile, journal::setCurrentFileNum,
() -> { journal.switchFiles(); return null; }, () -> { journal.clearBackupFiles(); return null; });
}
}