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