Refactored the way embedded objects are referenced in MHTML.

This commit is contained in:
Vincent Richard 2013-07-11 18:06:26 +02:00
parent 30d22deaf5
commit b886cd4864
4 changed files with 148 additions and 78 deletions

View File

@ -86,12 +86,13 @@ int main()
vmime::create <vmime::streamContentHandler> vmime::create <vmime::streamContentHandler>
(fileReader->getInputStream(), imageFile->getLength()); (fileReader->getInputStream(), imageFile->getLength());
const vmime::string cid = textPart.addObject(imageCts, vmime::ref <const vmime::htmlTextPart::embeddedObject> obj = textPart.addObject
vmime::mediaType(vmime::mediaTypes::IMAGE, vmime::mediaTypes::IMAGE_JPEG)); (imageCts, vmime::mediaType(vmime::mediaTypes::IMAGE, vmime::mediaTypes::IMAGE_JPEG));
// -- message text // -- message text
textPart.setText(vmime::create <vmime::stringContentHandler> textPart.setText(vmime::create <vmime::stringContentHandler>
(vmime::string("This is the <b>HTML text</b>.<br/><img src=\"") + cid + vmime::string("\"/>"))); (vmime::string("This is the <b>HTML text</b>.<br/>"
"<img src=\"") + obj->getReferenceId() + vmime::string("\"/>")));
textPart.setPlainText(vmime::create <vmime::stringContentHandler> textPart.setPlainText(vmime::create <vmime::stringContentHandler>
("This is the plain text (without HTML formatting).")); ("This is the plain text (without HTML formatting)."));

View File

