/*
* Copyright (c) 1998-2011 Caucho Technology -- all rights reserved
*
* This file is part of Resin(R) Open Source
*
* Each copy or derived work must preserve the copyright notice and this
* notice unmodified.
*
* Resin Open Source is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* Resin Open Source is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE, or any warranty
* of NON-INFRINGEMENT. See the GNU General Public License for more
* details.
*
* You should have received a copy of the GNU General Public License
* along with Resin Open Source; if not, write to the
*
* Free Software Foundation, Inc.
* 59 Temple Place, Suite 330
* Boston, MA 02111-1307 USA
*
* @author Scott Ferguson
*/
package com.caucho.quercus.lib.file;
import com.caucho.quercus.QuercusModuleException;
import com.caucho.quercus.env.*;
import com.caucho.vfs.ReadStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.UnsupportedEncodingException;
import java.util.logging.Logger;
/**
* Represents a Quercus file open for reading
*/
public class AbstractBinaryInput
implements BinaryInput
{
private static final Logger log
= Logger.getLogger(AbstractBinaryInput.class.getName());
private Env _env;
private final LineReader _lineReader;
private ReadStream _is;
// Set to true when EOF is read from the input stream.
private boolean _isEOF = false;
protected AbstractBinaryInput(Env env)
{
_env = env;
_lineReader = new LineReader(env);
}
protected AbstractBinaryInput(Env env, ReadStream is)
{
this(env);
init(is);
}
public void init(ReadStream is)
{
_is = is;
}
//
// read methods
//
/**
* Returns the input stream.
*/
public InputStream getInputStream()
{
return _is;
}
/**
* Opens a copy.
*/
public BinaryInput openCopy()
throws IOException
{
throw new UnsupportedOperationException(getClass().getName());
}
public void setEncoding(String encoding)
throws UnsupportedEncodingException
{
if (_is != null)
_is.setEncoding(encoding);
}
/**
* Unread the last byte.
*/
public void unread()
throws IOException
{
if (_is != null) {
_is.unread();
_isEOF = false;
}
}
/**
* Reads a character from a file, returning -1 on EOF.
*/
public int read()
throws IOException
{
if (_is != null) {
int c = _is.read();
if (c == -1)
_isEOF = true;
else
_isEOF = false;
return c;
}
else
return -1;
}
/**
* Reads a buffer from a file, returning -1 on EOF.
*/
public int read(byte []buffer, int offset, int length)
throws IOException
{
if (_is != null) {
int c = _is.read(buffer, offset, length);
if (c == -1)
_isEOF = true;
else
_isEOF = false;
return c;
}
else
return -1;
}
/**
* Reads a buffer from a file, returning -1 on EOF.
*/
public int read(char []buffer, int offset, int length)
throws IOException
{
if (_is != null) {
int c = _is.read(buffer, offset, length);
if (c == -1)
_isEOF = true;
else
_isEOF = false;
return c;
}
else
return -1;
}
/**
* Reads into a binary builder.
*/
public StringValue read(int length)
throws IOException
{
if (_is == null)
return null;
StringValue bb = _env.createBinaryBuilder();
if (bb.appendRead(_is, length) > 0)
return bb;
else
return null;
}
/**
* Reads the optional linefeed character from a \r\n
*/
public boolean readOptionalLinefeed()
throws IOException
{
if (_is == null)
return false;
int ch = read();
if (ch == '\n') {
return true;
}
else {
_is.unread();
return false;
}
}
public void writeToStream(OutputStream os, int length)
throws IOException
{
if (_is != null) {
_is.writeToStream(os, length);
}
}
/**
* Reads a line from a file, returning null on EOF.
*/
public StringValue readLine(long length)
throws IOException
{
return _lineReader.readLine(_env, this, length);
}
/**
* Appends to a string builder.
*/
public StringValue appendTo(StringValue builder)
throws IOException
{
if (_is != null)
return builder.append(_is);
else
return builder;
}
/**
* Returns true on the EOF.
*/
public boolean isEOF()
{
if (_is == null)
return true;
else {
return _isEOF;
}
}
/**
* Returns the current location in the file.
*/
public long getPosition()
{
if (_is == null)
return -1;
else
return _is.getPosition();
}
/**
* Sets the current location in the file.
*/
public boolean setPosition(long offset)
{
if (_is == null)
return false;
_isEOF = false;
try {
return _is.setPosition(offset);
} catch (IOException e) {
throw new QuercusModuleException(e);
}
}
public long seek(long offset, int whence)
{
long position;
switch (whence) {
case BinaryStream.SEEK_CUR:
position = getPosition() + offset;
break;
case BinaryStream.SEEK_END:
// don't necessarily have an end
position = getPosition();
break;
case BinaryStream.SEEK_SET:
default:
position = offset;
break;
}
if (! setPosition(position))
return -1L;
else
return position;
}
public Value stat()
{
return BooleanValue.FALSE;
}
/**
* Closes the stream for reading.
* The isEOF method will return true
* after this method has been invoked.
*/
public void closeRead()
{
ReadStream is = _is;
_is = null;
if (is != null)
is.close();
}
public Object toJavaObject()
{
return this;
}
public String getResourceType()
{
return "stream";
}
protected Env getEnv()
{
return _env;
}
/**
* Closes the file.
*/
public void close()
{
closeRead();
}
/**
* Converts to a string.
*/
public String toString()
{
if (_is != null)
return "AbstractBinaryInput[" + _is.getPath() + "]";
else
return "AbstractBinaryInput[closed]";
}
}