// MultipartParser.h // Declares the cMultipartParser class that parses messages in "multipart/*" encoding into the separate parts #pragma once #include "EnvelopeParser.h" class cMultipartParser : protected cEnvelopeParser::cCallbacks { public: class cCallbacks { public: // Force a virtual destructor in descendants: virtual ~cCallbacks() {} /** Called when a new part starts */ virtual void OnPartStart(void) = 0; /** Called when a complete header line is received for a part */ virtual void OnPartHeader(const AString & a_Key, const AString & a_Value) = 0; /** Called when body for a part is received */ virtual void OnPartData(const char * a_Data, size_t a_Size) = 0; /** Called when the current part ends */ virtual void OnPartEnd(void) = 0; } ; /** Creates the parser, expects to find the boundary in a_ContentType */ cMultipartParser(const AString & a_ContentType, cCallbacks & a_Callbacks); /** Parses more incoming data */ void Parse(const char * a_Data, size_t a_Size); protected: /** The callbacks to call for various parsing events */ cCallbacks & m_Callbacks; /** True if the data parsed so far is valid; if false, further parsing is skipped */ bool m_IsValid; /** Parser for each part's envelope */ cEnvelopeParser m_EnvelopeParser; /** Buffer for the incoming data until it is parsed */ AString m_IncomingData; /** The boundary, excluding both the initial "--" and the terminating CRLF */ AString m_Boundary; /** Set to true if some data for the current part has already been signalized to m_Callbacks. Used for proper CRLF inserting. */ bool m_HasHadData; /** Parse one line of incoming data. The CRLF has already been stripped from a_Data / a_Size */ void ParseLine(const char * a_Data, size_t a_Size); /** Parse one line of incoming data in the headers section of a part. The CRLF has already been stripped from a_Data / a_Size */ void ParseHeaderLine(const char * a_Data, size_t a_Size); // cEnvelopeParser overrides: virtual void OnHeaderLine(const AString & a_Key, const AString & a_Value) override; } ;