/*
 * 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();
		}
	}
}