| 1 | {==============================================================================|
|
|---|
| 2 | | Project : Ararat Synapse | 002.005.002 |
|
|---|
| 3 | |==============================================================================|
|
|---|
| 4 | | Content: MIME message object |
|
|---|
| 5 | |==============================================================================|
|
|---|
| 6 | | Copyright (c)1999-2006, Lukas Gebauer |
|
|---|
| 7 | | All rights reserved. |
|
|---|
| 8 | | |
|
|---|
| 9 | | Redistribution and use in source and binary forms, with or without |
|
|---|
| 10 | | modification, are permitted provided that the following conditions are met: |
|
|---|
| 11 | | |
|
|---|
| 12 | | Redistributions of source code must retain the above copyright notice, this |
|
|---|
| 13 | | list of conditions and the following disclaimer. |
|
|---|
| 14 | | |
|
|---|
| 15 | | Redistributions in binary form must reproduce the above copyright notice, |
|
|---|
| 16 | | this list of conditions and the following disclaimer in the documentation |
|
|---|
| 17 | | and/or other materials provided with the distribution. |
|
|---|
| 18 | | |
|
|---|
| 19 | | Neither the name of Lukas Gebauer nor the names of its contributors may |
|
|---|
| 20 | | be used to endorse or promote products derived from this software without |
|
|---|
| 21 | | specific prior written permission. |
|
|---|
| 22 | | |
|
|---|
| 23 | | THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" |
|
|---|
| 24 | | AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE |
|
|---|
| 25 | | IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE |
|
|---|
| 26 | | ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE FOR |
|
|---|
| 27 | | ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL |
|
|---|
| 28 | | DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR |
|
|---|
| 29 | | SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER |
|
|---|
| 30 | | CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT |
|
|---|
| 31 | | LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY |
|
|---|
| 32 | | OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH |
|
|---|
| 33 | | DAMAGE. |
|
|---|
| 34 | |==============================================================================|
|
|---|
| 35 | | The Initial Developer of the Original Code is Lukas Gebauer (Czech Republic).|
|
|---|
| 36 | | Portions created by Lukas Gebauer are Copyright (c)2000-2006. |
|
|---|
| 37 | | All Rights Reserved. |
|
|---|
| 38 | |==============================================================================|
|
|---|
| 39 | | Contributor(s): |
|
|---|
| 40 | |==============================================================================|
|
|---|
| 41 | | History: see HISTORY.HTM From distribution package |
|
|---|
| 42 | | (Found at URL: http://www.ararat.cz/synapse/) |
|
|---|
| 43 | |==============================================================================}
|
|---|
| 44 |
|
|---|
| 45 | {:@abstract(MIME message handling)
|
|---|
| 46 | Classes for easy handling with e-mail message.
|
|---|
| 47 | }
|
|---|
| 48 |
|
|---|
| 49 | {$IFDEF FPC}
|
|---|
| 50 | {$MODE DELPHI}
|
|---|
| 51 | {$ENDIF}
|
|---|
| 52 | {$H+}
|
|---|
| 53 |
|
|---|
| 54 | unit mimemess;
|
|---|
| 55 |
|
|---|
| 56 | interface
|
|---|
| 57 |
|
|---|
| 58 | uses
|
|---|
| 59 | Classes, SysUtils,
|
|---|
| 60 | mimepart, synachar, synautil, mimeinln;
|
|---|
| 61 |
|
|---|
| 62 | type
|
|---|
| 63 |
|
|---|
| 64 | {:Possible values for message priority}
|
|---|
| 65 | TMessPriority = (MP_unknown, MP_low, MP_normal, MP_high);
|
|---|
| 66 |
|
|---|
| 67 | {:@abstract(Object for basic e-mail header fields.)}
|
|---|
| 68 | TMessHeader = class(TObject)
|
|---|
| 69 | private
|
|---|
| 70 | FFrom: string;
|
|---|
| 71 | FToList: TStringList;
|
|---|
| 72 | FCCList: TStringList;
|
|---|
| 73 | FSubject: string;
|
|---|
| 74 | FOrganization: string;
|
|---|
| 75 | FCustomHeaders: TStringList;
|
|---|
| 76 | FDate: TDateTime;
|
|---|
| 77 | FXMailer: string;
|
|---|
| 78 | FCharsetCode: TMimeChar;
|
|---|
| 79 | FReplyTo: string;
|
|---|
| 80 | FMessageID: string;
|
|---|
| 81 | FPriority: TMessPriority;
|
|---|
| 82 | Fpri: TMessPriority;
|
|---|
| 83 | Fxpri: TMessPriority;
|
|---|
| 84 | Fxmspri: TMessPriority;
|
|---|
| 85 | protected
|
|---|
| 86 | function ParsePriority(value: string): TMessPriority;
|
|---|
| 87 | function DecodeHeader(value: string): boolean; virtual;
|
|---|
| 88 | public
|
|---|
| 89 | constructor Create; virtual;
|
|---|
| 90 | destructor Destroy; override;
|
|---|
| 91 |
|
|---|
| 92 | {:Clears all data fields.}
|
|---|
| 93 | procedure Clear; virtual;
|
|---|
| 94 |
|
|---|
| 95 | {Add headers from from this object to Value.}
|
|---|
| 96 | procedure EncodeHeaders(const Value: TStrings); virtual;
|
|---|
| 97 |
|
|---|
| 98 | {:Parse header from Value to this object.}
|
|---|
| 99 | procedure DecodeHeaders(const Value: TStrings);
|
|---|
| 100 |
|
|---|
| 101 | {:Try find specific header in CustomHeader. Search is case insensitive.
|
|---|
| 102 | This is good for reading any non-parsed header.}
|
|---|
| 103 | function FindHeader(Value: string): string;
|
|---|
| 104 |
|
|---|
| 105 | {:Try find specific headers in CustomHeader. This metod is for repeatly used
|
|---|
| 106 | headers like 'received' header, etc. Search is case insensitive.
|
|---|
| 107 | This is good for reading ano non-parsed header.}
|
|---|
| 108 | procedure FindHeaderList(Value: string; const HeaderList: TStrings);
|
|---|
| 109 | published
|
|---|
| 110 | {:Sender of message.}
|
|---|
| 111 | property From: string read FFrom Write FFrom;
|
|---|
| 112 |
|
|---|
| 113 | {:Stringlist with receivers of message. (one per line)}
|
|---|
| 114 | property ToList: TStringList read FToList;
|
|---|
| 115 |
|
|---|
| 116 | {:Stringlist with Carbon Copy receivers of message. (one per line)}
|
|---|
| 117 | property CCList: TStringList read FCCList;
|
|---|
| 118 |
|
|---|
| 119 | {:Subject of message.}
|
|---|
| 120 | property Subject: string read FSubject Write FSubject;
|
|---|
| 121 |
|
|---|
| 122 | {:Organization string.}
|
|---|
| 123 | property Organization: string read FOrganization Write FOrganization;
|
|---|
| 124 |
|
|---|
| 125 | {:After decoding contains all headers lines witch not have parsed to any
|
|---|
| 126 | other structures in this object. It mean: this conatins all other headers
|
|---|
| 127 | except:
|
|---|
| 128 |
|
|---|
| 129 | X-MAILER, FROM, SUBJECT, ORGANIZATION, TO, CC, DATE, MIME-VERSION,
|
|---|
| 130 | CONTENT-TYPE, CONTENT-DESCRIPTION, CONTENT-DISPOSITION, CONTENT-ID,
|
|---|
| 131 | CONTENT-TRANSFER-ENCODING, REPLY-TO, MESSAGE-ID, X-MSMAIL-PRIORITY,
|
|---|
| 132 | X-PRIORITY, PRIORITY
|
|---|
| 133 |
|
|---|
| 134 | When you encode headers, all this lines is added as headers. Be carefull
|
|---|
| 135 | for duplicites!}
|
|---|
| 136 | property CustomHeaders: TStringList read FCustomHeaders;
|
|---|
| 137 |
|
|---|
| 138 | {:Date and time of message.}
|
|---|
| 139 | property Date: TDateTime read FDate Write FDate;
|
|---|
| 140 |
|
|---|
| 141 | {:Mailer identification.}
|
|---|
| 142 | property XMailer: string read FXMailer Write FXMailer;
|
|---|
| 143 |
|
|---|
| 144 | {:Address for replies}
|
|---|
| 145 | property ReplyTo: string read FReplyTo Write FReplyTo;
|
|---|
| 146 |
|
|---|
| 147 | {:message indetifier}
|
|---|
| 148 | property MessageID: string read FMessageID Write FMessageID;
|
|---|
| 149 |
|
|---|
| 150 | {:message priority}
|
|---|
| 151 | property Priority: TMessPriority read FPriority Write FPriority;
|
|---|
| 152 |
|
|---|
| 153 | {:Specify base charset. By default is used system charset.}
|
|---|
| 154 | property CharsetCode: TMimeChar read FCharsetCode Write FCharsetCode;
|
|---|
| 155 | end;
|
|---|
| 156 |
|
|---|
| 157 | TMessHeaderClass = class of TMessHeader;
|
|---|
| 158 |
|
|---|
| 159 | {:@abstract(Object for handling of e-mail message.)}
|
|---|
| 160 | TMimeMess = class(TObject)
|
|---|
| 161 | private
|
|---|
| 162 | FMessagePart: TMimePart;
|
|---|
| 163 | FLines: TStringList;
|
|---|
| 164 | FHeader: TMessHeader;
|
|---|
| 165 | public
|
|---|
| 166 | constructor Create;
|
|---|
| 167 | {:create this object and assign your own descendant of @link(TMessHeader)
|
|---|
| 168 | object to @link(header) property. So, you can create your own message
|
|---|
| 169 | headers parser and use it by this object.}
|
|---|
| 170 | constructor CreateAltHeaders(HeadClass: TMessHeaderClass);
|
|---|
| 171 | destructor Destroy; override;
|
|---|
| 172 |
|
|---|
| 173 | {:Reset component to default state.}
|
|---|
| 174 | procedure Clear; virtual;
|
|---|
| 175 |
|
|---|
| 176 | {:Add MIME part as subpart of PartParent. If you need set root MIME part,
|
|---|
| 177 | then set as PartParent @NIL value. If you need set more then one subpart,
|
|---|
| 178 | you must have PartParent of multipart type!}
|
|---|
| 179 | function AddPart(const PartParent: TMimePart): TMimePart;
|
|---|
| 180 |
|
|---|
| 181 | {:Add MIME part as subpart of PartParent. If you need set root MIME part,
|
|---|
| 182 | then set as PartParent @NIL value. If you need set more then 1 subpart, you
|
|---|
| 183 | must have PartParent of multipart type!
|
|---|
| 184 |
|
|---|
| 185 | This part is marked as multipart with secondary MIME type specified by
|
|---|
| 186 | MultipartType parameter. (typical value is 'mixed')
|
|---|
| 187 |
|
|---|
| 188 | This part can be used as PartParent for another parts (include next
|
|---|
| 189 | multipart). If you need only one part, then you not need Multipart part.}
|
|---|
| 190 | function AddPartMultipart(const MultipartType: String; const PartParent: TMimePart): TMimePart;
|
|---|
| 191 |
|
|---|
| 192 | {:Add MIME part as subpart of PartParent. If you need set root MIME part,
|
|---|
| 193 | then set as PartParent @NIL value. If you need set more then 1 subpart, you
|
|---|
| 194 | must have PartParent of multipart type!
|
|---|
| 195 |
|
|---|
| 196 | After creation of part set type to text part and set all necessary
|
|---|
| 197 | properties. Content of part is readed from value stringlist.}
|
|---|
| 198 | function AddPartText(const Value: TStrings; const PartParent: TMimePart): TMimepart;
|
|---|
| 199 |
|
|---|
| 200 | {:Add MIME part as subpart of PartParent. If you need set root MIME part,
|
|---|
| 201 | then set as PartParent @NIL value. If you need set more then 1 subpart, you
|
|---|
| 202 | must have PartParent of multipart type!
|
|---|
| 203 |
|
|---|
| 204 | After creation of part set type to text part and set all necessary
|
|---|
| 205 | properties. Content of part is readed from value stringlist. You can select
|
|---|
| 206 | your charset and your encoding type. If Raw is @true, then it not doing
|
|---|
| 207 | charset conversion!}
|
|---|
| 208 | function AddPartTextEx(const Value: TStrings; const PartParent: TMimePart;
|
|---|
| 209 | PartCharset: TMimeChar; Raw: Boolean; PartEncoding: TMimeEncoding): TMimepart;
|
|---|
| 210 |
|
|---|
| 211 | {:Add MIME part as subpart of PartParent. If you need set root MIME part,
|
|---|
| 212 | then set as PartParent @NIL value. If you need set more then 1 subpart, you
|
|---|
| 213 | must have PartParent of multipart type!
|
|---|
| 214 |
|
|---|
| 215 | After creation of part set type to text part to HTML type and set all
|
|---|
| 216 | necessary properties. Content of HTML part is readed from Value stringlist.}
|
|---|
| 217 | function AddPartHTML(const Value: TStrings; const PartParent: TMimePart): TMimepart;
|
|---|
| 218 |
|
|---|
| 219 | {:Same as @link(AddPartText), but content is readed from file}
|
|---|
| 220 | function AddPartTextFromFile(const FileName: String; const PartParent: TMimePart): TMimepart;
|
|---|
| 221 |
|
|---|
| 222 | {:Same as @link(AddPartHTML), but content is readed from file}
|
|---|
| 223 | function AddPartHTMLFromFile(const FileName: String; const PartParent: TMimePart): TMimepart;
|
|---|
| 224 |
|
|---|
| 225 | {:Add MIME part as subpart of PartParent. If you need set root MIME part,
|
|---|
| 226 | then set as PartParent @NIL value. If you need set more then 1 subpart,
|
|---|
| 227 | you must have PartParent of multipart type!
|
|---|
| 228 |
|
|---|
| 229 | After creation of part set type to binary and set all necessary properties.
|
|---|
| 230 | MIME primary and secondary types defined automaticly by filename extension.
|
|---|
| 231 | Content of binary part is readed from Stream. This binary part is encoded
|
|---|
| 232 | as file attachment.}
|
|---|
| 233 | function AddPartBinary(const Stream: TStream; const FileName: string; const PartParent: TMimePart): TMimepart;
|
|---|
| 234 |
|
|---|
| 235 | {:Same as @link(AddPartBinary), but content is readed from file}
|
|---|
| 236 | function AddPartBinaryFromFile(const FileName: string; const PartParent: TMimePart): TMimepart;
|
|---|
| 237 |
|
|---|
| 238 | {:Add MIME part as subpart of PartParent. If you need set root MIME part,
|
|---|
| 239 | then set as PartParent @NIL value. If you need set more then 1 subpart, you
|
|---|
| 240 | must have PartParent of multipart type!
|
|---|
| 241 |
|
|---|
| 242 | After creation of part set type to binary and set all necessary properties.
|
|---|
| 243 | MIME primary and secondary types defined automaticly by filename extension.
|
|---|
| 244 | Content of binary part is readed from Stream.
|
|---|
| 245 |
|
|---|
| 246 | This binary part is encoded as inline data with given Conten ID (cid).
|
|---|
| 247 | Content ID can be used as reference ID in HTML source in HTML part.}
|
|---|
| 248 | function AddPartHTMLBinary(const Stream: TStream; const FileName, Cid: string; const PartParent: TMimePart): TMimepart;
|
|---|
| 249 |
|
|---|
| 250 | {:Same as @link(AddPartHTMLBinary), but content is readed from file}
|
|---|
| 251 | function AddPartHTMLBinaryFromFile(const FileName, Cid: string; const PartParent: TMimePart): TMimepart;
|
|---|
| 252 |
|
|---|
| 253 | {:Add MIME part as subpart of PartParent. If you need set root MIME part,
|
|---|
| 254 | then set as PartParent @NIL value. If you need set more then 1 subpart, you
|
|---|
| 255 | must have PartParent of multipart type!
|
|---|
| 256 |
|
|---|
| 257 | After creation of part set type to message and set all necessary properties.
|
|---|
| 258 | MIME primary and secondary types are setted to 'message/rfc822'.
|
|---|
| 259 | Content of raw RFC-822 message is readed from Stream.}
|
|---|
| 260 | function AddPartMess(const Value: TStrings; const PartParent: TMimePart): TMimepart;
|
|---|
| 261 |
|
|---|
| 262 | {:Same as @link(AddPartMess), but content is readed from file}
|
|---|
| 263 | function AddPartMessFromFile(const FileName: string; const PartParent: TMimePart): TMimepart;
|
|---|
| 264 |
|
|---|
| 265 | {:Compose message from @link(MessagePart) to @link(Lines). Headers from
|
|---|
| 266 | @link(Header) object is added also.}
|
|---|
| 267 | procedure EncodeMessage;
|
|---|
| 268 |
|
|---|
| 269 | {:Decode message from @link(Lines) to @link(MessagePart). Massage headers
|
|---|
| 270 | are parsed into @link(Header) object.}
|
|---|
| 271 | procedure DecodeMessage;
|
|---|
| 272 | published
|
|---|
| 273 | {:@link(TMimePart) object with decoded MIME message. This object can handle
|
|---|
| 274 | any number of nested @link(TMimePart) objects itself. It is used for handle
|
|---|
| 275 | any tree of MIME subparts.}
|
|---|
| 276 | property MessagePart: TMimePart read FMessagePart;
|
|---|
| 277 |
|
|---|
| 278 | {:Raw MIME encoded message.}
|
|---|
| 279 | property Lines: TStringList read FLines;
|
|---|
| 280 |
|
|---|
| 281 | {:Object for e-mail header fields. This object is created automaticly.
|
|---|
| 282 | Do not free this object!}
|
|---|
| 283 | property Header: TMessHeader read FHeader;
|
|---|
| 284 | end;
|
|---|
| 285 |
|
|---|
| 286 | implementation
|
|---|
| 287 |
|
|---|
| 288 | {==============================================================================}
|
|---|
| 289 |
|
|---|
| 290 | constructor TMessHeader.Create;
|
|---|
| 291 | begin
|
|---|
| 292 | inherited Create;
|
|---|
| 293 | FToList := TStringList.Create;
|
|---|
| 294 | FCCList := TStringList.Create;
|
|---|
| 295 | FCustomHeaders := TStringList.Create;
|
|---|
| 296 | FCharsetCode := GetCurCP;
|
|---|
| 297 | end;
|
|---|
| 298 |
|
|---|
| 299 | destructor TMessHeader.Destroy;
|
|---|
| 300 | begin
|
|---|
| 301 | FCustomHeaders.Free;
|
|---|
| 302 | FCCList.Free;
|
|---|
| 303 | FToList.Free;
|
|---|
| 304 | inherited Destroy;
|
|---|
| 305 | end;
|
|---|
| 306 |
|
|---|
| 307 | {==============================================================================}
|
|---|
| 308 |
|
|---|
| 309 | procedure TMessHeader.Clear;
|
|---|
| 310 | begin
|
|---|
| 311 | FFrom := '';
|
|---|
| 312 | FToList.Clear;
|
|---|
| 313 | FCCList.Clear;
|
|---|
| 314 | FSubject := '';
|
|---|
| 315 | FOrganization := '';
|
|---|
| 316 | FCustomHeaders.Clear;
|
|---|
| 317 | FDate := 0;
|
|---|
| 318 | FXMailer := '';
|
|---|
| 319 | FReplyTo := '';
|
|---|
| 320 | FMessageID := '';
|
|---|
| 321 | FPriority := MP_unknown;
|
|---|
| 322 | end;
|
|---|
| 323 |
|
|---|
| 324 | procedure TMessHeader.EncodeHeaders(const Value: TStrings);
|
|---|
| 325 | var
|
|---|
| 326 | n: Integer;
|
|---|
| 327 | s: string;
|
|---|
| 328 | begin
|
|---|
| 329 | if FDate = 0 then
|
|---|
| 330 | FDate := Now;
|
|---|
| 331 | for n := FCustomHeaders.Count - 1 downto 0 do
|
|---|
| 332 | if FCustomHeaders[n] <> '' then
|
|---|
| 333 | Value.Insert(0, FCustomHeaders[n]);
|
|---|
| 334 | if FPriority <> MP_unknown then
|
|---|
| 335 | case FPriority of
|
|---|
| 336 | MP_high:
|
|---|
| 337 | begin
|
|---|
| 338 | Value.Insert(0, 'X-MSMAIL-Priority: High');
|
|---|
| 339 | Value.Insert(0, 'X-Priority: 1');
|
|---|
| 340 | Value.Insert(0, 'Priority: urgent');
|
|---|
| 341 | end;
|
|---|
| 342 | MP_low:
|
|---|
| 343 | begin
|
|---|
| 344 | Value.Insert(0, 'X-MSMAIL-Priority: low');
|
|---|
| 345 | Value.Insert(0, 'X-Priority: 5');
|
|---|
| 346 | Value.Insert(0, 'Priority: non-urgent');
|
|---|
| 347 | end;
|
|---|
| 348 | end;
|
|---|
| 349 | if FReplyTo <> '' then
|
|---|
| 350 | Value.Insert(0, 'Reply-To: ' + GetEmailAddr(FReplyTo));
|
|---|
| 351 | if FMessageID <> '' then
|
|---|
| 352 | Value.Insert(0, 'Message-ID: <' + trim(FMessageID) + '>');
|
|---|
| 353 | if FXMailer = '' then
|
|---|
| 354 | Value.Insert(0, 'X-mailer: Synapse - Pascal TCP/IP library by Lukas Gebauer')
|
|---|
| 355 | else
|
|---|
| 356 | Value.Insert(0, 'X-mailer: ' + FXMailer);
|
|---|
| 357 | Value.Insert(0, 'MIME-Version: 1.0 (produced by Synapse)');
|
|---|
| 358 | if FOrganization <> '' then
|
|---|
| 359 | Value.Insert(0, 'Organization: ' + InlineCodeEx(FOrganization, FCharsetCode));
|
|---|
| 360 | s := '';
|
|---|
| 361 | for n := 0 to FCCList.Count - 1 do
|
|---|
| 362 | if s = '' then
|
|---|
| 363 | s := InlineEmailEx(FCCList[n], FCharsetCode)
|
|---|
| 364 | else
|
|---|
| 365 | s := s + ', ' + InlineEmailEx(FCCList[n], FCharsetCode);
|
|---|
| 366 | if s <> '' then
|
|---|
| 367 | Value.Insert(0, 'CC: ' + s);
|
|---|
| 368 | Value.Insert(0, 'Date: ' + Rfc822DateTime(FDate));
|
|---|
| 369 | if FSubject <> '' then
|
|---|
| 370 | Value.Insert(0, 'Subject: ' + InlineCodeEx(FSubject, FCharsetCode));
|
|---|
| 371 | s := '';
|
|---|
| 372 | for n := 0 to FToList.Count - 1 do
|
|---|
| 373 | if s = '' then
|
|---|
| 374 | s := InlineEmailEx(FToList[n], FCharsetCode)
|
|---|
| 375 | else
|
|---|
| 376 | s := s + ', ' + InlineEmailEx(FToList[n], FCharsetCode);
|
|---|
| 377 | if s <> '' then
|
|---|
| 378 | Value.Insert(0, 'To: ' + s);
|
|---|
| 379 | Value.Insert(0, 'From: ' + InlineEmailEx(FFrom, FCharsetCode));
|
|---|
| 380 | end;
|
|---|
| 381 |
|
|---|
| 382 | function TMessHeader.ParsePriority(value: string): TMessPriority;
|
|---|
| 383 | var
|
|---|
| 384 | s: string;
|
|---|
| 385 | x: integer;
|
|---|
| 386 | begin
|
|---|
| 387 | Result := MP_unknown;
|
|---|
| 388 | s := Trim(separateright(value, ':'));
|
|---|
| 389 | s := Separateleft(s, ' ');
|
|---|
| 390 | x := StrToIntDef(s, -1);
|
|---|
| 391 | if x >= 0 then
|
|---|
| 392 | case x of
|
|---|
| 393 | 1, 2:
|
|---|
| 394 | Result := MP_High;
|
|---|
| 395 | 3:
|
|---|
| 396 | Result := MP_Normal;
|
|---|
| 397 | 4, 5:
|
|---|
| 398 | Result := MP_Low;
|
|---|
| 399 | end
|
|---|
| 400 | else
|
|---|
| 401 | begin
|
|---|
| 402 | s := lowercase(s);
|
|---|
| 403 | if (s = 'urgent') or (s = 'high') or (s = 'highest') then
|
|---|
| 404 | Result := MP_High;
|
|---|
| 405 | if (s = 'normal') or (s = 'medium') then
|
|---|
| 406 | Result := MP_Normal;
|
|---|
| 407 | if (s = 'low') or (s = 'lowest')
|
|---|
| 408 | or (s = 'no-priority') or (s = 'non-urgent') then
|
|---|
| 409 | Result := MP_Low;
|
|---|
| 410 | end;
|
|---|
| 411 | end;
|
|---|
| 412 |
|
|---|
| 413 | function TMessHeader.DecodeHeader(value: string): boolean;
|
|---|
| 414 | var
|
|---|
| 415 | s, t: string;
|
|---|
| 416 | cp: TMimeChar;
|
|---|
| 417 | begin
|
|---|
| 418 | Result := True;
|
|---|
| 419 | cp := FCharsetCode;
|
|---|
| 420 | s := uppercase(value);
|
|---|
| 421 | if Pos('X-MAILER:', s) = 1 then
|
|---|
| 422 | begin
|
|---|
| 423 | FXMailer := Trim(SeparateRight(Value, ':'));
|
|---|
| 424 | Exit;
|
|---|
| 425 | end;
|
|---|
| 426 | if Pos('FROM:', s) = 1 then
|
|---|
| 427 | begin
|
|---|
| 428 | FFrom := InlineDecode(Trim(SeparateRight(Value, ':')), cp);
|
|---|
| 429 | Exit;
|
|---|
| 430 | end;
|
|---|
| 431 | if Pos('SUBJECT:', s) = 1 then
|
|---|
| 432 | begin
|
|---|
| 433 | FSubject := InlineDecode(Trim(SeparateRight(Value, ':')), cp);
|
|---|
| 434 | Exit;
|
|---|
| 435 | end;
|
|---|
| 436 | if Pos('ORGANIZATION:', s) = 1 then
|
|---|
| 437 | begin
|
|---|
| 438 | FOrganization := InlineDecode(Trim(SeparateRight(Value, ':')), cp);
|
|---|
| 439 | Exit;
|
|---|
| 440 | end;
|
|---|
| 441 | if Pos('TO:', s) = 1 then
|
|---|
| 442 | begin
|
|---|
| 443 | s := Trim(SeparateRight(Value, ':'));
|
|---|
| 444 | repeat
|
|---|
| 445 | t := InlineDecode(Trim(FetchEx(s, ',', '"')), cp);
|
|---|
| 446 | if t <> '' then
|
|---|
| 447 | FToList.Add(t);
|
|---|
| 448 | until s = '';
|
|---|
| 449 | Exit;
|
|---|
| 450 | end;
|
|---|
| 451 | if Pos('CC:', s) = 1 then
|
|---|
| 452 | begin
|
|---|
| 453 | s := Trim(SeparateRight(Value, ':'));
|
|---|
| 454 | repeat
|
|---|
| 455 | t := InlineDecode(Trim(FetchEx(s, ',', '"')), cp);
|
|---|
| 456 | if t <> '' then
|
|---|
| 457 | FCCList.Add(t);
|
|---|
| 458 | until s = '';
|
|---|
| 459 | Exit;
|
|---|
| 460 | end;
|
|---|
| 461 | if Pos('DATE:', s) = 1 then
|
|---|
| 462 | begin
|
|---|
| 463 | FDate := DecodeRfcDateTime(Trim(SeparateRight(Value, ':')));
|
|---|
| 464 | Exit;
|
|---|
| 465 | end;
|
|---|
| 466 | if Pos('REPLY-TO:', s) = 1 then
|
|---|
| 467 | begin
|
|---|
| 468 | FReplyTo := InlineDecode(Trim(SeparateRight(Value, ':')), cp);
|
|---|
| 469 | Exit;
|
|---|
| 470 | end;
|
|---|
| 471 | if Pos('MESSAGE-ID:', s) = 1 then
|
|---|
| 472 | begin
|
|---|
| 473 | FMessageID := GetEmailAddr(Trim(SeparateRight(Value, ':')));
|
|---|
| 474 | Exit;
|
|---|
| 475 | end;
|
|---|
| 476 | if Pos('PRIORITY:', s) = 1 then
|
|---|
| 477 | begin
|
|---|
| 478 | FPri := ParsePriority(value);
|
|---|
| 479 | Exit;
|
|---|
| 480 | end;
|
|---|
| 481 | if Pos('X-PRIORITY:', s) = 1 then
|
|---|
| 482 | begin
|
|---|
| 483 | FXPri := ParsePriority(value);
|
|---|
| 484 | Exit;
|
|---|
| 485 | end;
|
|---|
| 486 | if Pos('X-MSMAIL-PRIORITY:', s) = 1 then
|
|---|
| 487 | begin
|
|---|
| 488 | FXmsPri := ParsePriority(value);
|
|---|
| 489 | Exit;
|
|---|
| 490 | end;
|
|---|
| 491 | if Pos('MIME-VERSION:', s) = 1 then
|
|---|
| 492 | Exit;
|
|---|
| 493 | if Pos('CONTENT-TYPE:', s) = 1 then
|
|---|
| 494 | Exit;
|
|---|
| 495 | if Pos('CONTENT-DESCRIPTION:', s) = 1 then
|
|---|
| 496 | Exit;
|
|---|
| 497 | if Pos('CONTENT-DISPOSITION:', s) = 1 then
|
|---|
| 498 | Exit;
|
|---|
| 499 | if Pos('CONTENT-ID:', s) = 1 then
|
|---|
| 500 | Exit;
|
|---|
| 501 | if Pos('CONTENT-TRANSFER-ENCODING:', s) = 1 then
|
|---|
| 502 | Exit;
|
|---|
| 503 | Result := False;
|
|---|
| 504 | end;
|
|---|
| 505 |
|
|---|
| 506 | procedure TMessHeader.DecodeHeaders(const Value: TStrings);
|
|---|
| 507 | var
|
|---|
| 508 | s: string;
|
|---|
| 509 | x: Integer;
|
|---|
| 510 | begin
|
|---|
| 511 | Clear;
|
|---|
| 512 | Fpri := MP_unknown;
|
|---|
| 513 | Fxpri := MP_unknown;
|
|---|
| 514 | Fxmspri := MP_unknown;
|
|---|
| 515 | x := 0;
|
|---|
| 516 | while Value.Count > x do
|
|---|
| 517 | begin
|
|---|
| 518 | s := NormalizeHeader(Value, x);
|
|---|
| 519 | if s = '' then
|
|---|
| 520 | Break;
|
|---|
| 521 | if not DecodeHeader(s) then
|
|---|
| 522 | FCustomHeaders.Add(s);
|
|---|
| 523 | end;
|
|---|
| 524 | if Fpri <> MP_unknown then
|
|---|
| 525 | FPriority := Fpri
|
|---|
| 526 | else
|
|---|
| 527 | if Fxpri <> MP_unknown then
|
|---|
| 528 | FPriority := Fxpri
|
|---|
| 529 | else
|
|---|
| 530 | if Fxmspri <> MP_unknown then
|
|---|
| 531 | FPriority := Fxmspri
|
|---|
| 532 | end;
|
|---|
| 533 |
|
|---|
| 534 | function TMessHeader.FindHeader(Value: string): string;
|
|---|
| 535 | var
|
|---|
| 536 | n: integer;
|
|---|
| 537 | begin
|
|---|
| 538 | Result := '';
|
|---|
| 539 | for n := 0 to FCustomHeaders.Count - 1 do
|
|---|
| 540 | if Pos(UpperCase(Value), UpperCase(FCustomHeaders[n])) = 1 then
|
|---|
| 541 | begin
|
|---|
| 542 | Result := Trim(SeparateRight(FCustomHeaders[n], ':'));
|
|---|
| 543 | break;
|
|---|
| 544 | end;
|
|---|
| 545 | end;
|
|---|
| 546 |
|
|---|
| 547 | procedure TMessHeader.FindHeaderList(Value: string; const HeaderList: TStrings);
|
|---|
| 548 | var
|
|---|
| 549 | n: integer;
|
|---|
| 550 | begin
|
|---|
| 551 | HeaderList.Clear;
|
|---|
| 552 | for n := 0 to FCustomHeaders.Count - 1 do
|
|---|
| 553 | if Pos(UpperCase(Value), UpperCase(FCustomHeaders[n])) = 1 then
|
|---|
| 554 | begin
|
|---|
| 555 | HeaderList.Add(Trim(SeparateRight(FCustomHeaders[n], ':')));
|
|---|
| 556 | end;
|
|---|
| 557 | end;
|
|---|
| 558 |
|
|---|
| 559 | {==============================================================================}
|
|---|
| 560 |
|
|---|
| 561 | constructor TMimeMess.Create;
|
|---|
| 562 | begin
|
|---|
| 563 | CreateAltHeaders(TMessHeader);
|
|---|
| 564 | end;
|
|---|
| 565 |
|
|---|
| 566 | constructor TMimeMess.CreateAltHeaders(HeadClass: TMessHeaderClass);
|
|---|
| 567 | begin
|
|---|
| 568 | inherited Create;
|
|---|
| 569 | FMessagePart := TMimePart.Create;
|
|---|
| 570 | FLines := TStringList.Create;
|
|---|
| 571 | FHeader := HeadClass.Create;
|
|---|
| 572 | end;
|
|---|
| 573 |
|
|---|
| 574 | destructor TMimeMess.Destroy;
|
|---|
| 575 | begin
|
|---|
| 576 | FMessagePart.Free;
|
|---|
| 577 | FHeader.Free;
|
|---|
| 578 | FLines.Free;
|
|---|
| 579 | inherited Destroy;
|
|---|
| 580 | end;
|
|---|
| 581 |
|
|---|
| 582 | {==============================================================================}
|
|---|
| 583 |
|
|---|
| 584 | procedure TMimeMess.Clear;
|
|---|
| 585 | begin
|
|---|
| 586 | FMessagePart.Clear;
|
|---|
| 587 | FLines.Clear;
|
|---|
| 588 | FHeader.Clear;
|
|---|
| 589 | end;
|
|---|
| 590 |
|
|---|
| 591 | {==============================================================================}
|
|---|
| 592 |
|
|---|
| 593 | function TMimeMess.AddPart(const PartParent: TMimePart): TMimePart;
|
|---|
| 594 | begin
|
|---|
| 595 | if PartParent = nil then
|
|---|
| 596 | Result := FMessagePart
|
|---|
| 597 | else
|
|---|
| 598 | Result := PartParent.AddSubPart;
|
|---|
| 599 | Result.Clear;
|
|---|
| 600 | end;
|
|---|
| 601 |
|
|---|
| 602 | {==============================================================================}
|
|---|
| 603 |
|
|---|
| 604 | function TMimeMess.AddPartMultipart(const MultipartType: String; const PartParent: TMimePart): TMimePart;
|
|---|
| 605 | begin
|
|---|
| 606 | Result := AddPart(PartParent);
|
|---|
| 607 | with Result do
|
|---|
| 608 | begin
|
|---|
| 609 | Primary := 'Multipart';
|
|---|
| 610 | Secondary := MultipartType;
|
|---|
| 611 | Description := 'Multipart message';
|
|---|
| 612 | Boundary := GenerateBoundary;
|
|---|
| 613 | EncodePartHeader;
|
|---|
| 614 | end;
|
|---|
| 615 | end;
|
|---|
| 616 |
|
|---|
| 617 | function TMimeMess.AddPartText(const Value: TStrings; const PartParent: TMimePart): TMimepart;
|
|---|
| 618 | begin
|
|---|
| 619 | Result := AddPart(PartParent);
|
|---|
| 620 | with Result do
|
|---|
| 621 | begin
|
|---|
| 622 | Value.SaveToStream(DecodedLines);
|
|---|
| 623 | Primary := 'text';
|
|---|
| 624 | Secondary := 'plain';
|
|---|
| 625 | Description := 'Message text';
|
|---|
| 626 | Disposition := 'inline';
|
|---|
| 627 | CharsetCode := IdealCharsetCoding(Value.Text, TargetCharset, IdealCharsets);
|
|---|
| 628 | EncodingCode := ME_QUOTED_PRINTABLE;
|
|---|
| 629 | EncodePart;
|
|---|
| 630 | EncodePartHeader;
|
|---|
| 631 | end;
|
|---|
| 632 | end;
|
|---|
| 633 |
|
|---|
| 634 | function TMimeMess.AddPartTextEx(const Value: TStrings; const PartParent: TMimePart;
|
|---|
| 635 | PartCharset: TMimeChar; Raw: Boolean; PartEncoding: TMimeEncoding): TMimepart;
|
|---|
| 636 | begin
|
|---|
| 637 | Result := AddPart(PartParent);
|
|---|
| 638 | with Result do
|
|---|
| 639 | begin
|
|---|
| 640 | Value.SaveToStream(DecodedLines);
|
|---|
| 641 | Primary := 'text';
|
|---|
| 642 | Secondary := 'plain';
|
|---|
| 643 | Description := 'Message text';
|
|---|
| 644 | Disposition := 'inline';
|
|---|
| 645 | CharsetCode := PartCharset;
|
|---|
| 646 | EncodingCode := PartEncoding;
|
|---|
| 647 | ConvertCharset := not Raw;
|
|---|
| 648 | EncodePart;
|
|---|
| 649 | EncodePartHeader;
|
|---|
| 650 | end;
|
|---|
| 651 | end;
|
|---|
| 652 |
|
|---|
| 653 | function TMimeMess.AddPartHTML(const Value: TStrings; const PartParent: TMimePart): TMimepart;
|
|---|
| 654 | begin
|
|---|
| 655 | Result := AddPart(PartParent);
|
|---|
| 656 | with Result do
|
|---|
| 657 | begin
|
|---|
| 658 | Value.SaveToStream(DecodedLines);
|
|---|
| 659 | Primary := 'text';
|
|---|
| 660 | Secondary := 'html';
|
|---|
| 661 | Description := 'HTML text';
|
|---|
| 662 | Disposition := 'inline';
|
|---|
| 663 | CharsetCode := UTF_8;
|
|---|
| 664 | EncodingCode := ME_QUOTED_PRINTABLE;
|
|---|
| 665 | EncodePart;
|
|---|
| 666 | EncodePartHeader;
|
|---|
| 667 | end;
|
|---|
| 668 | end;
|
|---|
| 669 |
|
|---|
| 670 | function TMimeMess.AddPartTextFromFile(const FileName: String; const PartParent: TMimePart): TMimepart;
|
|---|
| 671 | var
|
|---|
| 672 | tmp: TStrings;
|
|---|
| 673 | begin
|
|---|
| 674 | tmp := TStringList.Create;
|
|---|
| 675 | try
|
|---|
| 676 | tmp.LoadFromFile(FileName);
|
|---|
| 677 | Result := AddPartText(tmp, PartParent);
|
|---|
| 678 | Finally
|
|---|
| 679 | tmp.Free;
|
|---|
| 680 | end;
|
|---|
| 681 | end;
|
|---|
| 682 |
|
|---|
| 683 | function TMimeMess.AddPartHTMLFromFile(const FileName: String; const PartParent: TMimePart): TMimepart;
|
|---|
| 684 | var
|
|---|
| 685 | tmp: TStrings;
|
|---|
| 686 | begin
|
|---|
| 687 | tmp := TStringList.Create;
|
|---|
| 688 | try
|
|---|
| 689 | tmp.LoadFromFile(FileName);
|
|---|
| 690 | Result := AddPartHTML(tmp, PartParent);
|
|---|
| 691 | Finally
|
|---|
| 692 | tmp.Free;
|
|---|
| 693 | end;
|
|---|
| 694 | end;
|
|---|
| 695 |
|
|---|
| 696 | function TMimeMess.AddPartBinary(const Stream: TStream; const FileName: string; const PartParent: TMimePart): TMimepart;
|
|---|
| 697 | begin
|
|---|
| 698 | Result := AddPart(PartParent);
|
|---|
| 699 | Result.DecodedLines.LoadFromStream(Stream);
|
|---|
| 700 | Result.MimeTypeFromExt(FileName);
|
|---|
| 701 | Result.Description := 'Attached file: ' + FileName;
|
|---|
| 702 | Result.Disposition := 'attachment';
|
|---|
| 703 | Result.FileName := FileName;
|
|---|
| 704 | Result.EncodingCode := ME_BASE64;
|
|---|
| 705 | Result.EncodePart;
|
|---|
| 706 | Result.EncodePartHeader;
|
|---|
| 707 | end;
|
|---|
| 708 |
|
|---|
| 709 | function TMimeMess.AddPartBinaryFromFile(const FileName: string; const PartParent: TMimePart): TMimepart;
|
|---|
| 710 | var
|
|---|
| 711 | tmp: TMemoryStream;
|
|---|
| 712 | begin
|
|---|
| 713 | tmp := TMemoryStream.Create;
|
|---|
| 714 | try
|
|---|
| 715 | tmp.LoadFromFile(FileName);
|
|---|
| 716 | Result := AddPartBinary(tmp, ExtractFileName(FileName), PartParent);
|
|---|
| 717 | finally
|
|---|
| 718 | tmp.Free;
|
|---|
| 719 | end;
|
|---|
| 720 | end;
|
|---|
| 721 |
|
|---|
| 722 | function TMimeMess.AddPartHTMLBinary(const Stream: TStream; const FileName, Cid: string; const PartParent: TMimePart): TMimepart;
|
|---|
| 723 | begin
|
|---|
| 724 | Result := AddPart(PartParent);
|
|---|
| 725 | Result.DecodedLines.LoadFromStream(Stream);
|
|---|
| 726 | Result.MimeTypeFromExt(FileName);
|
|---|
| 727 | Result.Description := 'Included file: ' + FileName;
|
|---|
| 728 | Result.Disposition := 'inline';
|
|---|
| 729 | Result.ContentID := Cid;
|
|---|
| 730 | Result.FileName := FileName;
|
|---|
| 731 | Result.EncodingCode := ME_BASE64;
|
|---|
| 732 | Result.EncodePart;
|
|---|
| 733 | Result.EncodePartHeader;
|
|---|
| 734 | end;
|
|---|
| 735 |
|
|---|
| 736 | function TMimeMess.AddPartHTMLBinaryFromFile(const FileName, Cid: string; const PartParent: TMimePart): TMimepart;
|
|---|
| 737 | var
|
|---|
| 738 | tmp: TMemoryStream;
|
|---|
| 739 | begin
|
|---|
| 740 | tmp := TMemoryStream.Create;
|
|---|
| 741 | try
|
|---|
| 742 | tmp.LoadFromFile(FileName);
|
|---|
| 743 | Result :=AddPartHTMLBinary(tmp, ExtractFileName(FileName), Cid, PartParent);
|
|---|
| 744 | finally
|
|---|
| 745 | tmp.Free;
|
|---|
| 746 | end;
|
|---|
| 747 | end;
|
|---|
| 748 |
|
|---|
| 749 | function TMimeMess.AddPartMess(const Value: TStrings; const PartParent: TMimePart): TMimepart;
|
|---|
| 750 | var
|
|---|
| 751 | part: Tmimepart;
|
|---|
| 752 | begin
|
|---|
| 753 | Result := AddPart(PartParent);
|
|---|
| 754 | part := AddPart(result);
|
|---|
| 755 | part.lines.addstrings(Value);
|
|---|
| 756 | part.DecomposeParts;
|
|---|
| 757 | with Result do
|
|---|
| 758 | begin
|
|---|
| 759 | Primary := 'message';
|
|---|
| 760 | Secondary := 'rfc822';
|
|---|
| 761 | Description := 'E-mail Message';
|
|---|
| 762 | EncodePart;
|
|---|
| 763 | EncodePartHeader;
|
|---|
| 764 | end;
|
|---|
| 765 | end;
|
|---|
| 766 |
|
|---|
| 767 | function TMimeMess.AddPartMessFromFile(const FileName: String; const PartParent: TMimePart): TMimepart;
|
|---|
| 768 | var
|
|---|
| 769 | tmp: TStrings;
|
|---|
| 770 | begin
|
|---|
| 771 | tmp := TStringList.Create;
|
|---|
| 772 | try
|
|---|
| 773 | tmp.LoadFromFile(FileName);
|
|---|
| 774 | Result := AddPartMess(tmp, PartParent);
|
|---|
| 775 | Finally
|
|---|
| 776 | tmp.Free;
|
|---|
| 777 | end;
|
|---|
| 778 | end;
|
|---|
| 779 |
|
|---|
| 780 | {==============================================================================}
|
|---|
| 781 |
|
|---|
| 782 | procedure TMimeMess.EncodeMessage;
|
|---|
| 783 | var
|
|---|
| 784 | l: TStringList;
|
|---|
| 785 | x: integer;
|
|---|
| 786 | begin
|
|---|
| 787 | //merge headers from THeaders and header field from MessagePart
|
|---|
| 788 | l := TStringList.Create;
|
|---|
| 789 | try
|
|---|
| 790 | FHeader.EncodeHeaders(l);
|
|---|
| 791 | x := IndexByBegin('CONTENT-TYPE', FMessagePart.Headers);
|
|---|
| 792 | if x >= 0 then
|
|---|
| 793 | l.add(FMessagePart.Headers[x]);
|
|---|
| 794 | x := IndexByBegin('CONTENT-DESCRIPTION', FMessagePart.Headers);
|
|---|
| 795 | if x >= 0 then
|
|---|
| 796 | l.add(FMessagePart.Headers[x]);
|
|---|
| 797 | x := IndexByBegin('CONTENT-DISPOSITION', FMessagePart.Headers);
|
|---|
| 798 | if x >= 0 then
|
|---|
| 799 | l.add(FMessagePart.Headers[x]);
|
|---|
| 800 | x := IndexByBegin('CONTENT-ID', FMessagePart.Headers);
|
|---|
| 801 | if x >= 0 then
|
|---|
| 802 | l.add(FMessagePart.Headers[x]);
|
|---|
| 803 | x := IndexByBegin('CONTENT-TRANSFER-ENCODING', FMessagePart.Headers);
|
|---|
| 804 | if x >= 0 then
|
|---|
| 805 | l.add(FMessagePart.Headers[x]);
|
|---|
| 806 | FMessagePart.Headers.Assign(l);
|
|---|
| 807 | finally
|
|---|
| 808 | l.Free;
|
|---|
| 809 | end;
|
|---|
| 810 | FMessagePart.ComposeParts;
|
|---|
| 811 | FLines.Assign(FMessagePart.Lines);
|
|---|
| 812 | end;
|
|---|
| 813 |
|
|---|
| 814 | {==============================================================================}
|
|---|
| 815 |
|
|---|
| 816 | procedure TMimeMess.DecodeMessage;
|
|---|
| 817 | begin
|
|---|
| 818 | FHeader.Clear;
|
|---|
| 819 | FHeader.DecodeHeaders(FLines);
|
|---|
| 820 | FMessagePart.Lines.Assign(FLines);
|
|---|
| 821 | FMessagePart.DecomposeParts;
|
|---|
| 822 | end;
|
|---|
| 823 |
|
|---|
| 824 | end.
|
|---|