vmime/src/headerField.cpp

275 lines
5.7 KiB
C++
Raw Normal View History

2004-10-05 10:28:21 +00:00
//
// VMime library (http://vmime.sourceforge.net)
// Copyright (C) 2002-2005 Vincent Richard <vincent@vincent-richard.net>
2004-10-05 10:28:21 +00:00
//
// This program 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.
//
// This program 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. See the GNU
// General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program; if not, write to the Free Software
// Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
//
#include "vmime/headerField.hpp"
#include "vmime/headerFieldFactory.hpp"
2004-10-05 10:28:21 +00:00
#include "vmime/parserHelpers.hpp"
2004-10-05 10:28:21 +00:00
namespace vmime
{
headerField::headerField()
2004-10-21 15:05:47 +00:00
: m_name("X-Undefined")
2004-10-05 10:28:21 +00:00
{
}
headerField::headerField(const string& fieldName)
2004-10-21 15:05:47 +00:00
: m_name(fieldName)
2004-10-05 10:28:21 +00:00
{
}
headerField::~headerField()
{
}
headerField* headerField::clone() const
{
2004-10-21 15:05:47 +00:00
headerField* field = headerFieldFactory::getInstance()->create(m_name);
2004-10-05 10:28:21 +00:00
field->copyFrom(*this);
return (field);
}
2004-10-21 15:05:47 +00:00
void headerField::copyFrom(const component& other)
2004-10-05 10:28:21 +00:00
{
2004-10-21 15:05:47 +00:00
const headerField& hf = dynamic_cast <const headerField&>(other);
getValue().copyFrom(hf.getValue());
2004-10-05 10:28:21 +00:00
}
2004-10-21 15:05:47 +00:00
headerField& headerField::operator=(const headerField& other)
2004-10-05 10:28:21 +00:00
{
2004-10-21 15:05:47 +00:00
copyFrom(other);
2004-10-05 10:28:21 +00:00
return (*this);
}
headerField* headerField::parseNext(const string& buffer, const string::size_type position,
const string::size_type end, string::size_type* newPosition)
{
string::size_type pos = position;
while (pos < end)
{
char_t c = buffer[pos];
// Check for end of headers (empty line): although RFC-822 recommends
// to use CRLF for header/body separator (see 4.1 SYNTAX), here, we
// also check for LF for compatibility with broken implementations...
if (c == '\n')
{
if (newPosition)
*newPosition = pos + 1; // LF: illegal
return (NULL);
}
else if (c == '\r' && pos + 1 < end && buffer[pos + 1] == '\n')
{
if (newPosition)
*newPosition = pos + 2; // CR+LF
return (NULL);
}
// This line may be a field description
if (!parserHelpers::isspace(c))
{
const string::size_type nameStart = pos; // remember the start position of the line
while (pos < end && (buffer[pos] != ':' && !parserHelpers::isspace(buffer[pos])))
++pos;
const string::size_type nameEnd = pos;
while (pos < end && (buffer[pos] == ' ' || buffer[pos] == '\t'))
++pos;
if (buffer[pos] != ':')
{
// Humm...does not seem to be a valid header line.
// Skip this error and advance to the next line
pos = nameStart;
while (pos < end && buffer[pos] != '\n')
++pos;
if (pos < end && buffer[pos] == '\n')
++pos;
}
else
{
// Extract the field name
const string name(buffer.begin() + nameStart,
buffer.begin() + nameEnd);
// Skip ':' character
++pos;
// Skip spaces between ':' and the field contents
while (pos < end && (buffer[pos] == ' ' || buffer[pos] == '\t'))
++pos;
// Extract the field value
string contents;
while (pos < end)
{
c = buffer[pos];
// Check for end of contents
if (c == '\r' && pos + 1 < end && buffer[pos + 1] == '\n')
{
pos += 2;
break;
}
else if (c == '\n')
{
++pos;
break;
}
const string::size_type ctsStart = pos;
string::size_type ctsEnd = pos;
while (pos < end)
{
c = buffer[pos];
// Check for end of line
if (c == '\r' && pos + 1 < end && buffer[pos + 1] == '\n')
{
ctsEnd = pos;
pos += 2;
break;
}
else if (c == '\n')
{
ctsEnd = pos;
++pos;
break;
}
++pos;
}
if (ctsEnd != ctsStart)
{
// Append this line to contents
contents.append(buffer.begin() + ctsStart,
buffer.begin() + ctsEnd);
}
// Handle the case of folded lines
if (buffer[pos] == ' ' || buffer[pos] == '\t')
{
// This is a folding white-space: we keep it as is and
// we continue with contents parsing...
}
else
{
// End of this field
break;
}
}
// Return a new field
headerField* field = headerFieldFactory::getInstance()->create(name);
field->parse(contents); // TODO: fix incorrect parsed bounds...
field->setParsedBounds(nameStart, pos);
if (newPosition)
*newPosition = pos;
return (field);
}
}
else
{
// Skip this error and advance to the next line
while (pos < end && buffer[pos] != '\n')
++pos;
if (buffer[pos] == '\n')
++pos;
}
}
if (newPosition)
*newPosition = pos;
return (NULL);
}
2004-10-21 15:05:47 +00:00
void headerField::parse(const string& buffer, const string::size_type position, const string::size_type end,
string::size_type* newPosition)
2004-10-05 10:28:21 +00:00
{
2004-10-21 15:05:47 +00:00
getValue().parse(buffer, position, end, newPosition);
2004-10-05 10:28:21 +00:00
}
2004-10-21 15:05:47 +00:00
void headerField::generate(utility::outputStream& os, const string::size_type maxLineLength,
2004-10-05 10:28:21 +00:00
const string::size_type curLinePos, string::size_type* newLinePos) const
{
2004-10-21 15:05:47 +00:00
os << m_name + ": ";
2004-10-05 10:28:21 +00:00
2004-10-21 15:05:47 +00:00
getValue().generate(os, maxLineLength, curLinePos + m_name.length() + 2, newLinePos);
2004-10-05 10:28:21 +00:00
}
2004-10-21 15:05:47 +00:00
const string headerField::getName() const
2004-10-05 10:28:21 +00:00
{
2004-10-21 15:05:47 +00:00
return (m_name);
2004-10-05 10:28:21 +00:00
}
const bool headerField::isCustom() const
{
2004-10-21 15:05:47 +00:00
return (m_name.length() > 2 && m_name[0] == 'X' && m_name[1] == '-');
2004-10-05 10:28:21 +00:00
}
const std::vector <const component*> headerField::getChildComponents() const
{
std::vector <const component*> list;
list.push_back(&getValue());
return (list);
}
2005-01-02 15:17:54 +00:00
void headerField::setValue(const string& value)
{
parse(value);
}
2004-10-05 10:28:21 +00:00
} // vmime