/* * Copyright (c) 2006-2011 Christian Plattner. All rights reserved. * Please refer to the LICENSE.txt for licensing details. */ package ch.ethz.ssh2; import java.io.IOException; import java.io.InputStream; /** * A <code>StreamGobbler</code> is an InputStream that uses an internal worker * thread to constantly consume input from another InputStream. It uses a buffer * to store the consumed data. The buffer size is automatically adjusted, if needed. * <p/> * This class is sometimes very convenient - if you wrap a session's STDOUT and STDERR * InputStreams with instances of this class, then you don't have to bother about * the shared window of STDOUT and STDERR in the low level SSH-2 protocol, * since all arriving data will be immediatelly consumed by the worker threads. * Also, as a side effect, the streams will be buffered (e.g., single byte * read() operations are faster). * <p/> * Other SSH for Java libraries include this functionality by default in * their STDOUT and STDERR InputStream implementations, however, please be aware * that this approach has also a downside: * <p/> * If you do not call the StreamGobbler's <code>read()</code> method often enough * and the peer is constantly sending huge amounts of data, then you will sooner or later * encounter a low memory situation due to the aggregated data (well, it also depends on the Java heap size). * Joe Average will like this class anyway - a paranoid programmer would never use such an approach. * <p/> * The term "StreamGobbler" was taken from an article called "When Runtime.exec() won't", * see http://www.javaworld.com/javaworld/jw-12-2000/jw-1229-traps.html. * * @author Christian Plattner * @version 2.50, 03/15/10 */ public class StreamGobbler extends InputStream { class GobblerThread extends Thread { @Override public void run() { byte[] buff = new byte[8192]; while (true) { try { int avail = is.read(buff); synchronized (synchronizer) { if (avail <= 0) { isEOF = true; synchronizer.notifyAll(); break; } int space_available = buffer.length - write_pos; if (space_available < avail) { /* compact/resize buffer */ int unread_size = write_pos - read_pos; int need_space = unread_size + avail; byte[] new_buffer = buffer; if (need_space > buffer.length) { int inc = need_space / 3; inc = (inc < 256) ? 256 : inc; inc = (inc > 8192) ? 8192 : inc; new_buffer = new byte[need_space + inc]; } if (unread_size > 0) System.arraycopy(buffer, read_pos, new_buffer, 0, unread_size); buffer = new_buffer; read_pos = 0; write_pos = unread_size; } System.arraycopy(buff, 0, buffer, write_pos, avail); write_pos += avail; synchronizer.notifyAll(); } } catch (IOException e) { synchronized (synchronizer) { exception = e; synchronizer.notifyAll(); break; } } } } } private InputStream is; private final Object synchronizer = new Object(); private boolean isEOF = false; private boolean isClosed = false; private IOException exception = null; private byte[] buffer = new byte[2048]; private int read_pos = 0; private int write_pos = 0; public StreamGobbler(InputStream is) { this.is = is; GobblerThread t = new GobblerThread(); t.setDaemon(true); t.start(); } @Override public int read() throws IOException { boolean wasInterrupted = false; try { synchronized (synchronizer) { if (isClosed) throw new IOException("This StreamGobbler is closed."); while (read_pos == write_pos) { if (exception != null) throw exception; if (isEOF) return -1; try { synchronizer.wait(); } catch (InterruptedException e) { wasInterrupted = true; } } return buffer[read_pos++] & 0xff; } } finally { if (wasInterrupted) Thread.currentThread().interrupt(); } } @Override public int available() throws IOException { synchronized (synchronizer) { if (isClosed) throw new IOException("This StreamGobbler is closed."); return write_pos - read_pos; } } @Override public int read(byte[] b) throws IOException { return read(b, 0, b.length); } @Override public void close() throws IOException { synchronized (synchronizer) { if (isClosed) return; isClosed = true; isEOF = true; synchronizer.notifyAll(); is.close(); } } @Override public int read(byte[] b, int off, int len) throws IOException { if (b == null) throw new NullPointerException(); if ((off < 0) || (len < 0) || ((off + len) > b.length) || ((off + len) < 0) || (off > b.length)) throw new IndexOutOfBoundsException(); if (len == 0) return 0; boolean wasInterrupted = false; try { synchronized (synchronizer) { if (isClosed) throw new IOException("This StreamGobbler is closed."); while (read_pos == write_pos) { if (exception != null) throw exception; if (isEOF) return -1; try { synchronizer.wait(); } catch (InterruptedException e) { wasInterrupted = true; } } int avail = write_pos - read_pos; avail = (avail > len) ? len : avail; System.arraycopy(buffer, read_pos, b, off, avail); read_pos += avail; return avail; } } finally { if (wasInterrupted) Thread.currentThread().interrupt(); } } }