// distribution boxbackup-0.10 (svn version: 494)
//
// Copyright (c) 2003 - 2006
// Ben Summers and contributors. All rights reserved.
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions
// are met:
// 1. Redistributions of source code must retain the above copyright
// notice, this list of conditions and the following disclaimer.
// 2. Redistributions in binary form must reproduce the above copyright
// notice, this list of conditions and the following disclaimer in the
// documentation and/or other materials provided with the distribution.
// 3. All use of this software and associated advertising materials must
// display the following acknowledgment:
// This product includes software developed by Ben Summers.
// 4. The names of the Authors may not be used to endorse or promote
// products derived from this software without specific prior written
// permission.
//
// [Where legally impermissible the Authors do not disclaim liability for
// direct physical injury or death caused solely by defects in the software
// unless it is modified by a third party.]
//
// THIS SOFTWARE IS PROVIDED BY THE AUTHORS ``AS IS'' AND ANY EXPRESS OR
// IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
// WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
// DISCLAIMED. IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY DIRECT,
// INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
// (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
// SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
// HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
// STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
// ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
// POSSIBILITY OF SUCH DAMAGE.
//
//
//
// --------------------------------------------------------------------------
//
// File
// Name: IOStreamGetLine.cpp
// Purpose: Line based file descriptor reading
// Created: 2003/07/24
//
// --------------------------------------------------------------------------
#include "Box.h"
#include "IOStreamGetLine.h"
#include "CommonException.h"
#include "MemLeakFindOn.h"
// utility whitespace function
inline bool iw(int c)
{
return (c == ' ' || c == '\t' || c == '\v' || c == '\f'); // \r, \n are already excluded
}
// --------------------------------------------------------------------------
//
// Function
// Name: IOStreamGetLine::IOStreamGetLine(int)
// Purpose: Constructor, taking file descriptor
// Created: 2003/07/24
//
// --------------------------------------------------------------------------
IOStreamGetLine::IOStreamGetLine(IOStream &Stream)
: mrStream(Stream),
mLineNumber(0),
mBufferBegin(0),
mBytesInBuffer(0),
mPendingEOF(false),
mEOF(false)
{
}
// --------------------------------------------------------------------------
//
// Function
// Name: IOStreamGetLine::~IOStreamGetLine()
// Purpose: Destructor
// Created: 2003/07/24
//
// --------------------------------------------------------------------------
IOStreamGetLine::~IOStreamGetLine()
{
}
// --------------------------------------------------------------------------
//
// Function
// Name: IOStreamGetLine::GetLine(std::string &, bool, int)
// Purpose: Gets a line from the file, returning it in rOutput. If Preprocess is true, leading
// and trailing whitespace is removed, and comments (after #)
// are deleted.
// Returns true if a line is available now, false if retrying may get a line (eg timeout, signal),
// and exceptions if it's EOF.
// Created: 2003/07/24
//
// --------------------------------------------------------------------------
bool IOStreamGetLine::GetLine(std::string &rOutput, bool Preprocess, int Timeout)
{
// EOF?
if(mEOF) {THROW_EXCEPTION(CommonException, GetLineEOF)}
// Initialise string to stored into
std::string r(mPendingString);
mPendingString.erase();
bool foundLineEnd = false;
while(!foundLineEnd && !mEOF)
{
// Use any bytes left in the buffer
while(mBufferBegin < mBytesInBuffer)
{
int c = mBuffer[mBufferBegin++];
if(c == '\r')
{
// Ignore nasty Windows line ending extra chars
}
else if(c == '\n')
{
// Line end!
foundLineEnd = true;
break;
}
else
{
// Add to string
r += c;
}
// Implicit line ending at EOF
if(mBufferBegin >= mBytesInBuffer && mPendingEOF)
{
foundLineEnd = true;
}
}
// Check size
if(r.size() > IOSTREAMGETLINE_MAX_LINE_SIZE)
{
THROW_EXCEPTION(CommonException, GetLineTooLarge)
}
// Read more in?
if(!foundLineEnd && mBufferBegin >= mBytesInBuffer && !mPendingEOF)
{
int bytes = mrStream.Read(mBuffer, sizeof(mBuffer), Timeout);
// Adjust buffer info
mBytesInBuffer = bytes;
mBufferBegin = 0;
// EOF / closed?
if(!mrStream.StreamDataLeft())
{
mPendingEOF = true;
}
// No data returned?
if(bytes == 0 && mrStream.StreamDataLeft())
{
// store string away
mPendingString = r;
// Return false;
return false;
}
}
// EOF?
if(mPendingEOF && mBufferBegin >= mBytesInBuffer)
{
// File is EOF, and now we've depleted the buffer completely, so tell caller as well.
mEOF = true;
}
}
if(!Preprocess)
{
rOutput = r;
return true;
}
else
{
// Check for comment char, but char before must be whitespace
int end = 0;
int size = r.size();
while(end < size)
{
if(r[end] == '#' && (end == 0 || (iw(r[end-1]))))
{
break;
}
end++;
}
// Remove whitespace
int begin = 0;
while(begin < size && iw(r[begin]))
{
begin++;
}
if(!iw(r[end])) end--;
while(end > begin && iw(r[end]))
{
end--;
}
// Return a sub string
rOutput = r.substr(begin, end - begin + 1);
return true;
}
}
// --------------------------------------------------------------------------
//
// Function
// Name: IOStreamGetLine::DetachFile()
// Purpose: Detaches the file handle, setting the file pointer correctly.
// Probably not good for sockets...
// Created: 2003/07/24
//
// --------------------------------------------------------------------------
void IOStreamGetLine::DetachFile()
{
// Adjust file pointer
int bytesOver = mBytesInBuffer - mBufferBegin;
ASSERT(bytesOver >= 0);
if(bytesOver > 0)
{
mrStream.Seek(0 - bytesOver, IOStream::SeekType_Relative);
}
}
// --------------------------------------------------------------------------
//
// Function
// Name: IOStreamGetLine::IgnoreBufferedData(int)
// Purpose: Ignore buffered bytes (effectively removing them from the
// beginning of the buffered data.)
// Cannot remove more bytes than are currently in the buffer.
// Be careful when this is used!
// Created: 22/12/04
//
// --------------------------------------------------------------------------
void IOStreamGetLine::IgnoreBufferedData(int BytesToIgnore)
{
int bytesInBuffer = mBytesInBuffer - mBufferBegin;
if(BytesToIgnore < 0 || BytesToIgnore > bytesInBuffer)
{
THROW_EXCEPTION(CommonException, IOStreamGetLineNotEnoughDataToIgnore)
}
mBufferBegin += BytesToIgnore;
}
syntax highlighted by Code2HTML, v. 0.9.1