All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.refcodes.codec.BaseDecoder Maven / Gradle / Ivy

// /////////////////////////////////////////////////////////////////////////////
// REFCODES.ORG
// /////////////////////////////////////////////////////////////////////////////
// This code is copyright (c) by Siegfried Steiner, Munich, Germany, distributed
// on an "AS IS" BASIS WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, and licen-
// sed under the following (see "http://en.wikipedia.org/wiki/Multi-licensing")
// licenses:
// -----------------------------------------------------------------------------
// GNU General Public License, v3.0 ("http://www.gnu.org/licenses/gpl-3.0.html")
// -----------------------------------------------------------------------------
// Apache License, v2.0 ("http://www.apache.org/licenses/TEXT-2.0")
// -----------------------------------------------------------------------------
// Please contact the copyright holding author(s) of the software artifacts in
// question for licensing issues not being covered by the above listed licenses,
// also regarding commercial licensing models or regarding the compatibility
// with other open source licenses.
// /////////////////////////////////////////////////////////////////////////////

package org.refcodes.codec;

import java.io.EOFException;
import java.io.IOException;
import java.io.InputStream;

import org.refcodes.codec.BaseMetricsAccessor.BaseMetricsBuilder;
import org.refcodes.codec.BaseMetricsAccessor.BaseMetricsProperty;
import org.refcodes.component.AbstractConnectableAutomaton;
import org.refcodes.component.ConnectionComponent;
import org.refcodes.component.Openable;
import org.refcodes.io.BytesDestination;
import org.refcodes.io.BytesReceiver;
import org.refcodes.io.BytesReceiverDecorator;
import org.refcodes.io.InputStreamByteReceiver;

/**
 * The {@link BaseDecoder} implements the {@link BaseBuilder} functionality in
 * terms of a {@link BytesReceiver}. It implements the {@link Decoder} interface
 * attaching to the "refcodes-io" artifact via the {@link BytesReceiver}
 * interface. This increases the scope of usage scenarios.
 */
public class BaseDecoder extends AbstractConnectableAutomaton implements BaseMetricsProperty, BaseMetricsBuilder, Decoder {

	// /////////////////////////////////////////////////////////////////////////
	// STATICS:
	// /////////////////////////////////////////////////////////////////////////

	private static final int BYTE_MASK = 0xFF;
	private static final int BITS_PER_BYTE = 8;

	// /////////////////////////////////////////////////////////////////////////
	// VARIABLES:
	// /////////////////////////////////////////////////////////////////////////

	private BytesReceiver _byteReceiver;
	private BaseMetrics _baseMetrics = BaseMetricsConfig.BASE64;
	private int _trailingBytes = 0;
	int _word = 0;
	private byte[] _decodedBytes = new byte[_baseMetrics.getBytesPerInt()];
	private int _readIndex = 0;

	// /////////////////////////////////////////////////////////////////////////
	// CONSTRUCTORS:
	// /////////////////////////////////////////////////////////////////////////

	/**
	 * Constructs the {@link BaseDecoder} reading the data to be decoded from
	 * the provided {@link InputStream}.
	 *
	 * @param aInputStream the input stream
	 * 
	 * @throws IOException throw in case using the {@link InputStream} caused
	 *         I/O related problems.
	 */
	public BaseDecoder( InputStream aInputStream ) throws IOException {
		this( new InputStreamByteReceiver( aInputStream ) );
	}

	/**
	 * Constructs the {@link BaseDecoder} reading the data to be decoded from
	 * the provided {@link InputStream}.
	 *
	 * @param aInputStream the input stream
	 * @param aBaseMetrics The {@link BaseMetrics} to use.
	 * 
	 * @throws IOException throw in case using the {@link InputStream} caused
	 *         I/O related problems.
	 */
	public BaseDecoder( InputStream aInputStream, BaseMetrics aBaseMetrics ) throws IOException {
		this( new InputStreamByteReceiver( aInputStream ), aBaseMetrics );
	}

	/**
	 * Constructs the {@link BaseDecoder} reading the data to be decoded from
	 * the provided {@link BytesDestination}.
	 * 
	 * @param aByteProvider The {@link BytesDestination} from which to read the
	 *        data.
	 */
	public BaseDecoder( BytesDestination aByteProvider ) {
		try {
			open( new BytesReceiverDecorator( aByteProvider ) );
		}
		catch ( IOException ignore ) {}
	}

	/**
	 * Constructs the {@link BaseDecoder} reading the data to be decoded from
	 * the provided {@link BytesReceiver}.
	 * 
	 * @param aByteReceiver The {@link BytesReceiver} from which to read the
	 *        data.
	 * 
	 * @throws IOException in case opening or accessing an open line
	 *         (connection, junction, link) caused problems.
	 */
	public BaseDecoder( BytesReceiver aByteReceiver ) throws IOException {
		open( aByteReceiver );
	}

	/**
	 * Constructs the {@link BaseDecoder} reading the data to be decoded from
	 * the provided {@link InputStreamByteReceiver}.
	 * 
	 * @param aInputStreamByteReceiver The {@link InputStreamByteReceiver} to
	 *        use.
	 * @param aBaseMetrics The {@link BaseMetrics} to use.
	 * 
	 * @throws IOException in case opening or accessing an open line
	 *         (connection, junction, link) caused problems.
	 */
	public BaseDecoder( InputStreamByteReceiver aInputStreamByteReceiver, BaseMetrics aBaseMetrics ) throws IOException {
		this( aInputStreamByteReceiver );
		setBaseMetrics( aBaseMetrics );
	}

