MultiReader.java
// Generated by delombok at Mon Jan 06 07:19:11 UTC 2025
package de.larssh.utils.io;
import java.io.IOException;
import java.io.Reader;
import java.util.Iterator;
import java.util.Optional;
import edu.umd.cs.findbugs.annotations.Nullable;
/**
* A {@code MultiReader} concatenates multiple {@link Reader}s into one. That
* might be useful for e.g. reading log files of multiple days.
*
* <p>
* The {@code readerIterator} needs to supply the {@link Reader}s to concatenate
* one after the other. {@link Reader}s, which have been read, are closed prior
* requesting the next.
*/
public class MultiReader extends Reader {
/**
* Value, that is returned by {@link Reader#read(char[], int, int)} on end of a
* {@link Reader}.
*/
private static final int END_OF_READER = -1;
/**
* If {@code true} this {@link MultiReader} reached its end, meaning
* {@link #readerIterator} has no next element.
*/
private boolean closed = false;
/**
* The current {@link Reader} or empty if the next needs to be supplied first or
* if this {@link MultiReader} is closed.
*/
private Optional<Reader> currentReader = Optional.empty();
/**
* {@link Iterator}, called whenever the next {@link Reader} needs to be
* supplied. If it has no next element, the {@link MultiReader} is closed.
*/
private final Iterator<Reader> readerIterator;
/**
* {@inheritDoc}
*/
@Override
public void close() throws IOException {
closed = true;
if (currentReader.isPresent()) {
currentReader.get().close();
currentReader = Optional.empty();
}
}
/**
* Determines the current {@link Reader} by updating the internal state of this
* {@link MultiReader} if necessary.
*
* @return the current {@link Reader} or empty if closed
*/
@SuppressWarnings({"checkstyle:SuppressWarnings", "resource"})
private Optional<Reader> getCurrentReader() {
if (closed) {
return Optional.empty();
}
if (currentReader.isPresent()) {
return currentReader;
}
currentReader = readerIterator.hasNext() ? Optional.of(readerIterator.next()) : Optional.empty();
if (!currentReader.isPresent()) {
closed = true;
}
return currentReader;
}
/**
* {@inheritDoc}
*/
@Override
@SuppressWarnings("checkstyle:SuppressWarnings")
public int read(@Nullable final char[] outputBuffer, final int offset, final int length) throws IOException {
final Optional<Reader> reader = getCurrentReader();
if (!reader.isPresent()) {
return END_OF_READER;
}
// Read from current reader
@SuppressWarnings("resource")
final int chars = reader.get().read(outputBuffer, offset, length);
if (chars == length) {
return chars;
}
// If less characters than expected have been read from the current reader,
// close it and continue with the next
currentReader = Optional.empty();
reader.get().close();
// Read additional chunk of characters
final int nonNegativeChars = chars == END_OF_READER ? 0 : chars;
final int nextChars = read(outputBuffer, offset + nonNegativeChars, length - nonNegativeChars);
return nonNegativeChars + (nextChars == END_OF_READER ? 0 : nextChars);
}
/**
* {@inheritDoc}
*/
@Override
@SuppressWarnings({"checkstyle:SuppressWarnings", "resource"})
public boolean ready() throws IOException {
final Optional<Reader> reader = getCurrentReader();
return reader.isPresent() && reader.get().ready();
}
/**
* Creates a new {@code MultiReader} instance.
*
* @param readerIterator {@link Iterator}, called whenever the next {@link Reader} needs to be
* supplied. If it has no next element, the {@link MultiReader} is closed.
*/
@java.lang.SuppressWarnings("all")
@edu.umd.cs.findbugs.annotations.SuppressFBWarnings(justification = "generated code")
@lombok.Generated
public MultiReader(final Iterator<Reader> readerIterator) {
this.readerIterator = readerIterator;
}
}