PeekableReader.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.NoSuchElementException;
import de.larssh.utils.Nullables;
import edu.umd.cs.findbugs.annotations.Nullable;

/**
 * A {@link Reader} that allows peeking up to one character without removing it,
 * from the logical I/O stream, supporting a one-element lookahead.
 *
 * <p>
 * The methods {@link #hasNext()} and {@link #next()} are implemented similar to
 * the methods of {@link java.util.Iterator} to simplify working with character
 * based readers.
 */
public class PeekableReader extends Reader {
	/**
	 * The wrapped reader
	 */
	private final Reader reader;
	/**
	 * The next character read if {@link #state} is {@link ReaderState#PEEKED}, else
	 * undefined.
	 */
	private int peekedCharacter = -1;
	/**
	 * The reader's current inner state
	 */
	private ReaderState state = ReaderState.CALL_FOR_NEXT;
	/**
	 * The peeked character at the time of calling {@link #mark(int)} the last time.
	 */
	private int markedPeekedCharacter = -1;
	/**
	 * The reader's current inner state at the time of calling {@link #mark(int)}
	 * the last time.
	 */
	private ReaderState markedState = ReaderState.CALL_FOR_NEXT;

	/**
	 * {@inheritDoc}
	 */
	@Override
	@SuppressWarnings("PMD.CloseResource")
	public void close() throws IOException {
		reader.close();
	}

	/**
	 * Returns {@code true} if the reader has more characters. (In other words,
	 * returns {@code true} if {@link #next()} would return a character rather than
	 * throwing an exception.)
	 *
	 * @return {@code true} if the reader has more characters, else {@code false}
	 * @throws IOException if an I/O error occurs
	 */
	public final boolean hasNext() throws IOException {
		if (state == ReaderState.CALL_FOR_NEXT) {
			peekedCharacter = reader.read();
			state = peekedCharacter == -1 ? ReaderState.END_OF_DATA : ReaderState.PEEKED;
		}
		return state == ReaderState.PEEKED;
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public void mark(final int readAheadLimit) throws IOException {
		reader.mark(readAheadLimit);
		markedPeekedCharacter = peekedCharacter;
		markedState = state;
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public boolean markSupported() {
		return reader.markSupported();
	}

	/**
	 * Returns the next character of the reader.
	 *
	 * @return the next character of the reader
	 * @throws IOException            if an I/O error occurs
	 * @throws NoSuchElementException if the reader has no more characters
	 */
	public final char next() throws IOException {
		// Method "hasNext" peeks the next character (if required)
		if (!hasNext()) {
			throw new NoSuchElementException();
		}
		if (state == ReaderState.PEEKED) {
			state = ReaderState.CALL_FOR_NEXT;
		}
		final char next = (char) peekedCharacter;
		peekedCharacter = -1;
		return next;
	}

	/**
	 * Returns the next character in the reader, returned by {@link #read()},
	 * without removing it from the I/O stream.
	 *
	 * @return the next element in the iteration
	 * @throws IOException            if an I/O error occurs
	 * @throws NoSuchElementException if the iteration has no more elements
	 */
	public char peek() throws IOException {
		// Method "hasNext" peeks the next character (if required)
		if (!hasNext()) {
			throw new NoSuchElementException();
		}
		return (char) peekedCharacter;
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public int read() throws IOException {
		if (state == ReaderState.PEEKED) {
			state = ReaderState.CALL_FOR_NEXT;
			return (char) peekedCharacter;
		}
		return super.read();
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	@SuppressWarnings("PMD.CyclomaticComplexity")
	public int read(@Nullable final char[] buffer, final int offset, final int length) throws IOException {
		if (state == ReaderState.CALL_FOR_NEXT) {
			return reader.read(buffer, offset, length);
		}
		if (state == ReaderState.END_OF_DATA) {
			return -1;
		}
		// Error handling as of JavaDoc
		final char[] nonNullableBuffer = Nullables.orElseThrow(buffer);
		if (offset < 0 || length < 0 || offset + length > nonNullableBuffer.length) {
			throw new IndexOutOfBoundsException();
		}
		// Early exit: Avoid further processing in case no character were requested
		if (length == 0) {
			return 0;
		}
		// Insert peeked character as first character
		nonNullableBuffer[offset] = (char) peekedCharacter;
		state = ReaderState.CALL_FOR_NEXT;
		// No need to read further characters if just one character were requested
		if (length == 1) {
			return 1;
		}
		// Read further characters and handle possible end of data
		final int noOfCharacters = reader.read(buffer, offset + 1, length - 1);
		if (noOfCharacters == -1) {
			state = ReaderState.END_OF_DATA;
			return 1;
		}
		return noOfCharacters + 1;
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public void reset() throws IOException {
		reader.reset();
		this.peekedCharacter = markedPeekedCharacter;
		state = markedState;
	}


	/**
	 * This enumeration contains the possible inner states of
	 * {@link PeekableReader}.
	 */
	@SuppressWarnings("PMD.UnnecessaryModifier")
	private enum ReaderState {
		/**
		 * Status of a reader, that needs to read once information about the next
		 * character is required.
		 */
		CALL_FOR_NEXT, /**
		 * Status representing the end of data. The reader must not be called any
		 * longer. This is an end state and must not change once reached.
		 */
		END_OF_DATA, /**
		 * Status of a reader, that peeked the next character already.
		 */
		PEEKED;
	}

	/**
	 * Creates a new {@code PeekableReader} instance.
	 *
	 * @param reader The wrapped reader
	 */
	@java.lang.SuppressWarnings("all")
	@edu.umd.cs.findbugs.annotations.SuppressFBWarnings(justification = "generated code")
	@lombok.Generated
	public PeekableReader(final Reader reader) {
		this.reader = reader;
	}
}