MultiInputStream.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.InputStream;
import java.util.Iterator;
import java.util.Optional;
import edu.umd.cs.findbugs.annotations.Nullable;

/**
 * A {@code MultiInputStream} concatenates multiple {@link InputStream}s into
 * one. That might be useful for e.g. reading log files of multiple days.
 *
 * <p>
 * The {@code inputStreamIterator} needs to supply the {@link InputStream}s to
 * concatenate one after the other. {@link InputStream}s, which have been read,
 * are closed prior requesting the next.
 */
public class MultiInputStream extends InputStream {
	/**
	 * Value, that is returned by {@link InputStream#read()} on end of an
	 * {@link InputStream}.
	 */
	private static final int END_OF_INPUT_STREAM = -1;
	/**
	 * If {@code true} this {@link MultiInputStream} reached its end, meaning
	 * {@link #inputStreamIterator} has no next element.
	 */
	private boolean closed = false;
	/**
	 * The current {@link InputStream} or empty if the next needs to be supplied
	 * first or if this {@link MultiInputStream} is closed.
	 */
	private Optional<InputStream> currentInputStream = Optional.empty();
	/**
	 * {@link Iterator}, called whenever the next {@link InputStream} needs to be
	 * supplied. If it has no next element, the {@link MultiInputStream} is closed.
	 */
	private final Iterator<InputStream> inputStreamIterator;

	/**
	 * {@inheritDoc}
	 */
	@Override
	public void close() throws IOException {
		closed = true;
		if (currentInputStream.isPresent()) {
			currentInputStream.get().close();
			currentInputStream = Optional.empty();
		}
	}

	/**
	 * Determines the current {@link InputStream} by updating the internal state of
	 * this {@link MultiInputStream} if necessary.
	 *
	 * @return the current {@link InputStream} or empty if closed
	 */
	@SuppressWarnings({"checkstyle:SuppressWarnings", "resource"})
	private Optional<InputStream> getCurrentInputStream() {
		if (closed) {
			return Optional.empty();
		}
		if (currentInputStream.isPresent()) {
			return currentInputStream;
		}
		currentInputStream = inputStreamIterator.hasNext() ? Optional.of(inputStreamIterator.next()) : Optional.empty();
		if (!currentInputStream.isPresent()) {
			closed = true;
		}
		return currentInputStream;
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	@SuppressWarnings("checkstyle:SuppressWarnings")
	public int read() throws IOException {
		final Optional<InputStream> inputStream = getCurrentInputStream();
		if (!inputStream.isPresent()) {
			return END_OF_INPUT_STREAM;
		}
		// Read from current input stream
		@SuppressWarnings("resource")
		final int value = inputStream.get().read();
		if (value != END_OF_INPUT_STREAM) {
			return value;
		}
		// At the end of the current input stream, close it and continue with the next
		currentInputStream = Optional.empty();
		inputStream.get().close();
		return read();
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	@SuppressWarnings("checkstyle:SuppressWarnings")
	public int read(@Nullable final byte[] outputBuffer, final int offset, final int length) throws IOException {
		final Optional<InputStream> inputStream = getCurrentInputStream();
		if (!inputStream.isPresent()) {
			return END_OF_INPUT_STREAM;
		}
		// Read from current input stream
		@SuppressWarnings("resource")
		final int chars = inputStream.get().read(outputBuffer, offset, length);
		if (chars == length) {
			return chars;
		}
		// If less bytes than expected have been read, closing the current input stream
		// and continuing with the next
		currentInputStream = Optional.empty();
		inputStream.get().close();
		// Read additional chunk of bytes
		final int nonNegativeChars = chars == END_OF_INPUT_STREAM ? 0 : chars;
		final int nextChars = read(outputBuffer, offset + nonNegativeChars, length - nonNegativeChars);
		return nonNegativeChars + (nextChars == END_OF_INPUT_STREAM ? 0 : nextChars);
	}

	/**
	 * Creates a new {@code MultiInputStream} instance.
	 *
	 * @param inputStreamIterator {@link Iterator}, called whenever the next {@link InputStream} needs to be
	 * supplied. If it has no next element, the {@link MultiInputStream} is closed.
	 */
	@java.lang.SuppressWarnings("all")
	@edu.umd.cs.findbugs.annotations.SuppressFBWarnings(justification = "generated code")
	@lombok.Generated
	public MultiInputStream(final Iterator<InputStream> inputStreamIterator) {
		this.inputStreamIterator = inputStreamIterator;
	}
}