1: | <?php
|
2: | |
3: | |
4: | |
5: | |
6: | |
7: | |
8: | |
9: | |
10: | |
11: | |
12: | |
13: | |
14: | |
15: | |
16: | |
17: | |
18: |
|
19: |
|
20: | |
21: | |
22: | |
23: | |
24: | |
25: | |
26: | |
27: |
|
28: | class PHPMailer
|
29: | {
|
30: | |
31: | |
32: | |
33: |
|
34: | public $Version = '5.2.28';
|
35: |
|
36: | |
37: | |
38: | |
39: | |
40: | |
41: |
|
42: | public $Priority = null;
|
43: |
|
44: | |
45: | |
46: | |
47: |
|
48: | public $CharSet = 'iso-8859-1';
|
49: |
|
50: | |
51: | |
52: | |
53: |
|
54: | public $ContentType = 'text/plain';
|
55: |
|
56: | |
57: | |
58: | |
59: | |
60: |
|
61: | public $Encoding = '8bit';
|
62: |
|
63: | |
64: | |
65: | |
66: |
|
67: | public $ErrorInfo = '';
|
68: |
|
69: | |
70: | |
71: | |
72: |
|
73: | public $From = 'root@localhost';
|
74: |
|
75: | |
76: | |
77: | |
78: |
|
79: | public $FromName = 'Root User';
|
80: |
|
81: | |
82: | |
83: | |
84: | |
85: |
|
86: | public $Sender = '';
|
87: |
|
88: | |
89: | |
90: | |
91: | |
92: | |
93: | |
94: | |
95: |
|
96: | public $ReturnPath = '';
|
97: |
|
98: | |
99: | |
100: | |
101: |
|
102: | public $Subject = '';
|
103: |
|
104: | |
105: | |
106: | |
107: | |
108: |
|
109: | public $Body = '';
|
110: |
|
111: | |
112: | |
113: | |
114: | |
115: | |
116: | |
117: |
|
118: | public $AltBody = '';
|
119: |
|
120: | |
121: | |
122: | |
123: | |
124: | |
125: | |
126: | |
127: |
|
128: | public $Ical = '';
|
129: |
|
130: | |
131: | |
132: | |
133: | |
134: |
|
135: | protected $MIMEBody = '';
|
136: |
|
137: | |
138: | |
139: | |
140: | |
141: |
|
142: | protected $MIMEHeader = '';
|
143: |
|
144: | |
145: | |
146: | |
147: | |
148: |
|
149: | protected $mailHeader = '';
|
150: |
|
151: | |
152: | |
153: | |
154: | |
155: |
|
156: | public $WordWrap = 0;
|
157: |
|
158: | |
159: | |
160: | |
161: | |
162: |
|
163: | public $Mailer = 'mail';
|
164: |
|
165: | |
166: | |
167: | |
168: |
|
169: | public $Sendmail = '/usr/sbin/sendmail';
|
170: |
|
171: | |
172: | |
173: | |
174: | |
175: |
|
176: | public $UseSendmailOptions = true;
|
177: |
|
178: | |
179: | |
180: | |
181: | |
182: | |
183: |
|
184: | public $PluginDir = '';
|
185: |
|
186: | |
187: | |
188: | |
189: |
|
190: | public $ConfirmReadingTo = '';
|
191: |
|
192: | |
193: | |
194: | |
195: | |
196: | |
197: | |
198: |
|
199: | public $Hostname = '';
|
200: |
|
201: | |
202: | |
203: | |
204: | |
205: | |
206: | |
207: | |
208: |
|
209: | public $MessageID = '';
|
210: |
|
211: | |
212: | |
213: | |
214: | |
215: |
|
216: | public $MessageDate = '';
|
217: |
|
218: | |
219: | |
220: | |
221: | |
222: | |
223: | |
224: | |
225: | |
226: | |
227: | |
228: |
|
229: | public $Host = 'localhost';
|
230: |
|
231: | |
232: | |
233: | |
234: | |
235: |
|
236: | public $Port = 25;
|
237: |
|
238: | |
239: | |
240: | |
241: | |
242: | |
243: | |
244: |
|
245: | public $Helo = '';
|
246: |
|
247: | |
248: | |
249: | |
250: | |
251: |
|
252: | public $SMTPSecure = '';
|
253: |
|
254: | |
255: | |
256: | |
257: | |
258: | |
259: |
|
260: | public $SMTPAutoTLS = true;
|
261: |
|
262: | |
263: | |
264: | |
265: | |
266: | |
267: | |
268: |
|
269: | public $SMTPAuth = false;
|
270: |
|
271: | |
272: | |
273: | |
274: |
|
275: | public $SMTPOptions = array();
|
276: |
|
277: | |
278: | |
279: | |
280: |
|
281: | public $Username = '';
|
282: |
|
283: | |
284: | |
285: | |
286: |
|
287: | public $Password = '';
|
288: |
|
289: | |
290: | |
291: | |
292: | |
293: |
|
294: | public $AuthType = '';
|
295: |
|
296: | |
297: | |
298: | |
299: | |
300: |
|
301: | public $Realm = '';
|
302: |
|
303: | |
304: | |
305: | |
306: | |
307: |
|
308: | public $Workstation = '';
|
309: |
|
310: | |
311: | |
312: | |
313: | |
314: |
|
315: | public $Timeout = 300;
|
316: |
|
317: | |
318: | |
319: | |
320: | |
321: | |
322: | |
323: | |
324: | |
325: | |
326: | |
327: | |
328: |
|
329: | public $SMTPDebug = 0;
|
330: |
|
331: | |
332: | |
333: | |
334: | |
335: | |
336: | |
337: | |
338: | |
339: | |
340: | |
341: | |
342: | |
343: | |
344: |
|
345: | public $Debugoutput = 'echo';
|
346: |
|
347: | |
348: | |
349: | |
350: | |
351: | |
352: |
|
353: | public $SMTPKeepAlive = false;
|
354: |
|
355: | |
356: | |
357: | |
358: | |
359: | |
360: |
|
361: | public $SingleTo = false;
|
362: |
|
363: | |
364: | |
365: | |
366: | |
367: |
|
368: | public $SingleToArray = array();
|
369: |
|
370: | |
371: | |
372: | |
373: | |
374: | |
375: | |
376: |
|
377: | public $do_verp = false;
|
378: |
|
379: | |
380: | |
381: | |
382: |
|
383: | public $AllowEmpty = false;
|
384: |
|
385: | |
386: | |
387: | |
388: | |
389: | |
390: |
|
391: | public $LE = "\n";
|
392: |
|
393: | |
394: | |
395: | |
396: |
|
397: | public $DKIM_selector = '';
|
398: |
|
399: | |
400: | |
401: | |
402: | |
403: |
|
404: | public $DKIM_identity = '';
|
405: |
|
406: | |
407: | |
408: | |
409: | |
410: |
|
411: | public $DKIM_passphrase = '';
|
412: |
|
413: | |
414: | |
415: | |
416: | |
417: |
|
418: | public $DKIM_domain = '';
|
419: |
|
420: | |
421: | |
422: | |
423: |
|
424: | public $DKIM_private = '';
|
425: |
|
426: | |
427: | |
428: | |
429: | |
430: |
|
431: | public $DKIM_private_string = '';
|
432: |
|
433: | |
434: | |
435: | |
436: | |
437: | |
438: | |
439: | |
440: | |
441: | |
442: | |
443: | |
444: | |
445: | |
446: | |
447: | |
448: | |
449: | |
450: |
|
451: | public $action_function = '';
|
452: |
|
453: | |
454: | |
455: | |
456: | |
457: |
|
458: | public $XMailer = '';
|
459: |
|
460: | |
461: | |
462: | |
463: | |
464: | |
465: | |
466: |
|
467: | public static $validator = 'auto';
|
468: |
|
469: | |
470: | |
471: | |
472: | |
473: |
|
474: | protected $smtp = null;
|
475: |
|
476: | |
477: | |
478: | |
479: | |
480: |
|
481: | protected $to = array();
|
482: |
|
483: | |
484: | |
485: | |
486: | |
487: |
|
488: | protected $cc = array();
|
489: |
|
490: | |
491: | |
492: | |
493: | |
494: |
|
495: | protected $bcc = array();
|
496: |
|
497: | |
498: | |
499: | |
500: | |
501: |
|
502: | protected $ReplyTo = array();
|
503: |
|
504: | |
505: | |
506: | |
507: | |
508: | |
509: | |
510: |
|
511: | protected $all_recipients = array();
|
512: |
|
513: | |
514: | |
515: | |
516: | |
517: | |
518: | |
519: | |
520: | |
521: | |
522: |
|
523: | protected $RecipientsQueue = array();
|
524: |
|
525: | |
526: | |
527: | |
528: | |
529: | |
530: | |
531: | |
532: |
|
533: | protected $ReplyToQueue = array();
|
534: |
|
535: | |
536: | |
537: | |
538: | |
539: |
|
540: | protected $attachment = array();
|
541: |
|
542: | |
543: | |
544: | |
545: | |
546: |
|
547: | protected $CustomHeader = array();
|
548: |
|
549: | |
550: | |
551: | |
552: | |
553: |
|
554: | protected $lastMessageID = '';
|
555: |
|
556: | |
557: | |
558: | |
559: | |
560: |
|
561: | protected $message_type = '';
|
562: |
|
563: | |
564: | |
565: | |
566: | |
567: |
|
568: | protected $boundary = array();
|
569: |
|
570: | |
571: | |
572: | |
573: | |
574: |
|
575: | protected $language = array();
|
576: |
|
577: | |
578: | |
579: | |
580: | |
581: |
|
582: | protected $error_count = 0;
|
583: |
|
584: | |
585: | |
586: | |
587: | |
588: |
|
589: | protected $sign_cert_file = '';
|
590: |
|
591: | |
592: | |
593: | |
594: | |
595: |
|
596: | protected $sign_key_file = '';
|
597: |
|
598: | |
599: | |
600: | |
601: | |
602: |
|
603: | protected $sign_extracerts_file = '';
|
604: |
|
605: | |
606: | |
607: | |
608: | |
609: | |
610: |
|
611: | protected $sign_key_pass = '';
|
612: |
|
613: | |
614: | |
615: | |
616: | |
617: |
|
618: | protected $exceptions = false;
|
619: |
|
620: | |
621: | |
622: | |
623: | |
624: |
|
625: | protected $uniqueid = '';
|
626: |
|
627: | |
628: | |
629: |
|
630: | const STOP_MESSAGE = 0;
|
631: |
|
632: | |
633: | |
634: |
|
635: | const STOP_CONTINUE = 1;
|
636: |
|
637: | |
638: | |
639: |
|
640: | const STOP_CRITICAL = 2;
|
641: |
|
642: | |
643: | |
644: |
|
645: | const CRLF = "\r\n";
|
646: |
|
647: | |
648: | |
649: | |
650: |
|
651: | const MAX_LINE_LENGTH = 998;
|
652: |
|
653: | |
654: | |
655: | |
656: |
|
657: | public function __construct($exceptions = null)
|
658: | {
|
659: | if ($exceptions !== null) {
|
660: | $this->exceptions = (boolean)$exceptions;
|
661: | }
|
662: |
|
663: | $this->Debugoutput = (strpos(PHP_SAPI, 'cli') !== false ? 'echo' : 'html');
|
664: | }
|
665: |
|
666: | |
667: | |
668: |
|
669: | public function __destruct()
|
670: | {
|
671: |
|
672: | $this->smtpClose();
|
673: | }
|
674: |
|
675: | |
676: | |
677: | |
678: | |
679: | |
680: | |
681: | |
682: | |
683: | |
684: | |
685: | |
686: | |
687: |
|
688: | private function mailPassthru($to, $subject, $body, $header, $params)
|
689: | {
|
690: |
|
691: | if (ini_get('mbstring.func_overload') & 1) {
|
692: | $subject = $this->secureHeader($subject);
|
693: | } else {
|
694: | $subject = $this->encodeHeader($this->secureHeader($subject));
|
695: | }
|
696: |
|
697: |
|
698: |
|
699: | if (ini_get('safe_mode') or !$this->UseSendmailOptions or is_null($params)) {
|
700: | $result = @mail($to, $subject, $body, $header);
|
701: | } else {
|
702: | $result = @mail($to, $subject, $body, $header, $params);
|
703: | }
|
704: | return $result;
|
705: | }
|
706: | |
707: | |
708: | |
709: | |
710: | |
711: | |
712: |
|
713: | protected function edebug($str)
|
714: | {
|
715: | if ($this->SMTPDebug <= 0) {
|
716: | return;
|
717: | }
|
718: |
|
719: | if (!in_array($this->Debugoutput, array('error_log', 'html', 'echo')) and is_callable($this->Debugoutput)) {
|
720: | call_user_func($this->Debugoutput, $str, $this->SMTPDebug);
|
721: | return;
|
722: | }
|
723: | switch ($this->Debugoutput) {
|
724: | case 'error_log':
|
725: |
|
726: | error_log($str);
|
727: | break;
|
728: | case 'html':
|
729: |
|
730: | echo htmlentities(
|
731: | preg_replace('/[\r\n]+/', '', $str),
|
732: | ENT_QUOTES,
|
733: | 'UTF-8'
|
734: | )
|
735: | . "<br>\n";
|
736: | break;
|
737: | case 'echo':
|
738: | default:
|
739: |
|
740: | $str = preg_replace('/\r\n?/ms', "\n", $str);
|
741: | echo gmdate('Y-m-d H:i:s') . "\t" . str_replace(
|
742: | "\n",
|
743: | "\n \t ",
|
744: | trim($str)
|
745: | ) . "\n";
|
746: | }
|
747: | }
|
748: |
|
749: | |
750: | |
751: | |
752: | |
753: |
|
754: | public function isHTML($isHtml = true)
|
755: | {
|
756: | if ($isHtml) {
|
757: | $this->ContentType = 'text/html';
|
758: | } else {
|
759: | $this->ContentType = 'text/plain';
|
760: | }
|
761: | }
|
762: |
|
763: | |
764: | |
765: | |
766: |
|
767: | public function isSMTP()
|
768: | {
|
769: | $this->Mailer = 'smtp';
|
770: | }
|
771: |
|
772: | |
773: | |
774: | |
775: |
|
776: | public function isMail()
|
777: | {
|
778: | $this->Mailer = 'mail';
|
779: | }
|
780: |
|
781: | |
782: | |
783: | |
784: |
|
785: | public function isSendmail()
|
786: | {
|
787: | $ini_sendmail_path = ini_get('sendmail_path');
|
788: |
|
789: | if (!stristr($ini_sendmail_path, 'sendmail')) {
|
790: | $this->Sendmail = '/usr/sbin/sendmail';
|
791: | } else {
|
792: | $this->Sendmail = $ini_sendmail_path;
|
793: | }
|
794: | $this->Mailer = 'sendmail';
|
795: | }
|
796: |
|
797: | |
798: | |
799: | |
800: |
|
801: | public function isQmail()
|
802: | {
|
803: | $ini_sendmail_path = ini_get('sendmail_path');
|
804: |
|
805: | if (!stristr($ini_sendmail_path, 'qmail')) {
|
806: | $this->Sendmail = '/var/qmail/bin/qmail-inject';
|
807: | } else {
|
808: | $this->Sendmail = $ini_sendmail_path;
|
809: | }
|
810: | $this->Mailer = 'qmail';
|
811: | }
|
812: |
|
813: | |
814: | |
815: | |
816: | |
817: | |
818: |
|
819: | public function addAddress($address, $name = '')
|
820: | {
|
821: | return $this->addOrEnqueueAnAddress('to', $address, $name);
|
822: | }
|
823: |
|
824: | |
825: | |
826: | |
827: | |
828: | |
829: | |
830: |
|
831: | public function addCC($address, $name = '')
|
832: | {
|
833: | return $this->addOrEnqueueAnAddress('cc', $address, $name);
|
834: | }
|
835: |
|
836: | |
837: | |
838: | |
839: | |
840: | |
841: | |
842: |
|
843: | public function addBCC($address, $name = '')
|
844: | {
|
845: | return $this->addOrEnqueueAnAddress('bcc', $address, $name);
|
846: | }
|
847: |
|
848: | |
849: | |
850: | |
851: | |
852: | |
853: |
|
854: | public function addReplyTo($address, $name = '')
|
855: | {
|
856: | return $this->addOrEnqueueAnAddress('Reply-To', $address, $name);
|
857: | }
|
858: |
|
859: | |
860: | |
861: | |
862: | |
863: | |
864: | |
865: | |
866: | |
867: | |
868: | |
869: | |
870: |
|
871: | protected function addOrEnqueueAnAddress($kind, $address, $name)
|
872: | {
|
873: | $address = trim($address);
|
874: | $name = trim(preg_replace('/[\r\n]+/', '', $name));
|
875: | if (($pos = strrpos($address, '@')) === false) {
|
876: |
|
877: | $error_message = $this->lang('invalid_address') . " (addAnAddress $kind): $address";
|
878: | $this->setError($error_message);
|
879: | $this->edebug($error_message);
|
880: | if ($this->exceptions) {
|
881: | throw new phpmailerException($error_message);
|
882: | }
|
883: | return false;
|
884: | }
|
885: | $params = array($kind, $address, $name);
|
886: |
|
887: | if ($this->has8bitChars(substr($address, ++$pos)) and $this->idnSupported()) {
|
888: | if ($kind != 'Reply-To') {
|
889: | if (!array_key_exists($address, $this->RecipientsQueue)) {
|
890: | $this->RecipientsQueue[$address] = $params;
|
891: | return true;
|
892: | }
|
893: | } else {
|
894: | if (!array_key_exists($address, $this->ReplyToQueue)) {
|
895: | $this->ReplyToQueue[$address] = $params;
|
896: | return true;
|
897: | }
|
898: | }
|
899: | return false;
|
900: | }
|
901: |
|
902: | return call_user_func_array(array($this, 'addAnAddress'), $params);
|
903: | }
|
904: |
|
905: | |
906: | |
907: | |
908: | |
909: | |
910: | |
911: | |
912: | |
913: | |
914: |
|
915: | protected function addAnAddress($kind, $address, $name = '')
|
916: | {
|
917: | if (!in_array($kind, array('to', 'cc', 'bcc', 'Reply-To'))) {
|
918: | $error_message = $this->lang('Invalid recipient kind: ') . $kind;
|
919: | $this->setError($error_message);
|
920: | $this->edebug($error_message);
|
921: | if ($this->exceptions) {
|
922: | throw new phpmailerException($error_message);
|
923: | }
|
924: | return false;
|
925: | }
|
926: | if (!$this->validateAddress($address)) {
|
927: | $error_message = $this->lang('invalid_address') . " (addAnAddress $kind): $address";
|
928: | $this->setError($error_message);
|
929: | $this->edebug($error_message);
|
930: | if ($this->exceptions) {
|
931: | throw new phpmailerException($error_message);
|
932: | }
|
933: | return false;
|
934: | }
|
935: | if ($kind != 'Reply-To') {
|
936: | if (!array_key_exists(strtolower($address), $this->all_recipients)) {
|
937: | array_push($this->$kind, array($address, $name));
|
938: | $this->all_recipients[strtolower($address)] = true;
|
939: | return true;
|
940: | }
|
941: | } else {
|
942: | if (!array_key_exists(strtolower($address), $this->ReplyTo)) {
|
943: | $this->ReplyTo[strtolower($address)] = array($address, $name);
|
944: | return true;
|
945: | }
|
946: | }
|
947: | return false;
|
948: | }
|
949: |
|
950: | |
951: | |
952: | |
953: | |
954: | |
955: | |
956: | |
957: | |
958: | |
959: |
|
960: | public function parseAddresses($addrstr, $useimap = true)
|
961: | {
|
962: | $addresses = array();
|
963: | if ($useimap and function_exists('imap_rfc822_parse_adrlist')) {
|
964: |
|
965: | $list = imap_rfc822_parse_adrlist($addrstr, '');
|
966: | foreach ($list as $address) {
|
967: | if ($address->host != '.SYNTAX-ERROR.') {
|
968: | if ($this->validateAddress($address->mailbox . '@' . $address->host)) {
|
969: | $addresses[] = array(
|
970: | 'name' => (property_exists($address, 'personal') ? $address->personal : ''),
|
971: | 'address' => $address->mailbox . '@' . $address->host
|
972: | );
|
973: | }
|
974: | }
|
975: | }
|
976: | } else {
|
977: |
|
978: | $list = explode(',', $addrstr);
|
979: | foreach ($list as $address) {
|
980: | $address = trim($address);
|
981: |
|
982: | if (strpos($address, '<') === false) {
|
983: |
|
984: | if ($this->validateAddress($address)) {
|
985: | $addresses[] = array(
|
986: | 'name' => '',
|
987: | 'address' => $address
|
988: | );
|
989: | }
|
990: | } else {
|
991: | list($name, $email) = explode('<', $address);
|
992: | $email = trim(str_replace('>', '', $email));
|
993: | if ($this->validateAddress($email)) {
|
994: | $addresses[] = array(
|
995: | 'name' => trim(str_replace(array('"', "'"), '', $name)),
|
996: | 'address' => $email
|
997: | );
|
998: | }
|
999: | }
|
1000: | }
|
1001: | }
|
1002: | return $addresses;
|
1003: | }
|
1004: |
|
1005: | |
1006: | |
1007: | |
1008: | |
1009: | |
1010: | |
1011: | |
1012: |
|
1013: | public function setFrom($address, $name = '', $auto = true)
|
1014: | {
|
1015: | $address = trim($address);
|
1016: | $name = trim(preg_replace('/[\r\n]+/', '', $name));
|
1017: |
|
1018: | if (($pos = strrpos($address, '@')) === false or
|
1019: | (!$this->has8bitChars(substr($address, ++$pos)) or !$this->idnSupported()) and
|
1020: | !$this->validateAddress($address)) {
|
1021: | $error_message = $this->lang('invalid_address') . " (setFrom) $address";
|
1022: | $this->setError($error_message);
|
1023: | $this->edebug($error_message);
|
1024: | if ($this->exceptions) {
|
1025: | throw new phpmailerException($error_message);
|
1026: | }
|
1027: | return false;
|
1028: | }
|
1029: | $this->From = $address;
|
1030: | $this->FromName = $name;
|
1031: | if ($auto) {
|
1032: | if (empty($this->Sender)) {
|
1033: | $this->Sender = $address;
|
1034: | }
|
1035: | }
|
1036: | return true;
|
1037: | }
|
1038: |
|
1039: | |
1040: | |
1041: | |
1042: | |
1043: | |
1044: | |
1045: |
|
1046: | public function getLastMessageID()
|
1047: | {
|
1048: | return $this->lastMessageID;
|
1049: | }
|
1050: |
|
1051: | |
1052: | |
1053: | |
1054: | |
1055: | |
1056: | |
1057: | |
1058: | |
1059: | |
1060: | |
1061: | |
1062: | |
1063: | |
1064: | |
1065: | |
1066: | |
1067: | |
1068: | |
1069: |
|
1070: | public static function validateAddress($address, $patternselect = null)
|
1071: | {
|
1072: | if (is_null($patternselect)) {
|
1073: | $patternselect = self::$validator;
|
1074: | }
|
1075: | if (is_callable($patternselect)) {
|
1076: | return call_user_func($patternselect, $address);
|
1077: | }
|
1078: |
|
1079: | if (strpos($address, "\n") !== false or strpos($address, "\r") !== false) {
|
1080: | return false;
|
1081: | }
|
1082: | if (!$patternselect or $patternselect == 'auto') {
|
1083: |
|
1084: |
|
1085: | if (defined('PCRE_VERSION')) {
|
1086: |
|
1087: | if (version_compare(PCRE_VERSION, '8.0.3') >= 0) {
|
1088: | $patternselect = 'pcre8';
|
1089: | } else {
|
1090: | $patternselect = 'pcre';
|
1091: | }
|
1092: | } elseif (function_exists('extension_loaded') and extension_loaded('pcre')) {
|
1093: |
|
1094: | $patternselect = 'pcre';
|
1095: | } else {
|
1096: |
|
1097: | if (version_compare(PHP_VERSION, '5.2.0') >= 0) {
|
1098: | $patternselect = 'php';
|
1099: | } else {
|
1100: | $patternselect = 'noregex';
|
1101: | }
|
1102: | }
|
1103: | }
|
1104: | switch ($patternselect) {
|
1105: | case 'pcre8':
|
1106: | |
1107: | |
1108: | |
1109: | |
1110: | |
1111: |
|
1112: | return (boolean)preg_match(
|
1113: | '/^(?!(?>(?1)"?(?>\\\[ -~]|[^"])"?(?1)){255,})(?!(?>(?1)"?(?>\\\[ -~]|[^"])"?(?1)){65,}@)' .
|
1114: | '((?>(?>(?>((?>(?>(?>\x0D\x0A)?[\t ])+|(?>[\t ]*\x0D\x0A)?[\t ]+)?)(\((?>(?2)' .
|
1115: | '(?>[\x01-\x08\x0B\x0C\x0E-\'*-\[\]-\x7F]|\\\[\x00-\x7F]|(?3)))*(?2)\)))+(?2))|(?2))?)' .
|
1116: | '([!#-\'*+\/-9=?^-~-]+|"(?>(?2)(?>[\x01-\x08\x0B\x0C\x0E-!#-\[\]-\x7F]|\\\[\x00-\x7F]))*' .
|
1117: | '(?2)")(?>(?1)\.(?1)(?4))*(?1)@(?!(?1)[a-z0-9-]{64,})(?1)(?>([a-z0-9](?>[a-z0-9-]*[a-z0-9])?)' .
|
1118: | '(?>(?1)\.(?!(?1)[a-z0-9-]{64,})(?1)(?5)){0,126}|\[(?:(?>IPv6:(?>([a-f0-9]{1,4})(?>:(?6)){7}' .
|
1119: | '|(?!(?:.*[a-f0-9][:\]]){8,})((?6)(?>:(?6)){0,6})?::(?7)?))|(?>(?>IPv6:(?>(?6)(?>:(?6)){5}:' .
|
1120: | '|(?!(?:.*[a-f0-9]:){6,})(?8)?::(?>((?6)(?>:(?6)){0,4}):)?))?(25[0-5]|2[0-4][0-9]|1[0-9]{2}' .
|
1121: | '|[1-9]?[0-9])(?>\.(?9)){3}))\])(?1)$/isD',
|
1122: | $address
|
1123: | );
|
1124: | case 'pcre':
|
1125: |
|
1126: | return (boolean)preg_match(
|
1127: | '/^(?!(?>"?(?>\\\[ -~]|[^"])"?){255,})(?!(?>"?(?>\\\[ -~]|[^"])"?){65,}@)(?>' .
|
1128: | '[!#-\'*+\/-9=?^-~-]+|"(?>(?>[\x01-\x08\x0B\x0C\x0E-!#-\[\]-\x7F]|\\\[\x00-\xFF]))*")' .
|
1129: | '(?>\.(?>[!#-\'*+\/-9=?^-~-]+|"(?>(?>[\x01-\x08\x0B\x0C\x0E-!#-\[\]-\x7F]|\\\[\x00-\xFF]))*"))*' .
|
1130: | '@(?>(?![a-z0-9-]{64,})(?>[a-z0-9](?>[a-z0-9-]*[a-z0-9])?)(?>\.(?![a-z0-9-]{64,})' .
|
1131: | '(?>[a-z0-9](?>[a-z0-9-]*[a-z0-9])?)){0,126}|\[(?:(?>IPv6:(?>(?>[a-f0-9]{1,4})(?>:' .
|
1132: | '[a-f0-9]{1,4}){7}|(?!(?:.*[a-f0-9][:\]]){8,})(?>[a-f0-9]{1,4}(?>:[a-f0-9]{1,4}){0,6})?' .
|
1133: | '::(?>[a-f0-9]{1,4}(?>:[a-f0-9]{1,4}){0,6})?))|(?>(?>IPv6:(?>[a-f0-9]{1,4}(?>:' .
|
1134: | '[a-f0-9]{1,4}){5}:|(?!(?:.*[a-f0-9]:){6,})(?>[a-f0-9]{1,4}(?>:[a-f0-9]{1,4}){0,4})?' .
|
1135: | '::(?>(?:[a-f0-9]{1,4}(?>:[a-f0-9]{1,4}){0,4}):)?))?(?>25[0-5]|2[0-4][0-9]|1[0-9]{2}' .
|
1136: | '|[1-9]?[0-9])(?>\.(?>25[0-5]|2[0-4][0-9]|1[0-9]{2}|[1-9]?[0-9])){3}))\])$/isD',
|
1137: | $address
|
1138: | );
|
1139: | case 'html5':
|
1140: | |
1141: | |
1142: | |
1143: |
|
1144: | return (boolean)preg_match(
|
1145: | '/^[a-zA-Z0-9.!#$%&\'*+\/=?^_`{|}~-]+@[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}' .
|
1146: | '[a-zA-Z0-9])?(?:\.[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?)*$/sD',
|
1147: | $address
|
1148: | );
|
1149: | case 'noregex':
|
1150: |
|
1151: |
|
1152: | return (strlen($address) >= 3
|
1153: | and strpos($address, '@') >= 1
|
1154: | and strpos($address, '@') != strlen($address) - 1);
|
1155: | case 'php':
|
1156: | default:
|
1157: | return (boolean)filter_var($address, FILTER_VALIDATE_EMAIL);
|
1158: | }
|
1159: | }
|
1160: |
|
1161: | |
1162: | |
1163: | |
1164: | |
1165: |
|
1166: | public function idnSupported()
|
1167: | {
|
1168: |
|
1169: | return function_exists('idn_to_ascii') and function_exists('mb_convert_encoding');
|
1170: | }
|
1171: |
|
1172: | |
1173: | |
1174: | |
1175: | |
1176: | |
1177: | |
1178: | |
1179: | |
1180: | |
1181: | |
1182: |
|
1183: | public function punyencodeAddress($address)
|
1184: | {
|
1185: |
|
1186: | if ($this->idnSupported() and
|
1187: | !empty($this->CharSet) and
|
1188: | ($pos = strrpos($address, '@')) !== false) {
|
1189: | $domain = substr($address, ++$pos);
|
1190: |
|
1191: | if ($this->has8bitChars($domain) and @mb_check_encoding($domain, $this->CharSet)) {
|
1192: | $domain = mb_convert_encoding($domain, 'UTF-8', $this->CharSet);
|
1193: | if (($punycode = defined('INTL_IDNA_VARIANT_UTS46') ?
|
1194: | idn_to_ascii($domain, 0, INTL_IDNA_VARIANT_UTS46) :
|
1195: | idn_to_ascii($domain)) !== false) {
|
1196: | return substr($address, 0, $pos) . $punycode;
|
1197: | }
|
1198: | }
|
1199: | }
|
1200: | return $address;
|
1201: | }
|
1202: |
|
1203: | |
1204: | |
1205: | |
1206: | |
1207: | |
1208: |
|
1209: | public function send()
|
1210: | {
|
1211: | try {
|
1212: | if (!$this->preSend()) {
|
1213: | return false;
|
1214: | }
|
1215: | return $this->postSend();
|
1216: | } catch (phpmailerException $exc) {
|
1217: | $this->mailHeader = '';
|
1218: | $this->setError($exc->getMessage());
|
1219: | if ($this->exceptions) {
|
1220: | throw $exc;
|
1221: | }
|
1222: | return false;
|
1223: | }
|
1224: | }
|
1225: |
|
1226: | |
1227: | |
1228: | |
1229: | |
1230: |
|
1231: | public function preSend()
|
1232: | {
|
1233: | try {
|
1234: | $this->error_count = 0;
|
1235: | $this->mailHeader = '';
|
1236: |
|
1237: |
|
1238: | foreach (array_merge($this->RecipientsQueue, $this->ReplyToQueue) as $params) {
|
1239: | $params[1] = $this->punyencodeAddress($params[1]);
|
1240: | call_user_func_array(array($this, 'addAnAddress'), $params);
|
1241: | }
|
1242: | if ((count($this->to) + count($this->cc) + count($this->bcc)) < 1) {
|
1243: | throw new phpmailerException($this->lang('provide_address'), self::STOP_CRITICAL);
|
1244: | }
|
1245: |
|
1246: |
|
1247: | foreach (array('From', 'Sender', 'ConfirmReadingTo') as $address_kind) {
|
1248: | $this->$address_kind = trim($this->$address_kind);
|
1249: | if (empty($this->$address_kind)) {
|
1250: | continue;
|
1251: | }
|
1252: | $this->$address_kind = $this->punyencodeAddress($this->$address_kind);
|
1253: | if (!$this->validateAddress($this->$address_kind)) {
|
1254: | $error_message = $this->lang('invalid_address') . ' (punyEncode) ' . $this->$address_kind;
|
1255: | $this->setError($error_message);
|
1256: | $this->edebug($error_message);
|
1257: | if ($this->exceptions) {
|
1258: | throw new phpmailerException($error_message);
|
1259: | }
|
1260: | return false;
|
1261: | }
|
1262: | }
|
1263: |
|
1264: |
|
1265: | if ($this->alternativeExists()) {
|
1266: | $this->ContentType = 'multipart/alternative';
|
1267: | }
|
1268: |
|
1269: | $this->setMessageType();
|
1270: |
|
1271: | if (!$this->AllowEmpty and empty($this->Body)) {
|
1272: | throw new phpmailerException($this->lang('empty_message'), self::STOP_CRITICAL);
|
1273: | }
|
1274: |
|
1275: |
|
1276: | $this->MIMEHeader = '';
|
1277: | $this->MIMEBody = $this->createBody();
|
1278: |
|
1279: | $tempheaders = $this->MIMEHeader;
|
1280: | $this->MIMEHeader = $this->createHeader();
|
1281: | $this->MIMEHeader .= $tempheaders;
|
1282: |
|
1283: |
|
1284: |
|
1285: | if ($this->Mailer == 'mail') {
|
1286: | if (count($this->to) > 0) {
|
1287: | $this->mailHeader .= $this->addrAppend('To', $this->to);
|
1288: | } else {
|
1289: | $this->mailHeader .= $this->headerLine('To', 'undisclosed-recipients:;');
|
1290: | }
|
1291: | $this->mailHeader .= $this->headerLine(
|
1292: | 'Subject',
|
1293: | $this->encodeHeader($this->secureHeader(trim($this->Subject)))
|
1294: | );
|
1295: | }
|
1296: |
|
1297: |
|
1298: | if (!empty($this->DKIM_domain)
|
1299: | and !empty($this->DKIM_selector)
|
1300: | and (!empty($this->DKIM_private_string)
|
1301: | or (!empty($this->DKIM_private)
|
1302: | and self::isPermittedPath($this->DKIM_private)
|
1303: | and file_exists($this->DKIM_private)
|
1304: | )
|
1305: | )
|
1306: | ) {
|
1307: | $header_dkim = $this->DKIM_Add(
|
1308: | $this->MIMEHeader . $this->mailHeader,
|
1309: | $this->encodeHeader($this->secureHeader($this->Subject)),
|
1310: | $this->MIMEBody
|
1311: | );
|
1312: | $this->MIMEHeader = rtrim($this->MIMEHeader, "\r\n ") . self::CRLF .
|
1313: | str_replace("\r\n", "\n", $header_dkim) . self::CRLF;
|
1314: | }
|
1315: | return true;
|
1316: | } catch (phpmailerException $exc) {
|
1317: | $this->setError($exc->getMessage());
|
1318: | if ($this->exceptions) {
|
1319: | throw $exc;
|
1320: | }
|
1321: | return false;
|
1322: | }
|
1323: | }
|
1324: |
|
1325: | |
1326: | |
1327: | |
1328: | |
1329: | |
1330: |
|
1331: | public function postSend()
|
1332: | {
|
1333: | try {
|
1334: |
|
1335: | switch ($this->Mailer) {
|
1336: | case 'sendmail':
|
1337: | case 'qmail':
|
1338: | return $this->sendmailSend($this->MIMEHeader, $this->MIMEBody);
|
1339: | case 'smtp':
|
1340: | return $this->smtpSend($this->MIMEHeader, $this->MIMEBody);
|
1341: | case 'mail':
|
1342: | return $this->mailSend($this->MIMEHeader, $this->MIMEBody);
|
1343: | default:
|
1344: | $sendMethod = $this->Mailer.'Send';
|
1345: | if (method_exists($this, $sendMethod)) {
|
1346: | return $this->$sendMethod($this->MIMEHeader, $this->MIMEBody);
|
1347: | }
|
1348: |
|
1349: | return $this->mailSend($this->MIMEHeader, $this->MIMEBody);
|
1350: | }
|
1351: | } catch (phpmailerException $exc) {
|
1352: | $this->setError($exc->getMessage());
|
1353: | $this->edebug($exc->getMessage());
|
1354: | if ($this->exceptions) {
|
1355: | throw $exc;
|
1356: | }
|
1357: | }
|
1358: | return false;
|
1359: | }
|
1360: |
|
1361: | |
1362: | |
1363: | |
1364: | |
1365: | |
1366: | |
1367: | |
1368: | |
1369: |
|
1370: | protected function sendmailSend($header, $body)
|
1371: | {
|
1372: |
|
1373: | if (!empty($this->Sender) and self::isShellSafe($this->Sender)) {
|
1374: | if ($this->Mailer == 'qmail') {
|
1375: | $sendmailFmt = '%s -f%s';
|
1376: | } else {
|
1377: | $sendmailFmt = '%s -oi -f%s -t';
|
1378: | }
|
1379: | } else {
|
1380: | if ($this->Mailer == 'qmail') {
|
1381: | $sendmailFmt = '%s';
|
1382: | } else {
|
1383: | $sendmailFmt = '%s -oi -t';
|
1384: | }
|
1385: | }
|
1386: |
|
1387: |
|
1388: | $sendmail = sprintf($sendmailFmt, escapeshellcmd($this->Sendmail), $this->Sender);
|
1389: |
|
1390: | if ($this->SingleTo) {
|
1391: | foreach ($this->SingleToArray as $toAddr) {
|
1392: | if (!@$mail = popen($sendmail, 'w')) {
|
1393: | throw new phpmailerException($this->lang('execute') . $this->Sendmail, self::STOP_CRITICAL);
|
1394: | }
|
1395: | fputs($mail, 'To: ' . $toAddr . "\n");
|
1396: | fputs($mail, $header);
|
1397: | fputs($mail, $body);
|
1398: | $result = pclose($mail);
|
1399: | $this->doCallback(
|
1400: | ($result == 0),
|
1401: | array($toAddr),
|
1402: | $this->cc,
|
1403: | $this->bcc,
|
1404: | $this->Subject,
|
1405: | $body,
|
1406: | $this->From
|
1407: | );
|
1408: | if ($result != 0) {
|
1409: | throw new phpmailerException($this->lang('execute') . $this->Sendmail, self::STOP_CRITICAL);
|
1410: | }
|
1411: | }
|
1412: | } else {
|
1413: | if (!@$mail = popen($sendmail, 'w')) {
|
1414: | throw new phpmailerException($this->lang('execute') . $this->Sendmail, self::STOP_CRITICAL);
|
1415: | }
|
1416: | fputs($mail, $header);
|
1417: | fputs($mail, $body);
|
1418: | $result = pclose($mail);
|
1419: | $this->doCallback(
|
1420: | ($result == 0),
|
1421: | $this->to,
|
1422: | $this->cc,
|
1423: | $this->bcc,
|
1424: | $this->Subject,
|
1425: | $body,
|
1426: | $this->From
|
1427: | );
|
1428: | if ($result != 0) {
|
1429: | throw new phpmailerException($this->lang('execute') . $this->Sendmail, self::STOP_CRITICAL);
|
1430: | }
|
1431: | }
|
1432: | return true;
|
1433: | }
|
1434: |
|
1435: | |
1436: | |
1437: | |
1438: | |
1439: | |
1440: | |
1441: | |
1442: | |
1443: |
|
1444: | protected static function isShellSafe($string)
|
1445: | {
|
1446: |
|
1447: | if (escapeshellcmd($string) !== $string
|
1448: | or !in_array(escapeshellarg($string), array("'$string'", "\"$string\""))
|
1449: | ) {
|
1450: | return false;
|
1451: | }
|
1452: |
|
1453: | $length = strlen($string);
|
1454: |
|
1455: | for ($i = 0; $i < $length; $i++) {
|
1456: | $c = $string[$i];
|
1457: |
|
1458: |
|
1459: |
|
1460: |
|
1461: | if (!ctype_alnum($c) && strpos('@_-.', $c) === false) {
|
1462: | return false;
|
1463: | }
|
1464: | }
|
1465: |
|
1466: | return true;
|
1467: | }
|
1468: |
|
1469: | |
1470: | |
1471: | |
1472: | |
1473: | |
1474: | |
1475: |
|
1476: | protected static function isPermittedPath($path)
|
1477: | {
|
1478: | return !preg_match('#^[a-z]+://#i', $path);
|
1479: | }
|
1480: |
|
1481: | |
1482: | |
1483: | |
1484: | |
1485: | |
1486: | |
1487: | |
1488: | |
1489: |
|
1490: | protected function mailSend($header, $body)
|
1491: | {
|
1492: | $toArr = array();
|
1493: | foreach ($this->to as $toaddr) {
|
1494: | $toArr[] = $this->addrFormat($toaddr);
|
1495: | }
|
1496: | $to = implode(', ', $toArr);
|
1497: |
|
1498: | $params = null;
|
1499: |
|
1500: | if (!empty($this->Sender) and $this->validateAddress($this->Sender)) {
|
1501: |
|
1502: | if (self::isShellSafe($this->Sender)) {
|
1503: | $params = sprintf('-f%s', $this->Sender);
|
1504: | }
|
1505: | }
|
1506: | if (!empty($this->Sender) and !ini_get('safe_mode') and $this->validateAddress($this->Sender)) {
|
1507: | $old_from = ini_get('sendmail_from');
|
1508: | ini_set('sendmail_from', $this->Sender);
|
1509: | }
|
1510: | $result = false;
|
1511: | if ($this->SingleTo and count($toArr) > 1) {
|
1512: | foreach ($toArr as $toAddr) {
|
1513: | $result = $this->mailPassthru($toAddr, $this->Subject, $body, $header, $params);
|
1514: | $this->doCallback($result, array($toAddr), $this->cc, $this->bcc, $this->Subject, $body, $this->From);
|
1515: | }
|
1516: | } else {
|
1517: | $result = $this->mailPassthru($to, $this->Subject, $body, $header, $params);
|
1518: | $this->doCallback($result, $this->to, $this->cc, $this->bcc, $this->Subject, $body, $this->From);
|
1519: | }
|
1520: | if (isset($old_from)) {
|
1521: | ini_set('sendmail_from', $old_from);
|
1522: | }
|
1523: | if (!$result) {
|
1524: | throw new phpmailerException($this->lang('instantiate'), self::STOP_CRITICAL);
|
1525: | }
|
1526: | return true;
|
1527: | }
|
1528: |
|
1529: | |
1530: | |
1531: | |
1532: | |
1533: |
|
1534: | public function getSMTPInstance()
|
1535: | {
|
1536: | if (!is_object($this->smtp)) {
|
1537: | $this->smtp = new SMTP;
|
1538: | }
|
1539: | return $this->smtp;
|
1540: | }
|
1541: |
|
1542: | |
1543: | |
1544: | |
1545: | |
1546: | |
1547: | |
1548: | |
1549: | |
1550: | |
1551: | |
1552: | |
1553: |
|
1554: | protected function smtpSend($header, $body)
|
1555: | {
|
1556: | $bad_rcpt = array();
|
1557: | if (!$this->smtpConnect($this->SMTPOptions)) {
|
1558: | throw new phpmailerException($this->lang('smtp_connect_failed'), self::STOP_CRITICAL);
|
1559: | }
|
1560: | if (!empty($this->Sender) and $this->validateAddress($this->Sender)) {
|
1561: | $smtp_from = $this->Sender;
|
1562: | } else {
|
1563: | $smtp_from = $this->From;
|
1564: | }
|
1565: | if (!$this->smtp->mail($smtp_from)) {
|
1566: | $this->setError($this->lang('from_failed') . $smtp_from . ' : ' . implode(',', $this->smtp->getError()));
|
1567: | throw new phpmailerException($this->ErrorInfo, self::STOP_CRITICAL);
|
1568: | }
|
1569: |
|
1570: |
|
1571: | foreach (array($this->to, $this->cc, $this->bcc) as $togroup) {
|
1572: | foreach ($togroup as $to) {
|
1573: | if (!$this->smtp->recipient($to[0])) {
|
1574: | $error = $this->smtp->getError();
|
1575: | $bad_rcpt[] = array('to' => $to[0], 'error' => $error['detail']);
|
1576: | $isSent = false;
|
1577: | } else {
|
1578: | $isSent = true;
|
1579: | }
|
1580: | $this->doCallback($isSent, array($to[0]), array(), array(), $this->Subject, $body, $this->From);
|
1581: | }
|
1582: | }
|
1583: |
|
1584: |
|
1585: | if ((count($this->all_recipients) > count($bad_rcpt)) and !$this->smtp->data($header . $body)) {
|
1586: | throw new phpmailerException($this->lang('data_not_accepted'), self::STOP_CRITICAL);
|
1587: | }
|
1588: | if ($this->SMTPKeepAlive) {
|
1589: | $this->smtp->reset();
|
1590: | } else {
|
1591: | $this->smtp->quit();
|
1592: | $this->smtp->close();
|
1593: | }
|
1594: |
|
1595: | if (count($bad_rcpt) > 0) {
|
1596: | $errstr = '';
|
1597: | foreach ($bad_rcpt as $bad) {
|
1598: | $errstr .= $bad['to'] . ': ' . $bad['error'];
|
1599: | }
|
1600: | throw new phpmailerException(
|
1601: | $this->lang('recipients_failed') . $errstr,
|
1602: | self::STOP_CONTINUE
|
1603: | );
|
1604: | }
|
1605: | return true;
|
1606: | }
|
1607: |
|
1608: | |
1609: | |
1610: | |
1611: | |
1612: | |
1613: | |
1614: | |
1615: | |
1616: |
|
1617: | public function smtpConnect($options = null)
|
1618: | {
|
1619: | if (is_null($this->smtp)) {
|
1620: | $this->smtp = $this->getSMTPInstance();
|
1621: | }
|
1622: |
|
1623: |
|
1624: | if (is_null($options)) {
|
1625: | $options = $this->SMTPOptions;
|
1626: | }
|
1627: |
|
1628: |
|
1629: | if ($this->smtp->connected()) {
|
1630: | return true;
|
1631: | }
|
1632: |
|
1633: | $this->smtp->setTimeout($this->Timeout);
|
1634: | $this->smtp->setDebugLevel($this->SMTPDebug);
|
1635: | $this->smtp->setDebugOutput($this->Debugoutput);
|
1636: | $this->smtp->setVerp($this->do_verp);
|
1637: | $hosts = explode(';', $this->Host);
|
1638: | $lastexception = null;
|
1639: |
|
1640: | foreach ($hosts as $hostentry) {
|
1641: | $hostinfo = array();
|
1642: | if (!preg_match(
|
1643: | '/^((ssl|tls):\/\/)*([a-zA-Z0-9\.-]*|\[[a-fA-F0-9:]+\]):?([0-9]*)$/',
|
1644: | trim($hostentry),
|
1645: | $hostinfo
|
1646: | )) {
|
1647: |
|
1648: | $this->edebug('Ignoring invalid host: ' . $hostentry);
|
1649: | continue;
|
1650: | }
|
1651: |
|
1652: |
|
1653: |
|
1654: |
|
1655: |
|
1656: | $prefix = '';
|
1657: | $secure = $this->SMTPSecure;
|
1658: | $tls = ($this->SMTPSecure == 'tls');
|
1659: | if ('ssl' == $hostinfo[2] or ('' == $hostinfo[2] and 'ssl' == $this->SMTPSecure)) {
|
1660: | $prefix = 'ssl://';
|
1661: | $tls = false;
|
1662: | $secure = 'ssl';
|
1663: | } elseif ($hostinfo[2] == 'tls') {
|
1664: | $tls = true;
|
1665: |
|
1666: | $secure = 'tls';
|
1667: | }
|
1668: |
|
1669: | $sslext = defined('OPENSSL_ALGO_SHA1');
|
1670: | if ('tls' === $secure or 'ssl' === $secure) {
|
1671: |
|
1672: | if (!$sslext) {
|
1673: | throw new phpmailerException($this->lang('extension_missing').'openssl', self::STOP_CRITICAL);
|
1674: | }
|
1675: | }
|
1676: | $host = $hostinfo[3];
|
1677: | $port = $this->Port;
|
1678: | $tport = (integer)$hostinfo[4];
|
1679: | if ($tport > 0 and $tport < 65536) {
|
1680: | $port = $tport;
|
1681: | }
|
1682: | if ($this->smtp->connect($prefix . $host, $port, $this->Timeout, $options)) {
|
1683: | try {
|
1684: | if ($this->Helo) {
|
1685: | $hello = $this->Helo;
|
1686: | } else {
|
1687: | $hello = $this->serverHostname();
|
1688: | }
|
1689: | $this->smtp->hello($hello);
|
1690: |
|
1691: |
|
1692: |
|
1693: |
|
1694: |
|
1695: | if ($this->SMTPAutoTLS and $sslext and $secure != 'ssl' and $this->smtp->getServerExt('STARTTLS')) {
|
1696: | $tls = true;
|
1697: | }
|
1698: | if ($tls) {
|
1699: | if (!$this->smtp->startTLS()) {
|
1700: | throw new phpmailerException($this->lang('connect_host'));
|
1701: | }
|
1702: |
|
1703: | $this->smtp->hello($hello);
|
1704: | }
|
1705: | if ($this->SMTPAuth) {
|
1706: | if (!$this->smtp->authenticate(
|
1707: | $this->Username,
|
1708: | $this->Password,
|
1709: | $this->AuthType,
|
1710: | $this->Realm,
|
1711: | $this->Workstation
|
1712: | )
|
1713: | ) {
|
1714: | throw new phpmailerException($this->lang('authenticate'));
|
1715: | }
|
1716: | }
|
1717: | return true;
|
1718: | } catch (phpmailerException $exc) {
|
1719: | $lastexception = $exc;
|
1720: | $this->edebug($exc->getMessage());
|
1721: |
|
1722: | $this->smtp->quit();
|
1723: | }
|
1724: | }
|
1725: | }
|
1726: |
|
1727: | $this->smtp->close();
|
1728: |
|
1729: | if ($this->exceptions and !is_null($lastexception)) {
|
1730: | throw $lastexception;
|
1731: | }
|
1732: | return false;
|
1733: | }
|
1734: |
|
1735: | |
1736: | |
1737: | |
1738: |
|
1739: | public function smtpClose()
|
1740: | {
|
1741: | if (is_a($this->smtp, 'SMTP')) {
|
1742: | if ($this->smtp->connected()) {
|
1743: | $this->smtp->quit();
|
1744: | $this->smtp->close();
|
1745: | }
|
1746: | }
|
1747: | }
|
1748: |
|
1749: | |
1750: | |
1751: | |
1752: | |
1753: | |
1754: | |
1755: | |
1756: | |
1757: |
|
1758: | public function setLanguage($langcode = 'en', $lang_path = '')
|
1759: | {
|
1760: |
|
1761: | $renamed_langcodes = array(
|
1762: | 'br' => 'pt_br',
|
1763: | 'cz' => 'cs',
|
1764: | 'dk' => 'da',
|
1765: | 'no' => 'nb',
|
1766: | 'se' => 'sv',
|
1767: | 'sr' => 'rs'
|
1768: | );
|
1769: |
|
1770: | if (isset($renamed_langcodes[$langcode])) {
|
1771: | $langcode = $renamed_langcodes[$langcode];
|
1772: | }
|
1773: |
|
1774: |
|
1775: | $PHPMAILER_LANG = array(
|
1776: | 'authenticate' => 'SMTP Error: Could not authenticate.',
|
1777: | 'connect_host' => 'SMTP Error: Could not connect to SMTP host.',
|
1778: | 'data_not_accepted' => 'SMTP Error: data not accepted.',
|
1779: | 'empty_message' => 'Message body empty',
|
1780: | 'encoding' => 'Unknown encoding: ',
|
1781: | 'execute' => 'Could not execute: ',
|
1782: | 'file_access' => 'Could not access file: ',
|
1783: | 'file_open' => 'File Error: Could not open file: ',
|
1784: | 'from_failed' => 'The following From address failed: ',
|
1785: | 'instantiate' => 'Could not instantiate mail function.',
|
1786: | 'invalid_address' => 'Invalid address: ',
|
1787: | 'mailer_not_supported' => ' mailer is not supported.',
|
1788: | 'provide_address' => 'You must provide at least one recipient email address.',
|
1789: | 'recipients_failed' => 'SMTP Error: The following recipients failed: ',
|
1790: | 'signing' => 'Signing Error: ',
|
1791: | 'smtp_connect_failed' => 'SMTP connect() failed.',
|
1792: | 'smtp_error' => 'SMTP server error: ',
|
1793: | 'variable_set' => 'Cannot set or reset variable: ',
|
1794: | 'extension_missing' => 'Extension missing: '
|
1795: | );
|
1796: | if (empty($lang_path)) {
|
1797: |
|
1798: | $lang_path = dirname(__FILE__). DIRECTORY_SEPARATOR . 'language'. DIRECTORY_SEPARATOR;
|
1799: | }
|
1800: |
|
1801: | if (!preg_match('/^[a-z]{2}(?:_[a-zA-Z]{2})?$/', $langcode)) {
|
1802: | $langcode = 'en';
|
1803: | }
|
1804: | $foundlang = true;
|
1805: | $lang_file = $lang_path . 'phpmailer.lang-' . $langcode . '.php';
|
1806: |
|
1807: | if ($langcode != 'en') {
|
1808: |
|
1809: | if (!self::isPermittedPath($lang_file) or !is_readable($lang_file)) {
|
1810: | $foundlang = false;
|
1811: | } else {
|
1812: |
|
1813: |
|
1814: | $foundlang = include $lang_file;
|
1815: | }
|
1816: | }
|
1817: | $this->language = $PHPMAILER_LANG;
|
1818: | return (boolean)$foundlang;
|
1819: | }
|
1820: |
|
1821: | |
1822: | |
1823: | |
1824: |
|
1825: | public function getTranslations()
|
1826: | {
|
1827: | return $this->language;
|
1828: | }
|
1829: |
|
1830: | |
1831: | |
1832: | |
1833: | |
1834: | |
1835: | |
1836: | |
1837: | |
1838: | |
1839: |
|
1840: | public function addrAppend($type, $addr)
|
1841: | {
|
1842: | $addresses = array();
|
1843: | foreach ($addr as $address) {
|
1844: | $addresses[] = $this->addrFormat($address);
|
1845: | }
|
1846: | return $type . ': ' . implode(', ', $addresses) . $this->LE;
|
1847: | }
|
1848: |
|
1849: | |
1850: | |
1851: | |
1852: | |
1853: | |
1854: | |
1855: |
|
1856: | public function addrFormat($addr)
|
1857: | {
|
1858: | if (empty($addr[1])) {
|
1859: | return $this->secureHeader($addr[0]);
|
1860: | } else {
|
1861: | return $this->encodeHeader($this->secureHeader($addr[1]), 'phrase') . ' <' . $this->secureHeader(
|
1862: | $addr[0]
|
1863: | ) . '>';
|
1864: | }
|
1865: | }
|
1866: |
|
1867: | |
1868: | |
1869: | |
1870: | |
1871: | |
1872: | |
1873: | |
1874: | |
1875: | |
1876: | |
1877: |
|
1878: | public function wrapText($message, $length, $qp_mode = false)
|
1879: | {
|
1880: | if ($qp_mode) {
|
1881: | $soft_break = sprintf(' =%s', $this->LE);
|
1882: | } else {
|
1883: | $soft_break = $this->LE;
|
1884: | }
|
1885: |
|
1886: |
|
1887: | $is_utf8 = (strtolower($this->CharSet) == 'utf-8');
|
1888: | $lelen = strlen($this->LE);
|
1889: | $crlflen = strlen(self::CRLF);
|
1890: |
|
1891: | $message = $this->fixEOL($message);
|
1892: |
|
1893: | if (substr($message, -$lelen) == $this->LE) {
|
1894: | $message = substr($message, 0, -$lelen);
|
1895: | }
|
1896: |
|
1897: |
|
1898: | $lines = explode($this->LE, $message);
|
1899: |
|
1900: | $message = '';
|
1901: | foreach ($lines as $line) {
|
1902: | $words = explode(' ', $line);
|
1903: | $buf = '';
|
1904: | $firstword = true;
|
1905: | foreach ($words as $word) {
|
1906: | if ($qp_mode and (strlen($word) > $length)) {
|
1907: | $space_left = $length - strlen($buf) - $crlflen;
|
1908: | if (!$firstword) {
|
1909: | if ($space_left > 20) {
|
1910: | $len = $space_left;
|
1911: | if ($is_utf8) {
|
1912: | $len = $this->utf8CharBoundary($word, $len);
|
1913: | } elseif (substr($word, $len - 1, 1) == '=') {
|
1914: | $len--;
|
1915: | } elseif (substr($word, $len - 2, 1) == '=') {
|
1916: | $len -= 2;
|
1917: | }
|
1918: | $part = substr($word, 0, $len);
|
1919: | $word = substr($word, $len);
|
1920: | $buf .= ' ' . $part;
|
1921: | $message .= $buf . sprintf('=%s', self::CRLF);
|
1922: | } else {
|
1923: | $message .= $buf . $soft_break;
|
1924: | }
|
1925: | $buf = '';
|
1926: | }
|
1927: | while (strlen($word) > 0) {
|
1928: | if ($length <= 0) {
|
1929: | break;
|
1930: | }
|
1931: | $len = $length;
|
1932: | if ($is_utf8) {
|
1933: | $len = $this->utf8CharBoundary($word, $len);
|
1934: | } elseif (substr($word, $len - 1, 1) == '=') {
|
1935: | $len--;
|
1936: | } elseif (substr($word, $len - 2, 1) == '=') {
|
1937: | $len -= 2;
|
1938: | }
|
1939: | $part = substr($word, 0, $len);
|
1940: | $word = substr($word, $len);
|
1941: |
|
1942: | if (strlen($word) > 0) {
|
1943: | $message .= $part . sprintf('=%s', self::CRLF);
|
1944: | } else {
|
1945: | $buf = $part;
|
1946: | }
|
1947: | }
|
1948: | } else {
|
1949: | $buf_o = $buf;
|
1950: | if (!$firstword) {
|
1951: | $buf .= ' ';
|
1952: | }
|
1953: | $buf .= $word;
|
1954: |
|
1955: | if (strlen($buf) > $length and $buf_o != '') {
|
1956: | $message .= $buf_o . $soft_break;
|
1957: | $buf = $word;
|
1958: | }
|
1959: | }
|
1960: | $firstword = false;
|
1961: | }
|
1962: | $message .= $buf . self::CRLF;
|
1963: | }
|
1964: |
|
1965: | return $message;
|
1966: | }
|
1967: |
|
1968: | |
1969: | |
1970: | |
1971: | |
1972: | |
1973: | |
1974: | |
1975: | |
1976: |
|
1977: | public function utf8CharBoundary($encodedText, $maxLength)
|
1978: | {
|
1979: | $foundSplitPos = false;
|
1980: | $lookBack = 3;
|
1981: | while (!$foundSplitPos) {
|
1982: | $lastChunk = substr($encodedText, $maxLength - $lookBack, $lookBack);
|
1983: | $encodedCharPos = strpos($lastChunk, '=');
|
1984: | if (false !== $encodedCharPos) {
|
1985: |
|
1986: |
|
1987: | $hex = substr($encodedText, $maxLength - $lookBack + $encodedCharPos + 1, 2);
|
1988: | $dec = hexdec($hex);
|
1989: | if ($dec < 128) {
|
1990: |
|
1991: |
|
1992: |
|
1993: | if ($encodedCharPos > 0) {
|
1994: | $maxLength = $maxLength - ($lookBack - $encodedCharPos);
|
1995: | }
|
1996: | $foundSplitPos = true;
|
1997: | } elseif ($dec >= 192) {
|
1998: |
|
1999: |
|
2000: | $maxLength = $maxLength - ($lookBack - $encodedCharPos);
|
2001: | $foundSplitPos = true;
|
2002: | } elseif ($dec < 192) {
|
2003: |
|
2004: | $lookBack += 3;
|
2005: | }
|
2006: | } else {
|
2007: |
|
2008: | $foundSplitPos = true;
|
2009: | }
|
2010: | }
|
2011: | return $maxLength;
|
2012: | }
|
2013: |
|
2014: | |
2015: | |
2016: | |
2017: | |
2018: | |
2019: | |
2020: | |
2021: |
|
2022: | public function setWordWrap()
|
2023: | {
|
2024: | if ($this->WordWrap < 1) {
|
2025: | return;
|
2026: | }
|
2027: |
|
2028: | switch ($this->message_type) {
|
2029: | case 'alt':
|
2030: | case 'alt_inline':
|
2031: | case 'alt_attach':
|
2032: | case 'alt_inline_attach':
|
2033: | $this->AltBody = $this->wrapText($this->AltBody, $this->WordWrap);
|
2034: | break;
|
2035: | default:
|
2036: | $this->Body = $this->wrapText($this->Body, $this->WordWrap);
|
2037: | break;
|
2038: | }
|
2039: | }
|
2040: |
|
2041: | |
2042: | |
2043: | |
2044: | |
2045: |
|
2046: | public function createHeader()
|
2047: | {
|
2048: | $result = '';
|
2049: |
|
2050: | $result .= $this->headerLine('Date', $this->MessageDate == '' ? self::rfcDate() : $this->MessageDate);
|
2051: |
|
2052: |
|
2053: | if ($this->SingleTo) {
|
2054: | if ($this->Mailer != 'mail') {
|
2055: | foreach ($this->to as $toaddr) {
|
2056: | $this->SingleToArray[] = $this->addrFormat($toaddr);
|
2057: | }
|
2058: | }
|
2059: | } else {
|
2060: | if (count($this->to) > 0) {
|
2061: | if ($this->Mailer != 'mail') {
|
2062: | $result .= $this->addrAppend('To', $this->to);
|
2063: | }
|
2064: | } elseif (count($this->cc) == 0) {
|
2065: | $result .= $this->headerLine('To', 'undisclosed-recipients:;');
|
2066: | }
|
2067: | }
|
2068: |
|
2069: | $result .= $this->addrAppend('From', array(array(trim($this->From), $this->FromName)));
|
2070: |
|
2071: |
|
2072: | if (count($this->cc) > 0) {
|
2073: | $result .= $this->addrAppend('Cc', $this->cc);
|
2074: | }
|
2075: |
|
2076: |
|
2077: | if ((
|
2078: | $this->Mailer == 'sendmail' or $this->Mailer == 'qmail' or $this->Mailer == 'mail'
|
2079: | )
|
2080: | and count($this->bcc) > 0
|
2081: | ) {
|
2082: | $result .= $this->addrAppend('Bcc', $this->bcc);
|
2083: | }
|
2084: |
|
2085: | if (count($this->ReplyTo) > 0) {
|
2086: | $result .= $this->addrAppend('Reply-To', $this->ReplyTo);
|
2087: | }
|
2088: |
|
2089: |
|
2090: | if ($this->Mailer != 'mail') {
|
2091: | $result .= $this->headerLine('Subject', $this->encodeHeader($this->secureHeader($this->Subject)));
|
2092: | }
|
2093: |
|
2094: |
|
2095: |
|
2096: | if ('' != $this->MessageID and preg_match('/^<.*@.*>$/', $this->MessageID)) {
|
2097: | $this->lastMessageID = $this->MessageID;
|
2098: | } else {
|
2099: | $this->lastMessageID = sprintf('<%s@%s>', $this->uniqueid, $this->serverHostname());
|
2100: | }
|
2101: | $result .= $this->headerLine('Message-ID', $this->lastMessageID);
|
2102: | if (!is_null($this->Priority)) {
|
2103: | $result .= $this->headerLine('X-Priority', $this->Priority);
|
2104: | }
|
2105: | if ($this->XMailer == '') {
|
2106: | $result .= $this->headerLine(
|
2107: | 'X-Mailer',
|
2108: | 'PHPMailer ' . $this->Version . ' (https://github.com/PHPMailer/PHPMailer)'
|
2109: | );
|
2110: | } else {
|
2111: | $myXmailer = trim($this->XMailer);
|
2112: | if ($myXmailer) {
|
2113: | $result .= $this->headerLine('X-Mailer', $myXmailer);
|
2114: | }
|
2115: | }
|
2116: |
|
2117: | if ($this->ConfirmReadingTo != '') {
|
2118: | $result .= $this->headerLine('Disposition-Notification-To', '<' . $this->ConfirmReadingTo . '>');
|
2119: | }
|
2120: |
|
2121: |
|
2122: | foreach ($this->CustomHeader as $header) {
|
2123: | $result .= $this->headerLine(
|
2124: | trim($header[0]),
|
2125: | $this->encodeHeader(trim($header[1]))
|
2126: | );
|
2127: | }
|
2128: | if (!$this->sign_key_file) {
|
2129: | $result .= $this->headerLine('MIME-Version', '1.0');
|
2130: | $result .= $this->getMailMIME();
|
2131: | }
|
2132: |
|
2133: | return $result;
|
2134: | }
|
2135: |
|
2136: | |
2137: | |
2138: | |
2139: | |
2140: |
|
2141: | public function getMailMIME()
|
2142: | {
|
2143: | $result = '';
|
2144: | $ismultipart = true;
|
2145: | switch ($this->message_type) {
|
2146: | case 'inline':
|
2147: | $result .= $this->headerLine('Content-Type', 'multipart/related;');
|
2148: | $result .= $this->textLine("\tboundary=\"" . $this->boundary[1] . '"');
|
2149: | break;
|
2150: | case 'attach':
|
2151: | case 'inline_attach':
|
2152: | case 'alt_attach':
|
2153: | case 'alt_inline_attach':
|
2154: | $result .= $this->headerLine('Content-Type', 'multipart/mixed;');
|
2155: | $result .= $this->textLine("\tboundary=\"" . $this->boundary[1] . '"');
|
2156: | break;
|
2157: | case 'alt':
|
2158: | case 'alt_inline':
|
2159: | $result .= $this->headerLine('Content-Type', 'multipart/alternative;');
|
2160: | $result .= $this->textLine("\tboundary=\"" . $this->boundary[1] . '"');
|
2161: | break;
|
2162: | default:
|
2163: |
|
2164: | $result .= $this->textLine('Content-Type: ' . $this->ContentType . '; charset=' . $this->CharSet);
|
2165: | $ismultipart = false;
|
2166: | break;
|
2167: | }
|
2168: |
|
2169: | if ($this->Encoding != '7bit') {
|
2170: |
|
2171: | if ($ismultipart) {
|
2172: | if ($this->Encoding == '8bit') {
|
2173: | $result .= $this->headerLine('Content-Transfer-Encoding', '8bit');
|
2174: | }
|
2175: |
|
2176: | } else {
|
2177: | $result .= $this->headerLine('Content-Transfer-Encoding', $this->Encoding);
|
2178: | }
|
2179: | }
|
2180: |
|
2181: | if ($this->Mailer != 'mail') {
|
2182: | $result .= $this->LE;
|
2183: | }
|
2184: |
|
2185: | return $result;
|
2186: | }
|
2187: |
|
2188: | |
2189: | |
2190: | |
2191: | |
2192: | |
2193: | |
2194: | |
2195: |
|
2196: | public function getSentMIMEMessage()
|
2197: | {
|
2198: | return rtrim($this->MIMEHeader . $this->mailHeader, "\n\r") . self::CRLF . self::CRLF . $this->MIMEBody;
|
2199: | }
|
2200: |
|
2201: | |
2202: | |
2203: | |
2204: |
|
2205: | protected function generateId() {
|
2206: | return md5(uniqid(time()));
|
2207: | }
|
2208: |
|
2209: | |
2210: | |
2211: | |
2212: | |
2213: | |
2214: | |
2215: |
|
2216: | public function createBody()
|
2217: | {
|
2218: | $body = '';
|
2219: |
|
2220: | $this->uniqueid = $this->generateId();
|
2221: | $this->boundary[1] = 'b1_' . $this->uniqueid;
|
2222: | $this->boundary[2] = 'b2_' . $this->uniqueid;
|
2223: | $this->boundary[3] = 'b3_' . $this->uniqueid;
|
2224: |
|
2225: | if ($this->sign_key_file) {
|
2226: | $body .= $this->getMailMIME() . $this->LE;
|
2227: | }
|
2228: |
|
2229: | $this->setWordWrap();
|
2230: |
|
2231: | $bodyEncoding = $this->Encoding;
|
2232: | $bodyCharSet = $this->CharSet;
|
2233: |
|
2234: | if ($bodyEncoding == '8bit' and !$this->has8bitChars($this->Body)) {
|
2235: | $bodyEncoding = '7bit';
|
2236: |
|
2237: | $bodyCharSet = 'us-ascii';
|
2238: | }
|
2239: |
|
2240: |
|
2241: | if ('base64' != $this->Encoding and self::hasLineLongerThanMax($this->Body)) {
|
2242: | $bodyEncoding = 'quoted-printable';
|
2243: | }
|
2244: |
|
2245: | $altBodyEncoding = $this->Encoding;
|
2246: | $altBodyCharSet = $this->CharSet;
|
2247: |
|
2248: | if ($altBodyEncoding == '8bit' and !$this->has8bitChars($this->AltBody)) {
|
2249: | $altBodyEncoding = '7bit';
|
2250: |
|
2251: | $altBodyCharSet = 'us-ascii';
|
2252: | }
|
2253: |
|
2254: |
|
2255: | if ('base64' != $altBodyEncoding and self::hasLineLongerThanMax($this->AltBody)) {
|
2256: | $altBodyEncoding = 'quoted-printable';
|
2257: | }
|
2258: |
|
2259: | $mimepre = "This is a multi-part message in MIME format." . $this->LE . $this->LE;
|
2260: | switch ($this->message_type) {
|
2261: | case 'inline':
|
2262: | $body .= $mimepre;
|
2263: | $body .= $this->getBoundary($this->boundary[1], $bodyCharSet, '', $bodyEncoding);
|
2264: | $body .= $this->encodeString($this->Body, $bodyEncoding);
|
2265: | $body .= $this->LE . $this->LE;
|
2266: | $body .= $this->attachAll('inline', $this->boundary[1]);
|
2267: | break;
|
2268: | case 'attach':
|
2269: | $body .= $mimepre;
|
2270: | $body .= $this->getBoundary($this->boundary[1], $bodyCharSet, '', $bodyEncoding);
|
2271: | $body .= $this->encodeString($this->Body, $bodyEncoding);
|
2272: | $body .= $this->LE . $this->LE;
|
2273: | $body .= $this->attachAll('attachment', $this->boundary[1]);
|
2274: | break;
|
2275: | case 'inline_attach':
|
2276: | $body .= $mimepre;
|
2277: | $body .= $this->textLine('--' . $this->boundary[1]);
|
2278: | $body .= $this->headerLine('Content-Type', 'multipart/related;');
|
2279: | $body .= $this->textLine("\tboundary=\"" . $this->boundary[2] . '"');
|
2280: | $body .= $this->LE;
|
2281: | $body .= $this->getBoundary($this->boundary[2], $bodyCharSet, '', $bodyEncoding);
|
2282: | $body .= $this->encodeString($this->Body, $bodyEncoding);
|
2283: | $body .= $this->LE . $this->LE;
|
2284: | $body .= $this->attachAll('inline', $this->boundary[2]);
|
2285: | $body .= $this->LE;
|
2286: | $body .= $this->attachAll('attachment', $this->boundary[1]);
|
2287: | break;
|
2288: | case 'alt':
|
2289: | $body .= $mimepre;
|
2290: | $body .= $this->getBoundary($this->boundary[1], $altBodyCharSet, 'text/plain', $altBodyEncoding);
|
2291: | $body .= $this->encodeString($this->AltBody, $altBodyEncoding);
|
2292: | $body .= $this->LE . $this->LE;
|
2293: | $body .= $this->getBoundary($this->boundary[1], $bodyCharSet, 'text/html', $bodyEncoding);
|
2294: | $body .= $this->encodeString($this->Body, $bodyEncoding);
|
2295: | $body .= $this->LE . $this->LE;
|
2296: | if (!empty($this->Ical)) {
|
2297: | $body .= $this->getBoundary($this->boundary[1], '', 'text/calendar; method=REQUEST', '');
|
2298: | $body .= $this->encodeString($this->Ical, $this->Encoding);
|
2299: | $body .= $this->LE . $this->LE;
|
2300: | }
|
2301: | $body .= $this->endBoundary($this->boundary[1]);
|
2302: | break;
|
2303: | case 'alt_inline':
|
2304: | $body .= $mimepre;
|
2305: | $body .= $this->getBoundary($this->boundary[1], $altBodyCharSet, 'text/plain', $altBodyEncoding);
|
2306: | $body .= $this->encodeString($this->AltBody, $altBodyEncoding);
|
2307: | $body .= $this->LE . $this->LE;
|
2308: | $body .= $this->textLine('--' . $this->boundary[1]);
|
2309: | $body .= $this->headerLine('Content-Type', 'multipart/related;');
|
2310: | $body .= $this->textLine("\tboundary=\"" . $this->boundary[2] . '"');
|
2311: | $body .= $this->LE;
|
2312: | $body .= $this->getBoundary($this->boundary[2], $bodyCharSet, 'text/html', $bodyEncoding);
|
2313: | $body .= $this->encodeString($this->Body, $bodyEncoding);
|
2314: | $body .= $this->LE . $this->LE;
|
2315: | $body .= $this->attachAll('inline', $this->boundary[2]);
|
2316: | $body .= $this->LE;
|
2317: | $body .= $this->endBoundary($this->boundary[1]);
|
2318: | break;
|
2319: | case 'alt_attach':
|
2320: | $body .= $mimepre;
|
2321: | $body .= $this->textLine('--' . $this->boundary[1]);
|
2322: | $body .= $this->headerLine('Content-Type', 'multipart/alternative;');
|
2323: | $body .= $this->textLine("\tboundary=\"" . $this->boundary[2] . '"');
|
2324: | $body .= $this->LE;
|
2325: | $body .= $this->getBoundary($this->boundary[2], $altBodyCharSet, 'text/plain', $altBodyEncoding);
|
2326: | $body .= $this->encodeString($this->AltBody, $altBodyEncoding);
|
2327: | $body .= $this->LE . $this->LE;
|
2328: | $body .= $this->getBoundary($this->boundary[2], $bodyCharSet, 'text/html', $bodyEncoding);
|
2329: | $body .= $this->encodeString($this->Body, $bodyEncoding);
|
2330: | $body .= $this->LE . $this->LE;
|
2331: | $body .= $this->endBoundary($this->boundary[2]);
|
2332: | $body .= $this->LE;
|
2333: | $body .= $this->attachAll('attachment', $this->boundary[1]);
|
2334: | break;
|
2335: | case 'alt_inline_attach':
|
2336: | $body .= $mimepre;
|
2337: | $body .= $this->textLine('--' . $this->boundary[1]);
|
2338: | $body .= $this->headerLine('Content-Type', 'multipart/alternative;');
|
2339: | $body .= $this->textLine("\tboundary=\"" . $this->boundary[2] . '"');
|
2340: | $body .= $this->LE;
|
2341: | $body .= $this->getBoundary($this->boundary[2], $altBodyCharSet, 'text/plain', $altBodyEncoding);
|
2342: | $body .= $this->encodeString($this->AltBody, $altBodyEncoding);
|
2343: | $body .= $this->LE . $this->LE;
|
2344: | $body .= $this->textLine('--' . $this->boundary[2]);
|
2345: | $body .= $this->headerLine('Content-Type', 'multipart/related;');
|
2346: | $body .= $this->textLine("\tboundary=\"" . $this->boundary[3] . '"');
|
2347: | $body .= $this->LE;
|
2348: | $body .= $this->getBoundary($this->boundary[3], $bodyCharSet, 'text/html', $bodyEncoding);
|
2349: | $body .= $this->encodeString($this->Body, $bodyEncoding);
|
2350: | $body .= $this->LE . $this->LE;
|
2351: | $body .= $this->attachAll('inline', $this->boundary[3]);
|
2352: | $body .= $this->LE;
|
2353: | $body .= $this->endBoundary($this->boundary[2]);
|
2354: | $body .= $this->LE;
|
2355: | $body .= $this->attachAll('attachment', $this->boundary[1]);
|
2356: | break;
|
2357: | default:
|
2358: |
|
2359: |
|
2360: | $this->Encoding = $bodyEncoding;
|
2361: | $body .= $this->encodeString($this->Body, $this->Encoding);
|
2362: | break;
|
2363: | }
|
2364: |
|
2365: | if ($this->isError()) {
|
2366: | $body = '';
|
2367: | } elseif ($this->sign_key_file) {
|
2368: | try {
|
2369: | if (!defined('PKCS7_TEXT')) {
|
2370: | throw new phpmailerException($this->lang('extension_missing') . 'openssl');
|
2371: | }
|
2372: |
|
2373: | $file = tempnam(sys_get_temp_dir(), 'mail');
|
2374: | if (false === file_put_contents($file, $body)) {
|
2375: | throw new phpmailerException($this->lang('signing') . ' Could not write temp file');
|
2376: | }
|
2377: | $signed = tempnam(sys_get_temp_dir(), 'signed');
|
2378: |
|
2379: | if (empty($this->sign_extracerts_file)) {
|
2380: | $sign = @openssl_pkcs7_sign(
|
2381: | $file,
|
2382: | $signed,
|
2383: | 'file://' . realpath($this->sign_cert_file),
|
2384: | array('file://' . realpath($this->sign_key_file), $this->sign_key_pass),
|
2385: | null
|
2386: | );
|
2387: | } else {
|
2388: | $sign = @openssl_pkcs7_sign(
|
2389: | $file,
|
2390: | $signed,
|
2391: | 'file://' . realpath($this->sign_cert_file),
|
2392: | array('file://' . realpath($this->sign_key_file), $this->sign_key_pass),
|
2393: | null,
|
2394: | PKCS7_DETACHED,
|
2395: | $this->sign_extracerts_file
|
2396: | );
|
2397: | }
|
2398: | if ($sign) {
|
2399: | @unlink($file);
|
2400: | $body = file_get_contents($signed);
|
2401: | @unlink($signed);
|
2402: |
|
2403: | $parts = explode("\n\n", $body, 2);
|
2404: | $this->MIMEHeader .= $parts[0] . $this->LE . $this->LE;
|
2405: | $body = $parts[1];
|
2406: | } else {
|
2407: | @unlink($file);
|
2408: | @unlink($signed);
|
2409: | throw new phpmailerException($this->lang('signing') . openssl_error_string());
|
2410: | }
|
2411: | } catch (phpmailerException $exc) {
|
2412: | $body = '';
|
2413: | if ($this->exceptions) {
|
2414: | throw $exc;
|
2415: | }
|
2416: | }
|
2417: | }
|
2418: | return $body;
|
2419: | }
|
2420: |
|
2421: | |
2422: | |
2423: | |
2424: | |
2425: | |
2426: | |
2427: | |
2428: | |
2429: |
|
2430: | protected function getBoundary($boundary, $charSet, $contentType, $encoding)
|
2431: | {
|
2432: | $result = '';
|
2433: | if ($charSet == '') {
|
2434: | $charSet = $this->CharSet;
|
2435: | }
|
2436: | if ($contentType == '') {
|
2437: | $contentType = $this->ContentType;
|
2438: | }
|
2439: | if ($encoding == '') {
|
2440: | $encoding = $this->Encoding;
|
2441: | }
|
2442: | $result .= $this->textLine('--' . $boundary);
|
2443: | $result .= sprintf('Content-Type: %s; charset=%s', $contentType, $charSet);
|
2444: | $result .= $this->LE;
|
2445: |
|
2446: | if ($encoding != '7bit') {
|
2447: | $result .= $this->headerLine('Content-Transfer-Encoding', $encoding);
|
2448: | }
|
2449: | $result .= $this->LE;
|
2450: |
|
2451: | return $result;
|
2452: | }
|
2453: |
|
2454: | |
2455: | |
2456: | |
2457: | |
2458: | |
2459: |
|
2460: | protected function endBoundary($boundary)
|
2461: | {
|
2462: | return $this->LE . '--' . $boundary . '--' . $this->LE;
|
2463: | }
|
2464: |
|
2465: | |
2466: | |
2467: | |
2468: | |
2469: | |
2470: |
|
2471: | protected function setMessageType()
|
2472: | {
|
2473: | $type = array();
|
2474: | if ($this->alternativeExists()) {
|
2475: | $type[] = 'alt';
|
2476: | }
|
2477: | if ($this->inlineImageExists()) {
|
2478: | $type[] = 'inline';
|
2479: | }
|
2480: | if ($this->attachmentExists()) {
|
2481: | $type[] = 'attach';
|
2482: | }
|
2483: | $this->message_type = implode('_', $type);
|
2484: | if ($this->message_type == '') {
|
2485: |
|
2486: | $this->message_type = 'plain';
|
2487: | }
|
2488: | }
|
2489: |
|
2490: | |
2491: | |
2492: | |
2493: | |
2494: | |
2495: | |
2496: |
|
2497: | public function headerLine($name, $value)
|
2498: | {
|
2499: | return $name . ': ' . $value . $this->LE;
|
2500: | }
|
2501: |
|
2502: | |
2503: | |
2504: | |
2505: | |
2506: | |
2507: |
|
2508: | public function textLine($value)
|
2509: | {
|
2510: | return $value . $this->LE;
|
2511: | }
|
2512: |
|
2513: | |
2514: | |
2515: | |
2516: | |
2517: | |
2518: | |
2519: | |
2520: | |
2521: | |
2522: | |
2523: | |
2524: | |
2525: | |
2526: |
|
2527: | public function addAttachment($path, $name = '', $encoding = 'base64', $type = '', $disposition = 'attachment')
|
2528: | {
|
2529: | try {
|
2530: | if (!self::isPermittedPath($path) or !@is_file($path)) {
|
2531: | throw new phpmailerException($this->lang('file_access') . $path, self::STOP_CONTINUE);
|
2532: | }
|
2533: |
|
2534: |
|
2535: | if ($type == '') {
|
2536: | $type = self::filenameToType($path);
|
2537: | }
|
2538: |
|
2539: | $filename = basename($path);
|
2540: | if ($name == '') {
|
2541: | $name = $filename;
|
2542: | }
|
2543: |
|
2544: | $this->attachment[] = array(
|
2545: | 0 => $path,
|
2546: | 1 => $filename,
|
2547: | 2 => $name,
|
2548: | 3 => $encoding,
|
2549: | 4 => $type,
|
2550: | 5 => false,
|
2551: | 6 => $disposition,
|
2552: | 7 => 0
|
2553: | );
|
2554: |
|
2555: | } catch (phpmailerException $exc) {
|
2556: | $this->setError($exc->getMessage());
|
2557: | $this->edebug($exc->getMessage());
|
2558: | if ($this->exceptions) {
|
2559: | throw $exc;
|
2560: | }
|
2561: | return false;
|
2562: | }
|
2563: | return true;
|
2564: | }
|
2565: |
|
2566: | |
2567: | |
2568: | |
2569: |
|
2570: | public function getAttachments()
|
2571: | {
|
2572: | return $this->attachment;
|
2573: | }
|
2574: |
|
2575: | |
2576: | |
2577: | |
2578: | |
2579: | |
2580: | |
2581: | |
2582: |
|
2583: | protected function attachAll($disposition_type, $boundary)
|
2584: | {
|
2585: |
|
2586: | $mime = array();
|
2587: | $cidUniq = array();
|
2588: | $incl = array();
|
2589: |
|
2590: |
|
2591: | foreach ($this->attachment as $attachment) {
|
2592: |
|
2593: | if ($attachment[6] == $disposition_type) {
|
2594: |
|
2595: | $string = '';
|
2596: | $path = '';
|
2597: | $bString = $attachment[5];
|
2598: | if ($bString) {
|
2599: | $string = $attachment[0];
|
2600: | } else {
|
2601: | $path = $attachment[0];
|
2602: | }
|
2603: |
|
2604: | $inclhash = md5(serialize($attachment));
|
2605: | if (in_array($inclhash, $incl)) {
|
2606: | continue;
|
2607: | }
|
2608: | $incl[] = $inclhash;
|
2609: | $name = $attachment[2];
|
2610: | $encoding = $attachment[3];
|
2611: | $type = $attachment[4];
|
2612: | $disposition = $attachment[6];
|
2613: | $cid = $attachment[7];
|
2614: | if ($disposition == 'inline' && array_key_exists($cid, $cidUniq)) {
|
2615: | continue;
|
2616: | }
|
2617: | $cidUniq[$cid] = true;
|
2618: |
|
2619: | $mime[] = sprintf('--%s%s', $boundary, $this->LE);
|
2620: |
|
2621: | if (!empty($name)) {
|
2622: | $mime[] = sprintf(
|
2623: | 'Content-Type: %s; name="%s"%s',
|
2624: | $type,
|
2625: | $this->encodeHeader($this->secureHeader($name)),
|
2626: | $this->LE
|
2627: | );
|
2628: | } else {
|
2629: | $mime[] = sprintf(
|
2630: | 'Content-Type: %s%s',
|
2631: | $type,
|
2632: | $this->LE
|
2633: | );
|
2634: | }
|
2635: |
|
2636: | if ($encoding != '7bit') {
|
2637: | $mime[] = sprintf('Content-Transfer-Encoding: %s%s', $encoding, $this->LE);
|
2638: | }
|
2639: |
|
2640: | if ($disposition == 'inline') {
|
2641: | $mime[] = sprintf('Content-ID: <%s>%s', $cid, $this->LE);
|
2642: | }
|
2643: |
|
2644: |
|
2645: |
|
2646: |
|
2647: |
|
2648: | if (!(empty($disposition))) {
|
2649: | $encoded_name = $this->encodeHeader($this->secureHeader($name));
|
2650: | if (preg_match('/[ \(\)<>@,;:\\"\/\[\]\?=]/', $encoded_name)) {
|
2651: | $mime[] = sprintf(
|
2652: | 'Content-Disposition: %s; filename="%s"%s',
|
2653: | $disposition,
|
2654: | $encoded_name,
|
2655: | $this->LE . $this->LE
|
2656: | );
|
2657: | } else {
|
2658: | if (!empty($encoded_name)) {
|
2659: | $mime[] = sprintf(
|
2660: | 'Content-Disposition: %s; filename=%s%s',
|
2661: | $disposition,
|
2662: | $encoded_name,
|
2663: | $this->LE . $this->LE
|
2664: | );
|
2665: | } else {
|
2666: | $mime[] = sprintf(
|
2667: | 'Content-Disposition: %s%s',
|
2668: | $disposition,
|
2669: | $this->LE . $this->LE
|
2670: | );
|
2671: | }
|
2672: | }
|
2673: | } else {
|
2674: | $mime[] = $this->LE;
|
2675: | }
|
2676: |
|
2677: |
|
2678: | if ($bString) {
|
2679: | $mime[] = $this->encodeString($string, $encoding);
|
2680: | if ($this->isError()) {
|
2681: | return '';
|
2682: | }
|
2683: | $mime[] = $this->LE . $this->LE;
|
2684: | } else {
|
2685: | $mime[] = $this->encodeFile($path, $encoding);
|
2686: | if ($this->isError()) {
|
2687: | return '';
|
2688: | }
|
2689: | $mime[] = $this->LE . $this->LE;
|
2690: | }
|
2691: | }
|
2692: | }
|
2693: |
|
2694: | $mime[] = sprintf('--%s--%s', $boundary, $this->LE);
|
2695: |
|
2696: | return implode('', $mime);
|
2697: | }
|
2698: |
|
2699: | |
2700: | |
2701: | |
2702: | |
2703: | |
2704: | |
2705: | |
2706: | |
2707: |
|
2708: | protected function encodeFile($path, $encoding = 'base64')
|
2709: | {
|
2710: | try {
|
2711: | if (!self::isPermittedPath($path) or !file_exists($path)) {
|
2712: | throw new phpmailerException($this->lang('file_open') . $path, self::STOP_CONTINUE);
|
2713: | }
|
2714: | $magic_quotes = false;
|
2715: | if( version_compare(PHP_VERSION, '7.4.0', '<') ) {
|
2716: | $magic_quotes = get_magic_quotes_runtime();
|
2717: | }
|
2718: | if ($magic_quotes) {
|
2719: | if (version_compare(PHP_VERSION, '5.3.0', '<')) {
|
2720: | set_magic_quotes_runtime(false);
|
2721: | } else {
|
2722: |
|
2723: |
|
2724: |
|
2725: | ini_set('magic_quotes_runtime', false);
|
2726: | }
|
2727: | }
|
2728: | $file_buffer = file_get_contents($path);
|
2729: | $file_buffer = $this->encodeString($file_buffer, $encoding);
|
2730: | if ($magic_quotes) {
|
2731: | if (version_compare(PHP_VERSION, '5.3.0', '<')) {
|
2732: | set_magic_quotes_runtime($magic_quotes);
|
2733: | } else {
|
2734: | ini_set('magic_quotes_runtime', $magic_quotes);
|
2735: | }
|
2736: | }
|
2737: | return $file_buffer;
|
2738: | } catch (Exception $exc) {
|
2739: | $this->setError($exc->getMessage());
|
2740: | return '';
|
2741: | }
|
2742: | }
|
2743: |
|
2744: | |
2745: | |
2746: | |
2747: | |
2748: | |
2749: | |
2750: | |
2751: |
|
2752: | public function encodeString($str, $encoding = 'base64')
|
2753: | {
|
2754: | $encoded = '';
|
2755: | switch (strtolower($encoding)) {
|
2756: | case 'base64':
|
2757: | $encoded = chunk_split(base64_encode($str), 76, $this->LE);
|
2758: | break;
|
2759: | case '7bit':
|
2760: | case '8bit':
|
2761: | $encoded = $this->fixEOL($str);
|
2762: |
|
2763: | if (substr($encoded, -(strlen($this->LE))) != $this->LE) {
|
2764: | $encoded .= $this->LE;
|
2765: | }
|
2766: | break;
|
2767: | case 'binary':
|
2768: | $encoded = $str;
|
2769: | break;
|
2770: | case 'quoted-printable':
|
2771: | $encoded = $this->encodeQP($str);
|
2772: | break;
|
2773: | default:
|
2774: | $this->setError($this->lang('encoding') . $encoding);
|
2775: | break;
|
2776: | }
|
2777: | return $encoded;
|
2778: | }
|
2779: |
|
2780: | |
2781: | |
2782: | |
2783: | |
2784: | |
2785: | |
2786: | |
2787: |
|
2788: | public function encodeHeader($str, $position = 'text')
|
2789: | {
|
2790: | $matchcount = 0;
|
2791: | switch (strtolower($position)) {
|
2792: | case 'phrase':
|
2793: | if (!preg_match('/[\200-\377]/', $str)) {
|
2794: |
|
2795: | $encoded = addcslashes($str, "\0..\37\177\\\"");
|
2796: | if (($str == $encoded) && !preg_match('/[^A-Za-z0-9!#$%&\'*+\/=?^_`{|}~ -]/', $str)) {
|
2797: | return ($encoded);
|
2798: | } else {
|
2799: | return ("\"$encoded\"");
|
2800: | }
|
2801: | }
|
2802: | $matchcount = preg_match_all('/[^\040\041\043-\133\135-\176]/', $str, $matches);
|
2803: | break;
|
2804: |
|
2805: | case 'comment':
|
2806: | $matchcount = preg_match_all('/[()"]/', $str, $matches);
|
2807: |
|
2808: | case 'text':
|
2809: | default:
|
2810: | $matchcount += preg_match_all('/[\000-\010\013\014\016-\037\177-\377]/', $str, $matches);
|
2811: | break;
|
2812: | }
|
2813: |
|
2814: |
|
2815: | if ($matchcount == 0) {
|
2816: | return ($str);
|
2817: | }
|
2818: |
|
2819: | $maxlen = 75 - 7 - strlen($this->CharSet);
|
2820: |
|
2821: | if ($matchcount > strlen($str) / 3) {
|
2822: |
|
2823: | $encoding = 'B';
|
2824: | if (function_exists('mb_strlen') && $this->hasMultiBytes($str)) {
|
2825: |
|
2826: |
|
2827: | $encoded = $this->base64EncodeWrapMB($str, "\n");
|
2828: | } else {
|
2829: | $encoded = base64_encode($str);
|
2830: | $maxlen -= $maxlen % 4;
|
2831: | $encoded = trim(chunk_split($encoded, $maxlen, "\n"));
|
2832: | }
|
2833: | } else {
|
2834: | $encoding = 'Q';
|
2835: | $encoded = $this->encodeQ($str, $position);
|
2836: | $encoded = $this->wrapText($encoded, $maxlen, true);
|
2837: | $encoded = str_replace('=' . self::CRLF, "\n", trim($encoded));
|
2838: | }
|
2839: |
|
2840: | $encoded = preg_replace('/^(.*)$/m', ' =?' . $this->CharSet . "?$encoding?\\1?=", $encoded);
|
2841: | $encoded = trim(str_replace("\n", $this->LE, $encoded));
|
2842: |
|
2843: | return $encoded;
|
2844: | }
|
2845: |
|
2846: | |
2847: | |
2848: | |
2849: | |
2850: | |
2851: |
|
2852: | public function hasMultiBytes($str)
|
2853: | {
|
2854: | if (function_exists('mb_strlen')) {
|
2855: | return (strlen($str) > mb_strlen($str, $this->CharSet));
|
2856: | } else {
|
2857: | return false;
|
2858: | }
|
2859: | }
|
2860: |
|
2861: | |
2862: | |
2863: | |
2864: | |
2865: |
|
2866: | public function has8bitChars($text)
|
2867: | {
|
2868: | return (boolean)preg_match('/[\x80-\xFF]/', $text);
|
2869: | }
|
2870: |
|
2871: | |
2872: | |
2873: | |
2874: | |
2875: | |
2876: | |
2877: | |
2878: | |
2879: | |
2880: |
|
2881: | public function base64EncodeWrapMB($str, $linebreak = null)
|
2882: | {
|
2883: | $start = '=?' . $this->CharSet . '?B?';
|
2884: | $end = '?=';
|
2885: | $encoded = '';
|
2886: | if ($linebreak === null) {
|
2887: | $linebreak = $this->LE;
|
2888: | }
|
2889: |
|
2890: | $mb_length = mb_strlen($str, $this->CharSet);
|
2891: |
|
2892: | $length = 75 - strlen($start) - strlen($end);
|
2893: |
|
2894: | $ratio = $mb_length / strlen($str);
|
2895: |
|
2896: | $avgLength = floor($length * $ratio * .75);
|
2897: |
|
2898: | for ($i = 0; $i < $mb_length; $i += $offset) {
|
2899: | $lookBack = 0;
|
2900: | do {
|
2901: | $offset = $avgLength - $lookBack;
|
2902: | $chunk = mb_substr($str, $i, $offset, $this->CharSet);
|
2903: | $chunk = base64_encode($chunk);
|
2904: | $lookBack++;
|
2905: | } while (strlen($chunk) > $length);
|
2906: | $encoded .= $chunk . $linebreak;
|
2907: | }
|
2908: |
|
2909: |
|
2910: | $encoded = substr($encoded, 0, -strlen($linebreak));
|
2911: | return $encoded;
|
2912: | }
|
2913: |
|
2914: | |
2915: | |
2916: | |
2917: | |
2918: | |
2919: | |
2920: | |
2921: | |
2922: |
|
2923: | public function encodeQP($string, $line_max = 76)
|
2924: | {
|
2925: |
|
2926: | if (function_exists('quoted_printable_encode')) {
|
2927: | return quoted_printable_encode($string);
|
2928: | }
|
2929: |
|
2930: | $string = str_replace(
|
2931: | array('%20', '%0D%0A.', '%0D%0A', '%'),
|
2932: | array(' ', "\r\n=2E", "\r\n", '='),
|
2933: | rawurlencode($string)
|
2934: | );
|
2935: | return preg_replace('/[^\r\n]{' . ($line_max - 3) . '}[^=\r\n]{2}/', "$0=\r\n", $string);
|
2936: | }
|
2937: |
|
2938: | |
2939: | |
2940: | |
2941: | |
2942: | |
2943: | |
2944: | |
2945: | |
2946: | |
2947: |
|
2948: | public function encodeQPphp(
|
2949: | $string,
|
2950: | $line_max = 76,
|
2951: | $space_conv = false
|
2952: | ) {
|
2953: | return $this->encodeQP($string, $line_max);
|
2954: | }
|
2955: |
|
2956: | |
2957: | |
2958: | |
2959: | |
2960: | |
2961: | |
2962: | |
2963: |
|
2964: | public function encodeQ($str, $position = 'text')
|
2965: | {
|
2966: |
|
2967: | $pattern = '';
|
2968: | $encoded = str_replace(array("\r", "\n"), '', $str);
|
2969: | switch (strtolower($position)) {
|
2970: | case 'phrase':
|
2971: |
|
2972: | $pattern = '^A-Za-z0-9!*+\/ -';
|
2973: | break;
|
2974: |
|
2975: | case 'comment':
|
2976: |
|
2977: | $pattern = '\(\)"';
|
2978: |
|
2979: |
|
2980: | case 'text':
|
2981: | default:
|
2982: |
|
2983: |
|
2984: | $pattern = '\000-\011\013\014\016-\037\075\077\137\177-\377' . $pattern;
|
2985: | break;
|
2986: | }
|
2987: | $matches = array();
|
2988: | if (preg_match_all("/[{$pattern}]/", $encoded, $matches)) {
|
2989: |
|
2990: |
|
2991: | $eqkey = array_search('=', $matches[0]);
|
2992: | if (false !== $eqkey) {
|
2993: | unset($matches[0][$eqkey]);
|
2994: | array_unshift($matches[0], '=');
|
2995: | }
|
2996: | foreach (array_unique($matches[0]) as $char) {
|
2997: | $encoded = str_replace($char, '=' . sprintf('%02X', ord($char)), $encoded);
|
2998: | }
|
2999: | }
|
3000: |
|
3001: | return str_replace(' ', '_', $encoded);
|
3002: | }
|
3003: |
|
3004: | |
3005: | |
3006: | |
3007: | |
3008: | |
3009: | |
3010: | |
3011: | |
3012: | |
3013: | |
3014: |
|
3015: | public function addStringAttachment(
|
3016: | $string,
|
3017: | $filename,
|
3018: | $encoding = 'base64',
|
3019: | $type = '',
|
3020: | $disposition = 'attachment'
|
3021: | ) {
|
3022: |
|
3023: | if ($type == '') {
|
3024: | $type = self::filenameToType($filename);
|
3025: | }
|
3026: |
|
3027: | $this->attachment[] = array(
|
3028: | 0 => $string,
|
3029: | 1 => $filename,
|
3030: | 2 => basename($filename),
|
3031: | 3 => $encoding,
|
3032: | 4 => $type,
|
3033: | 5 => true,
|
3034: | 6 => $disposition,
|
3035: | 7 => 0
|
3036: | );
|
3037: | }
|
3038: |
|
3039: | |
3040: | |
3041: | |
3042: | |
3043: | |
3044: | |
3045: | |
3046: | |
3047: | |
3048: | |
3049: | |
3050: | |
3051: | |
3052: | |
3053: | |
3054: | |
3055: |
|
3056: | public function addEmbeddedImage($path, $cid, $name = '', $encoding = 'base64', $type = '', $disposition = 'inline')
|
3057: | {
|
3058: | if (!self::isPermittedPath($path) or !@is_file($path)) {
|
3059: | $this->setError($this->lang('file_access') . $path);
|
3060: | return false;
|
3061: | }
|
3062: |
|
3063: |
|
3064: | if ($type == '') {
|
3065: | $type = self::filenameToType($path);
|
3066: | }
|
3067: |
|
3068: | $filename = basename($path);
|
3069: | if ($name == '') {
|
3070: | $name = $filename;
|
3071: | }
|
3072: |
|
3073: |
|
3074: | $this->attachment[] = array(
|
3075: | 0 => $path,
|
3076: | 1 => $filename,
|
3077: | 2 => $name,
|
3078: | 3 => $encoding,
|
3079: | 4 => $type,
|
3080: | 5 => false,
|
3081: | 6 => $disposition,
|
3082: | 7 => $cid
|
3083: | );
|
3084: | return true;
|
3085: | }
|
3086: |
|
3087: | |
3088: | |
3089: | |
3090: | |
3091: | |
3092: | |
3093: | |
3094: | |
3095: | |
3096: | |
3097: | |
3098: | |
3099: | |
3100: |
|
3101: | public function addStringEmbeddedImage(
|
3102: | $string,
|
3103: | $cid,
|
3104: | $name = '',
|
3105: | $encoding = 'base64',
|
3106: | $type = '',
|
3107: | $disposition = 'inline'
|
3108: | ) {
|
3109: |
|
3110: | if ($type == '' and !empty($name)) {
|
3111: | $type = self::filenameToType($name);
|
3112: | }
|
3113: |
|
3114: |
|
3115: | $this->attachment[] = array(
|
3116: | 0 => $string,
|
3117: | 1 => $name,
|
3118: | 2 => $name,
|
3119: | 3 => $encoding,
|
3120: | 4 => $type,
|
3121: | 5 => true,
|
3122: | 6 => $disposition,
|
3123: | 7 => $cid
|
3124: | );
|
3125: | return true;
|
3126: | }
|
3127: |
|
3128: | |
3129: | |
3130: | |
3131: | |
3132: |
|
3133: | public function inlineImageExists()
|
3134: | {
|
3135: | foreach ($this->attachment as $attachment) {
|
3136: | if ($attachment[6] == 'inline') {
|
3137: | return true;
|
3138: | }
|
3139: | }
|
3140: | return false;
|
3141: | }
|
3142: |
|
3143: | |
3144: | |
3145: | |
3146: |
|
3147: | public function attachmentExists()
|
3148: | {
|
3149: | foreach ($this->attachment as $attachment) {
|
3150: | if ($attachment[6] == 'attachment') {
|
3151: | return true;
|
3152: | }
|
3153: | }
|
3154: | return false;
|
3155: | }
|
3156: |
|
3157: | |
3158: | |
3159: | |
3160: |
|
3161: | public function alternativeExists()
|
3162: | {
|
3163: | return !empty($this->AltBody);
|
3164: | }
|
3165: |
|
3166: | |
3167: | |
3168: | |
3169: | |
3170: | |
3171: |
|
3172: | public function clearQueuedAddresses($kind)
|
3173: | {
|
3174: | $RecipientsQueue = $this->RecipientsQueue;
|
3175: | foreach ($RecipientsQueue as $address => $params) {
|
3176: | if ($params[0] == $kind) {
|
3177: | unset($this->RecipientsQueue[$address]);
|
3178: | }
|
3179: | }
|
3180: | }
|
3181: |
|
3182: | |
3183: | |
3184: | |
3185: |
|
3186: | public function clearAddresses()
|
3187: | {
|
3188: | foreach ($this->to as $to) {
|
3189: | unset($this->all_recipients[strtolower($to[0])]);
|
3190: | }
|
3191: | $this->to = array();
|
3192: | $this->clearQueuedAddresses('to');
|
3193: | }
|
3194: |
|
3195: | |
3196: | |
3197: | |
3198: |
|
3199: | public function clearCCs()
|
3200: | {
|
3201: | foreach ($this->cc as $cc) {
|
3202: | unset($this->all_recipients[strtolower($cc[0])]);
|
3203: | }
|
3204: | $this->cc = array();
|
3205: | $this->clearQueuedAddresses('cc');
|
3206: | }
|
3207: |
|
3208: | |
3209: | |
3210: | |
3211: |
|
3212: | public function clearBCCs()
|
3213: | {
|
3214: | foreach ($this->bcc as $bcc) {
|
3215: | unset($this->all_recipients[strtolower($bcc[0])]);
|
3216: | }
|
3217: | $this->bcc = array();
|
3218: | $this->clearQueuedAddresses('bcc');
|
3219: | }
|
3220: |
|
3221: | |
3222: | |
3223: | |
3224: |
|
3225: | public function clearReplyTos()
|
3226: | {
|
3227: | $this->ReplyTo = array();
|
3228: | $this->ReplyToQueue = array();
|
3229: | }
|
3230: |
|
3231: | |
3232: | |
3233: | |
3234: |
|
3235: | public function clearAllRecipients()
|
3236: | {
|
3237: | $this->to = array();
|
3238: | $this->cc = array();
|
3239: | $this->bcc = array();
|
3240: | $this->all_recipients = array();
|
3241: | $this->RecipientsQueue = array();
|
3242: | }
|
3243: |
|
3244: | |
3245: | |
3246: | |
3247: |
|
3248: | public function clearAttachments()
|
3249: | {
|
3250: | $this->attachment = array();
|
3251: | }
|
3252: |
|
3253: | |
3254: | |
3255: | |
3256: |
|
3257: | public function clearCustomHeaders()
|
3258: | {
|
3259: | $this->CustomHeader = array();
|
3260: | }
|
3261: |
|
3262: | |
3263: | |
3264: | |
3265: | |
3266: | |
3267: |
|
3268: | protected function setError($msg)
|
3269: | {
|
3270: | $this->error_count++;
|
3271: | if ($this->Mailer == 'smtp' and !is_null($this->smtp)) {
|
3272: | $lasterror = $this->smtp->getError();
|
3273: | if (!empty($lasterror['error'])) {
|
3274: | $msg .= $this->lang('smtp_error') . $lasterror['error'];
|
3275: | if (!empty($lasterror['detail'])) {
|
3276: | $msg .= ' Detail: '. $lasterror['detail'];
|
3277: | }
|
3278: | if (!empty($lasterror['smtp_code'])) {
|
3279: | $msg .= ' SMTP code: ' . $lasterror['smtp_code'];
|
3280: | }
|
3281: | if (!empty($lasterror['smtp_code_ex'])) {
|
3282: | $msg .= ' Additional SMTP info: ' . $lasterror['smtp_code_ex'];
|
3283: | }
|
3284: | }
|
3285: | }
|
3286: | $this->ErrorInfo = $msg;
|
3287: | }
|
3288: |
|
3289: | |
3290: | |
3291: | |
3292: | |
3293: | |
3294: |
|
3295: | public static function rfcDate()
|
3296: | {
|
3297: |
|
3298: |
|
3299: | date_default_timezone_set(@date_default_timezone_get());
|
3300: | return date('D, j M Y H:i:s O');
|
3301: | }
|
3302: |
|
3303: | |
3304: | |
3305: | |
3306: | |
3307: | |
3308: |
|
3309: | protected function serverHostname()
|
3310: | {
|
3311: | $result = 'localhost.localdomain';
|
3312: | if (!empty($this->Hostname)) {
|
3313: | $result = $this->Hostname;
|
3314: | } elseif (isset($_SERVER) and array_key_exists('SERVER_NAME', $_SERVER) and !empty($_SERVER['SERVER_NAME'])) {
|
3315: | $result = $_SERVER['SERVER_NAME'];
|
3316: | } elseif (function_exists('gethostname') && gethostname() !== false) {
|
3317: | $result = gethostname();
|
3318: | } elseif (php_uname('n') !== false) {
|
3319: | $result = php_uname('n');
|
3320: | }
|
3321: | return $result;
|
3322: | }
|
3323: |
|
3324: | |
3325: | |
3326: | |
3327: | |
3328: | |
3329: |
|
3330: | protected function lang($key)
|
3331: | {
|
3332: | if (count($this->language) < 1) {
|
3333: | $this->setLanguage('en');
|
3334: | }
|
3335: |
|
3336: | if (array_key_exists($key, $this->language)) {
|
3337: | if ($key == 'smtp_connect_failed') {
|
3338: |
|
3339: |
|
3340: |
|
3341: | return $this->language[$key] . ' https://github.com/PHPMailer/PHPMailer/wiki/Troubleshooting';
|
3342: | }
|
3343: | return $this->language[$key];
|
3344: | } else {
|
3345: |
|
3346: | return $key;
|
3347: | }
|
3348: | }
|
3349: |
|
3350: | |
3351: | |
3352: | |
3353: | |
3354: |
|
3355: | public function isError()
|
3356: | {
|
3357: | return ($this->error_count > 0);
|
3358: | }
|
3359: |
|
3360: | |
3361: | |
3362: | |
3363: | |
3364: | |
3365: | |
3366: |
|
3367: | public function fixEOL($str)
|
3368: | {
|
3369: |
|
3370: | $nstr = str_replace(array("\r\n", "\r"), "\n", $str);
|
3371: |
|
3372: | if ($this->LE !== "\n") {
|
3373: | $nstr = str_replace("\n", $this->LE, $nstr);
|
3374: | }
|
3375: | return $nstr;
|
3376: | }
|
3377: |
|
3378: | |
3379: | |
3380: | |
3381: | |
3382: | |
3383: | |
3384: | |
3385: | |
3386: |
|
3387: | public function addCustomHeader($name, $value = null)
|
3388: | {
|
3389: | if ($value === null) {
|
3390: |
|
3391: | $this->CustomHeader[] = explode(':', $name, 2);
|
3392: | } else {
|
3393: | $this->CustomHeader[] = array($name, $value);
|
3394: | }
|
3395: | }
|
3396: |
|
3397: | |
3398: | |
3399: | |
3400: |
|
3401: | public function getCustomHeaders()
|
3402: | {
|
3403: | return $this->CustomHeader;
|
3404: | }
|
3405: |
|
3406: | |
3407: | |
3408: | |
3409: | |
3410: | |
3411: | |
3412: | |
3413: | |
3414: | |
3415: | |
3416: | |
3417: | |
3418: | |
3419: | |
3420: | |
3421: |
|
3422: | public function msgHTML($message, $basedir = '', $advanced = false)
|
3423: | {
|
3424: | preg_match_all('/(src|background)=["\'](.*)["\']/Ui', $message, $images);
|
3425: | if (array_key_exists(2, $images)) {
|
3426: | if (strlen($basedir) > 1 && substr($basedir, -1) != '/') {
|
3427: |
|
3428: | $basedir .= '/';
|
3429: | }
|
3430: | foreach ($images[2] as $imgindex => $url) {
|
3431: |
|
3432: | if (preg_match('#^data:(image[^;,]*)(;base64)?,#', $url, $match)) {
|
3433: | $data = substr($url, strpos($url, ','));
|
3434: | if ($match[2]) {
|
3435: | $data = base64_decode($data);
|
3436: | } else {
|
3437: | $data = rawurldecode($data);
|
3438: | }
|
3439: | $cid = md5($url) . '@phpmailer.0';
|
3440: | if ($this->addStringEmbeddedImage($data, $cid, 'embed' . $imgindex, 'base64', $match[1])) {
|
3441: | $message = str_replace(
|
3442: | $images[0][$imgindex],
|
3443: | $images[1][$imgindex] . '="cid:' . $cid . '"',
|
3444: | $message
|
3445: | );
|
3446: | }
|
3447: | continue;
|
3448: | }
|
3449: | if (
|
3450: |
|
3451: | !empty($basedir)
|
3452: |
|
3453: | && (strpos($url, '..') === false)
|
3454: |
|
3455: | && substr($url, 0, 4) !== 'cid:'
|
3456: |
|
3457: | && !preg_match('#^[a-z][a-z0-9+.-]*:?//#i', $url)
|
3458: | ) {
|
3459: | $filename = basename($url);
|
3460: | $directory = dirname($url);
|
3461: | if ($directory == '.') {
|
3462: | $directory = '';
|
3463: | }
|
3464: | $cid = md5($url) . '@phpmailer.0';
|
3465: | if (strlen($directory) > 1 && substr($directory, -1) != '/') {
|
3466: | $directory .= '/';
|
3467: | }
|
3468: | if ($this->addEmbeddedImage(
|
3469: | $basedir . $directory . $filename,
|
3470: | $cid,
|
3471: | $filename,
|
3472: | 'base64',
|
3473: | self::_mime_types((string)self::mb_pathinfo($filename, PATHINFO_EXTENSION))
|
3474: | )
|
3475: | ) {
|
3476: | $message = preg_replace(
|
3477: | '/' . $images[1][$imgindex] . '=["\']' . preg_quote($url, '/') . '["\']/Ui',
|
3478: | $images[1][$imgindex] . '="cid:' . $cid . '"',
|
3479: | $message
|
3480: | );
|
3481: | }
|
3482: | }
|
3483: | }
|
3484: | }
|
3485: | $this->isHTML(true);
|
3486: |
|
3487: | $this->Body = $this->normalizeBreaks($message);
|
3488: | $this->AltBody = $this->normalizeBreaks($this->html2text($message, $advanced));
|
3489: | if (!$this->alternativeExists()) {
|
3490: | $this->AltBody = 'To view this email message, open it in a program that understands HTML!' .
|
3491: | self::CRLF . self::CRLF;
|
3492: | }
|
3493: | return $this->Body;
|
3494: | }
|
3495: |
|
3496: | |
3497: | |
3498: | |
3499: | |
3500: | |
3501: | |
3502: | |
3503: | |
3504: | |
3505: | |
3506: | |
3507: | |
3508: | |
3509: | |
3510: | |
3511: | |
3512: | |
3513: | |
3514: | |
3515: |
|
3516: | public function html2text($html, $advanced = false)
|
3517: | {
|
3518: | if (is_callable($advanced)) {
|
3519: | return call_user_func($advanced, $html);
|
3520: | }
|
3521: | return html_entity_decode(
|
3522: | trim(strip_tags(preg_replace('/<(head|title|style|script)[^>]*>.*?<\/\\1>/si', '', $html))),
|
3523: | ENT_QUOTES,
|
3524: | $this->CharSet
|
3525: | );
|
3526: | }
|
3527: |
|
3528: | |
3529: | |
3530: | |
3531: | |
3532: | |
3533: | |
3534: |
|
3535: | public static function _mime_types($ext = '')
|
3536: | {
|
3537: | $mimes = array(
|
3538: | 'xl' => 'application/excel',
|
3539: | 'js' => 'application/javascript',
|
3540: | 'hqx' => 'application/mac-binhex40',
|
3541: | 'cpt' => 'application/mac-compactpro',
|
3542: | 'bin' => 'application/macbinary',
|
3543: | 'doc' => 'application/msword',
|
3544: | 'word' => 'application/msword',
|
3545: | 'xlsx' => 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet',
|
3546: | 'xltx' => 'application/vnd.openxmlformats-officedocument.spreadsheetml.template',
|
3547: | 'potx' => 'application/vnd.openxmlformats-officedocument.presentationml.template',
|
3548: | 'ppsx' => 'application/vnd.openxmlformats-officedocument.presentationml.slideshow',
|
3549: | 'pptx' => 'application/vnd.openxmlformats-officedocument.presentationml.presentation',
|
3550: | 'sldx' => 'application/vnd.openxmlformats-officedocument.presentationml.slide',
|
3551: | 'docx' => 'application/vnd.openxmlformats-officedocument.wordprocessingml.document',
|
3552: | 'dotx' => 'application/vnd.openxmlformats-officedocument.wordprocessingml.template',
|
3553: | 'xlam' => 'application/vnd.ms-excel.addin.macroEnabled.12',
|
3554: | 'xlsb' => 'application/vnd.ms-excel.sheet.binary.macroEnabled.12',
|
3555: | 'class' => 'application/octet-stream',
|
3556: | 'dll' => 'application/octet-stream',
|
3557: | 'dms' => 'application/octet-stream',
|
3558: | 'exe' => 'application/octet-stream',
|
3559: | 'lha' => 'application/octet-stream',
|
3560: | 'lzh' => 'application/octet-stream',
|
3561: | 'psd' => 'application/octet-stream',
|
3562: | 'sea' => 'application/octet-stream',
|
3563: | 'so' => 'application/octet-stream',
|
3564: | 'oda' => 'application/oda',
|
3565: | 'pdf' => 'application/pdf',
|
3566: | 'ai' => 'application/postscript',
|
3567: | 'eps' => 'application/postscript',
|
3568: | 'ps' => 'application/postscript',
|
3569: | 'smi' => 'application/smil',
|
3570: | 'smil' => 'application/smil',
|
3571: | 'mif' => 'application/vnd.mif',
|
3572: | 'xls' => 'application/vnd.ms-excel',
|
3573: | 'ppt' => 'application/vnd.ms-powerpoint',
|
3574: | 'wbxml' => 'application/vnd.wap.wbxml',
|
3575: | 'wmlc' => 'application/vnd.wap.wmlc',
|
3576: | 'dcr' => 'application/x-director',
|
3577: | 'dir' => 'application/x-director',
|
3578: | 'dxr' => 'application/x-director',
|
3579: | 'dvi' => 'application/x-dvi',
|
3580: | 'gtar' => 'application/x-gtar',
|
3581: | 'php3' => 'application/x-httpd-php',
|
3582: | 'php4' => 'application/x-httpd-php',
|
3583: | 'php' => 'application/x-httpd-php',
|
3584: | 'phtml' => 'application/x-httpd-php',
|
3585: | 'phps' => 'application/x-httpd-php-source',
|
3586: | 'swf' => 'application/x-shockwave-flash',
|
3587: | 'sit' => 'application/x-stuffit',
|
3588: | 'tar' => 'application/x-tar',
|
3589: | 'tgz' => 'application/x-tar',
|
3590: | 'xht' => 'application/xhtml+xml',
|
3591: | 'xhtml' => 'application/xhtml+xml',
|
3592: | 'zip' => 'application/zip',
|
3593: | 'mid' => 'audio/midi',
|
3594: | 'midi' => 'audio/midi',
|
3595: | 'mp2' => 'audio/mpeg',
|
3596: | 'mp3' => 'audio/mpeg',
|
3597: | 'mpga' => 'audio/mpeg',
|
3598: | 'aif' => 'audio/x-aiff',
|
3599: | 'aifc' => 'audio/x-aiff',
|
3600: | 'aiff' => 'audio/x-aiff',
|
3601: | 'ram' => 'audio/x-pn-realaudio',
|
3602: | 'rm' => 'audio/x-pn-realaudio',
|
3603: | 'rpm' => 'audio/x-pn-realaudio-plugin',
|
3604: | 'ra' => 'audio/x-realaudio',
|
3605: | 'wav' => 'audio/x-wav',
|
3606: | 'bmp' => 'image/bmp',
|
3607: | 'gif' => 'image/gif',
|
3608: | 'jpeg' => 'image/jpeg',
|
3609: | 'jpe' => 'image/jpeg',
|
3610: | 'jpg' => 'image/jpeg',
|
3611: | 'png' => 'image/png',
|
3612: | 'tiff' => 'image/tiff',
|
3613: | 'tif' => 'image/tiff',
|
3614: | 'eml' => 'message/rfc822',
|
3615: | 'css' => 'text/css',
|
3616: | 'html' => 'text/html',
|
3617: | 'htm' => 'text/html',
|
3618: | 'shtml' => 'text/html',
|
3619: | 'log' => 'text/plain',
|
3620: | 'text' => 'text/plain',
|
3621: | 'txt' => 'text/plain',
|
3622: | 'rtx' => 'text/richtext',
|
3623: | 'rtf' => 'text/rtf',
|
3624: | 'vcf' => 'text/vcard',
|
3625: | 'vcard' => 'text/vcard',
|
3626: | 'xml' => 'text/xml',
|
3627: | 'xsl' => 'text/xml',
|
3628: | 'mpeg' => 'video/mpeg',
|
3629: | 'mpe' => 'video/mpeg',
|
3630: | 'mpg' => 'video/mpeg',
|
3631: | 'mov' => 'video/quicktime',
|
3632: | 'qt' => 'video/quicktime',
|
3633: | 'rv' => 'video/vnd.rn-realvideo',
|
3634: | 'avi' => 'video/x-msvideo',
|
3635: | 'movie' => 'video/x-sgi-movie'
|
3636: | );
|
3637: | if (array_key_exists(strtolower($ext), $mimes)) {
|
3638: | return $mimes[strtolower($ext)];
|
3639: | }
|
3640: | return 'application/octet-stream';
|
3641: | }
|
3642: |
|
3643: | |
3644: | |
3645: | |
3646: | |
3647: | |
3648: | |
3649: |
|
3650: | public static function filenameToType($filename)
|
3651: | {
|
3652: |
|
3653: | $qpos = strpos($filename, '?');
|
3654: | if (false !== $qpos) {
|
3655: | $filename = substr($filename, 0, $qpos);
|
3656: | }
|
3657: | $pathinfo = self::mb_pathinfo($filename);
|
3658: | return self::_mime_types($pathinfo['extension']);
|
3659: | }
|
3660: |
|
3661: | |
3662: | |
3663: | |
3664: | |
3665: | |
3666: | |
3667: | |
3668: | |
3669: | |
3670: | |
3671: |
|
3672: | public static function mb_pathinfo($path, $options = null)
|
3673: | {
|
3674: | $ret = array('dirname' => '', 'basename' => '', 'extension' => '', 'filename' => '');
|
3675: | $pathinfo = array();
|
3676: | if (preg_match('%^(.*?)[\\\\/]*(([^/\\\\]*?)(\.([^\.\\\\/]+?)|))[\\\\/\.]*$%im', $path, $pathinfo)) {
|
3677: | if (array_key_exists(1, $pathinfo)) {
|
3678: | $ret['dirname'] = $pathinfo[1];
|
3679: | }
|
3680: | if (array_key_exists(2, $pathinfo)) {
|
3681: | $ret['basename'] = $pathinfo[2];
|
3682: | }
|
3683: | if (array_key_exists(5, $pathinfo)) {
|
3684: | $ret['extension'] = $pathinfo[5];
|
3685: | }
|
3686: | if (array_key_exists(3, $pathinfo)) {
|
3687: | $ret['filename'] = $pathinfo[3];
|
3688: | }
|
3689: | }
|
3690: | switch ($options) {
|
3691: | case PATHINFO_DIRNAME:
|
3692: | case 'dirname':
|
3693: | return $ret['dirname'];
|
3694: | case PATHINFO_BASENAME:
|
3695: | case 'basename':
|
3696: | return $ret['basename'];
|
3697: | case PATHINFO_EXTENSION:
|
3698: | case 'extension':
|
3699: | return $ret['extension'];
|
3700: | case PATHINFO_FILENAME:
|
3701: | case 'filename':
|
3702: | return $ret['filename'];
|
3703: | default:
|
3704: | return $ret;
|
3705: | }
|
3706: | }
|
3707: |
|
3708: | |
3709: | |
3710: | |
3711: | |
3712: | |
3713: | |
3714: | |
3715: | |
3716: | |
3717: | |
3718: | |
3719: | |
3720: | |
3721: |
|
3722: | public function set($name, $value = '')
|
3723: | {
|
3724: | if (property_exists($this, $name)) {
|
3725: | $this->$name = $value;
|
3726: | return true;
|
3727: | } else {
|
3728: | $this->setError($this->lang('variable_set') . $name);
|
3729: | return false;
|
3730: | }
|
3731: | }
|
3732: |
|
3733: | |
3734: | |
3735: | |
3736: | |
3737: | |
3738: |
|
3739: | public function secureHeader($str)
|
3740: | {
|
3741: | return trim(str_replace(array("\r", "\n"), '', $str));
|
3742: | }
|
3743: |
|
3744: | |
3745: | |
3746: | |
3747: | |
3748: | |
3749: | |
3750: | |
3751: | |
3752: | |
3753: |
|
3754: | public static function normalizeBreaks($text, $breaktype = "\r\n")
|
3755: | {
|
3756: | return preg_replace('/(\r\n|\r|\n)/ms', $breaktype, $text);
|
3757: | }
|
3758: |
|
3759: | |
3760: | |
3761: | |
3762: | |
3763: | |
3764: | |
3765: | |
3766: |
|
3767: | public function sign($cert_filename, $key_filename, $key_pass, $extracerts_filename = '')
|
3768: | {
|
3769: | $this->sign_cert_file = $cert_filename;
|
3770: | $this->sign_key_file = $key_filename;
|
3771: | $this->sign_key_pass = $key_pass;
|
3772: | $this->sign_extracerts_file = $extracerts_filename;
|
3773: | }
|
3774: |
|
3775: | |
3776: | |
3777: | |
3778: | |
3779: | |
3780: |
|
3781: | public function DKIM_QP($txt)
|
3782: | {
|
3783: | $line = '';
|
3784: | for ($i = 0; $i < strlen($txt); $i++) {
|
3785: | $ord = ord($txt[$i]);
|
3786: | if (((0x21 <= $ord) && ($ord <= 0x3A)) || $ord == 0x3C || ((0x3E <= $ord) && ($ord <= 0x7E))) {
|
3787: | $line .= $txt[$i];
|
3788: | } else {
|
3789: | $line .= '=' . sprintf('%02X', $ord);
|
3790: | }
|
3791: | }
|
3792: | return $line;
|
3793: | }
|
3794: |
|
3795: | |
3796: | |
3797: | |
3798: | |
3799: | |
3800: | |
3801: |
|
3802: | public function DKIM_Sign($signHeader)
|
3803: | {
|
3804: | if (!defined('PKCS7_TEXT')) {
|
3805: | if ($this->exceptions) {
|
3806: | throw new phpmailerException($this->lang('extension_missing') . 'openssl');
|
3807: | }
|
3808: | return '';
|
3809: | }
|
3810: | $privKeyStr = !empty($this->DKIM_private_string) ? $this->DKIM_private_string : file_get_contents($this->DKIM_private);
|
3811: | if ('' != $this->DKIM_passphrase) {
|
3812: | $privKey = openssl_pkey_get_private($privKeyStr, $this->DKIM_passphrase);
|
3813: | } else {
|
3814: | $privKey = openssl_pkey_get_private($privKeyStr);
|
3815: | }
|
3816: |
|
3817: |
|
3818: | if (version_compare(PHP_VERSION, '5.3.0') >= 0 and
|
3819: | in_array('sha256WithRSAEncryption', openssl_get_md_methods(true))) {
|
3820: | if (openssl_sign($signHeader, $signature, $privKey, 'sha256WithRSAEncryption')) {
|
3821: | openssl_pkey_free($privKey);
|
3822: | return base64_encode($signature);
|
3823: | }
|
3824: | } else {
|
3825: | $pinfo = openssl_pkey_get_details($privKey);
|
3826: | $hash = hash('sha256', $signHeader);
|
3827: |
|
3828: |
|
3829: | $t = '3031300d060960864801650304020105000420' . $hash;
|
3830: | $pslen = $pinfo['bits'] / 8 - (strlen($t) / 2 + 3);
|
3831: | $eb = pack('H*', '0001' . str_repeat('FF', $pslen) . '00' . $t);
|
3832: |
|
3833: | if (openssl_private_encrypt($eb, $signature, $privKey, OPENSSL_NO_PADDING)) {
|
3834: | openssl_pkey_free($privKey);
|
3835: | return base64_encode($signature);
|
3836: | }
|
3837: | }
|
3838: | openssl_pkey_free($privKey);
|
3839: | return '';
|
3840: | }
|
3841: |
|
3842: | |
3843: | |
3844: | |
3845: | |
3846: | |
3847: |
|
3848: | public function DKIM_HeaderC($signHeader)
|
3849: | {
|
3850: | $signHeader = preg_replace('/\r\n\s+/', ' ', $signHeader);
|
3851: | $lines = explode("\r\n", $signHeader);
|
3852: | foreach ($lines as $key => $line) {
|
3853: | list($heading, $value) = explode(':', $line, 2);
|
3854: | $heading = strtolower($heading);
|
3855: | $value = preg_replace('/\s{2,}/', ' ', $value);
|
3856: | $lines[$key] = $heading . ':' . trim($value);
|
3857: | }
|
3858: | $signHeader = implode("\r\n", $lines);
|
3859: | return $signHeader;
|
3860: | }
|
3861: |
|
3862: | |
3863: | |
3864: | |
3865: | |
3866: | |
3867: |
|
3868: | public function DKIM_BodyC($body)
|
3869: | {
|
3870: | if ($body == '') {
|
3871: | return "\r\n";
|
3872: | }
|
3873: |
|
3874: | $body = str_replace("\r\n", "\n", $body);
|
3875: | $body = str_replace("\n", "\r\n", $body);
|
3876: |
|
3877: | while (substr($body, strlen($body) - 4, 4) == "\r\n\r\n") {
|
3878: | $body = substr($body, 0, strlen($body) - 2);
|
3879: | }
|
3880: | return $body;
|
3881: | }
|
3882: |
|
3883: | |
3884: | |
3885: | |
3886: | |
3887: | |
3888: | |
3889: | |
3890: |
|
3891: | public function DKIM_Add($headers_line, $subject, $body)
|
3892: | {
|
3893: | $DKIMsignatureType = 'rsa-sha256';
|
3894: | $DKIMcanonicalization = 'relaxed/simple';
|
3895: | $DKIMquery = 'dns/txt';
|
3896: | $DKIMtime = time();
|
3897: | $subject_header = "Subject: $subject";
|
3898: | $headers = explode($this->LE, $headers_line);
|
3899: | $from_header = '';
|
3900: | $to_header = '';
|
3901: | $date_header = '';
|
3902: | $current = '';
|
3903: | foreach ($headers as $header) {
|
3904: | if (strpos($header, 'From:') === 0) {
|
3905: | $from_header = $header;
|
3906: | $current = 'from_header';
|
3907: | } elseif (strpos($header, 'To:') === 0) {
|
3908: | $to_header = $header;
|
3909: | $current = 'to_header';
|
3910: | } elseif (strpos($header, 'Date:') === 0) {
|
3911: | $date_header = $header;
|
3912: | $current = 'date_header';
|
3913: | } else {
|
3914: | if (!empty($$current) && strpos($header, ' =?') === 0) {
|
3915: | $$current .= $header;
|
3916: | } else {
|
3917: | $current = '';
|
3918: | }
|
3919: | }
|
3920: | }
|
3921: | $from = str_replace('|', '=7C', $this->DKIM_QP($from_header));
|
3922: | $to = str_replace('|', '=7C', $this->DKIM_QP($to_header));
|
3923: | $date = str_replace('|', '=7C', $this->DKIM_QP($date_header));
|
3924: | $subject = str_replace(
|
3925: | '|',
|
3926: | '=7C',
|
3927: | $this->DKIM_QP($subject_header)
|
3928: | );
|
3929: | $body = $this->DKIM_BodyC($body);
|
3930: | $DKIMlen = strlen($body);
|
3931: | $DKIMb64 = base64_encode(pack('H*', hash('sha256', $body)));
|
3932: | if ('' == $this->DKIM_identity) {
|
3933: | $ident = '';
|
3934: | } else {
|
3935: | $ident = ' i=' . $this->DKIM_identity . ';';
|
3936: | }
|
3937: | $dkimhdrs = 'DKIM-Signature: v=1; a=' .
|
3938: | $DKIMsignatureType . '; q=' .
|
3939: | $DKIMquery . '; l=' .
|
3940: | $DKIMlen . '; s=' .
|
3941: | $this->DKIM_selector .
|
3942: | ";\r\n" .
|
3943: | "\tt=" . $DKIMtime . '; c=' . $DKIMcanonicalization . ";\r\n" .
|
3944: | "\th=From:To:Date:Subject;\r\n" .
|
3945: | "\td=" . $this->DKIM_domain . ';' . $ident . "\r\n" .
|
3946: | "\tz=$from\r\n" .
|
3947: | "\t|$to\r\n" .
|
3948: | "\t|$date\r\n" .
|
3949: | "\t|$subject;\r\n" .
|
3950: | "\tbh=" . $DKIMb64 . ";\r\n" .
|
3951: | "\tb=";
|
3952: | $toSign = $this->DKIM_HeaderC(
|
3953: | $from_header . "\r\n" .
|
3954: | $to_header . "\r\n" .
|
3955: | $date_header . "\r\n" .
|
3956: | $subject_header . "\r\n" .
|
3957: | $dkimhdrs
|
3958: | );
|
3959: | $signed = $this->DKIM_Sign($toSign);
|
3960: | return $dkimhdrs . $signed . "\r\n";
|
3961: | }
|
3962: |
|
3963: | |
3964: | |
3965: | |
3966: | |
3967: | |
3968: |
|
3969: | public static function hasLineLongerThanMax($str)
|
3970: | {
|
3971: |
|
3972: | return (boolean)preg_match('/^(.{'.(self::MAX_LINE_LENGTH + 2).',})/m', $str);
|
3973: | }
|
3974: |
|
3975: | |
3976: | |
3977: | |
3978: | |
3979: | |
3980: |
|
3981: | public function getToAddresses()
|
3982: | {
|
3983: | return $this->to;
|
3984: | }
|
3985: |
|
3986: | |
3987: | |
3988: | |
3989: | |
3990: | |
3991: |
|
3992: | public function getCcAddresses()
|
3993: | {
|
3994: | return $this->cc;
|
3995: | }
|
3996: |
|
3997: | |
3998: | |
3999: | |
4000: | |
4001: | |
4002: |
|
4003: | public function getBccAddresses()
|
4004: | {
|
4005: | return $this->bcc;
|
4006: | }
|
4007: |
|
4008: | |
4009: | |
4010: | |
4011: | |
4012: | |
4013: |
|
4014: | public function getReplyToAddresses()
|
4015: | {
|
4016: | return $this->ReplyTo;
|
4017: | }
|
4018: |
|
4019: | |
4020: | |
4021: | |
4022: | |
4023: | |
4024: |
|
4025: | public function getAllRecipientAddresses()
|
4026: | {
|
4027: | return $this->all_recipients;
|
4028: | }
|
4029: |
|
4030: | |
4031: | |
4032: | |
4033: | |
4034: | |
4035: | |
4036: | |
4037: | |
4038: | |
4039: |
|
4040: | protected function doCallback($isSent, $to, $cc, $bcc, $subject, $body, $from)
|
4041: | {
|
4042: | if (!empty($this->action_function) && is_callable($this->action_function)) {
|
4043: | $params = array($isSent, $to, $cc, $bcc, $subject, $body, $from);
|
4044: | call_user_func_array($this->action_function, $params);
|
4045: | }
|
4046: | }
|
4047: | }
|
4048: |
|
4049: | |
4050: | |
4051: | |
4052: |
|
4053: | class phpmailerException extends Exception
|
4054: | {
|
4055: | |
4056: | |
4057: | |
4058: |
|
4059: | public function errorMessage()
|
4060: | {
|
4061: | $errorMsg = '<strong>' . htmlspecialchars($this->getMessage()) . "</strong><br />\n";
|
4062: | return $errorMsg;
|
4063: | }
|
4064: | }
|
4065: | |