vmime/src/messageId.cpp

276 lines
5.2 KiB
C++
Raw Normal View History

2004-10-05 10:28:21 +00:00
//
2005-03-18 21:33:11 +00:00
// VMime library (http://www.vmime.org)
// 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/messageId.hpp"
#include "vmime/utility/random.hpp"
#include "vmime/platformDependant.hpp"
#include "vmime/parserHelpers.hpp"
2004-10-05 10:28:21 +00:00
namespace vmime
{
messageId::messageId()
{
}
messageId::messageId(const string& id)
{
parse(id);
}
messageId::messageId(const messageId& mid)
: component(), m_left(mid.m_left), m_right(mid.m_right)
{
}
messageId::messageId(const string& left, const string& right)
: m_left(left), m_right(right)
{
}
/*
RFC-2822:
3.6.4. Identification fields
msg-id = [CFWS] "<" id-left "@" id-right ">" [CFWS]
*/
void messageId::parse(const string& buffer, const string::size_type position,
const string::size_type end, string::size_type* newPosition)
{
const string::value_type* const pend = buffer.data() + end;
const string::value_type* const pstart = buffer.data() + position;
const string::value_type* p = pstart;
m_left.clear();
m_right.clear();
unsigned int commentLevel = 0;
bool escape = false;
bool stop = false;
for ( ; !stop && p < pend ; ++p)
{
if (escape)
{
// Ignore this character
}
else
{
switch (*p)
{
case '(': ++commentLevel; break;
case ')': --commentLevel; break;
case '\\': escape = true; break;
case '<':
{
if (commentLevel == 0)
{
stop = true;
break;
}
}
}
}
}
if (p < pend)
{
// Extract left part
const string::size_type leftStart = position + (p - pstart);
while (p < pend && *p != '@') ++p;
m_left = string(buffer.begin() + leftStart,
buffer.begin() + position + (p - pstart));
if (p < pend)
{
// Skip '@'
++p;
// Extract right part
const string::size_type rightStart = position + (p - pstart);
while (p < pend && *p != '>') ++p;
m_right = string(buffer.begin() + rightStart,
buffer.begin() + position + (p - pstart));
}
}
2004-12-15 20:28:09 +00:00
setParsedBounds(position, end);
2004-10-05 10:28:21 +00:00
if (newPosition)
*newPosition = end;
}
messageId* messageId::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 && parserHelpers::isSpace(buffer[pos]))
++pos;
if (pos != end)
{
const string::size_type begin = pos;
while (pos < end && !parserHelpers::isSpace(buffer[pos]))
++pos;
messageId* mid = new messageId();
mid->parse(buffer, begin, pos, NULL);
if (newPosition != NULL)
*newPosition = pos;
return (mid);
}
if (newPosition != NULL)
*newPosition = end;
return (NULL);
}
2004-10-21 15:05:47 +00:00
const string messageId::getId() const
2004-10-05 10:28:21 +00:00
{
return (m_left + '@' + m_right);
}
void messageId::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
{
string::size_type pos = curLinePos;
if (curLinePos + m_left.length() + m_right.length() + 3 > maxLineLength)
{
os << NEW_LINE_SEQUENCE;
pos = NEW_LINE_SEQUENCE_LENGTH;
}
2004-10-05 10:28:21 +00:00
os << '<' << m_left << '@' << m_right << '>';
if (newLinePos)
*newLinePos = pos + m_left.length() + m_right.length() + 3;
2004-10-05 10:28:21 +00:00
}
messageId& messageId::operator=(const string& id)
{
parse(id);
return (*this);
}
messageId messageId::generateId()
{
std::ostringstream left;
left << "vmime";
left << '.';
2004-10-21 15:05:47 +00:00
left << std::hex << utility::random::getTime();
2004-10-05 10:28:21 +00:00
left << '.';
2004-10-21 15:05:47 +00:00
left << std::hex << utility::random::getProcess();
2004-10-05 10:28:21 +00:00
left << '.';
2004-10-21 15:05:47 +00:00
left << std::hex << utility::random::getNext();
left << std::hex << utility::random::getNext();
2004-10-05 10:28:21 +00:00
return (messageId(left.str(), platformDependant::getHandler()->getHostName()));
}
const bool messageId::operator==(const messageId& mid) const
{
return (m_left == mid.m_left && m_right == mid.m_right);
}
2004-10-21 15:05:47 +00:00
const bool messageId::operator!=(const messageId& mid) const
{
return !(*this == mid);
}
messageId* messageId::clone() const
{
return new messageId(*this);
}
void messageId::copyFrom(const component& other)
{
const messageId& mid = dynamic_cast <const messageId&>(other);
m_left = mid.m_left;
m_right = mid.m_right;
}
messageId& messageId::operator=(const messageId& other)
{
copyFrom(other);
return (*this);
}
const string& messageId::getLeft() const
{
return (m_left);
}
void messageId::setLeft(const string& left)
{
m_left = left;
}
const string& messageId::getRight() const
{
return (m_right);
}
void messageId::setRight(const string& right)
{
m_right = right;
}
const std::vector <const component*> messageId::getChildComponents() const
{
return std::vector <const component*>();
}
2004-10-05 10:28:21 +00:00
} // vmime