	/**
	 * Explicit default constructor.
	 */
	protected BaseDecoder() {}

	// /////////////////////////////////////////////////////////////////////////
	// METHODS:
	// /////////////////////////////////////////////////////////////////////////

	/**
	 * {@inheritDoc}
	 */
	@Override
	public BaseMetrics getBaseMetrics() {
		return _baseMetrics;
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public void setBaseMetrics( BaseMetrics aBaseMetrics ) {
		_baseMetrics = aBaseMetrics;
		_decodedBytes = new byte[_baseMetrics.getBytesPerInt()];
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public BaseDecoder withBaseMetrics( BaseMetrics aBaseMetrics ) {
		setBaseMetrics( aBaseMetrics );
		return this;
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public int available() throws IOException {
		return _byteReceiver.available();
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public byte[] receiveAllBytes() throws IOException {
		if ( _readIndex > 0 ) {
			final byte[] theDecodedBytes = new byte[_decodedBytes.length - _readIndex];
			int l = 0;
			for ( int i = _readIndex; i < _decodedBytes.length; i++ ) {
				theDecodedBytes[l++] = _decodedBytes[i];
			}
			_readIndex = 0;
			return theDecodedBytes;
		}
		if ( !_byteReceiver.hasAvailable() ) {
			throw new EOFException( "Reached end of file (stream), no more data avaialble!" );
		}
		try {
			char eRead;
			for ( int l = 0; l < _baseMetrics.getDigitsPerInt(); l++ ) {
				eRead = (char) _byteReceiver.receiveByte();
				while ( eRead == '\n' || eRead == '\r' ) { // Skip any line breaks!
					eRead = (char) _byteReceiver.receiveByte();
				}
				_word <<= _baseMetrics.getBitsPerDigit();
				if ( eRead != _baseMetrics.getPaddingChar() ) {
					_word |= _baseMetrics.toValue( eRead ) & BYTE_MASK;
				}
				else {
					_trailingBytes++;
				}
			}
			for ( int i = 0; i < _baseMetrics.getBytesPerInt(); i++ ) {
				final int theIndex = _baseMetrics.getBytesPerInt() - i - 1;
				_decodedBytes[theIndex] = (byte) _word;
				_word >>= BITS_PER_BYTE;
			}
			_word = 0;
		}
		catch ( IOException e ) {
			throw new IOException( "Unable to read from the provided receiver <" + _byteReceiver + ">!", e );
		}

		if ( _trailingBytes != 0 ) {
			final int theLastBlockSize = _baseMetrics.getBytesPerInt() - _trailingBytes;
			final byte[] theDecodedBytes = new byte[theLastBlockSize];
			for ( int i = 0; i < theLastBlockSize; i++ ) {
				theDecodedBytes[i] = _decodedBytes[i];
			}
			return theDecodedBytes;
		}

		return _decodedBytes;
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public byte receiveByte() throws IOException {
		if ( _readIndex >= _baseMetrics.getBytesPerInt() - _trailingBytes ) {
			if ( !_byteReceiver.hasAvailable() ) {
				throw new EOFException( "Reached end of file (stream), no more data avaialble!" );
			}
			_trailingBytes = 0;
			_readIndex = 0;
		}
		if ( _readIndex == 0 ) {
			receiveAllBytes();
		}
		final byte theDecodedByte = _decodedBytes[_readIndex++];
		if ( _readIndex >= _decodedBytes.length ) {
			_readIndex = 0;
		}
		return theDecodedByte;
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public synchronized void close() throws IOException {
		try {
			_byteReceiver.close();
		}
		catch ( IOException e ) {
			throw new IOException( "Unable to close the receiver <" + _byteReceiver + ">", e );
		}
		super.close();
	}

	// /////////////////////////////////////////////////////////////////////////
	// HOOKS:
	// /////////////////////////////////////////////////////////////////////////

	/**
	 * Open.
	 *
	 * @param aConnection the connection
	 * 
	 * @throws IOException the open exception
	 */
	protected void open( BytesDestination aConnection ) throws IOException {
		if ( aConnection instanceof BytesReceiver ) {
			_byteReceiver = (BytesReceiver) aConnection;
		}
		else {
			_byteReceiver = new BytesReceiverDecorator( aConnection );
		}
		if ( !_byteReceiver.isOpened() ) {
			if ( _byteReceiver instanceof Openable ) {
				( (Openable) _byteReceiver ).open();
			}
			else {
				throw new IOException( "The provided connection is in status <" + _byteReceiver.getConnectionStatus() + "> but does not provide the <" + Openable.class.getName() + "> interface." );
			}
		}
		open();
	}

	// /////////////////////////////////////////////////////////////////////////
	// INNER CLASSES:
	// /////////////////////////////////////////////////////////////////////////

	/**
	 * The {@link BaseDecoderConnection} implements the {@link BaseDecoder}
	 * functionality in terms of a {@link ConnectionComponent}. In addition to
	 * the {@link BaseDecoder} it provides means to open a dedicated
	 * {@link BytesDestination} connection.
	 */
	public static class BaseDecoderConnection extends BaseDecoder implements ConnectionComponent {

		// /////////////////////////////////////////////////////////////////////
		// METHODS:
		// /////////////////////////////////////////////////////////////////////

		/**
		 * {@inheritDoc}
		 */
		@Override
		public void open( BytesDestination aConnection ) throws IOException {
			super.open( aConnection );
		}
	}
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy