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