@ -162,7 +162,8 @@ void htmlTextPart::findEmbeddedParts(const bodyPart& part,
} }
void htmlTextPart::addEmbeddedObject(const bodyPart& part, const string& id) void htmlTextPart::addEmbeddedObject(const bodyPart& part, const string& id,
const embeddedObject::ReferenceType refType)
{ {
// The object may already exists. This can happen if an object is // The object may already exists. This can happen if an object is
// identified by both a Content-Id and a Content-Location. In this // identified by both a Content-Id and a Content-Location. In this
@ -183,7 +184,7 @@ void htmlTextPart::addEmbeddedObject(const bodyPart& part, const string& id)
m_objects.push_back(vmime::create <embeddedObject> m_objects.push_back(vmime::create <embeddedObject>
(part.getBody()->getContents()->clone().dynamicCast <contentHandler>(), (part.getBody()->getContents()->clone().dynamicCast <contentHandler>(),
part.getBody()->getEncoding(), id, type)); part.getBody()->getEncoding(), id, type, refType));
} }
@ -235,7 +236,7 @@ void htmlTextPart::parse(ref <const bodyPart> message, ref <const bodyPart> pare
{ {
// This part is referenced in the HTML text. // This part is referenced in the HTML text.
// Add it to the embedded object list. // Add it to the embedded object list.
addEmbeddedObject(**p, mid.getId()); addEmbeddedObject(**p, mid.getId(), embeddedObject::REFERENCED_BY_ID);
} }
} }
@ -251,7 +252,7 @@ void htmlTextPart::parse(ref <const bodyPart> message, ref <const bodyPart> pare
{ {
// This part is referenced in the HTML text. // This part is referenced in the HTML text.
// Add it to the embedded object list. // Add it to the embedded object list.
addEmbeddedObject(**p, locStr); addEmbeddedObject(**p, locStr, embeddedObject::REFERENCED_BY_LOCATION);
} }
} }
@ -353,7 +354,7 @@ void htmlTextPart::setCharset(const charset& ch)
} }
const ref <const contentHandler> htmlTextPart::getPlainText() const ref <const contentHandler> htmlTextPart::getPlainText() const
{ {
return m_plainText; return m_plainText;
} }
@ -383,20 +384,18 @@ size_t htmlTextPart::getObjectCount() const
} }
const ref <const htmlTextPart::embeddedObject> htmlTextPart::getObjectAt(const size_t pos) const ref <const htmlTextPart::embeddedObject> htmlTextPart::getObjectAt(const size_t pos) const
{ {
return m_objects[pos]; return m_objects[pos];
} }
const ref <const htmlTextPart::embeddedObject> htmlTextPart::findObject(const string& id_) const ref <const htmlTextPart::embeddedObject> htmlTextPart::findObject(const string& id) const
{ {
const string id = cleanId(id_);
for (std::vector <ref <embeddedObject> >::const_iterator o = m_objects.begin() ; for (std::vector <ref <embeddedObject> >::const_iterator o = m_objects.begin() ;
o != m_objects.end() ; ++o) o != m_objects.end() ; ++o)
{ {
if ((*o)->getId() == id) if ((*o)->matchesId(id))
return *o; return *o;
} }
@ -404,14 +403,12 @@ const ref <const htmlTextPart::embeddedObject> htmlTextPart::findObject(const st
} }
bool htmlTextPart::hasObject(const string& id_) const bool htmlTextPart::hasObject(const string& id) const
{ {
const string id = cleanId(id_);
for (std::vector <ref <embeddedObject> >::const_iterator o = m_objects.begin() ; for (std::vector <ref <embeddedObject> >::const_iterator o = m_objects.begin() ;
o != m_objects.end() ; ++o) o != m_objects.end() ; ++o)
{ {
if ((*o)->getId() == id) if ((*o)->matchesId(id))
return true; return true;
} }
@ -419,33 +416,99 @@ bool htmlTextPart::hasObject(const string& id_) const
} }
const string htmlTextPart::addObject(ref <contentHandler> data, ref <const htmlTextPart::embeddedObject> htmlTextPart::addObject
const vmime::encoding& enc, const mediaType& type) (ref <contentHandler> data, const vmime::encoding& enc, const mediaType& type)
{ {
const messageId mid(messageId::generateId()); const messageId mid(messageId::generateId());
const string id = mid.getId();
m_objects.push_back(vmime::create <embeddedObject>(data, enc, id, type)); ref <embeddedObject> obj = vmime::create <embeddedObject>
(data, enc, mid.getId(), type, embeddedObject::REFERENCED_BY_ID);
return "CID:" + id; m_objects.push_back(obj);
return obj;
} }
const string htmlTextPart::addObject(ref <contentHandler> data, const mediaType& type) ref <const htmlTextPart::embeddedObject> htmlTextPart::addObject
(ref <contentHandler> data, const mediaType& type)
{ {
return addObject(data, encoding::decide(data), type); return addObject(data, encoding::decide(data), type);
} }
const string htmlTextPart::addObject(const string& data, const mediaType& type) ref <const htmlTextPart::embeddedObject> htmlTextPart::addObject
(const string& data, const mediaType& type)
{ {
ref <stringContentHandler> cts = vmime::create <stringContentHandler>(data); ref <stringContentHandler> cts = vmime::create <stringContentHandler>(data);
return addObject(cts, encoding::decide(cts), type); return addObject(cts, encoding::decide(cts), type);
} }
//
// htmlTextPart::embeddedObject
//
htmlTextPart::embeddedObject::embeddedObject
(ref <contentHandler> data, const encoding& enc,
const string& id, const mediaType& type, const ReferenceType refType)
: m_data(data->clone().dynamicCast <contentHandler>()),
m_encoding(enc), m_id(id), m_type(type), m_refType(refType)
{
}
ref <const contentHandler> htmlTextPart::embeddedObject::getData() const
{
return m_data;
}
const vmime::encoding htmlTextPart::embeddedObject::getEncoding() const
{
return m_encoding;
}
const string htmlTextPart::embeddedObject::getId() const
{
return m_id;
}
const string htmlTextPart::embeddedObject::getReferenceId() const
{
if (m_refType == REFERENCED_BY_ID)
return string("CID:") + m_id;
else
return m_id;
}
const mediaType htmlTextPart::embeddedObject::getType() const
{
return m_type;
}
htmlTextPart::embeddedObject::ReferenceType htmlTextPart::embeddedObject::getReferenceType() const
{
return m_refType;
}
bool htmlTextPart::embeddedObject::matchesId(const string& id) const
{
if (m_refType == REFERENCED_BY_ID)
return m_id == cleanId(id);
else
return m_id == id;
}
// static // static
const string htmlTextPart::cleanId(const string& id) const string htmlTextPart::embeddedObject::cleanId(const string& id)
{ {
if (id.length() >= 4 && if (id.length() >= 4 &&
(id[0] == 'c' || id[0] == 'C') && (id[0] == 'c' || id[0] == 'C') &&
@ -462,42 +525,4 @@ const string htmlTextPart::cleanId(const string& id)
} }
//
// htmlTextPart::embeddedObject
//
htmlTextPart::embeddedObject::embeddedObject
(ref <contentHandler> data, const encoding& enc,
const string& id, const mediaType& type)
: m_data(data->clone().dynamicCast <contentHandler>()),
m_encoding(enc), m_id(id), m_type(type)
{
}
const ref <const contentHandler> htmlTextPart::embeddedObject::getData() const
{
return m_data;
}
const vmime::encoding& htmlTextPart::embeddedObject::getEncoding() const
{
return m_encoding;
}
const string& htmlTextPart::embeddedObject::getId() const
{
return m_id;
}
const mediaType& htmlTextPart::embeddedObject::getType() const
{
return m_type;
}
} // vmime } // vmime

View File

@ -157,12 +157,14 @@ VMIME_TEST_SUITE_BEGIN(htmlTextPartTest)
obj = htmlPart.findObject("image1@test"); obj = htmlPart.findObject("image1@test");
VASSERT_EQ("ref-type1", vmime::htmlTextPart::embeddedObject::REFERENCED_BY_ID, obj->getReferenceType());
VASSERT_EQ("id-obj1", "image1@test", obj->getId()); VASSERT_EQ("id-obj1", "image1@test", obj->getId());
VASSERT_EQ("data-obj1", "Image1", extractContent(obj->getData())); VASSERT_EQ("data-obj1", "Image1", extractContent(obj->getData()));
VASSERT_EQ("type-obj1", "image/png", obj->getType().generate()); VASSERT_EQ("type-obj1", "image/png", obj->getType().generate());
obj = htmlPart.findObject("image2@test"); obj = htmlPart.findObject("image2@test");
VASSERT_EQ("ref-type2", vmime::htmlTextPart::embeddedObject::REFERENCED_BY_ID, obj->getReferenceType());
VASSERT_EQ("id-obj2", "image2@test", obj->getId()); VASSERT_EQ("id-obj2", "image2@test", obj->getId());
VASSERT_EQ("data-obj2", "Image2", extractContent(obj->getData())); VASSERT_EQ("data-obj2", "Image2", extractContent(obj->getData()));
VASSERT_EQ("type-obj2", "image/jpeg", obj->getType().generate()); VASSERT_EQ("type-obj2", "image/jpeg", obj->getType().generate());
@ -223,6 +225,7 @@ VMIME_TEST_SUITE_BEGIN(htmlTextPartTest)
obj = htmlPart.findObject("http://www.vmime.org/test/image1.png"); obj = htmlPart.findObject("http://www.vmime.org/test/image1.png");
VASSERT_EQ("ref-type", vmime::htmlTextPart::embeddedObject::REFERENCED_BY_LOCATION, obj->getReferenceType());
VASSERT_EQ("id-obj", "http://www.vmime.org/test/image1.png", obj->getId()); VASSERT_EQ("id-obj", "http://www.vmime.org/test/image1.png", obj->getId());
VASSERT_EQ("data-obj", "Image1", extractContent(obj->getData())); VASSERT_EQ("data-obj", "Image1", extractContent(obj->getData()));
VASSERT_EQ("type-obj", "image/png", obj->getType().generate()); VASSERT_EQ("type-obj", "image/png", obj->getType().generate());

View File

@ -51,7 +51,7 @@ public:
const charset& getCharset() const; const charset& getCharset() const;
void setCharset(const charset& ch); void setCharset(const charset& ch);
const ref <const contentHandler> getPlainText() const; ref <const contentHandler> getPlainText() const;
void setPlainText(ref <contentHandler> plainText); void setPlainText(ref <contentHandler> plainText);
const ref <const contentHandler> getText() const; const ref <const contentHandler> getText() const;
@ -63,41 +63,84 @@ public:
{ {
public: public:
/** The ways embedded objects can be referenced. */
enum ReferenceType
{
REFERENCED_BY_ID, /**< Referenced by Content-Id. */
REFERENCED_BY_LOCATION /**< Referenced by Content-Location. */
};
/** Constructs an embedded object.
*
* @param data content of the object
* @param enc encoding of the data
* @param id object identifier
* @param type object content type
* @param refType reference type
* @return a reference to a new embedded object
*/
embeddedObject(ref <contentHandler> data, const encoding& enc, embeddedObject(ref <contentHandler> data, const encoding& enc,
const string& id, const mediaType& type); const string& id, const mediaType& type,
const ReferenceType refType);
/** Return data stored in this embedded object. /** Return data stored in this embedded object.
* *
* @return stored data * @return stored data
*/ */
const ref <const contentHandler> getData() const; ref <const contentHandler> getData() const;
/** Return the encoding used for data in this /** Return the encoding used for data in this
* embedded object. * embedded object.
* *
* @return data encoding * @return data encoding
*/ */
const vmime::encoding& getEncoding() const; const vmime::encoding getEncoding() const;
/** Return the identifier of this embedded object. /** Returns the identifier of this embedded object (either a
* unique ID or a location).
* *
* @return object identifier * @return object identifier
*/ */
const string& getId() const; const string getId() const;
/** Return the identifier used to reference this embedded object
* in a text document (for example, you can use the result as
* the "src" attribute of an &lt;img> tag).
*
* @return object reference identifier
*/
const string getReferenceId() const;
/** Return the content type of data stored in /** Return the content type of data stored in
* this embedded object. * this embedded object.
* *
* @return data type * @return data type
*/ */
const mediaType& getType() const; const mediaType getType() const;
/** Returns the way this object is referenced.
*
* @return reference type (see ReferenceType enum)
*/
ReferenceType getReferenceType() const;
/** Returns whether this object matches the specified identifier.
*
* @param id identifier to test
* @return true if the specified identifier references this
* object, or false otherwise
*/
bool matchesId(const string& id) const;
private: private:
static const string cleanId(const string& id);
ref <contentHandler> m_data; ref <contentHandler> m_data;
encoding m_encoding; encoding m_encoding;
string m_id; string m_id;
mediaType m_type; mediaType m_type;
ReferenceType m_refType;
}; };
@ -115,7 +158,7 @@ public:
* @param id object identifier * @param id object identifier
* @return embedded object with the specified identifier * @return embedded object with the specified identifier
*/ */
const ref <const embeddedObject> findObject(const string& id) const; ref <const embeddedObject> findObject(const string& id) const;
/** Return the number of embedded objects. /** Return the number of embedded objects.
* *
@ -128,7 +171,7 @@ public:
* @param pos position of the embedded object * @param pos position of the embedded object
* @return embedded object at position 'pos' * @return embedded object at position 'pos'
*/ */
const ref <const embeddedObject> getObjectAt(const size_t pos) const; ref <const embeddedObject> getObjectAt(const size_t pos) const;
/** Embed an object and returns a string which identifies it. /** Embed an object and returns a string which identifies it.
* The returned identifier is suitable for use in the 'src' attribute * The returned identifier is suitable for use in the 'src' attribute
@ -142,7 +185,7 @@ public:
* @return an unique object identifier used to identify the new * @return an unique object identifier used to identify the new
* object among all other embedded objects * object among all other embedded objects
*/ */
const string addObject(const string& data, const mediaType& type); ref <const embeddedObject> addObject(const string& data, const mediaType& type);
/** Embed an object and returns a string which identifies it. /** Embed an object and returns a string which identifies it.
* The returned identifier is suitable for use in the 'src' attribute * The returned identifier is suitable for use in the 'src' attribute
@ -153,7 +196,7 @@ public:
* @return an unique object identifier used to identify the new * @return an unique object identifier used to identify the new
* object among all other embedded objects * object among all other embedded objects
*/ */
const string addObject(ref <contentHandler> data, const mediaType& type); ref <const embeddedObject> addObject(ref <contentHandler> data, const mediaType& type);
/** Embed an object and returns a string which identifies it. /** Embed an object and returns a string which identifies it.
* The returned identifier is suitable for use in the 'src' attribute * The returned identifier is suitable for use in the 'src' attribute
@ -165,7 +208,7 @@ public:
* @return an unique object identifier used to identify the new * @return an unique object identifier used to identify the new
* object among all other embedded objects * object among all other embedded objects
*/ */
const string addObject(ref <contentHandler> data, const encoding& enc, const mediaType& type); ref <const embeddedObject> addObject(ref <contentHandler> data, const encoding& enc, const mediaType& type);
size_t getPartCount() const; size_t getPartCount() const;
@ -182,11 +225,9 @@ private:
std::vector <ref <embeddedObject> > m_objects; std::vector <ref <embeddedObject> > m_objects;
void findEmbeddedParts(const bodyPart& part, std::vector <ref <const bodyPart> >& cidParts, std::vector <ref <const bodyPart> >& locParts); void findEmbeddedParts(const bodyPart& part, std::vector <ref <const bodyPart> >& cidParts, std::vector <ref <const bodyPart> >& locParts);
void addEmbeddedObject(const bodyPart& part, const string& id); void addEmbeddedObject(const bodyPart& part, const string& id, const embeddedObject::ReferenceType refType);
bool findPlainTextPart(const bodyPart& part, const bodyPart& parent, const bodyPart& textPart); bool findPlainTextPart(const bodyPart& part, const bodyPart& parent, const bodyPart& textPart);
static const string cleanId(const string& id);
}; };