/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.apache.jackrabbit.core.query.lucene;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.apache.lucene.store.Directory;
import org.apache.jackrabbit.core.query.lucene.directory.IndexOutputStream;
import org.apache.jackrabbit.core.query.lucene.directory.IndexInputStream;
import java.io.BufferedReader;
import java.io.BufferedWriter;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStreamWriter;
import java.io.Writer;
import java.io.OutputStream;
import java.util.ArrayList;
import java.util.List;
/**
* Implements a redo log for changes that have not been committed to disk. While
* nodes are added to and removed from the volatile index (held in memory) a
* redo log is written to keep track of the changes. In case the Jackrabbit
* process terminates unexpected the redo log is applied when Jackrabbit is
* restarted the next time.
* <p>
* This class is not thread-safe.
*/
class DefaultRedoLog implements RedoLog {
/**
* Logger instance for this class
*/
private static final Logger log = LoggerFactory.getLogger(DefaultRedoLog.class);
/**
* Default name of the redo log file
*/
static final String REDO_LOG = "redo.log";
/**
* Prefix of the redo log files.
*/
static final String REDO_LOG_PREFIX = "redo_";
/**
* The .log extension.
*/
static final String DOT_LOG = ".log";
/**
* Implements a {@link ActionCollector} that counts all entries and sets
* {@link #entryCount}.
*/
private final ActionCollector ENTRY_COUNTER = new ActionCollector() {
public void collect(MultiIndex.Action a) {
entryCount++;
}
};
/**
* The directory where the log file is stored.
*/
private final Directory dir;
/**
* The name of the log file.
*/
private final String fileName;
/**
* The number of log entries in the log file
*/
private int entryCount = 0;
/**
* Writer to the log file
*/
private Writer out;
/**
* Creates a new <code>RedoLog</code> instance, which stores its log in the
* given directory.
*
* @param dir the directory where the redo log file is located.
* @param fileName the name of the redo log file.
* @throws IOException if an error occurs while reading the redo log.
*/
private DefaultRedoLog(Directory dir, String fileName) throws IOException {
this.dir = dir;
this.fileName = fileName;
read(ENTRY_COUNTER);
}
/**
* Creates a new <code>RedoLog</code> instance, which stores its log in the
* given directory.
*
* @param dir the directory where the redo log file is located.
* @param generation the redo log generation number.
* @return the redo log.
* @throws IOException if the redo log cannot be created.
*/
static RedoLog create(Directory dir, long generation) throws IOException {
String fileName;
if (generation == 0) {
fileName = REDO_LOG;
} else {
fileName = REDO_LOG_PREFIX + Long.toString(
generation, Character.MAX_RADIX) + DOT_LOG;
}
return new DefaultRedoLog(dir, fileName);
}
/**
* Returns <code>true</code> if this redo log contains any entries,
* <code>false</code> otherwise.
* @return <code>true</code> if this redo log contains any entries,
* <code>false</code> otherwise.
*/
public boolean hasEntries() {
return entryCount > 0;
}
/**
* Returns the number of entries in this redo log.
* @return the number of entries in this redo log.
*/
public int getSize() {
return entryCount;
}
/**
* Returns a List with all {@link MultiIndex.Action} instances in the
* redo log.
*
* @return an List with all {@link MultiIndex.Action} instances in the
* redo log.
* @throws IOException if an error occurs while reading from the redo log.
*/
public List<MultiIndex.Action> getActions() throws IOException {
final List<MultiIndex.Action> actions = new ArrayList<MultiIndex.Action>();
read(new ActionCollector() {
public void collect(MultiIndex.Action a) {
actions.add(a);
}
});
return actions;
}
/**
* Appends an action to the log.
*
* @param action the action to append.
* @throws IOException if the node cannot be written to the redo
* log.
*/
public void append(MultiIndex.Action action) throws IOException {
initOut();
out.write(action.toString() + "\n");
entryCount++;
}
/**
* Flushes all pending writes to the underlying file.
* @throws IOException if an error occurs while writing.
*/
public void flush() throws IOException {
if (out != null) {
out.flush();
}
}
/**
* Closes this redo log.
*
* @throws IOException if an error occurs while flushing pending writes.
*/
public void close() throws IOException {
if (out != null) {
out.close();
out = null;
}
}
/**
* Initializes the {@link #out} stream if it is not yet set.
* @throws IOException if an error occurs while creating the
* output stream.
*/
private void initOut() throws IOException {
if (out == null) {
OutputStream os = new IndexOutputStream(dir.createOutput(fileName));
out = new BufferedWriter(new OutputStreamWriter(os));
}
}
/**
* Reads the log file and calls back {@link DefaultRedoLog.ActionCollector}.
*
* @param collector called back for each {@link MultiIndex.Action} read.
* @throws IOException if an error occurs while reading from the
* log file.
*/
private void read(ActionCollector collector) throws IOException {
if (!dir.fileExists(fileName)) {
return;
}
InputStream in = new IndexInputStream(dir.openInput(fileName));
try {
BufferedReader reader = new BufferedReader(new InputStreamReader(in));
String line;
while ((line = reader.readLine()) != null) {
try {
collector.collect(MultiIndex.Action.fromString(line));
} catch (IllegalArgumentException e) {
log.warn("Malformed redo entry: " + e.getMessage());
}
}
} finally {
if (in != null) {
try {
in.close();
} catch (IOException e) {
log.warn("Exception while closing redo log: " + e.toString());
}
}
}
}
//-----------------------< internal >---------------------------------------
/**
* Helper interface to collect Actions read from the redo log.
*/
interface ActionCollector {
/** Called when an action is created */
void collect(MultiIndex.Action action);
}
}