org.jline.utils.PumpReader Maven / Gradle / Ivy
/*
* Copyright (c) 2002-2017, the original author(s).
*
* This software is distributable under the BSD license. See the terms of the
* BSD license in the documentation provided with this software.
*
* https://opensource.org/licenses/BSD-3-Clause
*/
package org.jline.utils;
import java.io.IOException;
import java.io.InterruptedIOException;
import java.io.Reader;
import java.nio.ByteBuffer;
import java.nio.CharBuffer;
import java.nio.charset.Charset;
import java.nio.charset.CharsetEncoder;
import java.nio.charset.CoderResult;
import java.nio.charset.CodingErrorAction;
public class PumpReader extends Reader {
private static final int EOF = -1;
private static final int DEFAULT_BUFFER_SIZE = 4096;
// Read and write buffer are backed by the same array
private final CharBuffer readBuffer;
private final CharBuffer writeBuffer;
private final Writer writer;
private boolean closed;
public PumpReader() {
this(DEFAULT_BUFFER_SIZE);
}
public PumpReader(int bufferSize) {
char[] buf = new char[Math.max(bufferSize, 2)];
this.readBuffer = CharBuffer.wrap(buf);
this.writeBuffer = CharBuffer.wrap(buf);
this.writer = new Writer(this);
// There are no bytes available to read after initialization
readBuffer.limit(0);
}
public java.io.Writer getWriter() {
return this.writer;
}
public java.io.InputStream createInputStream(Charset charset) {
return new InputStream(this, charset);
}
/**
* Blocks until more input is available, even if {@link #readBuffer} already
* contains some chars; or until the reader is closed.
*
* @return true if more input is available, false if no additional input is
* available and the reader is closed
* @throws InterruptedIOException If {@link #wait()} is interrupted
*/
private boolean waitForMoreInput() throws InterruptedIOException {
if (!writeBuffer.hasRemaining()) {
throw new AssertionError("No space in write buffer");
}
int oldRemaining = readBuffer.remaining();
do {
if (closed) {
return false;
}
// Wake up waiting writers
notifyAll();
try {
wait();
} catch (InterruptedException e) {
throw new InterruptedIOException();
}
} while (readBuffer.remaining() <= oldRemaining);
return true;
}
/**
* Waits until {@code buffer.hasRemaining() == true}, or it is false and
* the reader is {@link #closed}.
*
* @return true if {@code buffer.hasRemaining() == true}; false otherwise
* when reader is closed
*/
private boolean wait(CharBuffer buffer) throws InterruptedIOException {
while (!buffer.hasRemaining()) {
if (closed) {
return false;
}
// Wake up waiting readers/writers
notifyAll();
try {
wait();
} catch (InterruptedException e) {
throw new InterruptedIOException();
}
}
return true;
}
/**
* Blocks until input is available or the reader is closed.
*
* @return true if input is available, false if no input is available and the reader is closed
* @throws InterruptedIOException If {@link #wait()} is interrupted
*/
private boolean waitForInput() throws InterruptedIOException {
return wait(readBuffer);
}
/**
* Blocks until there is new space available for buffering or the
* reader is closed.
*
* @throws InterruptedIOException If {@link #wait()} is interrupted
* @throws ClosedException If the reader was closed
*/
private void waitForBufferSpace() throws InterruptedIOException, ClosedException {
// Check `closed` to throw even if writer buffer has space available
if (!wait(writeBuffer) || closed) {
throw new ClosedException();
}
}
private static boolean rewind(CharBuffer buffer, CharBuffer other) {
// Extend limit of other buffer if there is additional input/output available
if (buffer.position() > other.position()) {
other.limit(buffer.position());
}
// If we have reached the end of the buffer, rewind and set the new limit
if (buffer.position() == buffer.capacity()) {
buffer.rewind();
buffer.limit(other.position());
return true;
} else {
return false;
}
}
/**
* Attempts to find additional input by rewinding the {@link #readBuffer}.
* Updates the {@link #writeBuffer} to make read bytes available for buffering.
*
* @return If more input is available
*/
private boolean rewindReadBuffer() {
boolean rw = rewind(readBuffer, writeBuffer) && readBuffer.hasRemaining();
notifyAll();
return rw;
}
/**
* Attempts to find additional buffer space by rewinding the {@link #writeBuffer}.
* Updates the {@link #readBuffer} to make written bytes available to the reader.
*/
private void rewindWriteBuffer() {
rewind(writeBuffer, readBuffer);
notifyAll();
}
@Override
public synchronized boolean ready() {
return readBuffer.hasRemaining();
}
public synchronized int available() {
int count = readBuffer.remaining();
if (writeBuffer.position() < readBuffer.position()) {
count += writeBuffer.position();
}
return count;
}
@Override
public synchronized int read() throws IOException {
if (!waitForInput()) {
return EOF;
}
int b = readBuffer.get();
rewindReadBuffer();
return b;
}
private int copyFromBuffer(char[] cbuf, int off, int len) {
len = Math.min(len, readBuffer.remaining());
readBuffer.get(cbuf, off, len);
return len;
}
@Override
public synchronized int read(char[] cbuf, int off, int len) throws IOException {
if (len == 0) {
return 0;
}
if (!waitForInput()) {
return EOF;
}
int count = copyFromBuffer(cbuf, off, len);
if (rewindReadBuffer() && count < len) {
count += copyFromBuffer(cbuf, off + count, len - count);
rewindReadBuffer();
}
return count;
}
@Override
public synchronized int read(CharBuffer target) throws IOException {
if (!target.hasRemaining()) {
return 0;
}
if (!waitForInput()) {
return EOF;
}
int count = readBuffer.read(target);
if (rewindReadBuffer() && target.hasRemaining()) {
count += readBuffer.read(target);
rewindReadBuffer();
}
return count;
}
private void encodeBytes(CharsetEncoder encoder, ByteBuffer output) throws IOException {
int oldPos = output.position();
CoderResult result = encoder.encode(readBuffer, output, false);
int encodedCount = output.position() - oldPos;
if (result.isUnderflow()) {
boolean hasMoreInput = rewindReadBuffer();
boolean reachedEndOfInput = false;
// If encoding did not make any progress must block for more input
if (encodedCount == 0 && !hasMoreInput) {
reachedEndOfInput = !waitForMoreInput();
}
result = encoder.encode(readBuffer, output, reachedEndOfInput);
if (result.isError()) {
result.throwException();
}
if (!reachedEndOfInput && output.position() - oldPos == 0) {
throw new AssertionError("Failed to encode any chars");
}
rewindReadBuffer();
} else if (result.isOverflow()) {
if (encodedCount == 0) {
throw new AssertionError("Output buffer has not enough space");
}
} else {
result.throwException();
}
}
synchronized int readBytes(CharsetEncoder encoder, byte[] b, int off, int len) throws IOException {
if (!waitForInput()) {
return 0;
}
ByteBuffer output = ByteBuffer.wrap(b, off, len);
encodeBytes(encoder, output);
return output.position() - off;
}
synchronized void readBytes(CharsetEncoder encoder, ByteBuffer output) throws IOException {
if (!waitForInput()) {
return;
}
encodeBytes(encoder, output);
}
synchronized void write(char c) throws IOException {
waitForBufferSpace();
writeBuffer.put(c);
rewindWriteBuffer();
}
synchronized void write(char[] cbuf, int off, int len) throws IOException {
while (len > 0) {
waitForBufferSpace();
// Copy as much characters as we can
int count = Math.min(len, writeBuffer.remaining());
writeBuffer.put(cbuf, off, count);
off += count;
len -= count;
// Update buffer states and rewind if necessary
rewindWriteBuffer();
}
}
synchronized void write(String str, int off, int len) throws IOException {
char[] buf = writeBuffer.array();
while (len > 0) {
waitForBufferSpace();
// Copy as much characters as we can
int count = Math.min(len, writeBuffer.remaining());
// CharBuffer.put(String) doesn't use getChars so do it manually
str.getChars(off, off + count, buf, writeBuffer.position());
writeBuffer.position(writeBuffer.position() + count);
off += count;
len -= count;
// Update buffer states and rewind if necessary
rewindWriteBuffer();
}
}
synchronized void flush() {
// Avoid waking up readers when there is nothing to read
if (readBuffer.hasRemaining()) {
// Notify readers
notifyAll();
}
}
@Override
public synchronized void close() throws IOException {
this.closed = true;
notifyAll();
}
private static class Writer extends java.io.Writer {
private final PumpReader reader;
private Writer(PumpReader reader) {
this.reader = reader;
}
@Override
public void write(int c) throws IOException {
reader.write((char) c);
}
@Override
public void write(char[] cbuf, int off, int len) throws IOException {
reader.write(cbuf, off, len);
}
@Override
public void write(String str, int off, int len) throws IOException {
reader.write(str, off, len);
}
@Override
public void flush() throws IOException {
reader.flush();
}
@Override
public void close() throws IOException {
reader.close();
}
}
private static class InputStream extends java.io.InputStream {
private final PumpReader reader;
private final CharsetEncoder encoder;
// To encode a character with multiple bytes (e.g. certain Unicode characters)
// we need enough space to encode them. Reading would fail if the read() method
// is used to read a single byte in these cases.
// Use this buffer to ensure we always have enough space to encode a character.
private final ByteBuffer buffer;
private InputStream(PumpReader reader, Charset charset) {
this.reader = reader;
this.encoder = charset.newEncoder()
.onUnmappableCharacter(CodingErrorAction.REPLACE)
.onMalformedInput(CodingErrorAction.REPLACE);
this.buffer = ByteBuffer.allocate((int) Math.ceil(encoder.maxBytesPerChar() * 2));
// No input available after initialization
buffer.limit(0);
}
@Override
public int available() throws IOException {
return (int) (reader.available() * (double) this.encoder.averageBytesPerChar()) + buffer.remaining();
}
@Override
public int read() throws IOException {
if (!buffer.hasRemaining() && !readUsingBuffer()) {
return EOF;
}
return buffer.get() & 0xFF;
}
private boolean readUsingBuffer() throws IOException {
buffer.clear(); // Reset buffer
reader.readBytes(encoder, buffer);
buffer.flip();
return buffer.hasRemaining();
}
private int copyFromBuffer(byte[] b, int off, int len) {
len = Math.min(len, buffer.remaining());
buffer.get(b, off, len);
return len;
}
@Override
public int read(byte[] b, int off, int len) throws IOException {
if (len == 0) {
return 0;
}
int read;
if (buffer.hasRemaining()) {
read = copyFromBuffer(b, off, len);
if (read == len) {
return len;
}
off += read;
len -= read;
} else {
read = 0;
}
// Do we have enough space to avoid buffering?
if (len >= buffer.capacity()) {
read += reader.readBytes(this.encoder, b, off, len);
} else if (readUsingBuffer()) {
read += copyFromBuffer(b, off, len);
}
// Return EOF if we didn't read any bytes
return read == 0 ? EOF : read;
}
@Override
public void close() throws IOException {
reader.close();
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy