blob: c8fc1a8243185affff6177b4ae8ad966231d3a95 [file] [log] [blame]
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +01001<?php
Matthias Andreas Benkarde39c4f82021-01-06 17:59:39 +01002
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +01003/**
4 * PHPMailer - PHP email creation and transport class.
5 * PHP Version 5.5.
6 *
7 * @see https://github.com/PHPMailer/PHPMailer/ The PHPMailer GitHub project
8 *
9 * @author Marcus Bointon (Synchro/coolbru) <phpmailer@synchromedia.co.uk>
10 * @author Jim Jagielski (jimjag) <jimjag@gmail.com>
11 * @author Andy Prevost (codeworxtech) <codeworxtech@users.sourceforge.net>
12 * @author Brent R. Matzelle (original founder)
Matthias Andreas Benkarde39c4f82021-01-06 17:59:39 +010013 * @copyright 2012 - 2020 Marcus Bointon
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +010014 * @copyright 2010 - 2012 Jim Jagielski
15 * @copyright 2004 - 2009 Andy Prevost
16 * @license http://www.gnu.org/copyleft/lesser.html GNU Lesser General Public License
17 * @note This program is distributed in the hope that it will be useful - WITHOUT
18 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
19 * FITNESS FOR A PARTICULAR PURPOSE.
20 */
21
22namespace PHPMailer\PHPMailer;
23
24/**
25 * PHPMailer - PHP email creation and transport class.
26 *
27 * @author Marcus Bointon (Synchro/coolbru) <phpmailer@synchromedia.co.uk>
28 * @author Jim Jagielski (jimjag) <jimjag@gmail.com>
29 * @author Andy Prevost (codeworxtech) <codeworxtech@users.sourceforge.net>
30 * @author Brent R. Matzelle (original founder)
31 */
32class PHPMailer
33{
34 const CHARSET_ASCII = 'us-ascii';
35 const CHARSET_ISO88591 = 'iso-8859-1';
36 const CHARSET_UTF8 = 'utf-8';
37
38 const CONTENT_TYPE_PLAINTEXT = 'text/plain';
39 const CONTENT_TYPE_TEXT_CALENDAR = 'text/calendar';
40 const CONTENT_TYPE_TEXT_HTML = 'text/html';
41 const CONTENT_TYPE_MULTIPART_ALTERNATIVE = 'multipart/alternative';
42 const CONTENT_TYPE_MULTIPART_MIXED = 'multipart/mixed';
43 const CONTENT_TYPE_MULTIPART_RELATED = 'multipart/related';
44
45 const ENCODING_7BIT = '7bit';
46 const ENCODING_8BIT = '8bit';
47 const ENCODING_BASE64 = 'base64';
48 const ENCODING_BINARY = 'binary';
49 const ENCODING_QUOTED_PRINTABLE = 'quoted-printable';
50
51 const ENCRYPTION_STARTTLS = 'tls';
52 const ENCRYPTION_SMTPS = 'ssl';
53
54 const ICAL_METHOD_REQUEST = 'REQUEST';
55 const ICAL_METHOD_PUBLISH = 'PUBLISH';
56 const ICAL_METHOD_REPLY = 'REPLY';
57 const ICAL_METHOD_ADD = 'ADD';
58 const ICAL_METHOD_CANCEL = 'CANCEL';
59 const ICAL_METHOD_REFRESH = 'REFRESH';
60 const ICAL_METHOD_COUNTER = 'COUNTER';
61 const ICAL_METHOD_DECLINECOUNTER = 'DECLINECOUNTER';
62
63 /**
64 * Email priority.
65 * Options: null (default), 1 = High, 3 = Normal, 5 = low.
66 * When null, the header is not set at all.
67 *
68 * @var int|null
69 */
70 public $Priority;
71
72 /**
73 * The character set of the message.
74 *
75 * @var string
76 */
77 public $CharSet = self::CHARSET_ISO88591;
78
79 /**
80 * The MIME Content-type of the message.
81 *
82 * @var string
83 */
84 public $ContentType = self::CONTENT_TYPE_PLAINTEXT;
85
86 /**
87 * The message encoding.
88 * Options: "8bit", "7bit", "binary", "base64", and "quoted-printable".
89 *
90 * @var string
91 */
92 public $Encoding = self::ENCODING_8BIT;
93
94 /**
95 * Holds the most recent mailer error message.
96 *
97 * @var string
98 */
99 public $ErrorInfo = '';
100
101 /**
102 * The From email address for the message.
103 *
104 * @var string
105 */
106 public $From = 'root@localhost';
107
108 /**
109 * The From name of the message.
110 *
111 * @var string
112 */
113 public $FromName = 'Root User';
114
115 /**
116 * The envelope sender of the message.
117 * This will usually be turned into a Return-Path header by the receiver,
118 * and is the address that bounces will be sent to.
119 * If not empty, will be passed via `-f` to sendmail or as the 'MAIL FROM' value over SMTP.
120 *
121 * @var string
122 */
123 public $Sender = '';
124
125 /**
126 * The Subject of the message.
127 *
128 * @var string
129 */
130 public $Subject = '';
131
132 /**
133 * An HTML or plain text message body.
134 * If HTML then call isHTML(true).
135 *
136 * @var string
137 */
138 public $Body = '';
139
140 /**
141 * The plain-text message body.
142 * This body can be read by mail clients that do not have HTML email
143 * capability such as mutt & Eudora.
144 * Clients that can read HTML will view the normal Body.
145 *
146 * @var string
147 */
148 public $AltBody = '';
149
150 /**
151 * An iCal message part body.
152 * Only supported in simple alt or alt_inline message types
153 * To generate iCal event structures, use classes like EasyPeasyICS or iCalcreator.
154 *
155 * @see http://sprain.ch/blog/downloads/php-class-easypeasyics-create-ical-files-with-php/
156 * @see http://kigkonsult.se/iCalcreator/
157 *
158 * @var string
159 */
160 public $Ical = '';
161
162 /**
163 * Value-array of "method" in Contenttype header "text/calendar"
164 *
165 * @var string[]
166 */
167 protected static $IcalMethods = [
168 self::ICAL_METHOD_REQUEST,
169 self::ICAL_METHOD_PUBLISH,
170 self::ICAL_METHOD_REPLY,
171 self::ICAL_METHOD_ADD,
172 self::ICAL_METHOD_CANCEL,
173 self::ICAL_METHOD_REFRESH,
174 self::ICAL_METHOD_COUNTER,
175 self::ICAL_METHOD_DECLINECOUNTER,
176 ];
177
178 /**
179 * The complete compiled MIME message body.
180 *
181 * @var string
182 */
183 protected $MIMEBody = '';
184
185 /**
186 * The complete compiled MIME message headers.
187 *
188 * @var string
189 */
190 protected $MIMEHeader = '';
191
192 /**
193 * Extra headers that createHeader() doesn't fold in.
194 *
195 * @var string
196 */
197 protected $mailHeader = '';
198
199 /**
200 * Word-wrap the message body to this number of chars.
201 * Set to 0 to not wrap. A useful value here is 78, for RFC2822 section 2.1.1 compliance.
202 *
203 * @see static::STD_LINE_LENGTH
204 *
205 * @var int
206 */
207 public $WordWrap = 0;
208
209 /**
210 * Which method to use to send mail.
211 * Options: "mail", "sendmail", or "smtp".
212 *
213 * @var string
214 */
215 public $Mailer = 'mail';
216
217 /**
218 * The path to the sendmail program.
219 *
220 * @var string
221 */
222 public $Sendmail = '/usr/sbin/sendmail';
223
224 /**
225 * Whether mail() uses a fully sendmail-compatible MTA.
226 * One which supports sendmail's "-oi -f" options.
227 *
228 * @var bool
229 */
230 public $UseSendmailOptions = true;
231
232 /**
233 * The email address that a reading confirmation should be sent to, also known as read receipt.
234 *
235 * @var string
236 */
237 public $ConfirmReadingTo = '';
238
239 /**
240 * The hostname to use in the Message-ID header and as default HELO string.
241 * If empty, PHPMailer attempts to find one with, in order,
242 * $_SERVER['SERVER_NAME'], gethostname(), php_uname('n'), or the value
243 * 'localhost.localdomain'.
244 *
245 * @see PHPMailer::$Helo
246 *
247 * @var string
248 */
249 public $Hostname = '';
250
251 /**
252 * An ID to be used in the Message-ID header.
253 * If empty, a unique id will be generated.
254 * You can set your own, but it must be in the format "<id@domain>",
255 * as defined in RFC5322 section 3.6.4 or it will be ignored.
256 *
257 * @see https://tools.ietf.org/html/rfc5322#section-3.6.4
258 *
259 * @var string
260 */
261 public $MessageID = '';
262
263 /**
264 * The message Date to be used in the Date header.
265 * If empty, the current date will be added.
266 *
267 * @var string
268 */
269 public $MessageDate = '';
270
271 /**
272 * SMTP hosts.
273 * Either a single hostname or multiple semicolon-delimited hostnames.
274 * You can also specify a different port
275 * for each host by using this format: [hostname:port]
276 * (e.g. "smtp1.example.com:25;smtp2.example.com").
277 * You can also specify encryption type, for example:
278 * (e.g. "tls://smtp1.example.com:587;ssl://smtp2.example.com:465").
279 * Hosts will be tried in order.
280 *
281 * @var string
282 */
283 public $Host = 'localhost';
284
285 /**
286 * The default SMTP server port.
287 *
288 * @var int
289 */
290 public $Port = 25;
291
292 /**
293 * The SMTP HELO/EHLO name used for the SMTP connection.
294 * Default is $Hostname. If $Hostname is empty, PHPMailer attempts to find
295 * one with the same method described above for $Hostname.
296 *
297 * @see PHPMailer::$Hostname
298 *
299 * @var string
300 */
301 public $Helo = '';
302
303 /**
304 * What kind of encryption to use on the SMTP connection.
305 * Options: '', static::ENCRYPTION_STARTTLS, or static::ENCRYPTION_SMTPS.
306 *
307 * @var string
308 */
309 public $SMTPSecure = '';
310
311 /**
312 * Whether to enable TLS encryption automatically if a server supports it,
313 * even if `SMTPSecure` is not set to 'tls'.
314 * Be aware that in PHP >= 5.6 this requires that the server's certificates are valid.
315 *
316 * @var bool
317 */
318 public $SMTPAutoTLS = true;
319
320 /**
321 * Whether to use SMTP authentication.
322 * Uses the Username and Password properties.
323 *
324 * @see PHPMailer::$Username
325 * @see PHPMailer::$Password
326 *
327 * @var bool
328 */
329 public $SMTPAuth = false;
330
331 /**
332 * Options array passed to stream_context_create when connecting via SMTP.
333 *
334 * @var array
335 */
336 public $SMTPOptions = [];
337
338 /**
339 * SMTP username.
340 *
341 * @var string
342 */
343 public $Username = '';
344
345 /**
346 * SMTP password.
347 *
348 * @var string
349 */
350 public $Password = '';
351
352 /**
353 * SMTP auth type.
354 * Options are CRAM-MD5, LOGIN, PLAIN, XOAUTH2, attempted in that order if not specified.
355 *
356 * @var string
357 */
358 public $AuthType = '';
359
360 /**
361 * An instance of the PHPMailer OAuth class.
362 *
363 * @var OAuth
364 */
365 protected $oauth;
366
367 /**
368 * The SMTP server timeout in seconds.
369 * Default of 5 minutes (300sec) is from RFC2821 section 4.5.3.2.
370 *
371 * @var int
372 */
373 public $Timeout = 300;
374
375 /**
376 * Comma separated list of DSN notifications
377 * 'NEVER' under no circumstances a DSN must be returned to the sender.
378 * If you use NEVER all other notifications will be ignored.
379 * 'SUCCESS' will notify you when your mail has arrived at its destination.
380 * 'FAILURE' will arrive if an error occurred during delivery.
381 * 'DELAY' will notify you if there is an unusual delay in delivery, but the actual
382 * delivery's outcome (success or failure) is not yet decided.
383 *
384 * @see https://tools.ietf.org/html/rfc3461 See section 4.1 for more information about NOTIFY
385 */
386 public $dsn = '';
387
388 /**
389 * SMTP class debug output mode.
390 * Debug output level.
391 * Options:
Matthias Andreas Benkarde39c4f82021-01-06 17:59:39 +0100392 * @see SMTP::DEBUG_OFF: No output
393 * @see SMTP::DEBUG_CLIENT: Client messages
394 * @see SMTP::DEBUG_SERVER: Client and server messages
395 * @see SMTP::DEBUG_CONNECTION: As SERVER plus connection status
396 * @see SMTP::DEBUG_LOWLEVEL: Noisy, low-level data output, rarely needed
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +0100397 *
398 * @see SMTP::$do_debug
399 *
400 * @var int
401 */
402 public $SMTPDebug = 0;
403
404 /**
405 * How to handle debug output.
406 * Options:
407 * * `echo` Output plain-text as-is, appropriate for CLI
408 * * `html` Output escaped, line breaks converted to `<br>`, appropriate for browser output
409 * * `error_log` Output to error log as configured in php.ini
410 * By default PHPMailer will use `echo` if run from a `cli` or `cli-server` SAPI, `html` otherwise.
411 * Alternatively, you can provide a callable expecting two params: a message string and the debug level:
412 *
413 * ```php
414 * $mail->Debugoutput = function($str, $level) {echo "debug level $level; message: $str";};
415 * ```
416 *
417 * Alternatively, you can pass in an instance of a PSR-3 compatible logger, though only `debug`
418 * level output is used:
419 *
420 * ```php
421 * $mail->Debugoutput = new myPsr3Logger;
422 * ```
423 *
424 * @see SMTP::$Debugoutput
425 *
426 * @var string|callable|\Psr\Log\LoggerInterface
427 */
428 public $Debugoutput = 'echo';
429
430 /**
431 * Whether to keep SMTP connection open after each message.
432 * If this is set to true then to close the connection
433 * requires an explicit call to smtpClose().
434 *
435 * @var bool
436 */
437 public $SMTPKeepAlive = false;
438
439 /**
440 * Whether to split multiple to addresses into multiple messages
441 * or send them all in one message.
442 * Only supported in `mail` and `sendmail` transports, not in SMTP.
443 *
444 * @var bool
Matthias Andreas Benkarde39c4f82021-01-06 17:59:39 +0100445 *
446 * @deprecated 6.0.0 PHPMailer isn't a mailing list manager!
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +0100447 */
448 public $SingleTo = false;
449
450 /**
451 * Storage for addresses when SingleTo is enabled.
452 *
453 * @var array
454 */
455 protected $SingleToArray = [];
456
457 /**
458 * Whether to generate VERP addresses on send.
459 * Only applicable when sending via SMTP.
460 *
461 * @see https://en.wikipedia.org/wiki/Variable_envelope_return_path
462 * @see http://www.postfix.org/VERP_README.html Postfix VERP info
463 *
464 * @var bool
465 */
466 public $do_verp = false;
467
468 /**
469 * Whether to allow sending messages with an empty body.
470 *
471 * @var bool
472 */
473 public $AllowEmpty = false;
474
475 /**
476 * DKIM selector.
477 *
478 * @var string
479 */
480 public $DKIM_selector = '';
481
482 /**
483 * DKIM Identity.
484 * Usually the email address used as the source of the email.
485 *
486 * @var string
487 */
488 public $DKIM_identity = '';
489
490 /**
491 * DKIM passphrase.
492 * Used if your key is encrypted.
493 *
494 * @var string
495 */
496 public $DKIM_passphrase = '';
497
498 /**
499 * DKIM signing domain name.
500 *
501 * @example 'example.com'
502 *
503 * @var string
504 */
505 public $DKIM_domain = '';
506
507 /**
508 * DKIM Copy header field values for diagnostic use.
509 *
510 * @var bool
511 */
512 public $DKIM_copyHeaderFields = true;
513
514 /**
515 * DKIM Extra signing headers.
516 *
517 * @example ['List-Unsubscribe', 'List-Help']
518 *
519 * @var array
520 */
521 public $DKIM_extraHeaders = [];
522
523 /**
524 * DKIM private key file path.
525 *
526 * @var string
527 */
528 public $DKIM_private = '';
529
530 /**
531 * DKIM private key string.
532 *
533 * If set, takes precedence over `$DKIM_private`.
534 *
535 * @var string
536 */
537 public $DKIM_private_string = '';
538
539 /**
540 * Callback Action function name.
541 *
542 * The function that handles the result of the send email action.
543 * It is called out by send() for each email sent.
544 *
545 * Value can be any php callable: http://www.php.net/is_callable
546 *
547 * Parameters:
548 * bool $result result of the send action
549 * array $to email addresses of the recipients
550 * array $cc cc email addresses
551 * array $bcc bcc email addresses
552 * string $subject the subject
553 * string $body the email body
554 * string $from email address of sender
555 * string $extra extra information of possible use
556 * "smtp_transaction_id' => last smtp transaction id
557 *
558 * @var string
559 */
560 public $action_function = '';
561
562 /**
563 * What to put in the X-Mailer header.
564 * Options: An empty string for PHPMailer default, whitespace/null for none, or a string to use.
565 *
566 * @var string|null
567 */
568 public $XMailer = '';
569
570 /**
571 * Which validator to use by default when validating email addresses.
572 * May be a callable to inject your own validator, but there are several built-in validators.
573 * The default validator uses PHP's FILTER_VALIDATE_EMAIL filter_var option.
574 *
575 * @see PHPMailer::validateAddress()
576 *
577 * @var string|callable
578 */
579 public static $validator = 'php';
580
581 /**
582 * An instance of the SMTP sender class.
583 *
584 * @var SMTP
585 */
586 protected $smtp;
587
588 /**
589 * The array of 'to' names and addresses.
590 *
591 * @var array
592 */
593 protected $to = [];
594
595 /**
596 * The array of 'cc' names and addresses.
597 *
598 * @var array
599 */
600 protected $cc = [];
601
602 /**
603 * The array of 'bcc' names and addresses.
604 *
605 * @var array
606 */
607 protected $bcc = [];
608
609 /**
610 * The array of reply-to names and addresses.
611 *
612 * @var array
613 */
614 protected $ReplyTo = [];
615
616 /**
617 * An array of all kinds of addresses.
618 * Includes all of $to, $cc, $bcc.
619 *
620 * @see PHPMailer::$to
621 * @see PHPMailer::$cc
622 * @see PHPMailer::$bcc
623 *
624 * @var array
625 */
626 protected $all_recipients = [];
627
628 /**
629 * An array of names and addresses queued for validation.
630 * In send(), valid and non duplicate entries are moved to $all_recipients
631 * and one of $to, $cc, or $bcc.
632 * This array is used only for addresses with IDN.
633 *
634 * @see PHPMailer::$to
635 * @see PHPMailer::$cc
636 * @see PHPMailer::$bcc
637 * @see PHPMailer::$all_recipients
638 *
639 * @var array
640 */
641 protected $RecipientsQueue = [];
642
643 /**
644 * An array of reply-to names and addresses queued for validation.
645 * In send(), valid and non duplicate entries are moved to $ReplyTo.
646 * This array is used only for addresses with IDN.
647 *
648 * @see PHPMailer::$ReplyTo
649 *
650 * @var array
651 */
652 protected $ReplyToQueue = [];
653
654 /**
655 * The array of attachments.
656 *
657 * @var array
658 */
659 protected $attachment = [];
660
661 /**
662 * The array of custom headers.
663 *
664 * @var array
665 */
666 protected $CustomHeader = [];
667
668 /**
669 * The most recent Message-ID (including angular brackets).
670 *
671 * @var string
672 */
673 protected $lastMessageID = '';
674
675 /**
676 * The message's MIME type.
677 *
678 * @var string
679 */
680 protected $message_type = '';
681
682 /**
683 * The array of MIME boundary strings.
684 *
685 * @var array
686 */
687 protected $boundary = [];
688
689 /**
690 * The array of available languages.
691 *
692 * @var array
693 */
694 protected $language = [];
695
696 /**
697 * The number of errors encountered.
698 *
699 * @var int
700 */
701 protected $error_count = 0;
702
703 /**
704 * The S/MIME certificate file path.
705 *
706 * @var string
707 */
708 protected $sign_cert_file = '';
709
710 /**
711 * The S/MIME key file path.
712 *
713 * @var string
714 */
715 protected $sign_key_file = '';
716
717 /**
718 * The optional S/MIME extra certificates ("CA Chain") file path.
719 *
720 * @var string
721 */
722 protected $sign_extracerts_file = '';
723
724 /**
725 * The S/MIME password for the key.
726 * Used only if the key is encrypted.
727 *
728 * @var string
729 */
730 protected $sign_key_pass = '';
731
732 /**
733 * Whether to throw exceptions for errors.
734 *
735 * @var bool
736 */
737 protected $exceptions = false;
738
739 /**
740 * Unique ID used for message ID and boundaries.
741 *
742 * @var string
743 */
744 protected $uniqueid = '';
745
746 /**
747 * The PHPMailer Version number.
748 *
749 * @var string
750 */
Matthias Andreas Benkarde39c4f82021-01-06 17:59:39 +0100751 const VERSION = '6.2.0';
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +0100752
753 /**
754 * Error severity: message only, continue processing.
755 *
756 * @var int
757 */
758 const STOP_MESSAGE = 0;
759
760 /**
761 * Error severity: message, likely ok to continue processing.
762 *
763 * @var int
764 */
765 const STOP_CONTINUE = 1;
766
767 /**
768 * Error severity: message, plus full stop, critical error reached.
769 *
770 * @var int
771 */
772 const STOP_CRITICAL = 2;
773
774 /**
775 * The SMTP standard CRLF line break.
776 * If you want to change line break format, change static::$LE, not this.
777 */
778 const CRLF = "\r\n";
779
780 /**
781 * "Folding White Space" a white space string used for line folding.
782 */
783 const FWS = ' ';
784
785 /**
786 * SMTP RFC standard line ending; Carriage Return, Line Feed.
787 *
788 * @var string
789 */
790 protected static $LE = self::CRLF;
791
792 /**
793 * The maximum line length supported by mail().
794 *
795 * Background: mail() will sometimes corrupt messages
796 * with headers headers longer than 65 chars, see #818.
797 *
798 * @var int
799 */
800 const MAIL_MAX_LINE_LENGTH = 63;
801
802 /**
803 * The maximum line length allowed by RFC 2822 section 2.1.1.
804 *
805 * @var int
806 */
807 const MAX_LINE_LENGTH = 998;
808
809 /**
810 * The lower maximum line length allowed by RFC 2822 section 2.1.1.
811 * This length does NOT include the line break
812 * 76 means that lines will be 77 or 78 chars depending on whether
813 * the line break format is LF or CRLF; both are valid.
814 *
815 * @var int
816 */
817 const STD_LINE_LENGTH = 76;
818
819 /**
820 * Constructor.
821 *
822 * @param bool $exceptions Should we throw external exceptions?
823 */
824 public function __construct($exceptions = null)
825 {
826 if (null !== $exceptions) {
827 $this->exceptions = (bool) $exceptions;
828 }
829 //Pick an appropriate debug output format automatically
830 $this->Debugoutput = (strpos(PHP_SAPI, 'cli') !== false ? 'echo' : 'html');
831 }
832
833 /**
834 * Destructor.
835 */
836 public function __destruct()
837 {
838 //Close any open SMTP connection nicely
839 $this->smtpClose();
840 }
841
842 /**
843 * Call mail() in a safe_mode-aware fashion.
844 * Also, unless sendmail_path points to sendmail (or something that
845 * claims to be sendmail), don't pass params (not a perfect fix,
846 * but it will do).
847 *
848 * @param string $to To
849 * @param string $subject Subject
850 * @param string $body Message Body
851 * @param string $header Additional Header(s)
852 * @param string|null $params Params
853 *
854 * @return bool
855 */
856 private function mailPassthru($to, $subject, $body, $header, $params)
857 {
858 //Check overloading of mail function to avoid double-encoding
859 if (ini_get('mbstring.func_overload') & 1) {
860 $subject = $this->secureHeader($subject);
861 } else {
862 $subject = $this->encodeHeader($this->secureHeader($subject));
863 }
864 //Calling mail() with null params breaks
865 if (!$this->UseSendmailOptions || null === $params) {
866 $result = @mail($to, $subject, $body, $header);
867 } else {
868 $result = @mail($to, $subject, $body, $header, $params);
869 }
870
871 return $result;
872 }
873
874 /**
875 * Output debugging info via user-defined method.
876 * Only generates output if SMTP debug output is enabled (@see SMTP::$do_debug).
877 *
878 * @see PHPMailer::$Debugoutput
879 * @see PHPMailer::$SMTPDebug
880 *
881 * @param string $str
882 */
883 protected function edebug($str)
884 {
885 if ($this->SMTPDebug <= 0) {
886 return;
887 }
888 //Is this a PSR-3 logger?
889 if ($this->Debugoutput instanceof \Psr\Log\LoggerInterface) {
890 $this->Debugoutput->debug($str);
891
892 return;
893 }
894 //Avoid clash with built-in function names
895 if (is_callable($this->Debugoutput) && !in_array($this->Debugoutput, ['error_log', 'html', 'echo'])) {
896 call_user_func($this->Debugoutput, $str, $this->SMTPDebug);
897
898 return;
899 }
900 switch ($this->Debugoutput) {
901 case 'error_log':
902 //Don't output, just log
Matthias Andreas Benkarde39c4f82021-01-06 17:59:39 +0100903 /** @noinspection ForgottenDebugOutputInspection */
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +0100904 error_log($str);
905 break;
906 case 'html':
907 //Cleans up output a bit for a better looking, HTML-safe output
908 echo htmlentities(
909 preg_replace('/[\r\n]+/', '', $str),
910 ENT_QUOTES,
911 'UTF-8'
912 ), "<br>\n";
913 break;
914 case 'echo':
915 default:
916 //Normalize line breaks
917 $str = preg_replace('/\r\n|\r/m', "\n", $str);
918 echo gmdate('Y-m-d H:i:s'),
919 "\t",
920 //Trim trailing space
921 trim(
922 //Indent for readability, except for trailing break
923 str_replace(
924 "\n",
925 "\n \t ",
926 trim($str)
927 )
928 ),
929 "\n";
930 }
931 }
932
933 /**
934 * Sets message type to HTML or plain.
935 *
936 * @param bool $isHtml True for HTML mode
937 */
938 public function isHTML($isHtml = true)
939 {
940 if ($isHtml) {
941 $this->ContentType = static::CONTENT_TYPE_TEXT_HTML;
942 } else {
943 $this->ContentType = static::CONTENT_TYPE_PLAINTEXT;
944 }
945 }
946
947 /**
948 * Send messages using SMTP.
949 */
950 public function isSMTP()
951 {
952 $this->Mailer = 'smtp';
953 }
954
955 /**
956 * Send messages using PHP's mail() function.
957 */
958 public function isMail()
959 {
960 $this->Mailer = 'mail';
961 }
962
963 /**
964 * Send messages using $Sendmail.
965 */
966 public function isSendmail()
967 {
968 $ini_sendmail_path = ini_get('sendmail_path');
969
970 if (false === stripos($ini_sendmail_path, 'sendmail')) {
971 $this->Sendmail = '/usr/sbin/sendmail';
972 } else {
973 $this->Sendmail = $ini_sendmail_path;
974 }
975 $this->Mailer = 'sendmail';
976 }
977
978 /**
979 * Send messages using qmail.
980 */
981 public function isQmail()
982 {
983 $ini_sendmail_path = ini_get('sendmail_path');
984
985 if (false === stripos($ini_sendmail_path, 'qmail')) {
986 $this->Sendmail = '/var/qmail/bin/qmail-inject';
987 } else {
988 $this->Sendmail = $ini_sendmail_path;
989 }
990 $this->Mailer = 'qmail';
991 }
992
993 /**
994 * Add a "To" address.
995 *
996 * @param string $address The email address to send to
997 * @param string $name
998 *
999 * @throws Exception
1000 *
1001 * @return bool true on success, false if address already used or invalid in some way
1002 */
1003 public function addAddress($address, $name = '')
1004 {
1005 return $this->addOrEnqueueAnAddress('to', $address, $name);
1006 }
1007
1008 /**
1009 * Add a "CC" address.
1010 *
1011 * @param string $address The email address to send to
1012 * @param string $name
1013 *
1014 * @throws Exception
1015 *
1016 * @return bool true on success, false if address already used or invalid in some way
1017 */
1018 public function addCC($address, $name = '')
1019 {
1020 return $this->addOrEnqueueAnAddress('cc', $address, $name);
1021 }
1022
1023 /**
1024 * Add a "BCC" address.
1025 *
1026 * @param string $address The email address to send to
1027 * @param string $name
1028 *
1029 * @throws Exception
1030 *
1031 * @return bool true on success, false if address already used or invalid in some way
1032 */
1033 public function addBCC($address, $name = '')
1034 {
1035 return $this->addOrEnqueueAnAddress('bcc', $address, $name);
1036 }
1037
1038 /**
1039 * Add a "Reply-To" address.
1040 *
1041 * @param string $address The email address to reply to
1042 * @param string $name
1043 *
1044 * @throws Exception
1045 *
1046 * @return bool true on success, false if address already used or invalid in some way
1047 */
1048 public function addReplyTo($address, $name = '')
1049 {
1050 return $this->addOrEnqueueAnAddress('Reply-To', $address, $name);
1051 }
1052
1053 /**
1054 * Add an address to one of the recipient arrays or to the ReplyTo array. Because PHPMailer
1055 * can't validate addresses with an IDN without knowing the PHPMailer::$CharSet (that can still
1056 * be modified after calling this function), addition of such addresses is delayed until send().
1057 * Addresses that have been added already return false, but do not throw exceptions.
1058 *
1059 * @param string $kind One of 'to', 'cc', 'bcc', or 'ReplyTo'
1060 * @param string $address The email address to send, resp. to reply to
1061 * @param string $name
1062 *
1063 * @throws Exception
1064 *
1065 * @return bool true on success, false if address already used or invalid in some way
1066 */
1067 protected function addOrEnqueueAnAddress($kind, $address, $name)
1068 {
1069 $address = trim($address);
1070 $name = trim(preg_replace('/[\r\n]+/', '', $name)); //Strip breaks and trim
1071 $pos = strrpos($address, '@');
1072 if (false === $pos) {
1073 // At-sign is missing.
1074 $error_message = sprintf(
1075 '%s (%s): %s',
1076 $this->lang('invalid_address'),
1077 $kind,
1078 $address
1079 );
1080 $this->setError($error_message);
1081 $this->edebug($error_message);
1082 if ($this->exceptions) {
1083 throw new Exception($error_message);
1084 }
1085
1086 return false;
1087 }
1088 $params = [$kind, $address, $name];
1089 // Enqueue addresses with IDN until we know the PHPMailer::$CharSet.
1090 if (static::idnSupported() && $this->has8bitChars(substr($address, ++$pos))) {
1091 if ('Reply-To' !== $kind) {
1092 if (!array_key_exists($address, $this->RecipientsQueue)) {
1093 $this->RecipientsQueue[$address] = $params;
1094
1095 return true;
1096 }
1097 } elseif (!array_key_exists($address, $this->ReplyToQueue)) {
1098 $this->ReplyToQueue[$address] = $params;
1099
1100 return true;
1101 }
1102
1103 return false;
1104 }
1105
1106 // Immediately add standard addresses without IDN.
1107 return call_user_func_array([$this, 'addAnAddress'], $params);
1108 }
1109
1110 /**
1111 * Add an address to one of the recipient arrays or to the ReplyTo array.
1112 * Addresses that have been added already return false, but do not throw exceptions.
1113 *
1114 * @param string $kind One of 'to', 'cc', 'bcc', or 'ReplyTo'
1115 * @param string $address The email address to send, resp. to reply to
1116 * @param string $name
1117 *
1118 * @throws Exception
1119 *
1120 * @return bool true on success, false if address already used or invalid in some way
1121 */
1122 protected function addAnAddress($kind, $address, $name = '')
1123 {
1124 if (!in_array($kind, ['to', 'cc', 'bcc', 'Reply-To'])) {
1125 $error_message = sprintf(
1126 '%s: %s',
1127 $this->lang('Invalid recipient kind'),
1128 $kind
1129 );
1130 $this->setError($error_message);
1131 $this->edebug($error_message);
1132 if ($this->exceptions) {
1133 throw new Exception($error_message);
1134 }
1135
1136 return false;
1137 }
1138 if (!static::validateAddress($address)) {
1139 $error_message = sprintf(
1140 '%s (%s): %s',
1141 $this->lang('invalid_address'),
1142 $kind,
1143 $address
1144 );
1145 $this->setError($error_message);
1146 $this->edebug($error_message);
1147 if ($this->exceptions) {
1148 throw new Exception($error_message);
1149 }
1150
1151 return false;
1152 }
1153 if ('Reply-To' !== $kind) {
1154 if (!array_key_exists(strtolower($address), $this->all_recipients)) {
1155 $this->{$kind}[] = [$address, $name];
1156 $this->all_recipients[strtolower($address)] = true;
1157
1158 return true;
1159 }
1160 } elseif (!array_key_exists(strtolower($address), $this->ReplyTo)) {
1161 $this->ReplyTo[strtolower($address)] = [$address, $name];
1162
1163 return true;
1164 }
1165
1166 return false;
1167 }
1168
1169 /**
1170 * Parse and validate a string containing one or more RFC822-style comma-separated email addresses
1171 * of the form "display name <address>" into an array of name/address pairs.
1172 * Uses the imap_rfc822_parse_adrlist function if the IMAP extension is available.
1173 * Note that quotes in the name part are removed.
1174 *
1175 * @see http://www.andrew.cmu.edu/user/agreen1/testing/mrbs/web/Mail/RFC822.php A more careful implementation
1176 *
1177 * @param string $addrstr The address list string
1178 * @param bool $useimap Whether to use the IMAP extension to parse the list
1179 *
1180 * @return array
1181 */
1182 public static function parseAddresses($addrstr, $useimap = true)
1183 {
1184 $addresses = [];
1185 if ($useimap && function_exists('imap_rfc822_parse_adrlist')) {
1186 //Use this built-in parser if it's available
1187 $list = imap_rfc822_parse_adrlist($addrstr, '');
1188 foreach ($list as $address) {
Matthias Andreas Benkarde39c4f82021-01-06 17:59:39 +01001189 if (
1190 ('.SYNTAX-ERROR.' !== $address->host) && static::validateAddress(
1191 $address->mailbox . '@' . $address->host
1192 )
1193 ) {
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +01001194 $addresses[] = [
1195 'name' => (property_exists($address, 'personal') ? $address->personal : ''),
1196 'address' => $address->mailbox . '@' . $address->host,
1197 ];
1198 }
1199 }
1200 } else {
1201 //Use this simpler parser
1202 $list = explode(',', $addrstr);
1203 foreach ($list as $address) {
1204 $address = trim($address);
1205 //Is there a separate name part?
1206 if (strpos($address, '<') === false) {
1207 //No separate name, just use the whole thing
1208 if (static::validateAddress($address)) {
1209 $addresses[] = [
1210 'name' => '',
1211 'address' => $address,
1212 ];
1213 }
1214 } else {
1215 list($name, $email) = explode('<', $address);
1216 $email = trim(str_replace('>', '', $email));
1217 if (static::validateAddress($email)) {
1218 $addresses[] = [
1219 'name' => trim(str_replace(['"', "'"], '', $name)),
1220 'address' => $email,
1221 ];
1222 }
1223 }
1224 }
1225 }
1226
1227 return $addresses;
1228 }
1229
1230 /**
1231 * Set the From and FromName properties.
1232 *
1233 * @param string $address
1234 * @param string $name
1235 * @param bool $auto Whether to also set the Sender address, defaults to true
1236 *
1237 * @throws Exception
1238 *
1239 * @return bool
1240 */
1241 public function setFrom($address, $name = '', $auto = true)
1242 {
1243 $address = trim($address);
1244 $name = trim(preg_replace('/[\r\n]+/', '', $name)); //Strip breaks and trim
1245 // Don't validate now addresses with IDN. Will be done in send().
1246 $pos = strrpos($address, '@');
Matthias Andreas Benkarde39c4f82021-01-06 17:59:39 +01001247 if (
1248 (false === $pos)
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +01001249 || ((!$this->has8bitChars(substr($address, ++$pos)) || !static::idnSupported())
1250 && !static::validateAddress($address))
1251 ) {
1252 $error_message = sprintf(
1253 '%s (From): %s',
1254 $this->lang('invalid_address'),
1255 $address
1256 );
1257 $this->setError($error_message);
1258 $this->edebug($error_message);
1259 if ($this->exceptions) {
1260 throw new Exception($error_message);
1261 }
1262
1263 return false;
1264 }
1265 $this->From = $address;
1266 $this->FromName = $name;
1267 if ($auto && empty($this->Sender)) {
1268 $this->Sender = $address;
1269 }
1270
1271 return true;
1272 }
1273
1274 /**
1275 * Return the Message-ID header of the last email.
1276 * Technically this is the value from the last time the headers were created,
1277 * but it's also the message ID of the last sent message except in
1278 * pathological cases.
1279 *
1280 * @return string
1281 */
1282 public function getLastMessageID()
1283 {
1284 return $this->lastMessageID;
1285 }
1286
1287 /**
1288 * Check that a string looks like an email address.
1289 * Validation patterns supported:
1290 * * `auto` Pick best pattern automatically;
1291 * * `pcre8` Use the squiloople.com pattern, requires PCRE > 8.0;
1292 * * `pcre` Use old PCRE implementation;
1293 * * `php` Use PHP built-in FILTER_VALIDATE_EMAIL;
1294 * * `html5` Use the pattern given by the HTML5 spec for 'email' type form input elements.
1295 * * `noregex` Don't use a regex: super fast, really dumb.
1296 * Alternatively you may pass in a callable to inject your own validator, for example:
1297 *
1298 * ```php
1299 * PHPMailer::validateAddress('user@example.com', function($address) {
1300 * return (strpos($address, '@') !== false);
1301 * });
1302 * ```
1303 *
1304 * You can also set the PHPMailer::$validator static to a callable, allowing built-in methods to use your validator.
1305 *
1306 * @param string $address The email address to check
1307 * @param string|callable $patternselect Which pattern to use
1308 *
1309 * @return bool
1310 */
1311 public static function validateAddress($address, $patternselect = null)
1312 {
1313 if (null === $patternselect) {
1314 $patternselect = static::$validator;
1315 }
1316 if (is_callable($patternselect)) {
Matthias Andreas Benkarde39c4f82021-01-06 17:59:39 +01001317 return call_user_func($patternselect, $address);
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +01001318 }
1319 //Reject line breaks in addresses; it's valid RFC5322, but not RFC5321
1320 if (strpos($address, "\n") !== false || strpos($address, "\r") !== false) {
1321 return false;
1322 }
1323 switch ($patternselect) {
1324 case 'pcre': //Kept for BC
1325 case 'pcre8':
1326 /*
1327 * A more complex and more permissive version of the RFC5322 regex on which FILTER_VALIDATE_EMAIL
1328 * is based.
1329 * In addition to the addresses allowed by filter_var, also permits:
1330 * * dotless domains: `a@b`
1331 * * comments: `1234 @ local(blah) .machine .example`
1332 * * quoted elements: `'"test blah"@example.org'`
1333 * * numeric TLDs: `a@b.123`
1334 * * unbracketed IPv4 literals: `a@192.168.0.1`
1335 * * IPv6 literals: 'first.last@[IPv6:a1::]'
1336 * Not all of these will necessarily work for sending!
1337 *
1338 * @see http://squiloople.com/2009/12/20/email-address-validation/
1339 * @copyright 2009-2010 Michael Rushton
1340 * Feel free to use and redistribute this code. But please keep this copyright notice.
1341 */
1342 return (bool) preg_match(
1343 '/^(?!(?>(?1)"?(?>\\\[ -~]|[^"])"?(?1)){255,})(?!(?>(?1)"?(?>\\\[ -~]|[^"])"?(?1)){65,}@)' .
1344 '((?>(?>(?>((?>(?>(?>\x0D\x0A)?[\t ])+|(?>[\t ]*\x0D\x0A)?[\t ]+)?)(\((?>(?2)' .
1345 '(?>[\x01-\x08\x0B\x0C\x0E-\'*-\[\]-\x7F]|\\\[\x00-\x7F]|(?3)))*(?2)\)))+(?2))|(?2))?)' .
1346 '([!#-\'*+\/-9=?^-~-]+|"(?>(?2)(?>[\x01-\x08\x0B\x0C\x0E-!#-\[\]-\x7F]|\\\[\x00-\x7F]))*' .
1347 '(?2)")(?>(?1)\.(?1)(?4))*(?1)@(?!(?1)[a-z0-9-]{64,})(?1)(?>([a-z0-9](?>[a-z0-9-]*[a-z0-9])?)' .
1348 '(?>(?1)\.(?!(?1)[a-z0-9-]{64,})(?1)(?5)){0,126}|\[(?:(?>IPv6:(?>([a-f0-9]{1,4})(?>:(?6)){7}' .
1349 '|(?!(?:.*[a-f0-9][:\]]){8,})((?6)(?>:(?6)){0,6})?::(?7)?))|(?>(?>IPv6:(?>(?6)(?>:(?6)){5}:' .
1350 '|(?!(?:.*[a-f0-9]:){6,})(?8)?::(?>((?6)(?>:(?6)){0,4}):)?))?(25[0-5]|2[0-4][0-9]|1[0-9]{2}' .
1351 '|[1-9]?[0-9])(?>\.(?9)){3}))\])(?1)$/isD',
1352 $address
1353 );
1354 case 'html5':
1355 /*
1356 * This is the pattern used in the HTML5 spec for validation of 'email' type form input elements.
1357 *
Matthias Andreas Benkarde39c4f82021-01-06 17:59:39 +01001358 * @see https://html.spec.whatwg.org/#e-mail-state-(type=email)
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +01001359 */
1360 return (bool) preg_match(
1361 '/^[a-zA-Z0-9.!#$%&\'*+\/=?^_`{|}~-]+@[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}' .
1362 '[a-zA-Z0-9])?(?:\.[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?)*$/sD',
1363 $address
1364 );
1365 case 'php':
1366 default:
1367 return filter_var($address, FILTER_VALIDATE_EMAIL) !== false;
1368 }
1369 }
1370
1371 /**
1372 * Tells whether IDNs (Internationalized Domain Names) are supported or not. This requires the
1373 * `intl` and `mbstring` PHP extensions.
1374 *
1375 * @return bool `true` if required functions for IDN support are present
1376 */
1377 public static function idnSupported()
1378 {
1379 return function_exists('idn_to_ascii') && function_exists('mb_convert_encoding');
1380 }
1381
1382 /**
1383 * Converts IDN in given email address to its ASCII form, also known as punycode, if possible.
1384 * Important: Address must be passed in same encoding as currently set in PHPMailer::$CharSet.
1385 * This function silently returns unmodified address if:
1386 * - No conversion is necessary (i.e. domain name is not an IDN, or is already in ASCII form)
1387 * - Conversion to punycode is impossible (e.g. required PHP functions are not available)
1388 * or fails for any reason (e.g. domain contains characters not allowed in an IDN).
1389 *
1390 * @see PHPMailer::$CharSet
1391 *
1392 * @param string $address The email address to convert
1393 *
1394 * @return string The encoded address in ASCII form
1395 */
1396 public function punyencodeAddress($address)
1397 {
1398 // Verify we have required functions, CharSet, and at-sign.
1399 $pos = strrpos($address, '@');
Matthias Andreas Benkarde39c4f82021-01-06 17:59:39 +01001400 if (
1401 !empty($this->CharSet) &&
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +01001402 false !== $pos &&
1403 static::idnSupported()
1404 ) {
1405 $domain = substr($address, ++$pos);
1406 // Verify CharSet string is a valid one, and domain properly encoded in this CharSet.
1407 if ($this->has8bitChars($domain) && @mb_check_encoding($domain, $this->CharSet)) {
1408 $domain = mb_convert_encoding($domain, 'UTF-8', $this->CharSet);
1409 //Ignore IDE complaints about this line - method signature changed in PHP 5.4
1410 $errorcode = 0;
1411 if (defined('INTL_IDNA_VARIANT_UTS46')) {
1412 $punycode = idn_to_ascii($domain, $errorcode, INTL_IDNA_VARIANT_UTS46);
1413 } elseif (defined('INTL_IDNA_VARIANT_2003')) {
1414 $punycode = idn_to_ascii($domain, $errorcode, INTL_IDNA_VARIANT_2003);
1415 } else {
1416 $punycode = idn_to_ascii($domain, $errorcode);
1417 }
1418 if (false !== $punycode) {
1419 return substr($address, 0, $pos) . $punycode;
1420 }
1421 }
1422 }
1423
1424 return $address;
1425 }
1426
1427 /**
1428 * Create a message and send it.
1429 * Uses the sending method specified by $Mailer.
1430 *
1431 * @throws Exception
1432 *
1433 * @return bool false on error - See the ErrorInfo property for details of the error
1434 */
1435 public function send()
1436 {
1437 try {
1438 if (!$this->preSend()) {
1439 return false;
1440 }
1441
1442 return $this->postSend();
1443 } catch (Exception $exc) {
1444 $this->mailHeader = '';
1445 $this->setError($exc->getMessage());
1446 if ($this->exceptions) {
1447 throw $exc;
1448 }
1449
1450 return false;
1451 }
1452 }
1453
1454 /**
1455 * Prepare a message for sending.
1456 *
1457 * @throws Exception
1458 *
1459 * @return bool
1460 */
1461 public function preSend()
1462 {
Matthias Andreas Benkarde39c4f82021-01-06 17:59:39 +01001463 if (
1464 'smtp' === $this->Mailer
1465 || ('mail' === $this->Mailer && (PHP_VERSION_ID >= 80000 || stripos(PHP_OS, 'WIN') === 0))
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +01001466 ) {
1467 //SMTP mandates RFC-compliant line endings
1468 //and it's also used with mail() on Windows
1469 static::setLE(self::CRLF);
1470 } else {
1471 //Maintain backward compatibility with legacy Linux command line mailers
1472 static::setLE(PHP_EOL);
1473 }
1474 //Check for buggy PHP versions that add a header with an incorrect line break
Matthias Andreas Benkarde39c4f82021-01-06 17:59:39 +01001475 if (
1476 'mail' === $this->Mailer
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +01001477 && ((PHP_VERSION_ID >= 70000 && PHP_VERSION_ID < 70017)
1478 || (PHP_VERSION_ID >= 70100 && PHP_VERSION_ID < 70103))
1479 && ini_get('mail.add_x_header') === '1'
1480 && stripos(PHP_OS, 'WIN') === 0
1481 ) {
1482 trigger_error(
1483 'Your version of PHP is affected by a bug that may result in corrupted messages.' .
1484 ' To fix it, switch to sending using SMTP, disable the mail.add_x_header option in' .
1485 ' your php.ini, switch to MacOS or Linux, or upgrade your PHP to version 7.0.17+ or 7.1.3+.',
1486 E_USER_WARNING
1487 );
1488 }
1489
1490 try {
1491 $this->error_count = 0; // Reset errors
1492 $this->mailHeader = '';
1493
1494 // Dequeue recipient and Reply-To addresses with IDN
1495 foreach (array_merge($this->RecipientsQueue, $this->ReplyToQueue) as $params) {
1496 $params[1] = $this->punyencodeAddress($params[1]);
1497 call_user_func_array([$this, 'addAnAddress'], $params);
1498 }
1499 if (count($this->to) + count($this->cc) + count($this->bcc) < 1) {
1500 throw new Exception($this->lang('provide_address'), self::STOP_CRITICAL);
1501 }
1502
1503 // Validate From, Sender, and ConfirmReadingTo addresses
1504 foreach (['From', 'Sender', 'ConfirmReadingTo'] as $address_kind) {
1505 $this->$address_kind = trim($this->$address_kind);
1506 if (empty($this->$address_kind)) {
1507 continue;
1508 }
1509 $this->$address_kind = $this->punyencodeAddress($this->$address_kind);
1510 if (!static::validateAddress($this->$address_kind)) {
1511 $error_message = sprintf(
1512 '%s (%s): %s',
1513 $this->lang('invalid_address'),
1514 $address_kind,
1515 $this->$address_kind
1516 );
1517 $this->setError($error_message);
1518 $this->edebug($error_message);
1519 if ($this->exceptions) {
1520 throw new Exception($error_message);
1521 }
1522
1523 return false;
1524 }
1525 }
1526
1527 // Set whether the message is multipart/alternative
1528 if ($this->alternativeExists()) {
1529 $this->ContentType = static::CONTENT_TYPE_MULTIPART_ALTERNATIVE;
1530 }
1531
1532 $this->setMessageType();
1533 // Refuse to send an empty message unless we are specifically allowing it
1534 if (!$this->AllowEmpty && empty($this->Body)) {
1535 throw new Exception($this->lang('empty_message'), self::STOP_CRITICAL);
1536 }
1537
1538 //Trim subject consistently
1539 $this->Subject = trim($this->Subject);
1540 // Create body before headers in case body makes changes to headers (e.g. altering transfer encoding)
1541 $this->MIMEHeader = '';
1542 $this->MIMEBody = $this->createBody();
1543 // createBody may have added some headers, so retain them
1544 $tempheaders = $this->MIMEHeader;
1545 $this->MIMEHeader = $this->createHeader();
1546 $this->MIMEHeader .= $tempheaders;
1547
1548 // To capture the complete message when using mail(), create
1549 // an extra header list which createHeader() doesn't fold in
1550 if ('mail' === $this->Mailer) {
1551 if (count($this->to) > 0) {
1552 $this->mailHeader .= $this->addrAppend('To', $this->to);
1553 } else {
1554 $this->mailHeader .= $this->headerLine('To', 'undisclosed-recipients:;');
1555 }
1556 $this->mailHeader .= $this->headerLine(
1557 'Subject',
1558 $this->encodeHeader($this->secureHeader($this->Subject))
1559 );
1560 }
1561
1562 // Sign with DKIM if enabled
Matthias Andreas Benkarde39c4f82021-01-06 17:59:39 +01001563 if (
1564 !empty($this->DKIM_domain)
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +01001565 && !empty($this->DKIM_selector)
1566 && (!empty($this->DKIM_private_string)
1567 || (!empty($this->DKIM_private)
1568 && static::isPermittedPath($this->DKIM_private)
1569 && file_exists($this->DKIM_private)
1570 )
1571 )
1572 ) {
1573 $header_dkim = $this->DKIM_Add(
1574 $this->MIMEHeader . $this->mailHeader,
1575 $this->encodeHeader($this->secureHeader($this->Subject)),
1576 $this->MIMEBody
1577 );
1578 $this->MIMEHeader = static::stripTrailingWSP($this->MIMEHeader) . static::$LE .
1579 static::normalizeBreaks($header_dkim) . static::$LE;
1580 }
1581
1582 return true;
1583 } catch (Exception $exc) {
1584 $this->setError($exc->getMessage());
1585 if ($this->exceptions) {
1586 throw $exc;
1587 }
1588
1589 return false;
1590 }
1591 }
1592
1593 /**
1594 * Actually send a message via the selected mechanism.
1595 *
1596 * @throws Exception
1597 *
1598 * @return bool
1599 */
1600 public function postSend()
1601 {
1602 try {
1603 // Choose the mailer and send through it
1604 switch ($this->Mailer) {
1605 case 'sendmail':
1606 case 'qmail':
1607 return $this->sendmailSend($this->MIMEHeader, $this->MIMEBody);
1608 case 'smtp':
1609 return $this->smtpSend($this->MIMEHeader, $this->MIMEBody);
1610 case 'mail':
1611 return $this->mailSend($this->MIMEHeader, $this->MIMEBody);
1612 default:
1613 $sendMethod = $this->Mailer . 'Send';
1614 if (method_exists($this, $sendMethod)) {
1615 return $this->$sendMethod($this->MIMEHeader, $this->MIMEBody);
1616 }
1617
1618 return $this->mailSend($this->MIMEHeader, $this->MIMEBody);
1619 }
1620 } catch (Exception $exc) {
Matthias Andreas Benkarde39c4f82021-01-06 17:59:39 +01001621 if ($this->Mailer === 'smtp' && $this->SMTPKeepAlive == true) {
1622 $this->smtp->reset();
1623 }
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +01001624 $this->setError($exc->getMessage());
1625 $this->edebug($exc->getMessage());
1626 if ($this->exceptions) {
1627 throw $exc;
1628 }
1629 }
1630
1631 return false;
1632 }
1633
1634 /**
1635 * Send mail using the $Sendmail program.
1636 *
1637 * @see PHPMailer::$Sendmail
1638 *
1639 * @param string $header The message headers
1640 * @param string $body The message body
1641 *
1642 * @throws Exception
1643 *
1644 * @return bool
1645 */
1646 protected function sendmailSend($header, $body)
1647 {
1648 $header = static::stripTrailingWSP($header) . static::$LE . static::$LE;
1649
1650 // CVE-2016-10033, CVE-2016-10045: Don't pass -f if characters will be escaped.
1651 if (!empty($this->Sender) && self::isShellSafe($this->Sender)) {
1652 if ('qmail' === $this->Mailer) {
1653 $sendmailFmt = '%s -f%s';
1654 } else {
1655 $sendmailFmt = '%s -oi -f%s -t';
1656 }
1657 } elseif ('qmail' === $this->Mailer) {
1658 $sendmailFmt = '%s';
1659 } else {
1660 $sendmailFmt = '%s -oi -t';
1661 }
1662
1663 $sendmail = sprintf($sendmailFmt, escapeshellcmd($this->Sendmail), $this->Sender);
1664
1665 if ($this->SingleTo) {
1666 foreach ($this->SingleToArray as $toAddr) {
1667 $mail = @popen($sendmail, 'w');
1668 if (!$mail) {
1669 throw new Exception($this->lang('execute') . $this->Sendmail, self::STOP_CRITICAL);
1670 }
1671 fwrite($mail, 'To: ' . $toAddr . "\n");
1672 fwrite($mail, $header);
1673 fwrite($mail, $body);
1674 $result = pclose($mail);
1675 $this->doCallback(
1676 ($result === 0),
1677 [$toAddr],
1678 $this->cc,
1679 $this->bcc,
1680 $this->Subject,
1681 $body,
1682 $this->From,
1683 []
1684 );
1685 if (0 !== $result) {
1686 throw new Exception($this->lang('execute') . $this->Sendmail, self::STOP_CRITICAL);
1687 }
1688 }
1689 } else {
1690 $mail = @popen($sendmail, 'w');
1691 if (!$mail) {
1692 throw new Exception($this->lang('execute') . $this->Sendmail, self::STOP_CRITICAL);
1693 }
1694 fwrite($mail, $header);
1695 fwrite($mail, $body);
1696 $result = pclose($mail);
1697 $this->doCallback(
1698 ($result === 0),
1699 $this->to,
1700 $this->cc,
1701 $this->bcc,
1702 $this->Subject,
1703 $body,
1704 $this->From,
1705 []
1706 );
1707 if (0 !== $result) {
1708 throw new Exception($this->lang('execute') . $this->Sendmail, self::STOP_CRITICAL);
1709 }
1710 }
1711
1712 return true;
1713 }
1714
1715 /**
1716 * Fix CVE-2016-10033 and CVE-2016-10045 by disallowing potentially unsafe shell characters.
1717 * Note that escapeshellarg and escapeshellcmd are inadequate for our purposes, especially on Windows.
1718 *
1719 * @see https://github.com/PHPMailer/PHPMailer/issues/924 CVE-2016-10045 bug report
1720 *
1721 * @param string $string The string to be validated
1722 *
1723 * @return bool
1724 */
1725 protected static function isShellSafe($string)
1726 {
1727 // Future-proof
Matthias Andreas Benkarde39c4f82021-01-06 17:59:39 +01001728 if (
1729 escapeshellcmd($string) !== $string
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +01001730 || !in_array(escapeshellarg($string), ["'$string'", "\"$string\""])
1731 ) {
1732 return false;
1733 }
1734
1735 $length = strlen($string);
1736
1737 for ($i = 0; $i < $length; ++$i) {
1738 $c = $string[$i];
1739
1740 // All other characters have a special meaning in at least one common shell, including = and +.
1741 // Full stop (.) has a special meaning in cmd.exe, but its impact should be negligible here.
1742 // Note that this does permit non-Latin alphanumeric characters based on the current locale.
1743 if (!ctype_alnum($c) && strpos('@_-.', $c) === false) {
1744 return false;
1745 }
1746 }
1747
1748 return true;
1749 }
1750
1751 /**
1752 * Check whether a file path is of a permitted type.
1753 * Used to reject URLs and phar files from functions that access local file paths,
1754 * such as addAttachment.
1755 *
1756 * @param string $path A relative or absolute path to a file
1757 *
1758 * @return bool
1759 */
1760 protected static function isPermittedPath($path)
1761 {
1762 return !preg_match('#^[a-z]+://#i', $path);
1763 }
1764
1765 /**
Matthias Andreas Benkarde39c4f82021-01-06 17:59:39 +01001766 * Check whether a file path is safe, accessible, and readable.
1767 *
1768 * @param string $path A relative or absolute path to a file
1769 *
1770 * @return bool
1771 */
1772 protected static function fileIsAccessible($path)
1773 {
1774 $readable = file_exists($path);
1775 //If not a UNC path (expected to start with \\), check read permission, see #2069
1776 if (strpos($path, '\\\\') !== 0) {
1777 $readable = $readable && is_readable($path);
1778 }
1779 return static::isPermittedPath($path) && $readable;
1780 }
1781
1782 /**
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +01001783 * Send mail using the PHP mail() function.
1784 *
1785 * @see http://www.php.net/manual/en/book.mail.php
1786 *
1787 * @param string $header The message headers
1788 * @param string $body The message body
1789 *
1790 * @throws Exception
1791 *
1792 * @return bool
1793 */
1794 protected function mailSend($header, $body)
1795 {
1796 $header = static::stripTrailingWSP($header) . static::$LE . static::$LE;
1797
1798 $toArr = [];
1799 foreach ($this->to as $toaddr) {
1800 $toArr[] = $this->addrFormat($toaddr);
1801 }
1802 $to = implode(', ', $toArr);
1803
1804 $params = null;
1805 //This sets the SMTP envelope sender which gets turned into a return-path header by the receiver
1806 //A space after `-f` is optional, but there is a long history of its presence
1807 //causing problems, so we don't use one
1808 //Exim docs: http://www.exim.org/exim-html-current/doc/html/spec_html/ch-the_exim_command_line.html
1809 //Sendmail docs: http://www.sendmail.org/~ca/email/man/sendmail.html
1810 //Qmail docs: http://www.qmail.org/man/man8/qmail-inject.html
1811 //Example problem: https://www.drupal.org/node/1057954
1812 // CVE-2016-10033, CVE-2016-10045: Don't pass -f if characters will be escaped.
1813 if (!empty($this->Sender) && static::validateAddress($this->Sender) && self::isShellSafe($this->Sender)) {
1814 $params = sprintf('-f%s', $this->Sender);
1815 }
1816 if (!empty($this->Sender) && static::validateAddress($this->Sender)) {
1817 $old_from = ini_get('sendmail_from');
1818 ini_set('sendmail_from', $this->Sender);
1819 }
1820 $result = false;
1821 if ($this->SingleTo && count($toArr) > 1) {
1822 foreach ($toArr as $toAddr) {
1823 $result = $this->mailPassthru($toAddr, $this->Subject, $body, $header, $params);
1824 $this->doCallback($result, [$toAddr], $this->cc, $this->bcc, $this->Subject, $body, $this->From, []);
1825 }
1826 } else {
1827 $result = $this->mailPassthru($to, $this->Subject, $body, $header, $params);
1828 $this->doCallback($result, $this->to, $this->cc, $this->bcc, $this->Subject, $body, $this->From, []);
1829 }
1830 if (isset($old_from)) {
1831 ini_set('sendmail_from', $old_from);
1832 }
1833 if (!$result) {
1834 throw new Exception($this->lang('instantiate'), self::STOP_CRITICAL);
1835 }
1836
1837 return true;
1838 }
1839
1840 /**
1841 * Get an instance to use for SMTP operations.
1842 * Override this function to load your own SMTP implementation,
1843 * or set one with setSMTPInstance.
1844 *
1845 * @return SMTP
1846 */
1847 public function getSMTPInstance()
1848 {
1849 if (!is_object($this->smtp)) {
1850 $this->smtp = new SMTP();
1851 }
1852
1853 return $this->smtp;
1854 }
1855
1856 /**
1857 * Provide an instance to use for SMTP operations.
1858 *
1859 * @return SMTP
1860 */
1861 public function setSMTPInstance(SMTP $smtp)
1862 {
1863 $this->smtp = $smtp;
1864
1865 return $this->smtp;
1866 }
1867
1868 /**
1869 * Send mail via SMTP.
1870 * Returns false if there is a bad MAIL FROM, RCPT, or DATA input.
1871 *
1872 * @see PHPMailer::setSMTPInstance() to use a different class.
1873 *
1874 * @uses \PHPMailer\PHPMailer\SMTP
1875 *
1876 * @param string $header The message headers
1877 * @param string $body The message body
1878 *
1879 * @throws Exception
1880 *
1881 * @return bool
1882 */
1883 protected function smtpSend($header, $body)
1884 {
1885 $header = static::stripTrailingWSP($header) . static::$LE . static::$LE;
1886 $bad_rcpt = [];
1887 if (!$this->smtpConnect($this->SMTPOptions)) {
1888 throw new Exception($this->lang('smtp_connect_failed'), self::STOP_CRITICAL);
1889 }
1890 //Sender already validated in preSend()
1891 if ('' === $this->Sender) {
1892 $smtp_from = $this->From;
1893 } else {
1894 $smtp_from = $this->Sender;
1895 }
1896 if (!$this->smtp->mail($smtp_from)) {
1897 $this->setError($this->lang('from_failed') . $smtp_from . ' : ' . implode(',', $this->smtp->getError()));
1898 throw new Exception($this->ErrorInfo, self::STOP_CRITICAL);
1899 }
1900
1901 $callbacks = [];
1902 // Attempt to send to all recipients
1903 foreach ([$this->to, $this->cc, $this->bcc] as $togroup) {
1904 foreach ($togroup as $to) {
1905 if (!$this->smtp->recipient($to[0], $this->dsn)) {
1906 $error = $this->smtp->getError();
1907 $bad_rcpt[] = ['to' => $to[0], 'error' => $error['detail']];
1908 $isSent = false;
1909 } else {
1910 $isSent = true;
1911 }
1912
Matthias Andreas Benkarde39c4f82021-01-06 17:59:39 +01001913 $callbacks[] = ['issent' => $isSent, 'to' => $to[0]];
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +01001914 }
1915 }
1916
1917 // Only send the DATA command if we have viable recipients
1918 if ((count($this->all_recipients) > count($bad_rcpt)) && !$this->smtp->data($header . $body)) {
1919 throw new Exception($this->lang('data_not_accepted'), self::STOP_CRITICAL);
1920 }
1921
1922 $smtp_transaction_id = $this->smtp->getLastTransactionID();
1923
1924 if ($this->SMTPKeepAlive) {
1925 $this->smtp->reset();
1926 } else {
1927 $this->smtp->quit();
1928 $this->smtp->close();
1929 }
1930
1931 foreach ($callbacks as $cb) {
1932 $this->doCallback(
1933 $cb['issent'],
1934 [$cb['to']],
1935 [],
1936 [],
1937 $this->Subject,
1938 $body,
1939 $this->From,
1940 ['smtp_transaction_id' => $smtp_transaction_id]
1941 );
1942 }
1943
1944 //Create error message for any bad addresses
1945 if (count($bad_rcpt) > 0) {
1946 $errstr = '';
1947 foreach ($bad_rcpt as $bad) {
1948 $errstr .= $bad['to'] . ': ' . $bad['error'];
1949 }
1950 throw new Exception($this->lang('recipients_failed') . $errstr, self::STOP_CONTINUE);
1951 }
1952
1953 return true;
1954 }
1955
1956 /**
1957 * Initiate a connection to an SMTP server.
1958 * Returns false if the operation failed.
1959 *
1960 * @param array $options An array of options compatible with stream_context_create()
1961 *
1962 * @throws Exception
1963 *
1964 * @uses \PHPMailer\PHPMailer\SMTP
1965 *
1966 * @return bool
1967 */
1968 public function smtpConnect($options = null)
1969 {
1970 if (null === $this->smtp) {
1971 $this->smtp = $this->getSMTPInstance();
1972 }
1973
1974 //If no options are provided, use whatever is set in the instance
1975 if (null === $options) {
1976 $options = $this->SMTPOptions;
1977 }
1978
1979 // Already connected?
1980 if ($this->smtp->connected()) {
1981 return true;
1982 }
1983
1984 $this->smtp->setTimeout($this->Timeout);
1985 $this->smtp->setDebugLevel($this->SMTPDebug);
1986 $this->smtp->setDebugOutput($this->Debugoutput);
1987 $this->smtp->setVerp($this->do_verp);
1988 $hosts = explode(';', $this->Host);
1989 $lastexception = null;
1990
1991 foreach ($hosts as $hostentry) {
1992 $hostinfo = [];
Matthias Andreas Benkarde39c4f82021-01-06 17:59:39 +01001993 if (
1994 !preg_match(
1995 '/^(?:(ssl|tls):\/\/)?(.+?)(?::(\d+))?$/',
1996 trim($hostentry),
1997 $hostinfo
1998 )
1999 ) {
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +01002000 $this->edebug($this->lang('invalid_hostentry') . ' ' . trim($hostentry));
2001 // Not a valid host entry
2002 continue;
2003 }
2004 // $hostinfo[1]: optional ssl or tls prefix
2005 // $hostinfo[2]: the hostname
2006 // $hostinfo[3]: optional port number
2007 // The host string prefix can temporarily override the current setting for SMTPSecure
2008 // If it's not specified, the default value is used
2009
2010 //Check the host name is a valid name or IP address before trying to use it
2011 if (!static::isValidHost($hostinfo[2])) {
2012 $this->edebug($this->lang('invalid_host') . ' ' . $hostinfo[2]);
2013 continue;
2014 }
2015 $prefix = '';
2016 $secure = $this->SMTPSecure;
2017 $tls = (static::ENCRYPTION_STARTTLS === $this->SMTPSecure);
2018 if ('ssl' === $hostinfo[1] || ('' === $hostinfo[1] && static::ENCRYPTION_SMTPS === $this->SMTPSecure)) {
2019 $prefix = 'ssl://';
2020 $tls = false; // Can't have SSL and TLS at the same time
2021 $secure = static::ENCRYPTION_SMTPS;
2022 } elseif ('tls' === $hostinfo[1]) {
2023 $tls = true;
2024 // tls doesn't use a prefix
2025 $secure = static::ENCRYPTION_STARTTLS;
2026 }
2027 //Do we need the OpenSSL extension?
2028 $sslext = defined('OPENSSL_ALGO_SHA256');
2029 if (static::ENCRYPTION_STARTTLS === $secure || static::ENCRYPTION_SMTPS === $secure) {
2030 //Check for an OpenSSL constant rather than using extension_loaded, which is sometimes disabled
2031 if (!$sslext) {
2032 throw new Exception($this->lang('extension_missing') . 'openssl', self::STOP_CRITICAL);
2033 }
2034 }
2035 $host = $hostinfo[2];
2036 $port = $this->Port;
Matthias Andreas Benkarde39c4f82021-01-06 17:59:39 +01002037 if (
2038 array_key_exists(3, $hostinfo) &&
2039 is_numeric($hostinfo[3]) &&
2040 $hostinfo[3] > 0 &&
2041 $hostinfo[3] < 65536
2042 ) {
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +01002043 $port = (int) $hostinfo[3];
2044 }
2045 if ($this->smtp->connect($prefix . $host, $port, $this->Timeout, $options)) {
2046 try {
2047 if ($this->Helo) {
2048 $hello = $this->Helo;
2049 } else {
2050 $hello = $this->serverHostname();
2051 }
2052 $this->smtp->hello($hello);
2053 //Automatically enable TLS encryption if:
2054 // * it's not disabled
2055 // * we have openssl extension
2056 // * we are not already using SSL
2057 // * the server offers STARTTLS
2058 if ($this->SMTPAutoTLS && $sslext && 'ssl' !== $secure && $this->smtp->getServerExt('STARTTLS')) {
2059 $tls = true;
2060 }
2061 if ($tls) {
2062 if (!$this->smtp->startTLS()) {
2063 throw new Exception($this->lang('connect_host'));
2064 }
2065 // We must resend EHLO after TLS negotiation
2066 $this->smtp->hello($hello);
2067 }
Matthias Andreas Benkarde39c4f82021-01-06 17:59:39 +01002068 if (
2069 $this->SMTPAuth && !$this->smtp->authenticate(
2070 $this->Username,
2071 $this->Password,
2072 $this->AuthType,
2073 $this->oauth
2074 )
2075 ) {
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +01002076 throw new Exception($this->lang('authenticate'));
2077 }
2078
2079 return true;
2080 } catch (Exception $exc) {
2081 $lastexception = $exc;
2082 $this->edebug($exc->getMessage());
2083 // We must have connected, but then failed TLS or Auth, so close connection nicely
2084 $this->smtp->quit();
2085 }
2086 }
2087 }
2088 // If we get here, all connection attempts have failed, so close connection hard
2089 $this->smtp->close();
2090 // As we've caught all exceptions, just report whatever the last one was
2091 if ($this->exceptions && null !== $lastexception) {
2092 throw $lastexception;
2093 }
2094
2095 return false;
2096 }
2097
2098 /**
2099 * Close the active SMTP session if one exists.
2100 */
2101 public function smtpClose()
2102 {
2103 if ((null !== $this->smtp) && $this->smtp->connected()) {
2104 $this->smtp->quit();
2105 $this->smtp->close();
2106 }
2107 }
2108
2109 /**
2110 * Set the language for error messages.
2111 * Returns false if it cannot load the language file.
2112 * The default language is English.
2113 *
2114 * @param string $langcode ISO 639-1 2-character language code (e.g. French is "fr")
2115 * @param string $lang_path Path to the language file directory, with trailing separator (slash)
2116 *
2117 * @return bool
2118 */
2119 public function setLanguage($langcode = 'en', $lang_path = '')
2120 {
2121 // Backwards compatibility for renamed language codes
2122 $renamed_langcodes = [
2123 'br' => 'pt_br',
2124 'cz' => 'cs',
2125 'dk' => 'da',
2126 'no' => 'nb',
2127 'se' => 'sv',
2128 'rs' => 'sr',
2129 'tg' => 'tl',
2130 'am' => 'hy',
2131 ];
2132
Matthias Andreas Benkarde39c4f82021-01-06 17:59:39 +01002133 if (array_key_exists($langcode, $renamed_langcodes)) {
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +01002134 $langcode = $renamed_langcodes[$langcode];
2135 }
2136
2137 // Define full set of translatable strings in English
2138 $PHPMAILER_LANG = [
2139 'authenticate' => 'SMTP Error: Could not authenticate.',
2140 'connect_host' => 'SMTP Error: Could not connect to SMTP host.',
2141 'data_not_accepted' => 'SMTP Error: data not accepted.',
2142 'empty_message' => 'Message body empty',
2143 'encoding' => 'Unknown encoding: ',
2144 'execute' => 'Could not execute: ',
2145 'file_access' => 'Could not access file: ',
2146 'file_open' => 'File Error: Could not open file: ',
2147 'from_failed' => 'The following From address failed: ',
2148 'instantiate' => 'Could not instantiate mail function.',
2149 'invalid_address' => 'Invalid address: ',
2150 'invalid_hostentry' => 'Invalid hostentry: ',
2151 'invalid_host' => 'Invalid host: ',
2152 'mailer_not_supported' => ' mailer is not supported.',
2153 'provide_address' => 'You must provide at least one recipient email address.',
2154 'recipients_failed' => 'SMTP Error: The following recipients failed: ',
2155 'signing' => 'Signing Error: ',
2156 'smtp_connect_failed' => 'SMTP connect() failed.',
2157 'smtp_error' => 'SMTP server error: ',
2158 'variable_set' => 'Cannot set or reset variable: ',
2159 'extension_missing' => 'Extension missing: ',
2160 ];
2161 if (empty($lang_path)) {
2162 // Calculate an absolute path so it can work if CWD is not here
2163 $lang_path = dirname(__DIR__) . DIRECTORY_SEPARATOR . 'language' . DIRECTORY_SEPARATOR;
2164 }
2165 //Validate $langcode
2166 if (!preg_match('/^[a-z]{2}(?:_[a-zA-Z]{2})?$/', $langcode)) {
2167 $langcode = 'en';
2168 }
2169 $foundlang = true;
2170 $lang_file = $lang_path . 'phpmailer.lang-' . $langcode . '.php';
2171 // There is no English translation file
2172 if ('en' !== $langcode) {
2173 // Make sure language file path is readable
Matthias Andreas Benkarde39c4f82021-01-06 17:59:39 +01002174 if (!static::fileIsAccessible($lang_file)) {
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +01002175 $foundlang = false;
2176 } else {
2177 // Overwrite language-specific strings.
2178 // This way we'll never have missing translation keys.
2179 $foundlang = include $lang_file;
2180 }
2181 }
2182 $this->language = $PHPMAILER_LANG;
2183
2184 return (bool) $foundlang; // Returns false if language not found
2185 }
2186
2187 /**
2188 * Get the array of strings for the current language.
2189 *
2190 * @return array
2191 */
2192 public function getTranslations()
2193 {
2194 return $this->language;
2195 }
2196
2197 /**
2198 * Create recipient headers.
2199 *
2200 * @param string $type
2201 * @param array $addr An array of recipients,
2202 * where each recipient is a 2-element indexed array with element 0 containing an address
2203 * and element 1 containing a name, like:
2204 * [['joe@example.com', 'Joe User'], ['zoe@example.com', 'Zoe User']]
2205 *
2206 * @return string
2207 */
2208 public function addrAppend($type, $addr)
2209 {
2210 $addresses = [];
2211 foreach ($addr as $address) {
2212 $addresses[] = $this->addrFormat($address);
2213 }
2214
2215 return $type . ': ' . implode(', ', $addresses) . static::$LE;
2216 }
2217
2218 /**
2219 * Format an address for use in a message header.
2220 *
2221 * @param array $addr A 2-element indexed array, element 0 containing an address, element 1 containing a name like
2222 * ['joe@example.com', 'Joe User']
2223 *
2224 * @return string
2225 */
2226 public function addrFormat($addr)
2227 {
2228 if (empty($addr[1])) { // No name provided
2229 return $this->secureHeader($addr[0]);
2230 }
2231
2232 return $this->encodeHeader($this->secureHeader($addr[1]), 'phrase') .
2233 ' <' . $this->secureHeader($addr[0]) . '>';
2234 }
2235
2236 /**
2237 * Word-wrap message.
2238 * For use with mailers that do not automatically perform wrapping
2239 * and for quoted-printable encoded messages.
2240 * Original written by philippe.
2241 *
2242 * @param string $message The message to wrap
2243 * @param int $length The line length to wrap to
2244 * @param bool $qp_mode Whether to run in Quoted-Printable mode
2245 *
2246 * @return string
2247 */
2248 public function wrapText($message, $length, $qp_mode = false)
2249 {
2250 if ($qp_mode) {
2251 $soft_break = sprintf(' =%s', static::$LE);
2252 } else {
2253 $soft_break = static::$LE;
2254 }
2255 // If utf-8 encoding is used, we will need to make sure we don't
2256 // split multibyte characters when we wrap
2257 $is_utf8 = static::CHARSET_UTF8 === strtolower($this->CharSet);
2258 $lelen = strlen(static::$LE);
2259 $crlflen = strlen(static::$LE);
2260
2261 $message = static::normalizeBreaks($message);
2262 //Remove a trailing line break
2263 if (substr($message, -$lelen) === static::$LE) {
2264 $message = substr($message, 0, -$lelen);
2265 }
2266
2267 //Split message into lines
2268 $lines = explode(static::$LE, $message);
2269 //Message will be rebuilt in here
2270 $message = '';
2271 foreach ($lines as $line) {
2272 $words = explode(' ', $line);
2273 $buf = '';
2274 $firstword = true;
2275 foreach ($words as $word) {
2276 if ($qp_mode && (strlen($word) > $length)) {
2277 $space_left = $length - strlen($buf) - $crlflen;
2278 if (!$firstword) {
2279 if ($space_left > 20) {
2280 $len = $space_left;
2281 if ($is_utf8) {
2282 $len = $this->utf8CharBoundary($word, $len);
2283 } elseif ('=' === substr($word, $len - 1, 1)) {
2284 --$len;
2285 } elseif ('=' === substr($word, $len - 2, 1)) {
2286 $len -= 2;
2287 }
2288 $part = substr($word, 0, $len);
2289 $word = substr($word, $len);
2290 $buf .= ' ' . $part;
2291 $message .= $buf . sprintf('=%s', static::$LE);
2292 } else {
2293 $message .= $buf . $soft_break;
2294 }
2295 $buf = '';
2296 }
2297 while ($word !== '') {
2298 if ($length <= 0) {
2299 break;
2300 }
2301 $len = $length;
2302 if ($is_utf8) {
2303 $len = $this->utf8CharBoundary($word, $len);
2304 } elseif ('=' === substr($word, $len - 1, 1)) {
2305 --$len;
2306 } elseif ('=' === substr($word, $len - 2, 1)) {
2307 $len -= 2;
2308 }
2309 $part = substr($word, 0, $len);
2310 $word = (string) substr($word, $len);
2311
2312 if ($word !== '') {
2313 $message .= $part . sprintf('=%s', static::$LE);
2314 } else {
2315 $buf = $part;
2316 }
2317 }
2318 } else {
2319 $buf_o = $buf;
2320 if (!$firstword) {
2321 $buf .= ' ';
2322 }
2323 $buf .= $word;
2324
2325 if ('' !== $buf_o && strlen($buf) > $length) {
2326 $message .= $buf_o . $soft_break;
2327 $buf = $word;
2328 }
2329 }
2330 $firstword = false;
2331 }
2332 $message .= $buf . static::$LE;
2333 }
2334
2335 return $message;
2336 }
2337
2338 /**
2339 * Find the last character boundary prior to $maxLength in a utf-8
2340 * quoted-printable encoded string.
2341 * Original written by Colin Brown.
2342 *
2343 * @param string $encodedText utf-8 QP text
2344 * @param int $maxLength Find the last character boundary prior to this length
2345 *
2346 * @return int
2347 */
2348 public function utf8CharBoundary($encodedText, $maxLength)
2349 {
2350 $foundSplitPos = false;
2351 $lookBack = 3;
2352 while (!$foundSplitPos) {
2353 $lastChunk = substr($encodedText, $maxLength - $lookBack, $lookBack);
2354 $encodedCharPos = strpos($lastChunk, '=');
2355 if (false !== $encodedCharPos) {
2356 // Found start of encoded character byte within $lookBack block.
2357 // Check the encoded byte value (the 2 chars after the '=')
2358 $hex = substr($encodedText, $maxLength - $lookBack + $encodedCharPos + 1, 2);
2359 $dec = hexdec($hex);
2360 if ($dec < 128) {
2361 // Single byte character.
2362 // If the encoded char was found at pos 0, it will fit
2363 // otherwise reduce maxLength to start of the encoded char
2364 if ($encodedCharPos > 0) {
2365 $maxLength -= $lookBack - $encodedCharPos;
2366 }
2367 $foundSplitPos = true;
2368 } elseif ($dec >= 192) {
2369 // First byte of a multi byte character
2370 // Reduce maxLength to split at start of character
2371 $maxLength -= $lookBack - $encodedCharPos;
2372 $foundSplitPos = true;
2373 } elseif ($dec < 192) {
2374 // Middle byte of a multi byte character, look further back
2375 $lookBack += 3;
2376 }
2377 } else {
2378 // No encoded character found
2379 $foundSplitPos = true;
2380 }
2381 }
2382
2383 return $maxLength;
2384 }
2385
2386 /**
2387 * Apply word wrapping to the message body.
2388 * Wraps the message body to the number of chars set in the WordWrap property.
2389 * You should only do this to plain-text bodies as wrapping HTML tags may break them.
2390 * This is called automatically by createBody(), so you don't need to call it yourself.
2391 */
2392 public function setWordWrap()
2393 {
2394 if ($this->WordWrap < 1) {
2395 return;
2396 }
2397
2398 switch ($this->message_type) {
2399 case 'alt':
2400 case 'alt_inline':
2401 case 'alt_attach':
2402 case 'alt_inline_attach':
2403 $this->AltBody = $this->wrapText($this->AltBody, $this->WordWrap);
2404 break;
2405 default:
2406 $this->Body = $this->wrapText($this->Body, $this->WordWrap);
2407 break;
2408 }
2409 }
2410
2411 /**
2412 * Assemble message headers.
2413 *
2414 * @return string The assembled headers
2415 */
2416 public function createHeader()
2417 {
2418 $result = '';
2419
2420 $result .= $this->headerLine('Date', '' === $this->MessageDate ? self::rfcDate() : $this->MessageDate);
2421
Matthias Andreas Benkarde39c4f82021-01-06 17:59:39 +01002422 // The To header is created automatically by mail(), so needs to be omitted here
2423 if ('mail' !== $this->Mailer) {
2424 if ($this->SingleTo) {
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +01002425 foreach ($this->to as $toaddr) {
2426 $this->SingleToArray[] = $this->addrFormat($toaddr);
2427 }
Matthias Andreas Benkarde39c4f82021-01-06 17:59:39 +01002428 } elseif (count($this->to) > 0) {
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +01002429 $result .= $this->addrAppend('To', $this->to);
Matthias Andreas Benkarde39c4f82021-01-06 17:59:39 +01002430 } elseif (count($this->cc) === 0) {
2431 $result .= $this->headerLine('To', 'undisclosed-recipients:;');
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +01002432 }
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +01002433 }
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +01002434 $result .= $this->addrAppend('From', [[trim($this->From), $this->FromName]]);
2435
2436 // sendmail and mail() extract Cc from the header before sending
2437 if (count($this->cc) > 0) {
2438 $result .= $this->addrAppend('Cc', $this->cc);
2439 }
2440
2441 // sendmail and mail() extract Bcc from the header before sending
Matthias Andreas Benkarde39c4f82021-01-06 17:59:39 +01002442 if (
2443 (
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +01002444 'sendmail' === $this->Mailer || 'qmail' === $this->Mailer || 'mail' === $this->Mailer
2445 )
2446 && count($this->bcc) > 0
2447 ) {
2448 $result .= $this->addrAppend('Bcc', $this->bcc);
2449 }
2450
2451 if (count($this->ReplyTo) > 0) {
2452 $result .= $this->addrAppend('Reply-To', $this->ReplyTo);
2453 }
2454
2455 // mail() sets the subject itself
2456 if ('mail' !== $this->Mailer) {
2457 $result .= $this->headerLine('Subject', $this->encodeHeader($this->secureHeader($this->Subject)));
2458 }
2459
2460 // Only allow a custom message ID if it conforms to RFC 5322 section 3.6.4
2461 // https://tools.ietf.org/html/rfc5322#section-3.6.4
2462 if ('' !== $this->MessageID && preg_match('/^<.*@.*>$/', $this->MessageID)) {
2463 $this->lastMessageID = $this->MessageID;
2464 } else {
2465 $this->lastMessageID = sprintf('<%s@%s>', $this->uniqueid, $this->serverHostname());
2466 }
2467 $result .= $this->headerLine('Message-ID', $this->lastMessageID);
2468 if (null !== $this->Priority) {
2469 $result .= $this->headerLine('X-Priority', $this->Priority);
2470 }
2471 if ('' === $this->XMailer) {
2472 $result .= $this->headerLine(
2473 'X-Mailer',
2474 'PHPMailer ' . self::VERSION . ' (https://github.com/PHPMailer/PHPMailer)'
2475 );
2476 } else {
2477 $myXmailer = trim($this->XMailer);
2478 if ($myXmailer) {
2479 $result .= $this->headerLine('X-Mailer', $myXmailer);
2480 }
2481 }
2482
2483 if ('' !== $this->ConfirmReadingTo) {
2484 $result .= $this->headerLine('Disposition-Notification-To', '<' . $this->ConfirmReadingTo . '>');
2485 }
2486
2487 // Add custom headers
2488 foreach ($this->CustomHeader as $header) {
2489 $result .= $this->headerLine(
2490 trim($header[0]),
2491 $this->encodeHeader(trim($header[1]))
2492 );
2493 }
2494 if (!$this->sign_key_file) {
2495 $result .= $this->headerLine('MIME-Version', '1.0');
2496 $result .= $this->getMailMIME();
2497 }
2498
2499 return $result;
2500 }
2501
2502 /**
2503 * Get the message MIME type headers.
2504 *
2505 * @return string
2506 */
2507 public function getMailMIME()
2508 {
2509 $result = '';
2510 $ismultipart = true;
2511 switch ($this->message_type) {
2512 case 'inline':
2513 $result .= $this->headerLine('Content-Type', static::CONTENT_TYPE_MULTIPART_RELATED . ';');
2514 $result .= $this->textLine(' boundary="' . $this->boundary[1] . '"');
2515 break;
2516 case 'attach':
2517 case 'inline_attach':
2518 case 'alt_attach':
2519 case 'alt_inline_attach':
2520 $result .= $this->headerLine('Content-Type', static::CONTENT_TYPE_MULTIPART_MIXED . ';');
2521 $result .= $this->textLine(' boundary="' . $this->boundary[1] . '"');
2522 break;
2523 case 'alt':
2524 case 'alt_inline':
2525 $result .= $this->headerLine('Content-Type', static::CONTENT_TYPE_MULTIPART_ALTERNATIVE . ';');
2526 $result .= $this->textLine(' boundary="' . $this->boundary[1] . '"');
2527 break;
2528 default:
2529 // Catches case 'plain': and case '':
2530 $result .= $this->textLine('Content-Type: ' . $this->ContentType . '; charset=' . $this->CharSet);
2531 $ismultipart = false;
2532 break;
2533 }
2534 // RFC1341 part 5 says 7bit is assumed if not specified
2535 if (static::ENCODING_7BIT !== $this->Encoding) {
2536 // RFC 2045 section 6.4 says multipart MIME parts may only use 7bit, 8bit or binary CTE
2537 if ($ismultipart) {
2538 if (static::ENCODING_8BIT === $this->Encoding) {
2539 $result .= $this->headerLine('Content-Transfer-Encoding', static::ENCODING_8BIT);
2540 }
2541 // The only remaining alternatives are quoted-printable and base64, which are both 7bit compatible
2542 } else {
2543 $result .= $this->headerLine('Content-Transfer-Encoding', $this->Encoding);
2544 }
2545 }
2546
2547 if ('mail' !== $this->Mailer) {
2548// $result .= static::$LE;
2549 }
2550
2551 return $result;
2552 }
2553
2554 /**
2555 * Returns the whole MIME message.
2556 * Includes complete headers and body.
2557 * Only valid post preSend().
2558 *
2559 * @see PHPMailer::preSend()
2560 *
2561 * @return string
2562 */
2563 public function getSentMIMEMessage()
2564 {
2565 return static::stripTrailingWSP($this->MIMEHeader . $this->mailHeader) .
2566 static::$LE . static::$LE . $this->MIMEBody;
2567 }
2568
2569 /**
2570 * Create a unique ID to use for boundaries.
2571 *
2572 * @return string
2573 */
2574 protected function generateId()
2575 {
2576 $len = 32; //32 bytes = 256 bits
2577 $bytes = '';
2578 if (function_exists('random_bytes')) {
2579 try {
2580 $bytes = random_bytes($len);
2581 } catch (\Exception $e) {
2582 //Do nothing
2583 }
2584 } elseif (function_exists('openssl_random_pseudo_bytes')) {
2585 /** @noinspection CryptographicallySecureRandomnessInspection */
2586 $bytes = openssl_random_pseudo_bytes($len);
2587 }
2588 if ($bytes === '') {
2589 //We failed to produce a proper random string, so make do.
2590 //Use a hash to force the length to the same as the other methods
2591 $bytes = hash('sha256', uniqid((string) mt_rand(), true), true);
2592 }
2593
2594 //We don't care about messing up base64 format here, just want a random string
2595 return str_replace(['=', '+', '/'], '', base64_encode(hash('sha256', $bytes, true)));
2596 }
2597
2598 /**
2599 * Assemble the message body.
2600 * Returns an empty string on failure.
2601 *
2602 * @throws Exception
2603 *
2604 * @return string The assembled message body
2605 */
2606 public function createBody()
2607 {
2608 $body = '';
2609 //Create unique IDs and preset boundaries
2610 $this->uniqueid = $this->generateId();
2611 $this->boundary[1] = 'b1_' . $this->uniqueid;
2612 $this->boundary[2] = 'b2_' . $this->uniqueid;
2613 $this->boundary[3] = 'b3_' . $this->uniqueid;
2614
2615 if ($this->sign_key_file) {
2616 $body .= $this->getMailMIME() . static::$LE;
2617 }
2618
2619 $this->setWordWrap();
2620
2621 $bodyEncoding = $this->Encoding;
2622 $bodyCharSet = $this->CharSet;
2623 //Can we do a 7-bit downgrade?
2624 if (static::ENCODING_8BIT === $bodyEncoding && !$this->has8bitChars($this->Body)) {
2625 $bodyEncoding = static::ENCODING_7BIT;
2626 //All ISO 8859, Windows codepage and UTF-8 charsets are ascii compatible up to 7-bit
2627 $bodyCharSet = static::CHARSET_ASCII;
2628 }
2629 //If lines are too long, and we're not already using an encoding that will shorten them,
2630 //change to quoted-printable transfer encoding for the body part only
2631 if (static::ENCODING_BASE64 !== $this->Encoding && static::hasLineLongerThanMax($this->Body)) {
2632 $bodyEncoding = static::ENCODING_QUOTED_PRINTABLE;
2633 }
2634
2635 $altBodyEncoding = $this->Encoding;
2636 $altBodyCharSet = $this->CharSet;
2637 //Can we do a 7-bit downgrade?
2638 if (static::ENCODING_8BIT === $altBodyEncoding && !$this->has8bitChars($this->AltBody)) {
2639 $altBodyEncoding = static::ENCODING_7BIT;
2640 //All ISO 8859, Windows codepage and UTF-8 charsets are ascii compatible up to 7-bit
2641 $altBodyCharSet = static::CHARSET_ASCII;
2642 }
2643 //If lines are too long, and we're not already using an encoding that will shorten them,
2644 //change to quoted-printable transfer encoding for the alt body part only
2645 if (static::ENCODING_BASE64 !== $altBodyEncoding && static::hasLineLongerThanMax($this->AltBody)) {
2646 $altBodyEncoding = static::ENCODING_QUOTED_PRINTABLE;
2647 }
2648 //Use this as a preamble in all multipart message types
2649 $mimepre = 'This is a multi-part message in MIME format.' . static::$LE . static::$LE;
2650 switch ($this->message_type) {
2651 case 'inline':
2652 $body .= $mimepre;
2653 $body .= $this->getBoundary($this->boundary[1], $bodyCharSet, '', $bodyEncoding);
2654 $body .= $this->encodeString($this->Body, $bodyEncoding);
2655 $body .= static::$LE;
2656 $body .= $this->attachAll('inline', $this->boundary[1]);
2657 break;
2658 case 'attach':
2659 $body .= $mimepre;
2660 $body .= $this->getBoundary($this->boundary[1], $bodyCharSet, '', $bodyEncoding);
2661 $body .= $this->encodeString($this->Body, $bodyEncoding);
2662 $body .= static::$LE;
2663 $body .= $this->attachAll('attachment', $this->boundary[1]);
2664 break;
2665 case 'inline_attach':
2666 $body .= $mimepre;
2667 $body .= $this->textLine('--' . $this->boundary[1]);
2668 $body .= $this->headerLine('Content-Type', static::CONTENT_TYPE_MULTIPART_RELATED . ';');
2669 $body .= $this->textLine(' boundary="' . $this->boundary[2] . '";');
2670 $body .= $this->textLine(' type="' . static::CONTENT_TYPE_TEXT_HTML . '"');
2671 $body .= static::$LE;
2672 $body .= $this->getBoundary($this->boundary[2], $bodyCharSet, '', $bodyEncoding);
2673 $body .= $this->encodeString($this->Body, $bodyEncoding);
2674 $body .= static::$LE;
2675 $body .= $this->attachAll('inline', $this->boundary[2]);
2676 $body .= static::$LE;
2677 $body .= $this->attachAll('attachment', $this->boundary[1]);
2678 break;
2679 case 'alt':
2680 $body .= $mimepre;
2681 $body .= $this->getBoundary(
2682 $this->boundary[1],
2683 $altBodyCharSet,
2684 static::CONTENT_TYPE_PLAINTEXT,
2685 $altBodyEncoding
2686 );
2687 $body .= $this->encodeString($this->AltBody, $altBodyEncoding);
2688 $body .= static::$LE;
2689 $body .= $this->getBoundary(
2690 $this->boundary[1],
2691 $bodyCharSet,
2692 static::CONTENT_TYPE_TEXT_HTML,
2693 $bodyEncoding
2694 );
2695 $body .= $this->encodeString($this->Body, $bodyEncoding);
2696 $body .= static::$LE;
2697 if (!empty($this->Ical)) {
2698 $method = static::ICAL_METHOD_REQUEST;
2699 foreach (static::$IcalMethods as $imethod) {
2700 if (stripos($this->Ical, 'METHOD:' . $imethod) !== false) {
2701 $method = $imethod;
2702 break;
2703 }
2704 }
2705 $body .= $this->getBoundary(
2706 $this->boundary[1],
2707 '',
2708 static::CONTENT_TYPE_TEXT_CALENDAR . '; method=' . $method,
2709 ''
2710 );
2711 $body .= $this->encodeString($this->Ical, $this->Encoding);
2712 $body .= static::$LE;
2713 }
2714 $body .= $this->endBoundary($this->boundary[1]);
2715 break;
2716 case 'alt_inline':
2717 $body .= $mimepre;
2718 $body .= $this->getBoundary(
2719 $this->boundary[1],
2720 $altBodyCharSet,
2721 static::CONTENT_TYPE_PLAINTEXT,
2722 $altBodyEncoding
2723 );
2724 $body .= $this->encodeString($this->AltBody, $altBodyEncoding);
2725 $body .= static::$LE;
2726 $body .= $this->textLine('--' . $this->boundary[1]);
2727 $body .= $this->headerLine('Content-Type', static::CONTENT_TYPE_MULTIPART_RELATED . ';');
2728 $body .= $this->textLine(' boundary="' . $this->boundary[2] . '";');
2729 $body .= $this->textLine(' type="' . static::CONTENT_TYPE_TEXT_HTML . '"');
2730 $body .= static::$LE;
2731 $body .= $this->getBoundary(
2732 $this->boundary[2],
2733 $bodyCharSet,
2734 static::CONTENT_TYPE_TEXT_HTML,
2735 $bodyEncoding
2736 );
2737 $body .= $this->encodeString($this->Body, $bodyEncoding);
2738 $body .= static::$LE;
2739 $body .= $this->attachAll('inline', $this->boundary[2]);
2740 $body .= static::$LE;
2741 $body .= $this->endBoundary($this->boundary[1]);
2742 break;
2743 case 'alt_attach':
2744 $body .= $mimepre;
2745 $body .= $this->textLine('--' . $this->boundary[1]);
2746 $body .= $this->headerLine('Content-Type', static::CONTENT_TYPE_MULTIPART_ALTERNATIVE . ';');
2747 $body .= $this->textLine(' boundary="' . $this->boundary[2] . '"');
2748 $body .= static::$LE;
2749 $body .= $this->getBoundary(
2750 $this->boundary[2],
2751 $altBodyCharSet,
2752 static::CONTENT_TYPE_PLAINTEXT,
2753 $altBodyEncoding
2754 );
2755 $body .= $this->encodeString($this->AltBody, $altBodyEncoding);
2756 $body .= static::$LE;
2757 $body .= $this->getBoundary(
2758 $this->boundary[2],
2759 $bodyCharSet,
2760 static::CONTENT_TYPE_TEXT_HTML,
2761 $bodyEncoding
2762 );
2763 $body .= $this->encodeString($this->Body, $bodyEncoding);
2764 $body .= static::$LE;
2765 if (!empty($this->Ical)) {
2766 $method = static::ICAL_METHOD_REQUEST;
2767 foreach (static::$IcalMethods as $imethod) {
2768 if (stripos($this->Ical, 'METHOD:' . $imethod) !== false) {
2769 $method = $imethod;
2770 break;
2771 }
2772 }
2773 $body .= $this->getBoundary(
2774 $this->boundary[2],
2775 '',
2776 static::CONTENT_TYPE_TEXT_CALENDAR . '; method=' . $method,
2777 ''
2778 );
2779 $body .= $this->encodeString($this->Ical, $this->Encoding);
2780 }
2781 $body .= $this->endBoundary($this->boundary[2]);
2782 $body .= static::$LE;
2783 $body .= $this->attachAll('attachment', $this->boundary[1]);
2784 break;
2785 case 'alt_inline_attach':
2786 $body .= $mimepre;
2787 $body .= $this->textLine('--' . $this->boundary[1]);
2788 $body .= $this->headerLine('Content-Type', static::CONTENT_TYPE_MULTIPART_ALTERNATIVE . ';');
2789 $body .= $this->textLine(' boundary="' . $this->boundary[2] . '"');
2790 $body .= static::$LE;
2791 $body .= $this->getBoundary(
2792 $this->boundary[2],
2793 $altBodyCharSet,
2794 static::CONTENT_TYPE_PLAINTEXT,
2795 $altBodyEncoding
2796 );
2797 $body .= $this->encodeString($this->AltBody, $altBodyEncoding);
2798 $body .= static::$LE;
2799 $body .= $this->textLine('--' . $this->boundary[2]);
2800 $body .= $this->headerLine('Content-Type', static::CONTENT_TYPE_MULTIPART_RELATED . ';');
2801 $body .= $this->textLine(' boundary="' . $this->boundary[3] . '";');
2802 $body .= $this->textLine(' type="' . static::CONTENT_TYPE_TEXT_HTML . '"');
2803 $body .= static::$LE;
2804 $body .= $this->getBoundary(
2805 $this->boundary[3],
2806 $bodyCharSet,
2807 static::CONTENT_TYPE_TEXT_HTML,
2808 $bodyEncoding
2809 );
2810 $body .= $this->encodeString($this->Body, $bodyEncoding);
2811 $body .= static::$LE;
2812 $body .= $this->attachAll('inline', $this->boundary[3]);
2813 $body .= static::$LE;
2814 $body .= $this->endBoundary($this->boundary[2]);
2815 $body .= static::$LE;
2816 $body .= $this->attachAll('attachment', $this->boundary[1]);
2817 break;
2818 default:
2819 // Catch case 'plain' and case '', applies to simple `text/plain` and `text/html` body content types
2820 //Reset the `Encoding` property in case we changed it for line length reasons
2821 $this->Encoding = $bodyEncoding;
2822 $body .= $this->encodeString($this->Body, $this->Encoding);
2823 break;
2824 }
2825
2826 if ($this->isError()) {
2827 $body = '';
2828 if ($this->exceptions) {
2829 throw new Exception($this->lang('empty_message'), self::STOP_CRITICAL);
2830 }
2831 } elseif ($this->sign_key_file) {
2832 try {
2833 if (!defined('PKCS7_TEXT')) {
2834 throw new Exception($this->lang('extension_missing') . 'openssl');
2835 }
2836
2837 $file = tempnam(sys_get_temp_dir(), 'srcsign');
2838 $signed = tempnam(sys_get_temp_dir(), 'mailsign');
2839 file_put_contents($file, $body);
2840
2841 //Workaround for PHP bug https://bugs.php.net/bug.php?id=69197
2842 if (empty($this->sign_extracerts_file)) {
2843 $sign = @openssl_pkcs7_sign(
2844 $file,
2845 $signed,
2846 'file://' . realpath($this->sign_cert_file),
2847 ['file://' . realpath($this->sign_key_file), $this->sign_key_pass],
2848 []
2849 );
2850 } else {
2851 $sign = @openssl_pkcs7_sign(
2852 $file,
2853 $signed,
2854 'file://' . realpath($this->sign_cert_file),
2855 ['file://' . realpath($this->sign_key_file), $this->sign_key_pass],
2856 [],
2857 PKCS7_DETACHED,
2858 $this->sign_extracerts_file
2859 );
2860 }
2861
2862 @unlink($file);
2863 if ($sign) {
2864 $body = file_get_contents($signed);
2865 @unlink($signed);
2866 //The message returned by openssl contains both headers and body, so need to split them up
2867 $parts = explode("\n\n", $body, 2);
2868 $this->MIMEHeader .= $parts[0] . static::$LE . static::$LE;
2869 $body = $parts[1];
2870 } else {
2871 @unlink($signed);
2872 throw new Exception($this->lang('signing') . openssl_error_string());
2873 }
2874 } catch (Exception $exc) {
2875 $body = '';
2876 if ($this->exceptions) {
2877 throw $exc;
2878 }
2879 }
2880 }
2881
2882 return $body;
2883 }
2884
2885 /**
2886 * Return the start of a message boundary.
2887 *
2888 * @param string $boundary
2889 * @param string $charSet
2890 * @param string $contentType
2891 * @param string $encoding
2892 *
2893 * @return string
2894 */
2895 protected function getBoundary($boundary, $charSet, $contentType, $encoding)
2896 {
2897 $result = '';
2898 if ('' === $charSet) {
2899 $charSet = $this->CharSet;
2900 }
2901 if ('' === $contentType) {
2902 $contentType = $this->ContentType;
2903 }
2904 if ('' === $encoding) {
2905 $encoding = $this->Encoding;
2906 }
2907 $result .= $this->textLine('--' . $boundary);
2908 $result .= sprintf('Content-Type: %s; charset=%s', $contentType, $charSet);
2909 $result .= static::$LE;
2910 // RFC1341 part 5 says 7bit is assumed if not specified
2911 if (static::ENCODING_7BIT !== $encoding) {
2912 $result .= $this->headerLine('Content-Transfer-Encoding', $encoding);
2913 }
2914 $result .= static::$LE;
2915
2916 return $result;
2917 }
2918
2919 /**
2920 * Return the end of a message boundary.
2921 *
2922 * @param string $boundary
2923 *
2924 * @return string
2925 */
2926 protected function endBoundary($boundary)
2927 {
2928 return static::$LE . '--' . $boundary . '--' . static::$LE;
2929 }
2930
2931 /**
2932 * Set the message type.
2933 * PHPMailer only supports some preset message types, not arbitrary MIME structures.
2934 */
2935 protected function setMessageType()
2936 {
2937 $type = [];
2938 if ($this->alternativeExists()) {
2939 $type[] = 'alt';
2940 }
2941 if ($this->inlineImageExists()) {
2942 $type[] = 'inline';
2943 }
2944 if ($this->attachmentExists()) {
2945 $type[] = 'attach';
2946 }
2947 $this->message_type = implode('_', $type);
2948 if ('' === $this->message_type) {
2949 //The 'plain' message_type refers to the message having a single body element, not that it is plain-text
2950 $this->message_type = 'plain';
2951 }
2952 }
2953
2954 /**
2955 * Format a header line.
2956 *
2957 * @param string $name
2958 * @param string|int $value
2959 *
2960 * @return string
2961 */
2962 public function headerLine($name, $value)
2963 {
2964 return $name . ': ' . $value . static::$LE;
2965 }
2966
2967 /**
2968 * Return a formatted mail line.
2969 *
2970 * @param string $value
2971 *
2972 * @return string
2973 */
2974 public function textLine($value)
2975 {
2976 return $value . static::$LE;
2977 }
2978
2979 /**
2980 * Add an attachment from a path on the filesystem.
2981 * Never use a user-supplied path to a file!
2982 * Returns false if the file could not be found or read.
2983 * Explicitly *does not* support passing URLs; PHPMailer is not an HTTP client.
2984 * If you need to do that, fetch the resource yourself and pass it in via a local file or string.
2985 *
2986 * @param string $path Path to the attachment
2987 * @param string $name Overrides the attachment name
2988 * @param string $encoding File encoding (see $Encoding)
Matthias Andreas Benkarde39c4f82021-01-06 17:59:39 +01002989 * @param string $type MIME type, e.g. `image/jpeg`; determined automatically from $path if not specified
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +01002990 * @param string $disposition Disposition to use
2991 *
2992 * @throws Exception
2993 *
2994 * @return bool
2995 */
2996 public function addAttachment(
2997 $path,
2998 $name = '',
2999 $encoding = self::ENCODING_BASE64,
3000 $type = '',
3001 $disposition = 'attachment'
3002 ) {
3003 try {
Matthias Andreas Benkarde39c4f82021-01-06 17:59:39 +01003004 if (!static::fileIsAccessible($path)) {
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +01003005 throw new Exception($this->lang('file_access') . $path, self::STOP_CONTINUE);
3006 }
3007
3008 // If a MIME type is not specified, try to work it out from the file name
3009 if ('' === $type) {
3010 $type = static::filenameToType($path);
3011 }
3012
3013 $filename = (string) static::mb_pathinfo($path, PATHINFO_BASENAME);
3014 if ('' === $name) {
3015 $name = $filename;
3016 }
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +01003017 if (!$this->validateEncoding($encoding)) {
3018 throw new Exception($this->lang('encoding') . $encoding);
3019 }
3020
3021 $this->attachment[] = [
3022 0 => $path,
3023 1 => $filename,
3024 2 => $name,
3025 3 => $encoding,
3026 4 => $type,
3027 5 => false, // isStringAttachment
3028 6 => $disposition,
3029 7 => $name,
3030 ];
3031 } catch (Exception $exc) {
3032 $this->setError($exc->getMessage());
3033 $this->edebug($exc->getMessage());
3034 if ($this->exceptions) {
3035 throw $exc;
3036 }
3037
3038 return false;
3039 }
3040
3041 return true;
3042 }
3043
3044 /**
3045 * Return the array of attachments.
3046 *
3047 * @return array
3048 */
3049 public function getAttachments()
3050 {
3051 return $this->attachment;
3052 }
3053
3054 /**
3055 * Attach all file, string, and binary attachments to the message.
3056 * Returns an empty string on failure.
3057 *
3058 * @param string $disposition_type
3059 * @param string $boundary
3060 *
3061 * @throws Exception
3062 *
3063 * @return string
3064 */
3065 protected function attachAll($disposition_type, $boundary)
3066 {
3067 // Return text of body
3068 $mime = [];
3069 $cidUniq = [];
3070 $incl = [];
3071
3072 // Add all attachments
3073 foreach ($this->attachment as $attachment) {
3074 // Check if it is a valid disposition_filter
3075 if ($attachment[6] === $disposition_type) {
3076 // Check for string attachment
3077 $string = '';
3078 $path = '';
3079 $bString = $attachment[5];
3080 if ($bString) {
3081 $string = $attachment[0];
3082 } else {
3083 $path = $attachment[0];
3084 }
3085
3086 $inclhash = hash('sha256', serialize($attachment));
3087 if (in_array($inclhash, $incl, true)) {
3088 continue;
3089 }
3090 $incl[] = $inclhash;
3091 $name = $attachment[2];
3092 $encoding = $attachment[3];
3093 $type = $attachment[4];
3094 $disposition = $attachment[6];
3095 $cid = $attachment[7];
3096 if ('inline' === $disposition && array_key_exists($cid, $cidUniq)) {
3097 continue;
3098 }
3099 $cidUniq[$cid] = true;
3100
3101 $mime[] = sprintf('--%s%s', $boundary, static::$LE);
3102 //Only include a filename property if we have one
3103 if (!empty($name)) {
3104 $mime[] = sprintf(
3105 'Content-Type: %s; name=%s%s',
3106 $type,
3107 static::quotedString($this->encodeHeader($this->secureHeader($name))),
3108 static::$LE
3109 );
3110 } else {
3111 $mime[] = sprintf(
3112 'Content-Type: %s%s',
3113 $type,
3114 static::$LE
3115 );
3116 }
3117 // RFC1341 part 5 says 7bit is assumed if not specified
3118 if (static::ENCODING_7BIT !== $encoding) {
3119 $mime[] = sprintf('Content-Transfer-Encoding: %s%s', $encoding, static::$LE);
3120 }
3121
3122 //Only set Content-IDs on inline attachments
3123 if ((string) $cid !== '' && $disposition === 'inline') {
3124 $mime[] = 'Content-ID: <' . $this->encodeHeader($this->secureHeader($cid)) . '>' . static::$LE;
3125 }
3126
3127 // Allow for bypassing the Content-Disposition header
3128 if (!empty($disposition)) {
3129 $encoded_name = $this->encodeHeader($this->secureHeader($name));
3130 if (!empty($encoded_name)) {
3131 $mime[] = sprintf(
3132 'Content-Disposition: %s; filename=%s%s',
3133 $disposition,
3134 static::quotedString($encoded_name),
3135 static::$LE . static::$LE
3136 );
3137 } else {
3138 $mime[] = sprintf(
3139 'Content-Disposition: %s%s',
3140 $disposition,
3141 static::$LE . static::$LE
3142 );
3143 }
3144 } else {
3145 $mime[] = static::$LE;
3146 }
3147
3148 // Encode as string attachment
3149 if ($bString) {
3150 $mime[] = $this->encodeString($string, $encoding);
3151 } else {
3152 $mime[] = $this->encodeFile($path, $encoding);
3153 }
3154 if ($this->isError()) {
3155 return '';
3156 }
3157 $mime[] = static::$LE;
3158 }
3159 }
3160
3161 $mime[] = sprintf('--%s--%s', $boundary, static::$LE);
3162
3163 return implode('', $mime);
3164 }
3165
3166 /**
3167 * Encode a file attachment in requested format.
3168 * Returns an empty string on failure.
3169 *
3170 * @param string $path The full path to the file
3171 * @param string $encoding The encoding to use; one of 'base64', '7bit', '8bit', 'binary', 'quoted-printable'
3172 *
3173 * @return string
3174 */
3175 protected function encodeFile($path, $encoding = self::ENCODING_BASE64)
3176 {
3177 try {
Matthias Andreas Benkarde39c4f82021-01-06 17:59:39 +01003178 if (!static::fileIsAccessible($path)) {
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +01003179 throw new Exception($this->lang('file_open') . $path, self::STOP_CONTINUE);
3180 }
3181 $file_buffer = file_get_contents($path);
3182 if (false === $file_buffer) {
3183 throw new Exception($this->lang('file_open') . $path, self::STOP_CONTINUE);
3184 }
3185 $file_buffer = $this->encodeString($file_buffer, $encoding);
3186
3187 return $file_buffer;
3188 } catch (Exception $exc) {
3189 $this->setError($exc->getMessage());
3190 $this->edebug($exc->getMessage());
3191 if ($this->exceptions) {
3192 throw $exc;
3193 }
3194
3195 return '';
3196 }
3197 }
3198
3199 /**
3200 * Encode a string in requested format.
3201 * Returns an empty string on failure.
3202 *
3203 * @param string $str The text to encode
3204 * @param string $encoding The encoding to use; one of 'base64', '7bit', '8bit', 'binary', 'quoted-printable'
3205 *
3206 * @throws Exception
3207 *
3208 * @return string
3209 */
3210 public function encodeString($str, $encoding = self::ENCODING_BASE64)
3211 {
3212 $encoded = '';
3213 switch (strtolower($encoding)) {
3214 case static::ENCODING_BASE64:
3215 $encoded = chunk_split(
3216 base64_encode($str),
3217 static::STD_LINE_LENGTH,
3218 static::$LE
3219 );
3220 break;
3221 case static::ENCODING_7BIT:
3222 case static::ENCODING_8BIT:
3223 $encoded = static::normalizeBreaks($str);
3224 // Make sure it ends with a line break
3225 if (substr($encoded, -(strlen(static::$LE))) !== static::$LE) {
3226 $encoded .= static::$LE;
3227 }
3228 break;
3229 case static::ENCODING_BINARY:
3230 $encoded = $str;
3231 break;
3232 case static::ENCODING_QUOTED_PRINTABLE:
3233 $encoded = $this->encodeQP($str);
3234 break;
3235 default:
3236 $this->setError($this->lang('encoding') . $encoding);
3237 if ($this->exceptions) {
3238 throw new Exception($this->lang('encoding') . $encoding);
3239 }
3240 break;
3241 }
3242
3243 return $encoded;
3244 }
3245
3246 /**
3247 * Encode a header value (not including its label) optimally.
3248 * Picks shortest of Q, B, or none. Result includes folding if needed.
3249 * See RFC822 definitions for phrase, comment and text positions.
3250 *
3251 * @param string $str The header value to encode
3252 * @param string $position What context the string will be used in
3253 *
3254 * @return string
3255 */
3256 public function encodeHeader($str, $position = 'text')
3257 {
3258 $matchcount = 0;
3259 switch (strtolower($position)) {
3260 case 'phrase':
3261 if (!preg_match('/[\200-\377]/', $str)) {
3262 // Can't use addslashes as we don't know the value of magic_quotes_sybase
3263 $encoded = addcslashes($str, "\0..\37\177\\\"");
3264 if (($str === $encoded) && !preg_match('/[^A-Za-z0-9!#$%&\'*+\/=?^_`{|}~ -]/', $str)) {
3265 return $encoded;
3266 }
3267
3268 return "\"$encoded\"";
3269 }
3270 $matchcount = preg_match_all('/[^\040\041\043-\133\135-\176]/', $str, $matches);
3271 break;
3272 /* @noinspection PhpMissingBreakStatementInspection */
3273 case 'comment':
3274 $matchcount = preg_match_all('/[()"]/', $str, $matches);
3275 //fallthrough
3276 case 'text':
3277 default:
3278 $matchcount += preg_match_all('/[\000-\010\013\014\016-\037\177-\377]/', $str, $matches);
3279 break;
3280 }
3281
3282 if ($this->has8bitChars($str)) {
3283 $charset = $this->CharSet;
3284 } else {
3285 $charset = static::CHARSET_ASCII;
3286 }
3287
3288 // Q/B encoding adds 8 chars and the charset ("` =?<charset>?[QB]?<content>?=`").
3289 $overhead = 8 + strlen($charset);
3290
3291 if ('mail' === $this->Mailer) {
3292 $maxlen = static::MAIL_MAX_LINE_LENGTH - $overhead;
3293 } else {
3294 $maxlen = static::MAX_LINE_LENGTH - $overhead;
3295 }
3296
3297 // Select the encoding that produces the shortest output and/or prevents corruption.
3298 if ($matchcount > strlen($str) / 3) {
3299 // More than 1/3 of the content needs encoding, use B-encode.
3300 $encoding = 'B';
3301 } elseif ($matchcount > 0) {
3302 // Less than 1/3 of the content needs encoding, use Q-encode.
3303 $encoding = 'Q';
3304 } elseif (strlen($str) > $maxlen) {
3305 // No encoding needed, but value exceeds max line length, use Q-encode to prevent corruption.
3306 $encoding = 'Q';
3307 } else {
3308 // No reformatting needed
3309 $encoding = false;
3310 }
3311
3312 switch ($encoding) {
3313 case 'B':
3314 if ($this->hasMultiBytes($str)) {
3315 // Use a custom function which correctly encodes and wraps long
3316 // multibyte strings without breaking lines within a character
3317 $encoded = $this->base64EncodeWrapMB($str, "\n");
3318 } else {
3319 $encoded = base64_encode($str);
3320 $maxlen -= $maxlen % 4;
3321 $encoded = trim(chunk_split($encoded, $maxlen, "\n"));
3322 }
3323 $encoded = preg_replace('/^(.*)$/m', ' =?' . $charset . "?$encoding?\\1?=", $encoded);
3324 break;
3325 case 'Q':
3326 $encoded = $this->encodeQ($str, $position);
3327 $encoded = $this->wrapText($encoded, $maxlen, true);
3328 $encoded = str_replace('=' . static::$LE, "\n", trim($encoded));
3329 $encoded = preg_replace('/^(.*)$/m', ' =?' . $charset . "?$encoding?\\1?=", $encoded);
3330 break;
3331 default:
3332 return $str;
3333 }
3334
3335 return trim(static::normalizeBreaks($encoded));
3336 }
3337
3338 /**
3339 * Check if a string contains multi-byte characters.
3340 *
3341 * @param string $str multi-byte text to wrap encode
3342 *
3343 * @return bool
3344 */
3345 public function hasMultiBytes($str)
3346 {
3347 if (function_exists('mb_strlen')) {
3348 return strlen($str) > mb_strlen($str, $this->CharSet);
3349 }
3350
3351 // Assume no multibytes (we can't handle without mbstring functions anyway)
3352 return false;
3353 }
3354
3355 /**
3356 * Does a string contain any 8-bit chars (in any charset)?
3357 *
3358 * @param string $text
3359 *
3360 * @return bool
3361 */
3362 public function has8bitChars($text)
3363 {
3364 return (bool) preg_match('/[\x80-\xFF]/', $text);
3365 }
3366
3367 /**
3368 * Encode and wrap long multibyte strings for mail headers
3369 * without breaking lines within a character.
3370 * Adapted from a function by paravoid.
3371 *
3372 * @see http://www.php.net/manual/en/function.mb-encode-mimeheader.php#60283
3373 *
3374 * @param string $str multi-byte text to wrap encode
3375 * @param string $linebreak string to use as linefeed/end-of-line
3376 *
3377 * @return string
3378 */
3379 public function base64EncodeWrapMB($str, $linebreak = null)
3380 {
3381 $start = '=?' . $this->CharSet . '?B?';
3382 $end = '?=';
3383 $encoded = '';
3384 if (null === $linebreak) {
3385 $linebreak = static::$LE;
3386 }
3387
3388 $mb_length = mb_strlen($str, $this->CharSet);
3389 // Each line must have length <= 75, including $start and $end
3390 $length = 75 - strlen($start) - strlen($end);
3391 // Average multi-byte ratio
3392 $ratio = $mb_length / strlen($str);
3393 // Base64 has a 4:3 ratio
3394 $avgLength = floor($length * $ratio * .75);
3395
3396 $offset = 0;
3397 for ($i = 0; $i < $mb_length; $i += $offset) {
3398 $lookBack = 0;
3399 do {
3400 $offset = $avgLength - $lookBack;
3401 $chunk = mb_substr($str, $i, $offset, $this->CharSet);
3402 $chunk = base64_encode($chunk);
3403 ++$lookBack;
3404 } while (strlen($chunk) > $length);
3405 $encoded .= $chunk . $linebreak;
3406 }
3407
3408 // Chomp the last linefeed
3409 return substr($encoded, 0, -strlen($linebreak));
3410 }
3411
3412 /**
3413 * Encode a string in quoted-printable format.
3414 * According to RFC2045 section 6.7.
3415 *
3416 * @param string $string The text to encode
3417 *
3418 * @return string
3419 */
3420 public function encodeQP($string)
3421 {
3422 return static::normalizeBreaks(quoted_printable_encode($string));
3423 }
3424
3425 /**
3426 * Encode a string using Q encoding.
3427 *
3428 * @see http://tools.ietf.org/html/rfc2047#section-4.2
3429 *
3430 * @param string $str the text to encode
3431 * @param string $position Where the text is going to be used, see the RFC for what that means
3432 *
3433 * @return string
3434 */
3435 public function encodeQ($str, $position = 'text')
3436 {
3437 // There should not be any EOL in the string
3438 $pattern = '';
3439 $encoded = str_replace(["\r", "\n"], '', $str);
3440 switch (strtolower($position)) {
3441 case 'phrase':
3442 // RFC 2047 section 5.3
3443 $pattern = '^A-Za-z0-9!*+\/ -';
3444 break;
3445 /*
3446 * RFC 2047 section 5.2.
3447 * Build $pattern without including delimiters and []
3448 */
3449 /* @noinspection PhpMissingBreakStatementInspection */
3450 case 'comment':
3451 $pattern = '\(\)"';
3452 /* Intentional fall through */
3453 case 'text':
3454 default:
3455 // RFC 2047 section 5.1
3456 // Replace every high ascii, control, =, ? and _ characters
3457 $pattern = '\000-\011\013\014\016-\037\075\077\137\177-\377' . $pattern;
3458 break;
3459 }
3460 $matches = [];
3461 if (preg_match_all("/[{$pattern}]/", $encoded, $matches)) {
3462 // If the string contains an '=', make sure it's the first thing we replace
3463 // so as to avoid double-encoding
3464 $eqkey = array_search('=', $matches[0], true);
3465 if (false !== $eqkey) {
3466 unset($matches[0][$eqkey]);
3467 array_unshift($matches[0], '=');
3468 }
3469 foreach (array_unique($matches[0]) as $char) {
3470 $encoded = str_replace($char, '=' . sprintf('%02X', ord($char)), $encoded);
3471 }
3472 }
3473 // Replace spaces with _ (more readable than =20)
3474 // RFC 2047 section 4.2(2)
3475 return str_replace(' ', '_', $encoded);
3476 }
3477
3478 /**
3479 * Add a string or binary attachment (non-filesystem).
3480 * This method can be used to attach ascii or binary data,
3481 * such as a BLOB record from a database.
3482 *
3483 * @param string $string String attachment data
3484 * @param string $filename Name of the attachment
3485 * @param string $encoding File encoding (see $Encoding)
3486 * @param string $type File extension (MIME) type
3487 * @param string $disposition Disposition to use
3488 *
3489 * @throws Exception
3490 *
3491 * @return bool True on successfully adding an attachment
3492 */
3493 public function addStringAttachment(
3494 $string,
3495 $filename,
3496 $encoding = self::ENCODING_BASE64,
3497 $type = '',
3498 $disposition = 'attachment'
3499 ) {
3500 try {
3501 // If a MIME type is not specified, try to work it out from the file name
3502 if ('' === $type) {
3503 $type = static::filenameToType($filename);
3504 }
3505
3506 if (!$this->validateEncoding($encoding)) {
3507 throw new Exception($this->lang('encoding') . $encoding);
3508 }
3509
3510 // Append to $attachment array
3511 $this->attachment[] = [
3512 0 => $string,
3513 1 => $filename,
3514 2 => static::mb_pathinfo($filename, PATHINFO_BASENAME),
3515 3 => $encoding,
3516 4 => $type,
3517 5 => true, // isStringAttachment
3518 6 => $disposition,
3519 7 => 0,
3520 ];
3521 } catch (Exception $exc) {
3522 $this->setError($exc->getMessage());
3523 $this->edebug($exc->getMessage());
3524 if ($this->exceptions) {
3525 throw $exc;
3526 }
3527
3528 return false;
3529 }
3530
3531 return true;
3532 }
3533
3534 /**
3535 * Add an embedded (inline) attachment from a file.
3536 * This can include images, sounds, and just about any other document type.
3537 * These differ from 'regular' attachments in that they are intended to be
3538 * displayed inline with the message, not just attached for download.
3539 * This is used in HTML messages that embed the images
3540 * the HTML refers to using the $cid value.
3541 * Never use a user-supplied path to a file!
3542 *
3543 * @param string $path Path to the attachment
3544 * @param string $cid Content ID of the attachment; Use this to reference
3545 * the content when using an embedded image in HTML
3546 * @param string $name Overrides the attachment name
3547 * @param string $encoding File encoding (see $Encoding)
3548 * @param string $type File MIME type
3549 * @param string $disposition Disposition to use
3550 *
3551 * @throws Exception
3552 *
3553 * @return bool True on successfully adding an attachment
3554 */
3555 public function addEmbeddedImage(
3556 $path,
3557 $cid,
3558 $name = '',
3559 $encoding = self::ENCODING_BASE64,
3560 $type = '',
3561 $disposition = 'inline'
3562 ) {
3563 try {
Matthias Andreas Benkarde39c4f82021-01-06 17:59:39 +01003564 if (!static::fileIsAccessible($path)) {
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +01003565 throw new Exception($this->lang('file_access') . $path, self::STOP_CONTINUE);
3566 }
3567
3568 // If a MIME type is not specified, try to work it out from the file name
3569 if ('' === $type) {
3570 $type = static::filenameToType($path);
3571 }
3572
3573 if (!$this->validateEncoding($encoding)) {
3574 throw new Exception($this->lang('encoding') . $encoding);
3575 }
3576
3577 $filename = (string) static::mb_pathinfo($path, PATHINFO_BASENAME);
3578 if ('' === $name) {
3579 $name = $filename;
3580 }
3581
3582 // Append to $attachment array
3583 $this->attachment[] = [
3584 0 => $path,
3585 1 => $filename,
3586 2 => $name,
3587 3 => $encoding,
3588 4 => $type,
3589 5 => false, // isStringAttachment
3590 6 => $disposition,
3591 7 => $cid,
3592 ];
3593 } catch (Exception $exc) {
3594 $this->setError($exc->getMessage());
3595 $this->edebug($exc->getMessage());
3596 if ($this->exceptions) {
3597 throw $exc;
3598 }
3599
3600 return false;
3601 }
3602
3603 return true;
3604 }
3605
3606 /**
3607 * Add an embedded stringified attachment.
3608 * This can include images, sounds, and just about any other document type.
3609 * If your filename doesn't contain an extension, be sure to set the $type to an appropriate MIME type.
3610 *
3611 * @param string $string The attachment binary data
3612 * @param string $cid Content ID of the attachment; Use this to reference
3613 * the content when using an embedded image in HTML
3614 * @param string $name A filename for the attachment. If this contains an extension,
3615 * PHPMailer will attempt to set a MIME type for the attachment.
3616 * For example 'file.jpg' would get an 'image/jpeg' MIME type.
3617 * @param string $encoding File encoding (see $Encoding), defaults to 'base64'
3618 * @param string $type MIME type - will be used in preference to any automatically derived type
3619 * @param string $disposition Disposition to use
3620 *
3621 * @throws Exception
3622 *
3623 * @return bool True on successfully adding an attachment
3624 */
3625 public function addStringEmbeddedImage(
3626 $string,
3627 $cid,
3628 $name = '',
3629 $encoding = self::ENCODING_BASE64,
3630 $type = '',
3631 $disposition = 'inline'
3632 ) {
3633 try {
3634 // If a MIME type is not specified, try to work it out from the name
3635 if ('' === $type && !empty($name)) {
3636 $type = static::filenameToType($name);
3637 }
3638
3639 if (!$this->validateEncoding($encoding)) {
3640 throw new Exception($this->lang('encoding') . $encoding);
3641 }
3642
3643 // Append to $attachment array
3644 $this->attachment[] = [
3645 0 => $string,
3646 1 => $name,
3647 2 => $name,
3648 3 => $encoding,
3649 4 => $type,
3650 5 => true, // isStringAttachment
3651 6 => $disposition,
3652 7 => $cid,
3653 ];
3654 } catch (Exception $exc) {
3655 $this->setError($exc->getMessage());
3656 $this->edebug($exc->getMessage());
3657 if ($this->exceptions) {
3658 throw $exc;
3659 }
3660
3661 return false;
3662 }
3663
3664 return true;
3665 }
3666
3667 /**
3668 * Validate encodings.
3669 *
3670 * @param string $encoding
3671 *
3672 * @return bool
3673 */
3674 protected function validateEncoding($encoding)
3675 {
3676 return in_array(
3677 $encoding,
3678 [
3679 self::ENCODING_7BIT,
3680 self::ENCODING_QUOTED_PRINTABLE,
3681 self::ENCODING_BASE64,
3682 self::ENCODING_8BIT,
3683 self::ENCODING_BINARY,
3684 ],
3685 true
3686 );
3687 }
3688
3689 /**
3690 * Check if an embedded attachment is present with this cid.
3691 *
3692 * @param string $cid
3693 *
3694 * @return bool
3695 */
3696 protected function cidExists($cid)
3697 {
3698 foreach ($this->attachment as $attachment) {
3699 if ('inline' === $attachment[6] && $cid === $attachment[7]) {
3700 return true;
3701 }
3702 }
3703
3704 return false;
3705 }
3706
3707 /**
3708 * Check if an inline attachment is present.
3709 *
3710 * @return bool
3711 */
3712 public function inlineImageExists()
3713 {
3714 foreach ($this->attachment as $attachment) {
3715 if ('inline' === $attachment[6]) {
3716 return true;
3717 }
3718 }
3719
3720 return false;
3721 }
3722
3723 /**
3724 * Check if an attachment (non-inline) is present.
3725 *
3726 * @return bool
3727 */
3728 public function attachmentExists()
3729 {
3730 foreach ($this->attachment as $attachment) {
3731 if ('attachment' === $attachment[6]) {
3732 return true;
3733 }
3734 }
3735
3736 return false;
3737 }
3738
3739 /**
3740 * Check if this message has an alternative body set.
3741 *
3742 * @return bool
3743 */
3744 public function alternativeExists()
3745 {
3746 return !empty($this->AltBody);
3747 }
3748
3749 /**
3750 * Clear queued addresses of given kind.
3751 *
3752 * @param string $kind 'to', 'cc', or 'bcc'
3753 */
3754 public function clearQueuedAddresses($kind)
3755 {
3756 $this->RecipientsQueue = array_filter(
3757 $this->RecipientsQueue,
3758 static function ($params) use ($kind) {
3759 return $params[0] !== $kind;
3760 }
3761 );
3762 }
3763
3764 /**
3765 * Clear all To recipients.
3766 */
3767 public function clearAddresses()
3768 {
3769 foreach ($this->to as $to) {
3770 unset($this->all_recipients[strtolower($to[0])]);
3771 }
3772 $this->to = [];
3773 $this->clearQueuedAddresses('to');
3774 }
3775
3776 /**
3777 * Clear all CC recipients.
3778 */
3779 public function clearCCs()
3780 {
3781 foreach ($this->cc as $cc) {
3782 unset($this->all_recipients[strtolower($cc[0])]);
3783 }
3784 $this->cc = [];
3785 $this->clearQueuedAddresses('cc');
3786 }
3787
3788 /**
3789 * Clear all BCC recipients.
3790 */
3791 public function clearBCCs()
3792 {
3793 foreach ($this->bcc as $bcc) {
3794 unset($this->all_recipients[strtolower($bcc[0])]);
3795 }
3796 $this->bcc = [];
3797 $this->clearQueuedAddresses('bcc');
3798 }
3799
3800 /**
3801 * Clear all ReplyTo recipients.
3802 */
3803 public function clearReplyTos()
3804 {
3805 $this->ReplyTo = [];
3806 $this->ReplyToQueue = [];
3807 }
3808
3809 /**
3810 * Clear all recipient types.
3811 */
3812 public function clearAllRecipients()
3813 {
3814 $this->to = [];
3815 $this->cc = [];
3816 $this->bcc = [];
3817 $this->all_recipients = [];
3818 $this->RecipientsQueue = [];
3819 }
3820
3821 /**
3822 * Clear all filesystem, string, and binary attachments.
3823 */
3824 public function clearAttachments()
3825 {
3826 $this->attachment = [];
3827 }
3828
3829 /**
3830 * Clear all custom headers.
3831 */
3832 public function clearCustomHeaders()
3833 {
3834 $this->CustomHeader = [];
3835 }
3836
3837 /**
3838 * Add an error message to the error container.
3839 *
3840 * @param string $msg
3841 */
3842 protected function setError($msg)
3843 {
3844 ++$this->error_count;
3845 if ('smtp' === $this->Mailer && null !== $this->smtp) {
3846 $lasterror = $this->smtp->getError();
3847 if (!empty($lasterror['error'])) {
3848 $msg .= $this->lang('smtp_error') . $lasterror['error'];
3849 if (!empty($lasterror['detail'])) {
3850 $msg .= ' Detail: ' . $lasterror['detail'];
3851 }
3852 if (!empty($lasterror['smtp_code'])) {
3853 $msg .= ' SMTP code: ' . $lasterror['smtp_code'];
3854 }
3855 if (!empty($lasterror['smtp_code_ex'])) {
3856 $msg .= ' Additional SMTP info: ' . $lasterror['smtp_code_ex'];
3857 }
3858 }
3859 }
3860 $this->ErrorInfo = $msg;
3861 }
3862
3863 /**
3864 * Return an RFC 822 formatted date.
3865 *
3866 * @return string
3867 */
3868 public static function rfcDate()
3869 {
3870 // Set the time zone to whatever the default is to avoid 500 errors
3871 // Will default to UTC if it's not set properly in php.ini
3872 date_default_timezone_set(@date_default_timezone_get());
3873
3874 return date('D, j M Y H:i:s O');
3875 }
3876
3877 /**
3878 * Get the server hostname.
3879 * Returns 'localhost.localdomain' if unknown.
3880 *
3881 * @return string
3882 */
3883 protected function serverHostname()
3884 {
3885 $result = '';
3886 if (!empty($this->Hostname)) {
3887 $result = $this->Hostname;
3888 } elseif (isset($_SERVER) && array_key_exists('SERVER_NAME', $_SERVER)) {
3889 $result = $_SERVER['SERVER_NAME'];
3890 } elseif (function_exists('gethostname') && gethostname() !== false) {
3891 $result = gethostname();
3892 } elseif (php_uname('n') !== false) {
3893 $result = php_uname('n');
3894 }
3895 if (!static::isValidHost($result)) {
3896 return 'localhost.localdomain';
3897 }
3898
3899 return $result;
3900 }
3901
3902 /**
3903 * Validate whether a string contains a valid value to use as a hostname or IP address.
3904 * IPv6 addresses must include [], e.g. `[::1]`, not just `::1`.
3905 *
3906 * @param string $host The host name or IP address to check
3907 *
3908 * @return bool
3909 */
3910 public static function isValidHost($host)
3911 {
3912 //Simple syntax limits
Matthias Andreas Benkarde39c4f82021-01-06 17:59:39 +01003913 if (
3914 empty($host)
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +01003915 || !is_string($host)
3916 || strlen($host) > 256
3917 || !preg_match('/^([a-zA-Z\d.-]*|\[[a-fA-F\d:]+])$/', $host)
3918 ) {
3919 return false;
3920 }
3921 //Looks like a bracketed IPv6 address
3922 if (strlen($host) > 2 && substr($host, 0, 1) === '[' && substr($host, -1, 1) === ']') {
3923 return filter_var(substr($host, 1, -1), FILTER_VALIDATE_IP, FILTER_FLAG_IPV6) !== false;
3924 }
3925 //If removing all the dots results in a numeric string, it must be an IPv4 address.
3926 //Need to check this first because otherwise things like `999.0.0.0` are considered valid host names
3927 if (is_numeric(str_replace('.', '', $host))) {
3928 //Is it a valid IPv4 address?
3929 return filter_var($host, FILTER_VALIDATE_IP, FILTER_FLAG_IPV4) !== false;
3930 }
3931 if (filter_var('http://' . $host, FILTER_VALIDATE_URL) !== false) {
3932 //Is it a syntactically valid hostname?
3933 return true;
3934 }
3935
3936 return false;
3937 }
3938
3939 /**
3940 * Get an error message in the current language.
3941 *
3942 * @param string $key
3943 *
3944 * @return string
3945 */
3946 protected function lang($key)
3947 {
3948 if (count($this->language) < 1) {
3949 $this->setLanguage(); // set the default language
3950 }
3951
3952 if (array_key_exists($key, $this->language)) {
3953 if ('smtp_connect_failed' === $key) {
3954 //Include a link to troubleshooting docs on SMTP connection failure
3955 //this is by far the biggest cause of support questions
3956 //but it's usually not PHPMailer's fault.
3957 return $this->language[$key] . ' https://github.com/PHPMailer/PHPMailer/wiki/Troubleshooting';
3958 }
3959
3960 return $this->language[$key];
3961 }
3962
3963 //Return the key as a fallback
3964 return $key;
3965 }
3966
3967 /**
3968 * Check if an error occurred.
3969 *
3970 * @return bool True if an error did occur
3971 */
3972 public function isError()
3973 {
3974 return $this->error_count > 0;
3975 }
3976
3977 /**
3978 * Add a custom header.
3979 * $name value can be overloaded to contain
3980 * both header name and value (name:value).
3981 *
3982 * @param string $name Custom header name
3983 * @param string|null $value Header value
3984 *
3985 * @throws Exception
3986 */
3987 public function addCustomHeader($name, $value = null)
3988 {
3989 if (null === $value && strpos($name, ':') !== false) {
3990 // Value passed in as name:value
3991 list($name, $value) = explode(':', $name, 2);
3992 }
3993 $name = trim($name);
3994 $value = trim($value);
3995 //Ensure name is not empty, and that neither name nor value contain line breaks
3996 if (empty($name) || strpbrk($name . $value, "\r\n") !== false) {
3997 if ($this->exceptions) {
3998 throw new Exception('Invalid header name or value');
3999 }
4000
4001 return false;
4002 }
4003 $this->CustomHeader[] = [$name, $value];
4004
4005 return true;
4006 }
4007
4008 /**
4009 * Returns all custom headers.
4010 *
4011 * @return array
4012 */
4013 public function getCustomHeaders()
4014 {
4015 return $this->CustomHeader;
4016 }
4017
4018 /**
4019 * Create a message body from an HTML string.
4020 * Automatically inlines images and creates a plain-text version by converting the HTML,
4021 * overwriting any existing values in Body and AltBody.
4022 * Do not source $message content from user input!
4023 * $basedir is prepended when handling relative URLs, e.g. <img src="/images/a.png"> and must not be empty
4024 * will look for an image file in $basedir/images/a.png and convert it to inline.
4025 * If you don't provide a $basedir, relative paths will be left untouched (and thus probably break in email)
4026 * Converts data-uri images into embedded attachments.
4027 * If you don't want to apply these transformations to your HTML, just set Body and AltBody directly.
4028 *
4029 * @param string $message HTML message string
4030 * @param string $basedir Absolute path to a base directory to prepend to relative paths to images
4031 * @param bool|callable $advanced Whether to use the internal HTML to text converter
Matthias Andreas Benkarde39c4f82021-01-06 17:59:39 +01004032 * or your own custom converter
4033 * @return string The transformed message body
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +01004034 *
4035 * @throws Exception
4036 *
4037 * @see PHPMailer::html2text()
4038 */
4039 public function msgHTML($message, $basedir = '', $advanced = false)
4040 {
4041 preg_match_all('/(?<!-)(src|background)=["\'](.*)["\']/Ui', $message, $images);
4042 if (array_key_exists(2, $images)) {
4043 if (strlen($basedir) > 1 && '/' !== substr($basedir, -1)) {
4044 // Ensure $basedir has a trailing /
4045 $basedir .= '/';
4046 }
4047 foreach ($images[2] as $imgindex => $url) {
4048 // Convert data URIs into embedded images
4049 //e.g. ""
4050 $match = [];
4051 if (preg_match('#^data:(image/(?:jpe?g|gif|png));?(base64)?,(.+)#', $url, $match)) {
4052 if (count($match) === 4 && static::ENCODING_BASE64 === $match[2]) {
4053 $data = base64_decode($match[3]);
4054 } elseif ('' === $match[2]) {
4055 $data = rawurldecode($match[3]);
4056 } else {
4057 //Not recognised so leave it alone
4058 continue;
4059 }
4060 //Hash the decoded data, not the URL, so that the same data-URI image used in multiple places
4061 //will only be embedded once, even if it used a different encoding
4062 $cid = substr(hash('sha256', $data), 0, 32) . '@phpmailer.0'; // RFC2392 S 2
4063
4064 if (!$this->cidExists($cid)) {
4065 $this->addStringEmbeddedImage(
4066 $data,
4067 $cid,
4068 'embed' . $imgindex,
4069 static::ENCODING_BASE64,
4070 $match[1]
4071 );
4072 }
4073 $message = str_replace(
4074 $images[0][$imgindex],
4075 $images[1][$imgindex] . '="cid:' . $cid . '"',
4076 $message
4077 );
4078 continue;
4079 }
Matthias Andreas Benkarde39c4f82021-01-06 17:59:39 +01004080 if (
4081 // Only process relative URLs if a basedir is provided (i.e. no absolute local paths)
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +01004082 !empty($basedir)
4083 // Ignore URLs containing parent dir traversal (..)
4084 && (strpos($url, '..') === false)
4085 // Do not change urls that are already inline images
4086 && 0 !== strpos($url, 'cid:')
4087 // Do not change absolute URLs, including anonymous protocol
4088 && !preg_match('#^[a-z][a-z0-9+.-]*:?//#i', $url)
4089 ) {
4090 $filename = static::mb_pathinfo($url, PATHINFO_BASENAME);
4091 $directory = dirname($url);
4092 if ('.' === $directory) {
4093 $directory = '';
4094 }
4095 // RFC2392 S 2
4096 $cid = substr(hash('sha256', $url), 0, 32) . '@phpmailer.0';
4097 if (strlen($basedir) > 1 && '/' !== substr($basedir, -1)) {
4098 $basedir .= '/';
4099 }
4100 if (strlen($directory) > 1 && '/' !== substr($directory, -1)) {
4101 $directory .= '/';
4102 }
Matthias Andreas Benkarde39c4f82021-01-06 17:59:39 +01004103 if (
4104 $this->addEmbeddedImage(
4105 $basedir . $directory . $filename,
4106 $cid,
4107 $filename,
4108 static::ENCODING_BASE64,
4109 static::_mime_types((string) static::mb_pathinfo($filename, PATHINFO_EXTENSION))
4110 )
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +01004111 ) {
4112 $message = preg_replace(
4113 '/' . $images[1][$imgindex] . '=["\']' . preg_quote($url, '/') . '["\']/Ui',
4114 $images[1][$imgindex] . '="cid:' . $cid . '"',
4115 $message
4116 );
4117 }
4118 }
4119 }
4120 }
4121 $this->isHTML();
4122 // Convert all message body line breaks to LE, makes quoted-printable encoding work much better
4123 $this->Body = static::normalizeBreaks($message);
4124 $this->AltBody = static::normalizeBreaks($this->html2text($message, $advanced));
4125 if (!$this->alternativeExists()) {
4126 $this->AltBody = 'This is an HTML-only message. To view it, activate HTML in your email application.'
4127 . static::$LE;
4128 }
4129
4130 return $this->Body;
4131 }
4132
4133 /**
4134 * Convert an HTML string into plain text.
4135 * This is used by msgHTML().
4136 * Note - older versions of this function used a bundled advanced converter
4137 * which was removed for license reasons in #232.
4138 * Example usage:
4139 *
4140 * ```php
4141 * // Use default conversion
4142 * $plain = $mail->html2text($html);
4143 * // Use your own custom converter
4144 * $plain = $mail->html2text($html, function($html) {
4145 * $converter = new MyHtml2text($html);
4146 * return $converter->get_text();
4147 * });
4148 * ```
4149 *
4150 * @param string $html The HTML text to convert
4151 * @param bool|callable $advanced Any boolean value to use the internal converter,
4152 * or provide your own callable for custom conversion
4153 *
4154 * @return string
4155 */
4156 public function html2text($html, $advanced = false)
4157 {
4158 if (is_callable($advanced)) {
Matthias Andreas Benkarde39c4f82021-01-06 17:59:39 +01004159 return call_user_func($advanced, $html);
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +01004160 }
4161
4162 return html_entity_decode(
4163 trim(strip_tags(preg_replace('/<(head|title|style|script)[^>]*>.*?<\/\\1>/si', '', $html))),
4164 ENT_QUOTES,
4165 $this->CharSet
4166 );
4167 }
4168
4169 /**
4170 * Get the MIME type for a file extension.
4171 *
4172 * @param string $ext File extension
4173 *
4174 * @return string MIME type of file
4175 */
4176 public static function _mime_types($ext = '')
4177 {
4178 $mimes = [
4179 'xl' => 'application/excel',
4180 'js' => 'application/javascript',
4181 'hqx' => 'application/mac-binhex40',
4182 'cpt' => 'application/mac-compactpro',
4183 'bin' => 'application/macbinary',
4184 'doc' => 'application/msword',
4185 'word' => 'application/msword',
4186 'xlsx' => 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet',
4187 'xltx' => 'application/vnd.openxmlformats-officedocument.spreadsheetml.template',
4188 'potx' => 'application/vnd.openxmlformats-officedocument.presentationml.template',
4189 'ppsx' => 'application/vnd.openxmlformats-officedocument.presentationml.slideshow',
4190 'pptx' => 'application/vnd.openxmlformats-officedocument.presentationml.presentation',
4191 'sldx' => 'application/vnd.openxmlformats-officedocument.presentationml.slide',
4192 'docx' => 'application/vnd.openxmlformats-officedocument.wordprocessingml.document',
4193 'dotx' => 'application/vnd.openxmlformats-officedocument.wordprocessingml.template',
4194 'xlam' => 'application/vnd.ms-excel.addin.macroEnabled.12',
4195 'xlsb' => 'application/vnd.ms-excel.sheet.binary.macroEnabled.12',
4196 'class' => 'application/octet-stream',
4197 'dll' => 'application/octet-stream',
4198 'dms' => 'application/octet-stream',
4199 'exe' => 'application/octet-stream',
4200 'lha' => 'application/octet-stream',
4201 'lzh' => 'application/octet-stream',
4202 'psd' => 'application/octet-stream',
4203 'sea' => 'application/octet-stream',
4204 'so' => 'application/octet-stream',
4205 'oda' => 'application/oda',
4206 'pdf' => 'application/pdf',
4207 'ai' => 'application/postscript',
4208 'eps' => 'application/postscript',
4209 'ps' => 'application/postscript',
4210 'smi' => 'application/smil',
4211 'smil' => 'application/smil',
4212 'mif' => 'application/vnd.mif',
4213 'xls' => 'application/vnd.ms-excel',
4214 'ppt' => 'application/vnd.ms-powerpoint',
4215 'wbxml' => 'application/vnd.wap.wbxml',
4216 'wmlc' => 'application/vnd.wap.wmlc',
4217 'dcr' => 'application/x-director',
4218 'dir' => 'application/x-director',
4219 'dxr' => 'application/x-director',
4220 'dvi' => 'application/x-dvi',
4221 'gtar' => 'application/x-gtar',
4222 'php3' => 'application/x-httpd-php',
4223 'php4' => 'application/x-httpd-php',
4224 'php' => 'application/x-httpd-php',
4225 'phtml' => 'application/x-httpd-php',
4226 'phps' => 'application/x-httpd-php-source',
4227 'swf' => 'application/x-shockwave-flash',
4228 'sit' => 'application/x-stuffit',
4229 'tar' => 'application/x-tar',
4230 'tgz' => 'application/x-tar',
4231 'xht' => 'application/xhtml+xml',
4232 'xhtml' => 'application/xhtml+xml',
4233 'zip' => 'application/zip',
4234 'mid' => 'audio/midi',
4235 'midi' => 'audio/midi',
4236 'mp2' => 'audio/mpeg',
4237 'mp3' => 'audio/mpeg',
4238 'm4a' => 'audio/mp4',
4239 'mpga' => 'audio/mpeg',
4240 'aif' => 'audio/x-aiff',
4241 'aifc' => 'audio/x-aiff',
4242 'aiff' => 'audio/x-aiff',
4243 'ram' => 'audio/x-pn-realaudio',
4244 'rm' => 'audio/x-pn-realaudio',
4245 'rpm' => 'audio/x-pn-realaudio-plugin',
4246 'ra' => 'audio/x-realaudio',
4247 'wav' => 'audio/x-wav',
4248 'mka' => 'audio/x-matroska',
4249 'bmp' => 'image/bmp',
4250 'gif' => 'image/gif',
4251 'jpeg' => 'image/jpeg',
4252 'jpe' => 'image/jpeg',
4253 'jpg' => 'image/jpeg',
4254 'png' => 'image/png',
4255 'tiff' => 'image/tiff',
4256 'tif' => 'image/tiff',
4257 'webp' => 'image/webp',
Matthias Andreas Benkarde39c4f82021-01-06 17:59:39 +01004258 'avif' => 'image/avif',
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +01004259 'heif' => 'image/heif',
4260 'heifs' => 'image/heif-sequence',
4261 'heic' => 'image/heic',
4262 'heics' => 'image/heic-sequence',
4263 'eml' => 'message/rfc822',
4264 'css' => 'text/css',
4265 'html' => 'text/html',
4266 'htm' => 'text/html',
4267 'shtml' => 'text/html',
4268 'log' => 'text/plain',
4269 'text' => 'text/plain',
4270 'txt' => 'text/plain',
4271 'rtx' => 'text/richtext',
4272 'rtf' => 'text/rtf',
4273 'vcf' => 'text/vcard',
4274 'vcard' => 'text/vcard',
4275 'ics' => 'text/calendar',
4276 'xml' => 'text/xml',
4277 'xsl' => 'text/xml',
4278 'wmv' => 'video/x-ms-wmv',
4279 'mpeg' => 'video/mpeg',
4280 'mpe' => 'video/mpeg',
4281 'mpg' => 'video/mpeg',
4282 'mp4' => 'video/mp4',
4283 'm4v' => 'video/mp4',
4284 'mov' => 'video/quicktime',
4285 'qt' => 'video/quicktime',
4286 'rv' => 'video/vnd.rn-realvideo',
4287 'avi' => 'video/x-msvideo',
4288 'movie' => 'video/x-sgi-movie',
4289 'webm' => 'video/webm',
4290 'mkv' => 'video/x-matroska',
4291 ];
4292 $ext = strtolower($ext);
4293 if (array_key_exists($ext, $mimes)) {
4294 return $mimes[$ext];
4295 }
4296
4297 return 'application/octet-stream';
4298 }
4299
4300 /**
4301 * Map a file name to a MIME type.
4302 * Defaults to 'application/octet-stream', i.e.. arbitrary binary data.
4303 *
4304 * @param string $filename A file name or full path, does not need to exist as a file
4305 *
4306 * @return string
4307 */
4308 public static function filenameToType($filename)
4309 {
4310 // In case the path is a URL, strip any query string before getting extension
4311 $qpos = strpos($filename, '?');
4312 if (false !== $qpos) {
4313 $filename = substr($filename, 0, $qpos);
4314 }
4315 $ext = static::mb_pathinfo($filename, PATHINFO_EXTENSION);
4316
4317 return static::_mime_types($ext);
4318 }
4319
4320 /**
4321 * Multi-byte-safe pathinfo replacement.
4322 * Drop-in replacement for pathinfo(), but multibyte- and cross-platform-safe.
4323 *
4324 * @see http://www.php.net/manual/en/function.pathinfo.php#107461
4325 *
4326 * @param string $path A filename or path, does not need to exist as a file
4327 * @param int|string $options Either a PATHINFO_* constant,
4328 * or a string name to return only the specified piece
4329 *
4330 * @return string|array
4331 */
4332 public static function mb_pathinfo($path, $options = null)
4333 {
4334 $ret = ['dirname' => '', 'basename' => '', 'extension' => '', 'filename' => ''];
4335 $pathinfo = [];
4336 if (preg_match('#^(.*?)[\\\\/]*(([^/\\\\]*?)(\.([^.\\\\/]+?)|))[\\\\/.]*$#m', $path, $pathinfo)) {
4337 if (array_key_exists(1, $pathinfo)) {
4338 $ret['dirname'] = $pathinfo[1];
4339 }
4340 if (array_key_exists(2, $pathinfo)) {
4341 $ret['basename'] = $pathinfo[2];
4342 }
4343 if (array_key_exists(5, $pathinfo)) {
4344 $ret['extension'] = $pathinfo[5];
4345 }
4346 if (array_key_exists(3, $pathinfo)) {
4347 $ret['filename'] = $pathinfo[3];
4348 }
4349 }
4350 switch ($options) {
4351 case PATHINFO_DIRNAME:
4352 case 'dirname':
4353 return $ret['dirname'];
4354 case PATHINFO_BASENAME:
4355 case 'basename':
4356 return $ret['basename'];
4357 case PATHINFO_EXTENSION:
4358 case 'extension':
4359 return $ret['extension'];
4360 case PATHINFO_FILENAME:
4361 case 'filename':
4362 return $ret['filename'];
4363 default:
4364 return $ret;
4365 }
4366 }
4367
4368 /**
4369 * Set or reset instance properties.
4370 * You should avoid this function - it's more verbose, less efficient, more error-prone and
4371 * harder to debug than setting properties directly.
4372 * Usage Example:
4373 * `$mail->set('SMTPSecure', static::ENCRYPTION_STARTTLS);`
4374 * is the same as:
4375 * `$mail->SMTPSecure = static::ENCRYPTION_STARTTLS;`.
4376 *
4377 * @param string $name The property name to set
4378 * @param mixed $value The value to set the property to
4379 *
4380 * @return bool
4381 */
4382 public function set($name, $value = '')
4383 {
4384 if (property_exists($this, $name)) {
4385 $this->$name = $value;
4386
4387 return true;
4388 }
4389 $this->setError($this->lang('variable_set') . $name);
4390
4391 return false;
4392 }
4393
4394 /**
4395 * Strip newlines to prevent header injection.
4396 *
4397 * @param string $str
4398 *
4399 * @return string
4400 */
4401 public function secureHeader($str)
4402 {
4403 return trim(str_replace(["\r", "\n"], '', $str));
4404 }
4405
4406 /**
4407 * Normalize line breaks in a string.
4408 * Converts UNIX LF, Mac CR and Windows CRLF line breaks into a single line break format.
4409 * Defaults to CRLF (for message bodies) and preserves consecutive breaks.
4410 *
4411 * @param string $text
4412 * @param string $breaktype What kind of line break to use; defaults to static::$LE
4413 *
4414 * @return string
4415 */
4416 public static function normalizeBreaks($text, $breaktype = null)
4417 {
4418 if (null === $breaktype) {
4419 $breaktype = static::$LE;
4420 }
4421 // Normalise to \n
4422 $text = str_replace([self::CRLF, "\r"], "\n", $text);
4423 // Now convert LE as needed
4424 if ("\n" !== $breaktype) {
4425 $text = str_replace("\n", $breaktype, $text);
4426 }
4427
4428 return $text;
4429 }
4430
4431 /**
4432 * Remove trailing breaks from a string.
4433 *
4434 * @param string $text
4435 *
4436 * @return string The text to remove breaks from
4437 */
4438 public static function stripTrailingWSP($text)
4439 {
4440 return rtrim($text, " \r\n\t");
4441 }
4442
4443 /**
4444 * Return the current line break format string.
4445 *
4446 * @return string
4447 */
4448 public static function getLE()
4449 {
4450 return static::$LE;
4451 }
4452
4453 /**
4454 * Set the line break format string, e.g. "\r\n".
4455 *
4456 * @param string $le
4457 */
4458 protected static function setLE($le)
4459 {
4460 static::$LE = $le;
4461 }
4462
4463 /**
4464 * Set the public and private key files and password for S/MIME signing.
4465 *
4466 * @param string $cert_filename
4467 * @param string $key_filename
4468 * @param string $key_pass Password for private key
4469 * @param string $extracerts_filename Optional path to chain certificate
4470 */
4471 public function sign($cert_filename, $key_filename, $key_pass, $extracerts_filename = '')
4472 {
4473 $this->sign_cert_file = $cert_filename;
4474 $this->sign_key_file = $key_filename;
4475 $this->sign_key_pass = $key_pass;
4476 $this->sign_extracerts_file = $extracerts_filename;
4477 }
4478
4479 /**
4480 * Quoted-Printable-encode a DKIM header.
4481 *
4482 * @param string $txt
4483 *
4484 * @return string
4485 */
4486 public function DKIM_QP($txt)
4487 {
4488 $line = '';
4489 $len = strlen($txt);
4490 for ($i = 0; $i < $len; ++$i) {
4491 $ord = ord($txt[$i]);
4492 if (((0x21 <= $ord) && ($ord <= 0x3A)) || $ord === 0x3C || ((0x3E <= $ord) && ($ord <= 0x7E))) {
4493 $line .= $txt[$i];
4494 } else {
4495 $line .= '=' . sprintf('%02X', $ord);
4496 }
4497 }
4498
4499 return $line;
4500 }
4501
4502 /**
4503 * Generate a DKIM signature.
4504 *
4505 * @param string $signHeader
4506 *
4507 * @throws Exception
4508 *
4509 * @return string The DKIM signature value
4510 */
4511 public function DKIM_Sign($signHeader)
4512 {
4513 if (!defined('PKCS7_TEXT')) {
4514 if ($this->exceptions) {
4515 throw new Exception($this->lang('extension_missing') . 'openssl');
4516 }
4517
4518 return '';
4519 }
4520 $privKeyStr = !empty($this->DKIM_private_string) ?
4521 $this->DKIM_private_string :
4522 file_get_contents($this->DKIM_private);
4523 if ('' !== $this->DKIM_passphrase) {
4524 $privKey = openssl_pkey_get_private($privKeyStr, $this->DKIM_passphrase);
4525 } else {
4526 $privKey = openssl_pkey_get_private($privKeyStr);
4527 }
4528 if (openssl_sign($signHeader, $signature, $privKey, 'sha256WithRSAEncryption')) {
Matthias Andreas Benkarde39c4f82021-01-06 17:59:39 +01004529 if (PHP_MAJOR_VERSION < 8) {
4530 openssl_pkey_free($privKey);
4531 }
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +01004532
4533 return base64_encode($signature);
4534 }
Matthias Andreas Benkarde39c4f82021-01-06 17:59:39 +01004535 if (PHP_MAJOR_VERSION < 8) {
4536 openssl_pkey_free($privKey);
4537 }
Matthias Andreas Benkardb382b102021-01-02 15:32:21 +01004538
4539 return '';
4540 }
4541
4542 /**
4543 * Generate a DKIM canonicalization header.
4544 * Uses the 'relaxed' algorithm from RFC6376 section 3.4.2.
4545 * Canonicalized headers should *always* use CRLF, regardless of mailer setting.
4546 *
4547 * @see https://tools.ietf.org/html/rfc6376#section-3.4.2
4548 *
4549 * @param string $signHeader Header
4550 *
4551 * @return string
4552 */
4553 public function DKIM_HeaderC($signHeader)
4554 {
4555 //Normalize breaks to CRLF (regardless of the mailer)
4556 $signHeader = static::normalizeBreaks($signHeader, self::CRLF);
4557 //Unfold header lines
4558 //Note PCRE \s is too broad a definition of whitespace; RFC5322 defines it as `[ \t]`
4559 //@see https://tools.ietf.org/html/rfc5322#section-2.2
4560 //That means this may break if you do something daft like put vertical tabs in your headers.
4561 $signHeader = preg_replace('/\r\n[ \t]+/', ' ', $signHeader);
4562 //Break headers out into an array
4563 $lines = explode(self::CRLF, $signHeader);
4564 foreach ($lines as $key => $line) {
4565 //If the header is missing a :, skip it as it's invalid
4566 //This is likely to happen because the explode() above will also split
4567 //on the trailing LE, leaving an empty line
4568 if (strpos($line, ':') === false) {
4569 continue;
4570 }
4571 list($heading, $value) = explode(':', $line, 2);
4572 //Lower-case header name
4573 $heading = strtolower($heading);
4574 //Collapse white space within the value, also convert WSP to space
4575 $value = preg_replace('/[ \t]+/', ' ', $value);
4576 //RFC6376 is slightly unclear here - it says to delete space at the *end* of each value
4577 //But then says to delete space before and after the colon.
4578 //Net result is the same as trimming both ends of the value.
4579 //By elimination, the same applies to the field name
4580 $lines[$key] = trim($heading, " \t") . ':' . trim($value, " \t");
4581 }
4582
4583 return implode(self::CRLF, $lines);
4584 }
4585
4586 /**
4587 * Generate a DKIM canonicalization body.
4588 * Uses the 'simple' algorithm from RFC6376 section 3.4.3.
4589 * Canonicalized bodies should *always* use CRLF, regardless of mailer setting.
4590 *
4591 * @see https://tools.ietf.org/html/rfc6376#section-3.4.3
4592 *
4593 * @param string $body Message Body
4594 *
4595 * @return string
4596 */
4597 public function DKIM_BodyC($body)
4598 {
4599 if (empty($body)) {
4600 return self::CRLF;
4601 }
4602 // Normalize line endings to CRLF
4603 $body = static::normalizeBreaks($body, self::CRLF);
4604
4605 //Reduce multiple trailing line breaks to a single one
4606 return static::stripTrailingWSP($body) . self::CRLF;
4607 }
4608
4609 /**
4610 * Create the DKIM header and body in a new message header.
4611 *
4612 * @param string $headers_line Header lines
4613 * @param string $subject Subject
4614 * @param string $body Body
4615 *
4616 * @throws Exception
4617 *
4618 * @return string
4619 */
4620 public function DKIM_Add($headers_line, $subject, $body)
4621 {
4622 $DKIMsignatureType = 'rsa-sha256'; // Signature & hash algorithms
4623 $DKIMcanonicalization = 'relaxed/simple'; // Canonicalization methods of header & body
4624 $DKIMquery = 'dns/txt'; // Query method
4625 $DKIMtime = time();
4626 //Always sign these headers without being asked
4627 //Recommended list from https://tools.ietf.org/html/rfc6376#section-5.4.1
4628 $autoSignHeaders = [
4629 'from',
4630 'to',
4631 'cc',
4632 'date',
4633 'subject',
4634 'reply-to',
4635 'message-id',
4636 'content-type',
4637 'mime-version',
4638 'x-mailer',
4639 ];
4640 if (stripos($headers_line, 'Subject') === false) {
4641 $headers_line .= 'Subject: ' . $subject . static::$LE;
4642 }
4643 $headerLines = explode(static::$LE, $headers_line);
4644 $currentHeaderLabel = '';
4645 $currentHeaderValue = '';
4646 $parsedHeaders = [];
4647 $headerLineIndex = 0;
4648 $headerLineCount = count($headerLines);
4649 foreach ($headerLines as $headerLine) {
4650 $matches = [];
4651 if (preg_match('/^([^ \t]*?)(?::[ \t]*)(.*)$/', $headerLine, $matches)) {
4652 if ($currentHeaderLabel !== '') {
4653 //We were previously in another header; This is the start of a new header, so save the previous one
4654 $parsedHeaders[] = ['label' => $currentHeaderLabel, 'value' => $currentHeaderValue];
4655 }
4656 $currentHeaderLabel = $matches[1];
4657 $currentHeaderValue = $matches[2];
4658 } elseif (preg_match('/^[ \t]+(.*)$/', $headerLine, $matches)) {
4659 //This is a folded continuation of the current header, so unfold it
4660 $currentHeaderValue .= ' ' . $matches[1];
4661 }
4662 ++$headerLineIndex;
4663 if ($headerLineIndex >= $headerLineCount) {
4664 //This was the last line, so finish off this header
4665 $parsedHeaders[] = ['label' => $currentHeaderLabel, 'value' => $currentHeaderValue];
4666 }
4667 }
4668 $copiedHeaders = [];
4669 $headersToSignKeys = [];
4670 $headersToSign = [];
4671 foreach ($parsedHeaders as $header) {
4672 //Is this header one that must be included in the DKIM signature?
4673 if (in_array(strtolower($header['label']), $autoSignHeaders, true)) {
4674 $headersToSignKeys[] = $header['label'];
4675 $headersToSign[] = $header['label'] . ': ' . $header['value'];
4676 if ($this->DKIM_copyHeaderFields) {
4677 $copiedHeaders[] = $header['label'] . ':' . //Note no space after this, as per RFC
4678 str_replace('|', '=7C', $this->DKIM_QP($header['value']));
4679 }
4680 continue;
4681 }
4682 //Is this an extra custom header we've been asked to sign?
4683 if (in_array($header['label'], $this->DKIM_extraHeaders, true)) {
4684 //Find its value in custom headers
4685 foreach ($this->CustomHeader as $customHeader) {
4686 if ($customHeader[0] === $header['label']) {
4687 $headersToSignKeys[] = $header['label'];
4688 $headersToSign[] = $header['label'] . ': ' . $header['value'];
4689 if ($this->DKIM_copyHeaderFields) {
4690 $copiedHeaders[] = $header['label'] . ':' . //Note no space after this, as per RFC
4691 str_replace('|', '=7C', $this->DKIM_QP($header['value']));
4692 }
4693 //Skip straight to the next header
4694 continue 2;
4695 }
4696 }
4697 }
4698 }
4699 $copiedHeaderFields = '';
4700 if ($this->DKIM_copyHeaderFields && count($copiedHeaders) > 0) {
4701 //Assemble a DKIM 'z' tag
4702 $copiedHeaderFields = ' z=';
4703 $first = true;
4704 foreach ($copiedHeaders as $copiedHeader) {
4705 if (!$first) {
4706 $copiedHeaderFields .= static::$LE . ' |';
4707 }
4708 //Fold long values
4709 if (strlen($copiedHeader) > self::STD_LINE_LENGTH - 3) {
4710 $copiedHeaderFields .= substr(
4711 chunk_split($copiedHeader, self::STD_LINE_LENGTH - 3, static::$LE . self::FWS),
4712 0,
4713 -strlen(static::$LE . self::FWS)
4714 );
4715 } else {
4716 $copiedHeaderFields .= $copiedHeader;
4717 }
4718 $first = false;
4719 }
4720 $copiedHeaderFields .= ';' . static::$LE;
4721 }
4722 $headerKeys = ' h=' . implode(':', $headersToSignKeys) . ';' . static::$LE;
4723 $headerValues = implode(static::$LE, $headersToSign);
4724 $body = $this->DKIM_BodyC($body);
4725 $DKIMb64 = base64_encode(pack('H*', hash('sha256', $body))); // Base64 of packed binary SHA-256 hash of body
4726 $ident = '';
4727 if ('' !== $this->DKIM_identity) {
4728 $ident = ' i=' . $this->DKIM_identity . ';' . static::$LE;
4729 }
4730 //The DKIM-Signature header is included in the signature *except for* the value of the `b` tag
4731 //which is appended after calculating the signature
4732 //https://tools.ietf.org/html/rfc6376#section-3.5
4733 $dkimSignatureHeader = 'DKIM-Signature: v=1;' .
4734 ' d=' . $this->DKIM_domain . ';' .
4735 ' s=' . $this->DKIM_selector . ';' . static::$LE .
4736 ' a=' . $DKIMsignatureType . ';' .
4737 ' q=' . $DKIMquery . ';' .
4738 ' t=' . $DKIMtime . ';' .
4739 ' c=' . $DKIMcanonicalization . ';' . static::$LE .
4740 $headerKeys .
4741 $ident .
4742 $copiedHeaderFields .
4743 ' bh=' . $DKIMb64 . ';' . static::$LE .
4744 ' b=';
4745 //Canonicalize the set of headers
4746 $canonicalizedHeaders = $this->DKIM_HeaderC(
4747 $headerValues . static::$LE . $dkimSignatureHeader
4748 );
4749 $signature = $this->DKIM_Sign($canonicalizedHeaders);
4750 $signature = trim(chunk_split($signature, self::STD_LINE_LENGTH - 3, static::$LE . self::FWS));
4751
4752 return static::normalizeBreaks($dkimSignatureHeader . $signature);
4753 }
4754
4755 /**
4756 * Detect if a string contains a line longer than the maximum line length
4757 * allowed by RFC 2822 section 2.1.1.
4758 *
4759 * @param string $str
4760 *
4761 * @return bool
4762 */
4763 public static function hasLineLongerThanMax($str)
4764 {
4765 return (bool) preg_match('/^(.{' . (self::MAX_LINE_LENGTH + strlen(static::$LE)) . ',})/m', $str);
4766 }
4767
4768 /**
4769 * If a string contains any "special" characters, double-quote the name,
4770 * and escape any double quotes with a backslash.
4771 *
4772 * @param string $str
4773 *
4774 * @return string
4775 *
4776 * @see RFC822 3.4.1
4777 */
4778 public static function quotedString($str)
4779 {
4780 if (preg_match('/[ ()<>@,;:"\/\[\]?=]/', $str)) {
4781 //If the string contains any of these chars, it must be double-quoted
4782 //and any double quotes must be escaped with a backslash
4783 return '"' . str_replace('"', '\\"', $str) . '"';
4784 }
4785
4786 //Return the string untouched, it doesn't need quoting
4787 return $str;
4788 }
4789
4790 /**
4791 * Allows for public read access to 'to' property.
4792 * Before the send() call, queued addresses (i.e. with IDN) are not yet included.
4793 *
4794 * @return array
4795 */
4796 public function getToAddresses()
4797 {
4798 return $this->to;
4799 }
4800
4801 /**
4802 * Allows for public read access to 'cc' property.
4803 * Before the send() call, queued addresses (i.e. with IDN) are not yet included.
4804 *
4805 * @return array
4806 */
4807 public function getCcAddresses()
4808 {
4809 return $this->cc;
4810 }
4811
4812 /**
4813 * Allows for public read access to 'bcc' property.
4814 * Before the send() call, queued addresses (i.e. with IDN) are not yet included.
4815 *
4816 * @return array
4817 */
4818 public function getBccAddresses()
4819 {
4820 return $this->bcc;
4821 }
4822
4823 /**
4824 * Allows for public read access to 'ReplyTo' property.
4825 * Before the send() call, queued addresses (i.e. with IDN) are not yet included.
4826 *
4827 * @return array
4828 */
4829 public function getReplyToAddresses()
4830 {
4831 return $this->ReplyTo;
4832 }
4833
4834 /**
4835 * Allows for public read access to 'all_recipients' property.
4836 * Before the send() call, queued addresses (i.e. with IDN) are not yet included.
4837 *
4838 * @return array
4839 */
4840 public function getAllRecipientAddresses()
4841 {
4842 return $this->all_recipients;
4843 }
4844
4845 /**
4846 * Perform a callback.
4847 *
4848 * @param bool $isSent
4849 * @param array $to
4850 * @param array $cc
4851 * @param array $bcc
4852 * @param string $subject
4853 * @param string $body
4854 * @param string $from
4855 * @param array $extra
4856 */
4857 protected function doCallback($isSent, $to, $cc, $bcc, $subject, $body, $from, $extra)
4858 {
4859 if (!empty($this->action_function) && is_callable($this->action_function)) {
4860 call_user_func($this->action_function, $isSent, $to, $cc, $bcc, $subject, $body, $from, $extra);
4861 }
4862 }
4863
4864 /**
4865 * Get the OAuth instance.
4866 *
4867 * @return OAuth
4868 */
4869 public function getOAuth()
4870 {
4871 return $this->oauth;
4872 }
4873
4874 /**
4875 * Set an OAuth instance.
4876 */
4877 public function setOAuth(OAuth $oauth)
4878 {
4879 $this->oauth = $oauth;
4880 }
4881}