blob: 71740ee5e13ec8526c9597fdc4098797f221ed7e [file] [log] [blame]
Andrey Andreevc5536aa2012-11-01 17:33:58 +02001<?php
Derek Allard2067d1a2008-11-13 22:59:24 +00002/**
3 * CodeIgniter
4 *
Andrey Andreevfe9309d2015-01-09 17:48:58 +02005 * An open source application development framework for PHP
Derek Allard2067d1a2008-11-13 22:59:24 +00006 *
Andrey Andreevbdb96ca2014-10-28 00:13:31 +02007 * This content is released under the MIT License (MIT)
Andrey Andreev1bd3d882011-12-22 15:38:20 +02008 *
Andrey Andreevcce6bd12018-01-09 11:32:02 +02009 * Copyright (c) 2014 - 2018, British Columbia Institute of Technology
Andrey Andreev1bd3d882011-12-22 15:38:20 +020010 *
Andrey Andreevbdb96ca2014-10-28 00:13:31 +020011 * Permission is hereby granted, free of charge, to any person obtaining a copy
12 * of this software and associated documentation files (the "Software"), to deal
13 * in the Software without restriction, including without limitation the rights
14 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
15 * copies of the Software, and to permit persons to whom the Software is
16 * furnished to do so, subject to the following conditions:
Derek Jonesf4a4bd82011-10-20 12:18:42 -050017 *
Andrey Andreevbdb96ca2014-10-28 00:13:31 +020018 * The above copyright notice and this permission notice shall be included in
19 * all copies or substantial portions of the Software.
20 *
21 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
22 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
23 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
24 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
25 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
26 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
27 * THE SOFTWARE.
28 *
29 * @package CodeIgniter
30 * @author EllisLab Dev Team
Andrey Andreev1924e872016-01-11 12:55:34 +020031 * @copyright Copyright (c) 2008 - 2014, EllisLab, Inc. (https://ellislab.com/)
Andrey Andreevcce6bd12018-01-09 11:32:02 +020032 * @copyright Copyright (c) 2014 - 2018, British Columbia Institute of Technology (http://bcit.ca/)
Andrey Andreevbdb96ca2014-10-28 00:13:31 +020033 * @license http://opensource.org/licenses/MIT MIT License
Andrey Andreevbd202c92016-01-11 12:50:18 +020034 * @link https://codeigniter.com
Andrey Andreevbdb96ca2014-10-28 00:13:31 +020035 * @since Version 1.0.0
Derek Allard2067d1a2008-11-13 22:59:24 +000036 * @filesource
37 */
Andrey Andreevc5536aa2012-11-01 17:33:58 +020038defined('BASEPATH') OR exit('No direct script access allowed');
Derek Allard2067d1a2008-11-13 22:59:24 +000039
Derek Allard2067d1a2008-11-13 22:59:24 +000040/**
41 * CodeIgniter Email Class
42 *
43 * Permits email to be sent using Mail, Sendmail, or SMTP.
44 *
45 * @package CodeIgniter
46 * @subpackage Libraries
47 * @category Libraries
Derek Jonesf4a4bd82011-10-20 12:18:42 -050048 * @author EllisLab Dev Team
Andrey Andreevbd202c92016-01-11 12:50:18 +020049 * @link https://codeigniter.com/user_guide/libraries/email.html
Derek Allard2067d1a2008-11-13 22:59:24 +000050 */
trita15dd4f2011-11-23 07:40:05 -050051class CI_Email {
Derek Allard2067d1a2008-11-13 22:59:24 +000052
Andrey Andreev597ea272012-11-01 22:56:26 +020053 /**
54 * Used as the User-Agent and X-Mailer headers' value.
55 *
56 * @var string
57 */
Andrey Andreev59c5b532012-03-01 14:09:51 +020058 public $useragent = 'CodeIgniter';
Andrey Andreev50814092012-03-01 12:36:12 +020059
Andrey Andreev597ea272012-11-01 22:56:26 +020060 /**
61 * Path to the Sendmail binary.
62 *
63 * @var string
64 */
65 public $mailpath = '/usr/sbin/sendmail'; // Sendmail path
66
67 /**
68 * Which method to use for sending e-mails.
69 *
70 * @var string 'mail', 'sendmail' or 'smtp'
71 */
72 public $protocol = 'mail'; // mail/sendmail/smtp
73
74 /**
75 * STMP Server host
76 *
77 * @var string
78 */
79 public $smtp_host = '';
80
81 /**
82 * SMTP Username
83 *
84 * @var string
85 */
86 public $smtp_user = '';
87
88 /**
89 * SMTP Password
90 *
91 * @var string
92 */
93 public $smtp_pass = '';
94
95 /**
96 * SMTP Server port
97 *
98 * @var int
99 */
100 public $smtp_port = 25;
101
102 /**
103 * SMTP connection timeout in seconds
104 *
105 * @var int
106 */
107 public $smtp_timeout = 5;
vlakofff3994252013-02-19 01:45:23 +0100108
nisheeth-barthwalcf225572013-02-18 01:08:04 +0530109 /**
nisheeth-barthwal59209de2013-02-18 17:02:13 +0530110 * SMTP persistent connection
111 *
112 * @var bool
113 */
114 public $smtp_keepalive = FALSE;
Andrey Andreev597ea272012-11-01 22:56:26 +0200115
116 /**
117 * SMTP Encryption
118 *
Andrey Andreev89b67b42013-03-12 19:34:23 +0200119 * @var string empty, 'tls' or 'ssl'
Andrey Andreev597ea272012-11-01 22:56:26 +0200120 */
Andrey Andreev89b67b42013-03-12 19:34:23 +0200121 public $smtp_crypto = '';
Andrey Andreev597ea272012-11-01 22:56:26 +0200122
123 /**
124 * Whether to apply word-wrapping to the message body.
125 *
126 * @var bool
127 */
128 public $wordwrap = TRUE;
129
130 /**
131 * Number of characters to wrap at.
132 *
133 * @see CI_Email::$wordwrap
134 * @var int
135 */
136 public $wrapchars = 76;
137
138 /**
139 * Message format.
140 *
141 * @var string 'text' or 'html'
142 */
143 public $mailtype = 'text';
144
145 /**
146 * Character set (default: utf-8)
147 *
148 * @var string
149 */
Andrey Andreev8c95c3d2016-05-09 12:35:41 +0300150 public $charset = 'UTF-8';
Andrey Andreev597ea272012-11-01 22:56:26 +0200151
152 /**
Andrey Andreev597ea272012-11-01 22:56:26 +0200153 * Alternative message (for HTML messages only)
154 *
155 * @var string
156 */
157 public $alt_message = '';
158
159 /**
160 * Whether to validate e-mail addresses.
161 *
162 * @var bool
163 */
164 public $validate = FALSE;
165
166 /**
167 * X-Priority header value.
168 *
169 * @var int 1-5
170 */
171 public $priority = 3; // Default priority (1 - 5)
172
173 /**
174 * Newline character sequence.
175 * Use "\r\n" to comply with RFC 822.
176 *
177 * @link http://www.ietf.org/rfc/rfc822.txt
178 * @var string "\r\n" or "\n"
179 */
180 public $newline = "\n"; // Default newline. "\r\n" or "\n" (Use "\r\n" to comply with RFC 822)
181
182 /**
183 * CRLF character sequence
184 *
185 * RFC 2045 specifies that for 'quoted-printable' encoding,
186 * "\r\n" must be used. However, it appears that some servers
187 * (even on the receiving end) don't handle it properly and
188 * switching to "\n", while improper, is the only solution
189 * that seems to work for all environments.
190 *
191 * @link http://www.ietf.org/rfc/rfc822.txt
192 * @var string
193 */
194 public $crlf = "\n";
195
196 /**
197 * Whether to use Delivery Status Notification.
198 *
199 * @var bool
200 */
201 public $dsn = FALSE;
202
203 /**
204 * Whether to send multipart alternatives.
205 * Yahoo! doesn't seem to like these.
206 *
207 * @var bool
208 */
209 public $send_multipart = TRUE;
210
211 /**
212 * Whether to send messages to BCC recipients in batches.
213 *
214 * @var bool
215 */
216 public $bcc_batch_mode = FALSE;
217
218 /**
219 * BCC Batch max number size.
220 *
221 * @see CI_Email::$bcc_batch_mode
222 * @var int
223 */
224 public $bcc_batch_size = 200;
225
226 // --------------------------------------------------------------------
227
228 /**
229 * Whether PHP is running in safe mode. Initialized by the class constructor.
230 *
231 * @var bool
232 */
Andrey Andreev50814092012-03-01 12:36:12 +0200233 protected $_safe_mode = FALSE;
Andrey Andreev597ea272012-11-01 22:56:26 +0200234
235 /**
236 * Subject header
237 *
238 * @var string
239 */
Andrey Andreev50814092012-03-01 12:36:12 +0200240 protected $_subject = '';
Andrey Andreev597ea272012-11-01 22:56:26 +0200241
242 /**
243 * Message body
244 *
245 * @var string
246 */
Andrey Andreev50814092012-03-01 12:36:12 +0200247 protected $_body = '';
Andrey Andreev597ea272012-11-01 22:56:26 +0200248
249 /**
250 * Final message body to be sent.
251 *
252 * @var string
253 */
Andrey Andreev50814092012-03-01 12:36:12 +0200254 protected $_finalbody = '';
Andrey Andreev597ea272012-11-01 22:56:26 +0200255
256 /**
Andrey Andreev597ea272012-11-01 22:56:26 +0200257 * Final headers to send
258 *
259 * @var string
260 */
Andrey Andreev50814092012-03-01 12:36:12 +0200261 protected $_header_str = '';
Andrey Andreev597ea272012-11-01 22:56:26 +0200262
263 /**
264 * SMTP Connection socket placeholder
265 *
266 * @var resource
267 */
Andrey Andreev50814092012-03-01 12:36:12 +0200268 protected $_smtp_connect = '';
Andrey Andreev597ea272012-11-01 22:56:26 +0200269
270 /**
271 * Mail encoding
272 *
273 * @var string '8bit' or '7bit'
274 */
Andrey Andreev50814092012-03-01 12:36:12 +0200275 protected $_encoding = '8bit';
Andrey Andreev597ea272012-11-01 22:56:26 +0200276
277 /**
278 * Whether to perform SMTP authentication
279 *
280 * @var bool
281 */
Andrey Andreev50814092012-03-01 12:36:12 +0200282 protected $_smtp_auth = FALSE;
Andrey Andreev597ea272012-11-01 22:56:26 +0200283
284 /**
285 * Whether to send a Reply-To header
286 *
287 * @var bool
288 */
Andrey Andreev50814092012-03-01 12:36:12 +0200289 protected $_replyto_flag = FALSE;
Andrey Andreev597ea272012-11-01 22:56:26 +0200290
291 /**
292 * Debug messages
293 *
294 * @see CI_Email::print_debugger()
295 * @var string
296 */
Andrey Andreev50814092012-03-01 12:36:12 +0200297 protected $_debug_msg = array();
Andrey Andreev597ea272012-11-01 22:56:26 +0200298
299 /**
300 * Recipients
301 *
302 * @var string[]
303 */
Andrey Andreev50814092012-03-01 12:36:12 +0200304 protected $_recipients = array();
Andrey Andreev597ea272012-11-01 22:56:26 +0200305
306 /**
307 * CC Recipients
308 *
309 * @var string[]
310 */
Andrey Andreev50814092012-03-01 12:36:12 +0200311 protected $_cc_array = array();
Andrey Andreev597ea272012-11-01 22:56:26 +0200312
313 /**
314 * BCC Recipients
315 *
316 * @var string[]
317 */
Andrey Andreev50814092012-03-01 12:36:12 +0200318 protected $_bcc_array = array();
Andrey Andreev597ea272012-11-01 22:56:26 +0200319
320 /**
321 * Message headers
322 *
323 * @var string[]
324 */
Andrey Andreev50814092012-03-01 12:36:12 +0200325 protected $_headers = array();
Andrey Andreev597ea272012-11-01 22:56:26 +0200326
327 /**
328 * Attachment data
329 *
330 * @var array
331 */
Andrey Andreevb8f9a152012-10-27 01:36:51 +0300332 protected $_attachments = array();
Andrey Andreev597ea272012-11-01 22:56:26 +0200333
334 /**
335 * Valid $protocol values
336 *
337 * @see CI_Email::$protocol
338 * @var string[]
339 */
Andrey Andreev50814092012-03-01 12:36:12 +0200340 protected $_protocols = array('mail', 'sendmail', 'smtp');
Andrey Andreev597ea272012-11-01 22:56:26 +0200341
342 /**
343 * Base charsets
344 *
345 * Character sets valid for 7-bit encoding,
346 * excluding language suffix.
347 *
348 * @var string[]
349 */
350 protected $_base_charsets = array('us-ascii', 'iso-2022-');
351
352 /**
353 * Bit depths
354 *
355 * Valid mail encodings
356 *
357 * @see CI_Email::$_encoding
358 * @var string[]
359 */
Andrey Andreev50814092012-03-01 12:36:12 +0200360 protected $_bit_depths = array('7bit', '8bit');
Andrey Andreev597ea272012-11-01 22:56:26 +0200361
362 /**
363 * $priority translations
364 *
365 * Actual values to send with the X-Priority header
366 *
367 * @var string[]
368 */
Andrey Andreevdea61772014-02-24 16:36:25 +0200369 protected $_priorities = array(
370 1 => '1 (Highest)',
371 2 => '2 (High)',
372 3 => '3 (Normal)',
373 4 => '4 (Low)',
374 5 => '5 (Lowest)'
375 );
Derek Allard2067d1a2008-11-13 22:59:24 +0000376
Andrey Andreev4e2cdec2016-10-28 14:19:08 +0300377 /**
Andrey Andreevc0c74d52017-01-19 15:26:35 +0200378 * mbstring.func_overload flag
Andrey Andreev4e2cdec2016-10-28 14:19:08 +0300379 *
380 * @var bool
381 */
Andrey Andreevc0c74d52017-01-19 15:26:35 +0200382 protected static $func_overload;
Andrey Andreev4e2cdec2016-10-28 14:19:08 +0300383
Andrey Andreev597ea272012-11-01 22:56:26 +0200384 // --------------------------------------------------------------------
385
Derek Allard2067d1a2008-11-13 22:59:24 +0000386 /**
387 * Constructor - Sets Email Preferences
388 *
389 * The constructor can be passed an array of config values
Andrey Andreev56454792012-05-17 14:32:19 +0300390 *
Andrey Andreev5fd3ae82012-10-24 14:55:35 +0300391 * @param array $config = array()
Andrey Andreev56454792012-05-17 14:32:19 +0300392 * @return void
Derek Allard2067d1a2008-11-13 22:59:24 +0000393 */
Andrey Andreev0b1fd2c2015-03-10 20:00:19 +0200394 public function __construct(array $config = array())
Derek Allard2067d1a2008-11-13 22:59:24 +0000395 {
Andrey Andreev925dd902012-10-19 11:06:31 +0300396 $this->charset = config_item('charset');
Andrey Andreev8c95c3d2016-05-09 12:35:41 +0300397 $this->initialize($config);
Andrey Andreevf6274742014-02-20 18:05:58 +0200398 $this->_safe_mode = ( ! is_php('5.4') && ini_get('safe_mode'));
Andrey Andreev925dd902012-10-19 11:06:31 +0300399
Andrey Andreevc0c74d52017-01-19 15:26:35 +0200400 isset(self::$func_overload) OR self::$func_overload = (extension_loaded('mbstring') && ini_get('mbstring.func_overload'));
Andrey Andreev4e2cdec2016-10-28 14:19:08 +0300401
Andrey Andreev90726b82015-01-20 12:39:22 +0200402 log_message('info', 'Email Class Initialized');
Derek Allard2067d1a2008-11-13 22:59:24 +0000403 }
nisheeth-barthwal59209de2013-02-18 17:02:13 +0530404
nisheeth-barthwalcf225572013-02-18 01:08:04 +0530405 // --------------------------------------------------------------------
vlakofff3994252013-02-19 01:45:23 +0100406
nisheeth-barthwalcf225572013-02-18 01:08:04 +0530407 /**
Derek Allard2067d1a2008-11-13 22:59:24 +0000408 * Initialize preferences
409 *
Andrey Andreev8c95c3d2016-05-09 12:35:41 +0300410 * @param array $config
Andrew Podner4296a652012-12-17 07:51:15 -0500411 * @return CI_Email
Derek Allard2067d1a2008-11-13 22:59:24 +0000412 */
Andrey Andreev8c95c3d2016-05-09 12:35:41 +0300413 public function initialize(array $config = array())
Derek Allard2067d1a2008-11-13 22:59:24 +0000414 {
Andrey Andreev8c95c3d2016-05-09 12:35:41 +0300415 $this->clear();
416
Derek Allard2067d1a2008-11-13 22:59:24 +0000417 foreach ($config as $key => $val)
418 {
419 if (isset($this->$key))
420 {
421 $method = 'set_'.$key;
422
423 if (method_exists($this, $method))
424 {
425 $this->$method($val);
426 }
427 else
428 {
429 $this->$key = $val;
430 }
431 }
432 }
433
Andrey Andreev8c95c3d2016-05-09 12:35:41 +0300434 $this->charset = strtoupper($this->charset);
435 $this->_smtp_auth = isset($this->smtp_user[0], $this->smtp_pass[0]);
Phil Sturgeona0f980e2011-01-13 10:59:12 +0000436
437 return $this;
Derek Allard2067d1a2008-11-13 22:59:24 +0000438 }
Barry Mienydd671972010-10-04 16:33:58 +0200439
Derek Allard2067d1a2008-11-13 22:59:24 +0000440 // --------------------------------------------------------------------
441
442 /**
443 * Initialize the Email Data
444 *
Bo-Yi Wu83320eb2011-09-15 13:28:02 +0800445 * @param bool
Andrew Podner4296a652012-12-17 07:51:15 -0500446 * @return CI_Email
Derek Allard2067d1a2008-11-13 22:59:24 +0000447 */
Phil Sturgeona0f980e2011-01-13 10:59:12 +0000448 public function clear($clear_attachments = FALSE)
Derek Allard2067d1a2008-11-13 22:59:24 +0000449 {
Andrey Andreev59c5b532012-03-01 14:09:51 +0200450 $this->_subject = '';
451 $this->_body = '';
452 $this->_finalbody = '';
453 $this->_header_str = '';
454 $this->_replyto_flag = FALSE;
Derek Allard2067d1a2008-11-13 22:59:24 +0000455 $this->_recipients = array();
Derek Jonesd1606352010-09-01 11:16:07 -0500456 $this->_cc_array = array();
457 $this->_bcc_array = array();
Derek Allard2067d1a2008-11-13 22:59:24 +0000458 $this->_headers = array();
459 $this->_debug_msg = array();
460
Mickey Wubfc1cad2012-05-31 22:28:40 -0700461 $this->set_header('Date', $this->_set_date());
Derek Allard2067d1a2008-11-13 22:59:24 +0000462
463 if ($clear_attachments !== FALSE)
464 {
Andrey Andreevb8f9a152012-10-27 01:36:51 +0300465 $this->_attachments = array();
Derek Allard2067d1a2008-11-13 22:59:24 +0000466 }
Eric Barnes6113f542010-12-29 13:36:12 -0500467
Greg Akera769deb2010-11-10 15:47:29 -0600468 return $this;
Derek Allard2067d1a2008-11-13 22:59:24 +0000469 }
Barry Mienydd671972010-10-04 16:33:58 +0200470
Derek Allard2067d1a2008-11-13 22:59:24 +0000471 // --------------------------------------------------------------------
472
473 /**
474 * Set FROM
475 *
Andrey Andreev5fd3ae82012-10-24 14:55:35 +0300476 * @param string $from
477 * @param string $name
478 * @param string $return_path = NULL Return-Path
Andrew Podner4296a652012-12-17 07:51:15 -0500479 * @return CI_Email
Derek Allard2067d1a2008-11-13 22:59:24 +0000480 */
Andrey Andreev925dd902012-10-19 11:06:31 +0300481 public function from($from, $name = '', $return_path = NULL)
Derek Allard2067d1a2008-11-13 22:59:24 +0000482 {
Andrey Andreev59c5b532012-03-01 14:09:51 +0200483 if (preg_match('/\<(.*)\>/', $from, $match))
Derek Allard2067d1a2008-11-13 22:59:24 +0000484 {
Andrey Andreev1bd3d882011-12-22 15:38:20 +0200485 $from = $match[1];
Derek Allard2067d1a2008-11-13 22:59:24 +0000486 }
487
488 if ($this->validate)
489 {
490 $this->validate_email($this->_str_to_array($from));
Andrey Andreevccd01c72012-10-05 17:12:55 +0300491 if ($return_path)
Melounek58dfc082012-06-29 08:43:47 +0200492 {
493 $this->validate_email($this->_str_to_array($return_path));
494 }
Derek Allard2067d1a2008-11-13 22:59:24 +0000495 }
496
497 // prepare the display name
Alex Bilbied261b1e2012-06-02 11:12:16 +0100498 if ($name !== '')
Derek Allard2067d1a2008-11-13 22:59:24 +0000499 {
500 // only use Q encoding if there are characters that would require it
501 if ( ! preg_match('/[\200-\377]/', $name))
502 {
503 // add slashes for non-printing characters, slashes, and double quotes, and surround it in double quotes
Derek Jonesc630bcf2008-11-17 21:09:45 +0000504 $name = '"'.addcslashes($name, "\0..\37\177'\"\\").'"';
Derek Allard2067d1a2008-11-13 22:59:24 +0000505 }
506 else
507 {
Andrey Andreev925dd902012-10-19 11:06:31 +0300508 $name = $this->_prep_q_encoding($name);
Derek Allard2067d1a2008-11-13 22:59:24 +0000509 }
510 }
511
Mickey Wubfc1cad2012-05-31 22:28:40 -0700512 $this->set_header('From', $name.' <'.$from.'>');
Melounek58dfc082012-06-29 08:43:47 +0200513
Andrey Andreev925dd902012-10-19 11:06:31 +0300514 isset($return_path) OR $return_path = $from;
Melounek58dfc082012-06-29 08:43:47 +0200515 $this->set_header('Return-Path', '<'.$return_path.'>');
Eric Barnes6113f542010-12-29 13:36:12 -0500516
Greg Akera769deb2010-11-10 15:47:29 -0600517 return $this;
Derek Allard2067d1a2008-11-13 22:59:24 +0000518 }
Barry Mienydd671972010-10-04 16:33:58 +0200519
Derek Allard2067d1a2008-11-13 22:59:24 +0000520 // --------------------------------------------------------------------
521
522 /**
523 * Set Reply-to
524 *
Derek Allard2067d1a2008-11-13 22:59:24 +0000525 * @param string
526 * @param string
Andrew Podner4296a652012-12-17 07:51:15 -0500527 * @return CI_Email
Derek Allard2067d1a2008-11-13 22:59:24 +0000528 */
Phil Sturgeona0f980e2011-01-13 10:59:12 +0000529 public function reply_to($replyto, $name = '')
Derek Allard2067d1a2008-11-13 22:59:24 +0000530 {
Andrey Andreev59c5b532012-03-01 14:09:51 +0200531 if (preg_match('/\<(.*)\>/', $replyto, $match))
Derek Allard2067d1a2008-11-13 22:59:24 +0000532 {
Andrey Andreev1bd3d882011-12-22 15:38:20 +0200533 $replyto = $match[1];
Derek Allard2067d1a2008-11-13 22:59:24 +0000534 }
535
536 if ($this->validate)
537 {
538 $this->validate_email($this->_str_to_array($replyto));
539 }
540
Andrey Andreevc70216d2016-01-20 17:25:13 +0200541 if ($name !== '')
Derek Allard2067d1a2008-11-13 22:59:24 +0000542 {
Andrey Andreevc70216d2016-01-20 17:25:13 +0200543 // only use Q encoding if there are characters that would require it
544 if ( ! preg_match('/[\200-\377]/', $name))
545 {
546 // add slashes for non-printing characters, slashes, and double quotes, and surround it in double quotes
547 $name = '"'.addcslashes($name, "\0..\37\177'\"\\").'"';
548 }
549 else
550 {
551 $name = $this->_prep_q_encoding($name);
552 }
Derek Allard2067d1a2008-11-13 22:59:24 +0000553 }
554
Mickey Wubfc1cad2012-05-31 22:28:40 -0700555 $this->set_header('Reply-To', $name.' <'.$replyto.'>');
Derek Allard2067d1a2008-11-13 22:59:24 +0000556 $this->_replyto_flag = TRUE;
Greg Akera769deb2010-11-10 15:47:29 -0600557
558 return $this;
Derek Allard2067d1a2008-11-13 22:59:24 +0000559 }
Barry Mienydd671972010-10-04 16:33:58 +0200560
Derek Allard2067d1a2008-11-13 22:59:24 +0000561 // --------------------------------------------------------------------
562
563 /**
564 * Set Recipients
565 *
Derek Allard2067d1a2008-11-13 22:59:24 +0000566 * @param string
Andrew Podner4296a652012-12-17 07:51:15 -0500567 * @return CI_Email
Derek Allard2067d1a2008-11-13 22:59:24 +0000568 */
Phil Sturgeona0f980e2011-01-13 10:59:12 +0000569 public function to($to)
Derek Allard2067d1a2008-11-13 22:59:24 +0000570 {
571 $to = $this->_str_to_array($to);
572 $to = $this->clean_email($to);
573
574 if ($this->validate)
575 {
576 $this->validate_email($to);
577 }
578
Andrey Andreev1bd3d882011-12-22 15:38:20 +0200579 if ($this->_get_protocol() !== 'mail')
Derek Allard2067d1a2008-11-13 22:59:24 +0000580 {
Mickey Wubfc1cad2012-05-31 22:28:40 -0700581 $this->set_header('To', implode(', ', $to));
Derek Allard2067d1a2008-11-13 22:59:24 +0000582 }
583
Andrey Andreev8a7078b2012-10-17 10:52:49 +0300584 $this->_recipients = $to;
Eric Barnes6113f542010-12-29 13:36:12 -0500585
Greg Akera769deb2010-11-10 15:47:29 -0600586 return $this;
Derek Allard2067d1a2008-11-13 22:59:24 +0000587 }
Barry Mienydd671972010-10-04 16:33:58 +0200588
Derek Allard2067d1a2008-11-13 22:59:24 +0000589 // --------------------------------------------------------------------
590
591 /**
592 * Set CC
593 *
Derek Allard2067d1a2008-11-13 22:59:24 +0000594 * @param string
Andrew Podner4296a652012-12-17 07:51:15 -0500595 * @return CI_Email
Derek Allard2067d1a2008-11-13 22:59:24 +0000596 */
Phil Sturgeona0f980e2011-01-13 10:59:12 +0000597 public function cc($cc)
Derek Allard2067d1a2008-11-13 22:59:24 +0000598 {
Andrey Andreev56454792012-05-17 14:32:19 +0300599 $cc = $this->clean_email($this->_str_to_array($cc));
Derek Allard2067d1a2008-11-13 22:59:24 +0000600
601 if ($this->validate)
602 {
603 $this->validate_email($cc);
604 }
605
Mickey Wubfc1cad2012-05-31 22:28:40 -0700606 $this->set_header('Cc', implode(', ', $cc));
Derek Allard2067d1a2008-11-13 22:59:24 +0000607
Andrey Andreev1bd3d882011-12-22 15:38:20 +0200608 if ($this->_get_protocol() === 'smtp')
Derek Allard2067d1a2008-11-13 22:59:24 +0000609 {
610 $this->_cc_array = $cc;
611 }
Eric Barnes6113f542010-12-29 13:36:12 -0500612
Greg Akera769deb2010-11-10 15:47:29 -0600613 return $this;
Derek Allard2067d1a2008-11-13 22:59:24 +0000614 }
Barry Mienydd671972010-10-04 16:33:58 +0200615
Derek Allard2067d1a2008-11-13 22:59:24 +0000616 // --------------------------------------------------------------------
617
618 /**
619 * Set BCC
620 *
Derek Allard2067d1a2008-11-13 22:59:24 +0000621 * @param string
622 * @param string
Andrew Podner4296a652012-12-17 07:51:15 -0500623 * @return CI_Email
Derek Allard2067d1a2008-11-13 22:59:24 +0000624 */
Phil Sturgeona0f980e2011-01-13 10:59:12 +0000625 public function bcc($bcc, $limit = '')
Derek Allard2067d1a2008-11-13 22:59:24 +0000626 {
Alex Bilbied261b1e2012-06-02 11:12:16 +0100627 if ($limit !== '' && is_numeric($limit))
Derek Allard2067d1a2008-11-13 22:59:24 +0000628 {
629 $this->bcc_batch_mode = TRUE;
630 $this->bcc_batch_size = $limit;
631 }
632
Andrey Andreev56454792012-05-17 14:32:19 +0300633 $bcc = $this->clean_email($this->_str_to_array($bcc));
Derek Allard2067d1a2008-11-13 22:59:24 +0000634
635 if ($this->validate)
636 {
637 $this->validate_email($bcc);
638 }
639
Andrey Andreev1bd3d882011-12-22 15:38:20 +0200640 if ($this->_get_protocol() === 'smtp' OR ($this->bcc_batch_mode && count($bcc) > $this->bcc_batch_size))
Derek Allard2067d1a2008-11-13 22:59:24 +0000641 {
642 $this->_bcc_array = $bcc;
643 }
644 else
645 {
Mickey Wubfc1cad2012-05-31 22:28:40 -0700646 $this->set_header('Bcc', implode(', ', $bcc));
Derek Allard2067d1a2008-11-13 22:59:24 +0000647 }
Eric Barnes6113f542010-12-29 13:36:12 -0500648
Greg Akera769deb2010-11-10 15:47:29 -0600649 return $this;
Derek Allard2067d1a2008-11-13 22:59:24 +0000650 }
Barry Mienydd671972010-10-04 16:33:58 +0200651
Derek Allard2067d1a2008-11-13 22:59:24 +0000652 // --------------------------------------------------------------------
653
654 /**
655 * Set Email Subject
656 *
Derek Allard2067d1a2008-11-13 22:59:24 +0000657 * @param string
Andrew Podner4296a652012-12-17 07:51:15 -0500658 * @return CI_Email
Derek Allard2067d1a2008-11-13 22:59:24 +0000659 */
Phil Sturgeona0f980e2011-01-13 10:59:12 +0000660 public function subject($subject)
Derek Allard2067d1a2008-11-13 22:59:24 +0000661 {
662 $subject = $this->_prep_q_encoding($subject);
Mickey Wubfc1cad2012-05-31 22:28:40 -0700663 $this->set_header('Subject', $subject);
Greg Akera769deb2010-11-10 15:47:29 -0600664 return $this;
Derek Allard2067d1a2008-11-13 22:59:24 +0000665 }
Barry Mienydd671972010-10-04 16:33:58 +0200666
Derek Allard2067d1a2008-11-13 22:59:24 +0000667 // --------------------------------------------------------------------
668
669 /**
670 * Set Body
671 *
Derek Allard2067d1a2008-11-13 22:59:24 +0000672 * @param string
Andrew Podner4296a652012-12-17 07:51:15 -0500673 * @return CI_Email
Derek Allard2067d1a2008-11-13 22:59:24 +0000674 */
Phil Sturgeona0f980e2011-01-13 10:59:12 +0000675 public function message($body)
Derek Allard2067d1a2008-11-13 22:59:24 +0000676 {
Andrey Andreev59c5b532012-03-01 14:09:51 +0200677 $this->_body = rtrim(str_replace("\r", '', $body));
diegorivera33c32802011-10-19 02:56:15 -0200678
Andrey Andreevaf728622011-10-20 10:11:59 +0300679 /* strip slashes only if magic quotes is ON
680 if we do it with magic quotes OFF, it strips real, user-inputted chars.
681
682 NOTE: In PHP 5.4 get_magic_quotes_gpc() will always return 0 and
683 it will probably not exist in future versions at all.
684 */
685 if ( ! is_php('5.4') && get_magic_quotes_gpc())
diegorivera9fca6152011-10-19 11:18:45 -0200686 {
diegorivera33c32802011-10-19 02:56:15 -0200687 $this->_body = stripslashes($this->_body);
diegorivera9fca6152011-10-19 11:18:45 -0200688 }
diegorivera33c32802011-10-19 02:56:15 -0200689
Greg Akera769deb2010-11-10 15:47:29 -0600690 return $this;
Derek Allard2067d1a2008-11-13 22:59:24 +0000691 }
Barry Mienydd671972010-10-04 16:33:58 +0200692
Derek Allard2067d1a2008-11-13 22:59:24 +0000693 // --------------------------------------------------------------------
694
695 /**
696 * Assign file attachments
697 *
Petr Heralecky1dbdf72c2014-01-10 16:54:51 +0100698 * @param string $file Can be local path, URL or buffered content
Andrey Andreev5fd3ae82012-10-24 14:55:35 +0300699 * @param string $disposition = 'attachment'
700 * @param string $newname = NULL
701 * @param string $mime = ''
Andrew Podner4296a652012-12-17 07:51:15 -0500702 * @return CI_Email
Derek Allard2067d1a2008-11-13 22:59:24 +0000703 */
Petr Heraleckyceb03af2014-01-10 16:40:54 +0100704 public function attach($file, $disposition = '', $newname = NULL, $mime = '')
Derek Allard2067d1a2008-11-13 22:59:24 +0000705 {
Petr Heralecky300e3f02014-01-10 11:49:11 +0100706 if ($mime === '')
707 {
Petr Heralecky1dbdf72c2014-01-10 16:54:51 +0100708 if (strpos($file, '://') === FALSE && ! file_exists($file))
Petr Heralecky300e3f02014-01-10 11:49:11 +0100709 {
Petr Heraleckyceb03af2014-01-10 16:40:54 +0100710 $this->_set_error_message('lang:email_attachment_missing', $file);
Petr Heralecky300e3f02014-01-10 11:49:11 +0100711 return FALSE;
712 }
Andrey Andreevc8097262014-01-10 14:45:31 +0200713
Andrey Andreev7cf682a2014-03-13 14:55:45 +0200714 if ( ! $fp = @fopen($file, 'rb'))
Petr Heralecky300e3f02014-01-10 11:49:11 +0100715 {
Petr Heraleckyceb03af2014-01-10 16:40:54 +0100716 $this->_set_error_message('lang:email_attachment_unreadable', $file);
Petr Heralecky300e3f02014-01-10 11:49:11 +0100717 return FALSE;
718 }
Petr Heralecky1dbdf72c2014-01-10 16:54:51 +0100719
Petr Heralecky300e3f02014-01-10 11:49:11 +0100720 $file_content = stream_get_contents($fp);
Petr Heraleckyceb03af2014-01-10 16:40:54 +0100721 $mime = $this->_mime_types(pathinfo($file, PATHINFO_EXTENSION));
Petr Heraleckyde886152014-01-10 12:52:56 +0100722 fclose($fp);
Petr Heralecky300e3f02014-01-10 11:49:11 +0100723 }
724 else
725 {
Petr Heraleckyceb03af2014-01-10 16:40:54 +0100726 $file_content =& $file; // buffered file
Petr Heralecky300e3f02014-01-10 11:49:11 +0100727 }
Andrey Andreevc8097262014-01-10 14:45:31 +0200728
Andrey Andreevb8f9a152012-10-27 01:36:51 +0300729 $this->_attachments[] = array(
Petr Heraleckyceb03af2014-01-10 16:40:54 +0100730 'name' => array($file, $newname),
Andrey Andreevb8f9a152012-10-27 01:36:51 +0300731 'disposition' => empty($disposition) ? 'attachment' : $disposition, // Can also be 'inline' Not sure if it matters
Petr Heralecky300e3f02014-01-10 11:49:11 +0100732 'type' => $mime,
Andrey Andreev3c422792016-06-06 09:44:50 +0300733 'content' => chunk_split(base64_encode($file_content)),
734 'multipart' => 'mixed'
Andrey Andreevb8f9a152012-10-27 01:36:51 +0300735 );
Petr Heralecky9ad2fff2014-01-10 13:25:34 +0100736
Greg Akera769deb2010-11-10 15:47:29 -0600737 return $this;
Derek Allard2067d1a2008-11-13 22:59:24 +0000738 }
Andrey Andreevc8097262014-01-10 14:45:31 +0200739
Petr Heralecky300e3f02014-01-10 11:49:11 +0100740 // --------------------------------------------------------------------
Andrey Andreevc8097262014-01-10 14:45:31 +0200741
Petr Heralecky300e3f02014-01-10 11:49:11 +0100742 /**
Andrey Andreevc8097262014-01-10 14:45:31 +0200743 * Set and return attachment Content-ID
744 *
745 * Useful for attached inline pictures
Petr Heralecky300e3f02014-01-10 11:49:11 +0100746 *
Petr Heraleckyde886152014-01-10 12:52:56 +0100747 * @param string $filename
748 * @return string
Petr Heralecky300e3f02014-01-10 11:49:11 +0100749 */
Andrey Andreevc8097262014-01-10 14:45:31 +0200750 public function attachment_cid($filename)
Petr Heralecky300e3f02014-01-10 11:49:11 +0100751 {
Petr Heraleckyde886152014-01-10 12:52:56 +0100752 for ($i = 0, $c = count($this->_attachments); $i < $c; $i++)
Petr Heralecky300e3f02014-01-10 11:49:11 +0100753 {
Petr Heralecky9ad2fff2014-01-10 13:25:34 +0100754 if ($this->_attachments[$i]['name'][0] === $filename)
Petr Heralecky300e3f02014-01-10 11:49:11 +0100755 {
Andrey Andreev3c422792016-06-06 09:44:50 +0300756 $this->_attachments[$i]['multipart'] = 'related';
Petr Heralecky230fca32014-01-10 12:55:57 +0100757 $this->_attachments[$i]['cid'] = uniqid(basename($this->_attachments[$i]['name'][0]).'@');
Petr Heraleckyde886152014-01-10 12:52:56 +0100758 return $this->_attachments[$i]['cid'];
Petr Heralecky300e3f02014-01-10 11:49:11 +0100759 }
760 }
Andrey Andreevc8097262014-01-10 14:45:31 +0200761
Petr Heralecky300e3f02014-01-10 11:49:11 +0100762 return FALSE;
763 }
Derek Allard2067d1a2008-11-13 22:59:24 +0000764
765 // --------------------------------------------------------------------
766
767 /**
768 * Add a Header Item
769 *
Derek Allard2067d1a2008-11-13 22:59:24 +0000770 * @param string
771 * @param string
Andrey Andreev6cefc6b2015-08-03 10:22:19 +0300772 * @return CI_Email
Derek Allard2067d1a2008-11-13 22:59:24 +0000773 */
Mickey Wubfc1cad2012-05-31 22:28:40 -0700774 public function set_header($header, $value)
Derek Allard2067d1a2008-11-13 22:59:24 +0000775 {
florisluitenf55d5142013-06-07 17:20:06 +0300776 $this->_headers[$header] = str_replace(array("\n", "\r"), '', $value);
Andrey Andreev6cefc6b2015-08-03 10:22:19 +0300777 return $this;
Derek Allard2067d1a2008-11-13 22:59:24 +0000778 }
Barry Mienydd671972010-10-04 16:33:58 +0200779
Derek Allard2067d1a2008-11-13 22:59:24 +0000780 // --------------------------------------------------------------------
781
782 /**
783 * Convert a String to an Array
784 *
Derek Allard2067d1a2008-11-13 22:59:24 +0000785 * @param string
786 * @return array
787 */
Phil Sturgeon6d2f13a2011-07-20 10:04:52 -0600788 protected function _str_to_array($email)
Derek Allard2067d1a2008-11-13 22:59:24 +0000789 {
790 if ( ! is_array($email))
791 {
Andrey Andreev56454792012-05-17 14:32:19 +0300792 return (strpos($email, ',') !== FALSE)
793 ? preg_split('/[\s,]/', $email, -1, PREG_SPLIT_NO_EMPTY)
794 : (array) trim($email);
Derek Allard2067d1a2008-11-13 22:59:24 +0000795 }
Andrey Andreev56454792012-05-17 14:32:19 +0300796
Derek Allard2067d1a2008-11-13 22:59:24 +0000797 return $email;
798 }
Barry Mienydd671972010-10-04 16:33:58 +0200799
Derek Allard2067d1a2008-11-13 22:59:24 +0000800 // --------------------------------------------------------------------
801
802 /**
803 * Set Multipart Value
804 *
Derek Allard2067d1a2008-11-13 22:59:24 +0000805 * @param string
Andrew Podner4296a652012-12-17 07:51:15 -0500806 * @return CI_Email
Derek Allard2067d1a2008-11-13 22:59:24 +0000807 */
Andrey Andreev505b3d62014-02-08 18:24:00 +0200808 public function set_alt_message($str)
Derek Allard2067d1a2008-11-13 22:59:24 +0000809 {
Dan Horrigand0ddeaf2011-08-21 09:07:27 -0400810 $this->alt_message = (string) $str;
Greg Akera769deb2010-11-10 15:47:29 -0600811 return $this;
Derek Allard2067d1a2008-11-13 22:59:24 +0000812 }
Barry Mienydd671972010-10-04 16:33:58 +0200813
Derek Allard2067d1a2008-11-13 22:59:24 +0000814 // --------------------------------------------------------------------
815
816 /**
817 * Set Mailtype
818 *
Derek Allard2067d1a2008-11-13 22:59:24 +0000819 * @param string
Andrew Podner4296a652012-12-17 07:51:15 -0500820 * @return CI_Email
Derek Allard2067d1a2008-11-13 22:59:24 +0000821 */
Phil Sturgeona0f980e2011-01-13 10:59:12 +0000822 public function set_mailtype($type = 'text')
Derek Allard2067d1a2008-11-13 22:59:24 +0000823 {
Andrey Andreev56454792012-05-17 14:32:19 +0300824 $this->mailtype = ($type === 'html') ? 'html' : 'text';
Greg Akera769deb2010-11-10 15:47:29 -0600825 return $this;
Derek Allard2067d1a2008-11-13 22:59:24 +0000826 }
Barry Mienydd671972010-10-04 16:33:58 +0200827
Derek Allard2067d1a2008-11-13 22:59:24 +0000828 // --------------------------------------------------------------------
nisheeth-barthwalcf225572013-02-18 01:08:04 +0530829
Derek Allard2067d1a2008-11-13 22:59:24 +0000830 /**
831 * Set Wordwrap
832 *
Dan Horrigan628e6602011-08-21 09:08:31 -0400833 * @param bool
Andrew Podner4296a652012-12-17 07:51:15 -0500834 * @return CI_Email
Derek Allard2067d1a2008-11-13 22:59:24 +0000835 */
Phil Sturgeona0f980e2011-01-13 10:59:12 +0000836 public function set_wordwrap($wordwrap = TRUE)
Derek Allard2067d1a2008-11-13 22:59:24 +0000837 {
Dan Horrigan628e6602011-08-21 09:08:31 -0400838 $this->wordwrap = (bool) $wordwrap;
Greg Akera769deb2010-11-10 15:47:29 -0600839 return $this;
Derek Allard2067d1a2008-11-13 22:59:24 +0000840 }
Barry Mienydd671972010-10-04 16:33:58 +0200841
Derek Allard2067d1a2008-11-13 22:59:24 +0000842 // --------------------------------------------------------------------
843
844 /**
845 * Set Protocol
846 *
Derek Allard2067d1a2008-11-13 22:59:24 +0000847 * @param string
Andrew Podner4296a652012-12-17 07:51:15 -0500848 * @return CI_Email
Derek Allard2067d1a2008-11-13 22:59:24 +0000849 */
Phil Sturgeona0f980e2011-01-13 10:59:12 +0000850 public function set_protocol($protocol = 'mail')
Derek Allard2067d1a2008-11-13 22:59:24 +0000851 {
Andrey Andreev59c5b532012-03-01 14:09:51 +0200852 $this->protocol = in_array($protocol, $this->_protocols, TRUE) ? strtolower($protocol) : 'mail';
Greg Akera769deb2010-11-10 15:47:29 -0600853 return $this;
Derek Allard2067d1a2008-11-13 22:59:24 +0000854 }
Barry Mienydd671972010-10-04 16:33:58 +0200855
Derek Allard2067d1a2008-11-13 22:59:24 +0000856 // --------------------------------------------------------------------
857
858 /**
859 * Set Priority
860 *
Andrey Andreev081c9462012-03-01 12:58:11 +0200861 * @param int
Andrew Podner4296a652012-12-17 07:51:15 -0500862 * @return CI_Email
Derek Allard2067d1a2008-11-13 22:59:24 +0000863 */
Phil Sturgeona0f980e2011-01-13 10:59:12 +0000864 public function set_priority($n = 3)
Derek Allard2067d1a2008-11-13 22:59:24 +0000865 {
Andrey Andreev59c5b532012-03-01 14:09:51 +0200866 $this->priority = preg_match('/^[1-5]$/', $n) ? (int) $n : 3;
Greg Akera769deb2010-11-10 15:47:29 -0600867 return $this;
Derek Allard2067d1a2008-11-13 22:59:24 +0000868 }
Barry Mienydd671972010-10-04 16:33:58 +0200869
Derek Allard2067d1a2008-11-13 22:59:24 +0000870 // --------------------------------------------------------------------
871
872 /**
873 * Set Newline Character
874 *
Derek Allard2067d1a2008-11-13 22:59:24 +0000875 * @param string
Andrew Podner4296a652012-12-17 07:51:15 -0500876 * @return CI_Email
Derek Allard2067d1a2008-11-13 22:59:24 +0000877 */
Phil Sturgeona0f980e2011-01-13 10:59:12 +0000878 public function set_newline($newline = "\n")
Derek Allard2067d1a2008-11-13 22:59:24 +0000879 {
Andrey Andreevb71e06b2011-12-22 19:22:50 +0200880 $this->newline = in_array($newline, array("\n", "\r\n", "\r")) ? $newline : "\n";
Greg Akera769deb2010-11-10 15:47:29 -0600881 return $this;
Derek Allard2067d1a2008-11-13 22:59:24 +0000882 }
Barry Mienydd671972010-10-04 16:33:58 +0200883
Derek Allard2067d1a2008-11-13 22:59:24 +0000884 // --------------------------------------------------------------------
885
886 /**
887 * Set CRLF
888 *
Derek Allard2067d1a2008-11-13 22:59:24 +0000889 * @param string
Andrew Podner4296a652012-12-17 07:51:15 -0500890 * @return CI_Email
Derek Allard2067d1a2008-11-13 22:59:24 +0000891 */
Phil Sturgeona0f980e2011-01-13 10:59:12 +0000892 public function set_crlf($crlf = "\n")
Derek Allard2067d1a2008-11-13 22:59:24 +0000893 {
Andrey Andreev76f15c92012-03-01 13:05:07 +0200894 $this->crlf = ($crlf !== "\n" && $crlf !== "\r\n" && $crlf !== "\r") ? "\n" : $crlf;
Greg Akera769deb2010-11-10 15:47:29 -0600895 return $this;
Derek Allard2067d1a2008-11-13 22:59:24 +0000896 }
Barry Mienydd671972010-10-04 16:33:58 +0200897
Derek Allard2067d1a2008-11-13 22:59:24 +0000898 // --------------------------------------------------------------------
899
900 /**
Derek Allard2067d1a2008-11-13 22:59:24 +0000901 * Get the Message ID
902 *
Derek Allard2067d1a2008-11-13 22:59:24 +0000903 * @return string
904 */
Phil Sturgeon6d2f13a2011-07-20 10:04:52 -0600905 protected function _get_message_id()
Derek Allard2067d1a2008-11-13 22:59:24 +0000906 {
Andrey Andreevb71e06b2011-12-22 19:22:50 +0200907 $from = str_replace(array('>', '<'), '', $this->_headers['Return-Path']);
Andrey Andreev56454792012-05-17 14:32:19 +0300908 return '<'.uniqid('').strstr($from, '@').'>';
Derek Allard2067d1a2008-11-13 22:59:24 +0000909 }
Barry Mienydd671972010-10-04 16:33:58 +0200910
Derek Allard2067d1a2008-11-13 22:59:24 +0000911 // --------------------------------------------------------------------
912
913 /**
914 * Get Mail Protocol
915 *
Andrey Andreev59c5b532012-03-01 14:09:51 +0200916 * @return mixed
Derek Allard2067d1a2008-11-13 22:59:24 +0000917 */
Andrey Andreevb9e45e32017-04-06 14:36:19 +0300918 protected function _get_protocol()
Derek Allard2067d1a2008-11-13 22:59:24 +0000919 {
920 $this->protocol = strtolower($this->protocol);
Andrey Andreev59c5b532012-03-01 14:09:51 +0200921 in_array($this->protocol, $this->_protocols, TRUE) OR $this->protocol = 'mail';
Andrey Andreevb9e45e32017-04-06 14:36:19 +0300922 return $this->protocol;
Derek Allard2067d1a2008-11-13 22:59:24 +0000923 }
Barry Mienydd671972010-10-04 16:33:58 +0200924
Derek Allard2067d1a2008-11-13 22:59:24 +0000925 // --------------------------------------------------------------------
926
927 /**
928 * Get Mail Encoding
929 *
Derek Allard2067d1a2008-11-13 22:59:24 +0000930 * @return string
931 */
Andrey Andreevb9e45e32017-04-06 14:36:19 +0300932 protected function _get_encoding()
Derek Allard2067d1a2008-11-13 22:59:24 +0000933 {
Andrey Andreev59c5b532012-03-01 14:09:51 +0200934 in_array($this->_encoding, $this->_bit_depths) OR $this->_encoding = '8bit';
Derek Allard2067d1a2008-11-13 22:59:24 +0000935
936 foreach ($this->_base_charsets as $charset)
937 {
Andrey Andreevbdcafdf2017-03-28 17:31:09 +0300938 if (strpos($this->charset, $charset) === 0)
Derek Allard2067d1a2008-11-13 22:59:24 +0000939 {
940 $this->_encoding = '7bit';
941 }
942 }
943
Andrey Andreevb9e45e32017-04-06 14:36:19 +0300944 return $this->_encoding;
Derek Allard2067d1a2008-11-13 22:59:24 +0000945 }
946
947 // --------------------------------------------------------------------
948
949 /**
950 * Get content type (text/html/attachment)
951 *
Derek Allard2067d1a2008-11-13 22:59:24 +0000952 * @return string
953 */
Phil Sturgeon6d2f13a2011-07-20 10:04:52 -0600954 protected function _get_content_type()
Derek Allard2067d1a2008-11-13 22:59:24 +0000955 {
Andrey Andreev56454792012-05-17 14:32:19 +0300956 if ($this->mailtype === 'html')
Derek Allard2067d1a2008-11-13 22:59:24 +0000957 {
Andrey Andreev3c422792016-06-06 09:44:50 +0300958 return empty($this->_attachments) ? 'html' : 'html-attach';
Derek Allard2067d1a2008-11-13 22:59:24 +0000959 }
Andrey Andreev3c422792016-06-06 09:44:50 +0300960 elseif ($this->mailtype === 'text' && ! empty($this->_attachments))
Derek Allard2067d1a2008-11-13 22:59:24 +0000961 {
962 return 'plain-attach';
963 }
Andrey Andreevfbe4d792017-12-27 19:49:03 +0200964
965 return 'plain';
Derek Allard2067d1a2008-11-13 22:59:24 +0000966 }
Barry Mienydd671972010-10-04 16:33:58 +0200967
Derek Allard2067d1a2008-11-13 22:59:24 +0000968 // --------------------------------------------------------------------
969
970 /**
971 * Set RFC 822 Date
972 *
Derek Allard2067d1a2008-11-13 22:59:24 +0000973 * @return string
974 */
Phil Sturgeon6d2f13a2011-07-20 10:04:52 -0600975 protected function _set_date()
Derek Allard2067d1a2008-11-13 22:59:24 +0000976 {
Andrey Andreev59c5b532012-03-01 14:09:51 +0200977 $timezone = date('Z');
Andrey Andreev3dad2e72012-06-14 15:10:56 +0300978 $operator = ($timezone[0] === '-') ? '-' : '+';
Derek Allard2067d1a2008-11-13 22:59:24 +0000979 $timezone = abs($timezone);
Andrey Andreev1bd3d882011-12-22 15:38:20 +0200980 $timezone = floor($timezone/3600) * 100 + ($timezone % 3600) / 60;
Derek Allard2067d1a2008-11-13 22:59:24 +0000981
Andrey Andreev59c5b532012-03-01 14:09:51 +0200982 return sprintf('%s %s%04d', date('D, j M Y H:i:s'), $operator, $timezone);
Derek Allard2067d1a2008-11-13 22:59:24 +0000983 }
Barry Mienydd671972010-10-04 16:33:58 +0200984
Derek Allard2067d1a2008-11-13 22:59:24 +0000985 // --------------------------------------------------------------------
986
987 /**
988 * Mime message
989 *
Derek Allard2067d1a2008-11-13 22:59:24 +0000990 * @return string
991 */
Phil Sturgeon6d2f13a2011-07-20 10:04:52 -0600992 protected function _get_mime_message()
Derek Allard2067d1a2008-11-13 22:59:24 +0000993 {
Andrey Andreev59c5b532012-03-01 14:09:51 +0200994 return 'This is a multi-part message in MIME format.'.$this->newline.'Your email application may not support this format.';
Derek Allard2067d1a2008-11-13 22:59:24 +0000995 }
Barry Mienydd671972010-10-04 16:33:58 +0200996
Derek Allard2067d1a2008-11-13 22:59:24 +0000997 // --------------------------------------------------------------------
998
999 /**
1000 * Validate Email Address
1001 *
Derek Allard2067d1a2008-11-13 22:59:24 +00001002 * @param string
1003 * @return bool
1004 */
Phil Sturgeona0f980e2011-01-13 10:59:12 +00001005 public function validate_email($email)
Derek Allard2067d1a2008-11-13 22:59:24 +00001006 {
1007 if ( ! is_array($email))
1008 {
patworkb0707982011-04-08 15:10:05 +02001009 $this->_set_error_message('lang:email_must_be_array');
Derek Allard2067d1a2008-11-13 22:59:24 +00001010 return FALSE;
1011 }
1012
1013 foreach ($email as $val)
1014 {
1015 if ( ! $this->valid_email($val))
1016 {
patworkb0707982011-04-08 15:10:05 +02001017 $this->_set_error_message('lang:email_invalid_address', $val);
Derek Allard2067d1a2008-11-13 22:59:24 +00001018 return FALSE;
1019 }
1020 }
1021
1022 return TRUE;
1023 }
Barry Mienydd671972010-10-04 16:33:58 +02001024
Derek Allard2067d1a2008-11-13 22:59:24 +00001025 // --------------------------------------------------------------------
1026
1027 /**
1028 * Email Validation
1029 *
Derek Allard2067d1a2008-11-13 22:59:24 +00001030 * @param string
1031 * @return bool
1032 */
Timothy Warrend37f0e92012-06-27 08:21:59 -04001033 public function valid_email($email)
Derek Allard2067d1a2008-11-13 22:59:24 +00001034 {
Andrey Andreev4fd2d492017-11-09 20:38:33 +02001035 if (function_exists('idn_to_ascii') && strpos($email, '@'))
Andrey Andreev95496662014-06-01 00:00:13 +03001036 {
Andrey Andreev4fd2d492017-11-09 20:38:33 +02001037 list($account, $domain) = explode('@', $email, 2);
Andrey Andreev221c0952018-01-22 10:29:19 +02001038 $domain = defined('INTL_IDNA_VARIANT_UTS46')
Andrey Andreev4fd2d492017-11-09 20:38:33 +02001039 ? idn_to_ascii($domain, 0, INTL_IDNA_VARIANT_UTS46)
1040 : idn_to_ascii($domain);
1041 $email = $account.'@'.$domain;
Andrey Andreev95496662014-06-01 00:00:13 +03001042 }
1043
Andrey Andreev580388b2012-06-27 15:43:46 +03001044 return (bool) filter_var($email, FILTER_VALIDATE_EMAIL);
Derek Allard2067d1a2008-11-13 22:59:24 +00001045 }
Barry Mienydd671972010-10-04 16:33:58 +02001046
Derek Allard2067d1a2008-11-13 22:59:24 +00001047 // --------------------------------------------------------------------
1048
1049 /**
1050 * Clean Extended Email Address: Joe Smith <joe@smith.com>
1051 *
Derek Allard2067d1a2008-11-13 22:59:24 +00001052 * @param string
1053 * @return string
1054 */
Phil Sturgeona0f980e2011-01-13 10:59:12 +00001055 public function clean_email($email)
Derek Allard2067d1a2008-11-13 22:59:24 +00001056 {
1057 if ( ! is_array($email))
1058 {
Andrey Andreev56454792012-05-17 14:32:19 +03001059 return preg_match('/\<(.*)\>/', $email, $match) ? $match[1] : $email;
Derek Allard2067d1a2008-11-13 22:59:24 +00001060 }
1061
1062 $clean_email = array();
1063
1064 foreach ($email as $addy)
1065 {
Andrey Andreev59c5b532012-03-01 14:09:51 +02001066 $clean_email[] = preg_match('/\<(.*)\>/', $addy, $match) ? $match[1] : $addy;
Derek Allard2067d1a2008-11-13 22:59:24 +00001067 }
1068
1069 return $clean_email;
1070 }
Barry Mienydd671972010-10-04 16:33:58 +02001071
Derek Allard2067d1a2008-11-13 22:59:24 +00001072 // --------------------------------------------------------------------
1073
1074 /**
1075 * Build alternative plain text message
1076 *
Andrey Andreev8df1ae22012-10-19 11:20:54 +03001077 * Provides the raw message for use in plain-text headers of
1078 * HTML-formatted emails.
Derek Allard2067d1a2008-11-13 22:59:24 +00001079 * If the user hasn't specified his own alternative message
1080 * it creates one by stripping the HTML
1081 *
Derek Allard2067d1a2008-11-13 22:59:24 +00001082 * @return string
1083 */
Phil Sturgeon6d2f13a2011-07-20 10:04:52 -06001084 protected function _get_alt_message()
Derek Allard2067d1a2008-11-13 22:59:24 +00001085 {
Andrey Andreev8df1ae22012-10-19 11:20:54 +03001086 if ( ! empty($this->alt_message))
Derek Allard2067d1a2008-11-13 22:59:24 +00001087 {
Andrey Andreev8df1ae22012-10-19 11:20:54 +03001088 return ($this->wordwrap)
1089 ? $this->word_wrap($this->alt_message, 76)
1090 : $this->alt_message;
Derek Allard2067d1a2008-11-13 22:59:24 +00001091 }
1092
Andrey Andreev56454792012-05-17 14:32:19 +03001093 $body = preg_match('/\<body.*?\>(.*)\<\/body\>/si', $this->_body, $match) ? $match[1] : $this->_body;
Andrey Andreev1bd3d882011-12-22 15:38:20 +02001094 $body = str_replace("\t", '', preg_replace('#<!--(.*)--\>#', '', trim(strip_tags($body))));
Derek Allard2067d1a2008-11-13 22:59:24 +00001095
1096 for ($i = 20; $i >= 3; $i--)
1097 {
Andrey Andreev1bd3d882011-12-22 15:38:20 +02001098 $body = str_replace(str_repeat("\n", $i), "\n\n", $body);
Derek Allard2067d1a2008-11-13 22:59:24 +00001099 }
1100
GDmac9bea4be2012-10-30 06:14:19 +01001101 // Reduce multiple spaces
Andrey Andreev8a203f62012-11-02 20:40:43 +02001102 $body = preg_replace('| +|', ' ', $body);
GDmac9bea4be2012-10-30 06:14:19 +01001103
Andrey Andreev8df1ae22012-10-19 11:20:54 +03001104 return ($this->wordwrap)
1105 ? $this->word_wrap($body, 76)
1106 : $body;
Derek Allard2067d1a2008-11-13 22:59:24 +00001107 }
Barry Mienydd671972010-10-04 16:33:58 +02001108
Derek Allard2067d1a2008-11-13 22:59:24 +00001109 // --------------------------------------------------------------------
1110
1111 /**
1112 * Word Wrap
1113 *
Derek Allard2067d1a2008-11-13 22:59:24 +00001114 * @param string
Andrey Andreev8df1ae22012-10-19 11:20:54 +03001115 * @param int line-length limit
Derek Allard2067d1a2008-11-13 22:59:24 +00001116 * @return string
1117 */
Andrey Andreev00ea2a92012-10-18 14:59:29 +03001118 public function word_wrap($str, $charlim = NULL)
Derek Allard2067d1a2008-11-13 22:59:24 +00001119 {
Andrey Andreev00ea2a92012-10-18 14:59:29 +03001120 // Set the character limit, if not already present
1121 if (empty($charlim))
Derek Allard2067d1a2008-11-13 22:59:24 +00001122 {
Andrey Andreev00ea2a92012-10-18 14:59:29 +03001123 $charlim = empty($this->wrapchars) ? 76 : $this->wrapchars;
Derek Allard2067d1a2008-11-13 22:59:24 +00001124 }
1125
Derek Allard2067d1a2008-11-13 22:59:24 +00001126 // Standardize newlines
1127 if (strpos($str, "\r") !== FALSE)
1128 {
Andrey Andreevb71e06b2011-12-22 19:22:50 +02001129 $str = str_replace(array("\r\n", "\r"), "\n", $str);
Derek Allard2067d1a2008-11-13 22:59:24 +00001130 }
1131
GDmac9bea4be2012-10-30 06:14:19 +01001132 // Reduce multiple spaces at end of line
1133 $str = preg_replace('| +\n|', "\n", $str);
1134
Derek Allard2067d1a2008-11-13 22:59:24 +00001135 // If the current word is surrounded by {unwrap} tags we'll
1136 // strip the entire chunk and replace it with a marker.
1137 $unwrap = array();
vlakoff0f7eba22014-05-20 10:32:44 +02001138 if (preg_match_all('|\{unwrap\}(.+?)\{/unwrap\}|s', $str, $matches))
Derek Allard2067d1a2008-11-13 22:59:24 +00001139 {
Andrey Andreev1bd3d882011-12-22 15:38:20 +02001140 for ($i = 0, $c = count($matches[0]); $i < $c; $i++)
Derek Allard2067d1a2008-11-13 22:59:24 +00001141 {
Andrey Andreev1bd3d882011-12-22 15:38:20 +02001142 $unwrap[] = $matches[1][$i];
vlakoff0f7eba22014-05-20 10:32:44 +02001143 $str = str_replace($matches[0][$i], '{{unwrapped'.$i.'}}', $str);
Derek Allard2067d1a2008-11-13 22:59:24 +00001144 }
1145 }
1146
vlakofff0ab8132014-05-20 10:32:53 +02001147 // Use PHP's native function to do the initial wordwrap.
Derek Allard2067d1a2008-11-13 22:59:24 +00001148 // We set the cut flag to FALSE so that any individual words that are
Andrey Andreev56454792012-05-17 14:32:19 +03001149 // too long get left alone. In the next step we'll deal with them.
Derek Allard2067d1a2008-11-13 22:59:24 +00001150 $str = wordwrap($str, $charlim, "\n", FALSE);
1151
1152 // Split the string into individual lines of text and cycle through them
Andrey Andreev56454792012-05-17 14:32:19 +03001153 $output = '';
Derek Allard2067d1a2008-11-13 22:59:24 +00001154 foreach (explode("\n", $str) as $line)
1155 {
1156 // Is the line within the allowed character count?
1157 // If so we'll join it to the output and continue
Andrey Andreev4e2cdec2016-10-28 14:19:08 +03001158 if (self::strlen($line) <= $charlim)
Derek Allard2067d1a2008-11-13 22:59:24 +00001159 {
1160 $output .= $line.$this->newline;
1161 continue;
1162 }
1163
1164 $temp = '';
Andrey Andreev1bd3d882011-12-22 15:38:20 +02001165 do
Derek Allard2067d1a2008-11-13 22:59:24 +00001166 {
1167 // If the over-length word is a URL we won't wrap it
vlakoff2a8560c2014-05-20 10:32:35 +02001168 if (preg_match('!\[url.+\]|://|www\.!', $line))
Derek Allard2067d1a2008-11-13 22:59:24 +00001169 {
1170 break;
1171 }
1172
1173 // Trim the word down
Andrey Andreev4e2cdec2016-10-28 14:19:08 +03001174 $temp .= self::substr($line, 0, $charlim - 1);
1175 $line = self::substr($line, $charlim - 1);
Derek Allard2067d1a2008-11-13 22:59:24 +00001176 }
Andrey Andreev4e2cdec2016-10-28 14:19:08 +03001177 while (self::strlen($line) > $charlim);
Derek Allard2067d1a2008-11-13 22:59:24 +00001178
1179 // If $temp contains data it means we had to split up an over-length
1180 // word into smaller chunks so we'll add it back to our current line
Alex Bilbied261b1e2012-06-02 11:12:16 +01001181 if ($temp !== '')
Derek Allard2067d1a2008-11-13 22:59:24 +00001182 {
Andrey Andreev1bd3d882011-12-22 15:38:20 +02001183 $output .= $temp.$this->newline;
Derek Allard2067d1a2008-11-13 22:59:24 +00001184 }
1185
Andrey Andreev1bd3d882011-12-22 15:38:20 +02001186 $output .= $line.$this->newline;
Derek Allard2067d1a2008-11-13 22:59:24 +00001187 }
1188
1189 // Put our markers back
1190 if (count($unwrap) > 0)
1191 {
1192 foreach ($unwrap as $key => $val)
1193 {
Andrey Andreev56454792012-05-17 14:32:19 +03001194 $output = str_replace('{{unwrapped'.$key.'}}', $val, $output);
Derek Allard2067d1a2008-11-13 22:59:24 +00001195 }
1196 }
1197
1198 return $output;
1199 }
Barry Mienydd671972010-10-04 16:33:58 +02001200
Derek Allard2067d1a2008-11-13 22:59:24 +00001201 // --------------------------------------------------------------------
1202
1203 /**
1204 * Build final headers
1205 *
Andrey Andreev19277092016-08-22 11:34:56 +03001206 * @return void
Derek Allard2067d1a2008-11-13 22:59:24 +00001207 */
Phil Sturgeon6d2f13a2011-07-20 10:04:52 -06001208 protected function _build_headers()
Derek Allard2067d1a2008-11-13 22:59:24 +00001209 {
Andrey Andreev6a5b5e72017-01-06 13:45:55 +02001210 $this->set_header('User-Agent', $this->useragent);
Mickey Wubfc1cad2012-05-31 22:28:40 -07001211 $this->set_header('X-Sender', $this->clean_email($this->_headers['From']));
1212 $this->set_header('X-Mailer', $this->useragent);
Andrey Andreevdea61772014-02-24 16:36:25 +02001213 $this->set_header('X-Priority', $this->_priorities[$this->priority]);
Mickey Wubfc1cad2012-05-31 22:28:40 -07001214 $this->set_header('Message-ID', $this->_get_message_id());
1215 $this->set_header('Mime-Version', '1.0');
Derek Allard2067d1a2008-11-13 22:59:24 +00001216 }
Barry Mienydd671972010-10-04 16:33:58 +02001217
Derek Allard2067d1a2008-11-13 22:59:24 +00001218 // --------------------------------------------------------------------
1219
1220 /**
1221 * Write Headers as a string
1222 *
Derek Allard2067d1a2008-11-13 22:59:24 +00001223 * @return void
1224 */
Phil Sturgeon6d2f13a2011-07-20 10:04:52 -06001225 protected function _write_headers()
Derek Allard2067d1a2008-11-13 22:59:24 +00001226 {
Andrey Andreev1bd3d882011-12-22 15:38:20 +02001227 if ($this->protocol === 'mail')
Derek Allard2067d1a2008-11-13 22:59:24 +00001228 {
nisheeth-barthwal6bb98902013-02-19 18:11:14 +05301229 if (isset($this->_headers['Subject']))
1230 {
1231 $this->_subject = $this->_headers['Subject'];
1232 unset($this->_headers['Subject']);
1233 }
Derek Allard2067d1a2008-11-13 22:59:24 +00001234 }
1235
1236 reset($this->_headers);
Andrey Andreev56454792012-05-17 14:32:19 +03001237 $this->_header_str = '';
Derek Allard2067d1a2008-11-13 22:59:24 +00001238
Pascal Kriete14287f32011-02-14 13:39:34 -05001239 foreach ($this->_headers as $key => $val)
Derek Allard2067d1a2008-11-13 22:59:24 +00001240 {
1241 $val = trim($val);
1242
Alex Bilbied261b1e2012-06-02 11:12:16 +01001243 if ($val !== '')
Derek Allard2067d1a2008-11-13 22:59:24 +00001244 {
Andrey Andreev56454792012-05-17 14:32:19 +03001245 $this->_header_str .= $key.': '.$val.$this->newline;
Derek Allard2067d1a2008-11-13 22:59:24 +00001246 }
1247 }
1248
Andrey Andreev1bd3d882011-12-22 15:38:20 +02001249 if ($this->_get_protocol() === 'mail')
Derek Allard2067d1a2008-11-13 22:59:24 +00001250 {
Derek Jones1d890882009-02-10 20:32:14 +00001251 $this->_header_str = rtrim($this->_header_str);
Derek Allard2067d1a2008-11-13 22:59:24 +00001252 }
1253 }
Barry Mienydd671972010-10-04 16:33:58 +02001254
Derek Allard2067d1a2008-11-13 22:59:24 +00001255 // --------------------------------------------------------------------
1256
1257 /**
1258 * Build Final Body and attachments
1259 *
buhayc6da1ef2013-04-18 09:06:49 -07001260 * @return bool
Derek Allard2067d1a2008-11-13 22:59:24 +00001261 */
Phil Sturgeon6d2f13a2011-07-20 10:04:52 -06001262 protected function _build_message()
Derek Allard2067d1a2008-11-13 22:59:24 +00001263 {
Andrey Andreev76f15c92012-03-01 13:05:07 +02001264 if ($this->wordwrap === TRUE && $this->mailtype !== 'html')
Derek Allard2067d1a2008-11-13 22:59:24 +00001265 {
1266 $this->_body = $this->word_wrap($this->_body);
1267 }
1268
Derek Allard2067d1a2008-11-13 22:59:24 +00001269 $this->_write_headers();
1270
Andrey Andreev1bd3d882011-12-22 15:38:20 +02001271 $hdr = ($this->_get_protocol() === 'mail') ? $this->newline : '';
Brandon Jones485d7412010-11-09 16:38:17 -05001272 $body = '';
Derek Allard2067d1a2008-11-13 22:59:24 +00001273
1274 switch ($this->_get_content_type())
1275 {
Andrey Andreev3c422792016-06-06 09:44:50 +03001276 case 'plain':
Derek Allard2067d1a2008-11-13 22:59:24 +00001277
Andrey Andreev56454792012-05-17 14:32:19 +03001278 $hdr .= 'Content-Type: text/plain; charset='.$this->charset.$this->newline
1279 .'Content-Transfer-Encoding: '.$this->_get_encoding();
Derek Allard2067d1a2008-11-13 22:59:24 +00001280
Andrey Andreev1bd3d882011-12-22 15:38:20 +02001281 if ($this->_get_protocol() === 'mail')
Derek Allard2067d1a2008-11-13 22:59:24 +00001282 {
1283 $this->_header_str .= $hdr;
1284 $this->_finalbody = $this->_body;
Derek Allard2067d1a2008-11-13 22:59:24 +00001285 }
Brandon Jones485d7412010-11-09 16:38:17 -05001286 else
1287 {
Andrey Andreev2b956af2013-08-07 14:32:56 +03001288 $this->_finalbody = $hdr.$this->newline.$this->newline.$this->_body;
Brandon Jones485d7412010-11-09 16:38:17 -05001289 }
Eric Barnes6113f542010-12-29 13:36:12 -05001290
Derek Allard2067d1a2008-11-13 22:59:24 +00001291 return;
1292
Andrey Andreev3c422792016-06-06 09:44:50 +03001293 case 'html':
Derek Allard2067d1a2008-11-13 22:59:24 +00001294
1295 if ($this->send_multipart === FALSE)
1296 {
Andrey Andreev56454792012-05-17 14:32:19 +03001297 $hdr .= 'Content-Type: text/html; charset='.$this->charset.$this->newline
Andrey Andreev2b956af2013-08-07 14:32:56 +03001298 .'Content-Transfer-Encoding: quoted-printable';
Derek Allard2067d1a2008-11-13 22:59:24 +00001299 }
1300 else
1301 {
Andrey Andreev3c422792016-06-06 09:44:50 +03001302 $boundary = uniqid('B_ALT_');
1303 $hdr .= 'Content-Type: multipart/alternative; boundary="'.$boundary.'"';
Derek Allard2067d1a2008-11-13 22:59:24 +00001304
Andrey Andreev56454792012-05-17 14:32:19 +03001305 $body .= $this->_get_mime_message().$this->newline.$this->newline
Andrey Andreev3c422792016-06-06 09:44:50 +03001306 .'--'.$boundary.$this->newline
Derek Allard2067d1a2008-11-13 22:59:24 +00001307
Andrey Andreev56454792012-05-17 14:32:19 +03001308 .'Content-Type: text/plain; charset='.$this->charset.$this->newline
1309 .'Content-Transfer-Encoding: '.$this->_get_encoding().$this->newline.$this->newline
Andrey Andreev3c422792016-06-06 09:44:50 +03001310 .$this->_get_alt_message().$this->newline.$this->newline
1311 .'--'.$boundary.$this->newline
Brandon Jones485d7412010-11-09 16:38:17 -05001312
Andrey Andreev56454792012-05-17 14:32:19 +03001313 .'Content-Type: text/html; charset='.$this->charset.$this->newline
1314 .'Content-Transfer-Encoding: quoted-printable'.$this->newline.$this->newline;
Derek Allard2067d1a2008-11-13 22:59:24 +00001315 }
Eric Barnes6113f542010-12-29 13:36:12 -05001316
Andrey Andreev56454792012-05-17 14:32:19 +03001317 $this->_finalbody = $body.$this->_prep_quoted_printable($this->_body).$this->newline.$this->newline;
Eric Barnes6113f542010-12-29 13:36:12 -05001318
Andrey Andreev1bd3d882011-12-22 15:38:20 +02001319 if ($this->_get_protocol() === 'mail')
Derek Allard2067d1a2008-11-13 22:59:24 +00001320 {
1321 $this->_header_str .= $hdr;
Brandon Jones485d7412010-11-09 16:38:17 -05001322 }
1323 else
1324 {
Andrey Andreev2b956af2013-08-07 14:32:56 +03001325 $this->_finalbody = $hdr.$this->newline.$this->newline.$this->_finalbody;
Derek Allard2067d1a2008-11-13 22:59:24 +00001326 }
1327
Derek Allard2067d1a2008-11-13 22:59:24 +00001328 if ($this->send_multipart !== FALSE)
1329 {
Andrey Andreev3c422792016-06-06 09:44:50 +03001330 $this->_finalbody .= '--'.$boundary.'--';
Derek Allard2067d1a2008-11-13 22:59:24 +00001331 }
1332
Derek Allard2067d1a2008-11-13 22:59:24 +00001333 return;
1334
Andrey Andreev3c422792016-06-06 09:44:50 +03001335 case 'plain-attach':
Derek Allard2067d1a2008-11-13 22:59:24 +00001336
Andrey Andreev3c422792016-06-06 09:44:50 +03001337 $boundary = uniqid('B_ATC_');
1338 $hdr .= 'Content-Type: multipart/mixed; boundary="'.$boundary.'"';
Derek Allard2067d1a2008-11-13 22:59:24 +00001339
Andrey Andreev1bd3d882011-12-22 15:38:20 +02001340 if ($this->_get_protocol() === 'mail')
Derek Allard2067d1a2008-11-13 22:59:24 +00001341 {
1342 $this->_header_str .= $hdr;
Eric Barnes6113f542010-12-29 13:36:12 -05001343 }
1344
Andrey Andreev2b956af2013-08-07 14:32:56 +03001345 $body .= $this->_get_mime_message().$this->newline
1346 .$this->newline
Andrey Andreev3c422792016-06-06 09:44:50 +03001347 .'--'.$boundary.$this->newline
Andrey Andreev56454792012-05-17 14:32:19 +03001348 .'Content-Type: text/plain; charset='.$this->charset.$this->newline
Andrey Andreev2b956af2013-08-07 14:32:56 +03001349 .'Content-Transfer-Encoding: '.$this->_get_encoding().$this->newline
1350 .$this->newline
Andrey Andreev56454792012-05-17 14:32:19 +03001351 .$this->_body.$this->newline.$this->newline;
Derek Allard2067d1a2008-11-13 22:59:24 +00001352
Andrey Andreev3c422792016-06-06 09:44:50 +03001353 $this->_append_attachments($body, $boundary);
Derek Allard2067d1a2008-11-13 22:59:24 +00001354
Andrey Andreev3c422792016-06-06 09:44:50 +03001355 break;
1356 case 'html-attach':
1357
1358 $alt_boundary = uniqid('B_ALT_');
1359 $last_boundary = NULL;
1360
1361 if ($this->_attachments_have_multipart('mixed'))
1362 {
1363 $atc_boundary = uniqid('B_ATC_');
1364 $hdr .= 'Content-Type: multipart/mixed; boundary="'.$atc_boundary.'"';
1365 $last_boundary = $atc_boundary;
1366 }
1367
1368 if ($this->_attachments_have_multipart('related'))
1369 {
1370 $rel_boundary = uniqid('B_REL_');
1371 $rel_boundary_header = 'Content-Type: multipart/related; boundary="'.$rel_boundary.'"';
1372
1373 if (isset($last_boundary))
1374 {
1375 $body .= '--'.$last_boundary.$this->newline.$rel_boundary_header;
1376 }
1377 else
1378 {
1379 $hdr .= $rel_boundary_header;
1380 }
1381
1382 $last_boundary = $rel_boundary;
1383 }
Eric Barnes6113f542010-12-29 13:36:12 -05001384
Andrey Andreev1bd3d882011-12-22 15:38:20 +02001385 if ($this->_get_protocol() === 'mail')
Derek Allard2067d1a2008-11-13 22:59:24 +00001386 {
1387 $this->_header_str .= $hdr;
Derek Allard2067d1a2008-11-13 22:59:24 +00001388 }
1389
Andrey Andreev4e2cdec2016-10-28 14:19:08 +03001390 self::strlen($body) && $body .= $this->newline.$this->newline;
Andrey Andreev56454792012-05-17 14:32:19 +03001391 $body .= $this->_get_mime_message().$this->newline.$this->newline
Andrey Andreev3c422792016-06-06 09:44:50 +03001392 .'--'.$last_boundary.$this->newline
Brandon Jones485d7412010-11-09 16:38:17 -05001393
Andrey Andreev3c422792016-06-06 09:44:50 +03001394 .'Content-Type: multipart/alternative; boundary="'.$alt_boundary.'"'.$this->newline.$this->newline
1395 .'--'.$alt_boundary.$this->newline
Brandon Jones485d7412010-11-09 16:38:17 -05001396
Andrey Andreev56454792012-05-17 14:32:19 +03001397 .'Content-Type: text/plain; charset='.$this->charset.$this->newline
1398 .'Content-Transfer-Encoding: '.$this->_get_encoding().$this->newline.$this->newline
Andrey Andreev3c422792016-06-06 09:44:50 +03001399 .$this->_get_alt_message().$this->newline.$this->newline
1400 .'--'.$alt_boundary.$this->newline
Brandon Jones485d7412010-11-09 16:38:17 -05001401
Andrey Andreev56454792012-05-17 14:32:19 +03001402 .'Content-Type: text/html; charset='.$this->charset.$this->newline
1403 .'Content-Transfer-Encoding: quoted-printable'.$this->newline.$this->newline
Brandon Jones485d7412010-11-09 16:38:17 -05001404
Andrey Andreev56454792012-05-17 14:32:19 +03001405 .$this->_prep_quoted_printable($this->_body).$this->newline.$this->newline
Andrey Andreev3c422792016-06-06 09:44:50 +03001406 .'--'.$alt_boundary.'--'.$this->newline.$this->newline;
Derek Allard2067d1a2008-11-13 22:59:24 +00001407
Andrey Andreev3c422792016-06-06 09:44:50 +03001408 if ( ! empty($rel_boundary))
1409 {
1410 $body .= $this->newline.$this->newline;
1411 $this->_append_attachments($body, $rel_boundary, 'related');
1412 }
1413
1414 // multipart/mixed attachments
1415 if ( ! empty($atc_boundary))
1416 {
1417 $body .= $this->newline.$this->newline;
1418 $this->_append_attachments($body, $atc_boundary, 'mixed');
1419 }
1420
1421 break;
Derek Allard2067d1a2008-11-13 22:59:24 +00001422 }
1423
Andrey Andreev2b956af2013-08-07 14:32:56 +03001424 $this->_finalbody = ($this->_get_protocol() === 'mail')
1425 ? $body
1426 : $hdr.$this->newline.$this->newline.$body;
Andrey Andreevc8097262014-01-10 14:45:31 +02001427
buhay466e8082013-04-17 14:25:34 -07001428 return TRUE;
Derek Allard2067d1a2008-11-13 22:59:24 +00001429 }
Barry Mienydd671972010-10-04 16:33:58 +02001430
Derek Allard2067d1a2008-11-13 22:59:24 +00001431 // --------------------------------------------------------------------
1432
Andrey Andreev3c422792016-06-06 09:44:50 +03001433 protected function _attachments_have_multipart($type)
1434 {
1435 foreach ($this->_attachments as &$attachment)
1436 {
1437 if ($attachment['multipart'] === $type)
1438 {
1439 return TRUE;
1440 }
1441 }
1442
1443 return FALSE;
1444 }
1445
1446 // --------------------------------------------------------------------
1447
1448 /**
1449 * Prepares attachment string
1450 *
1451 * @param string $body Message body to append to
1452 * @param string $boundary Multipart boundary
1453 * @param string $multipart When provided, only attachments of this type will be processed
1454 * @return string
1455 */
1456 protected function _append_attachments(&$body, $boundary, $multipart = null)
1457 {
1458 for ($i = 0, $c = count($this->_attachments); $i < $c; $i++)
1459 {
1460 if (isset($multipart) && $this->_attachments[$i]['multipart'] !== $multipart)
1461 {
1462 continue;
1463 }
1464
1465 $name = isset($this->_attachments[$i]['name'][1])
1466 ? $this->_attachments[$i]['name'][1]
1467 : basename($this->_attachments[$i]['name'][0]);
1468
1469 $body .= '--'.$boundary.$this->newline
1470 .'Content-Type: '.$this->_attachments[$i]['type'].'; name="'.$name.'"'.$this->newline
1471 .'Content-Disposition: '.$this->_attachments[$i]['disposition'].';'.$this->newline
1472 .'Content-Transfer-Encoding: base64'.$this->newline
Andrey Andreev9b0f5fa2016-08-01 13:54:06 +03001473 .(empty($this->_attachments[$i]['cid']) ? '' : 'Content-ID: <'.$this->_attachments[$i]['cid'].'>'.$this->newline)
1474 .$this->newline
Andrey Andreev3c422792016-06-06 09:44:50 +03001475 .$this->_attachments[$i]['content'].$this->newline;
1476 }
1477
1478 // $name won't be set if no attachments were appended,
1479 // and therefore a boundary wouldn't be necessary
1480 empty($name) OR $body .= '--'.$boundary.'--';
1481 }
1482
1483 // --------------------------------------------------------------------
1484
Derek Allard2067d1a2008-11-13 22:59:24 +00001485 /**
1486 * Prep Quoted Printable
1487 *
1488 * Prepares string for Quoted-Printable Content-Transfer-Encoding
1489 * Refer to RFC 2045 http://www.ietf.org/rfc/rfc2045.txt
1490 *
Derek Allard2067d1a2008-11-13 22:59:24 +00001491 * @param string
Derek Allard2067d1a2008-11-13 22:59:24 +00001492 * @return string
1493 */
Andrey Andreev683b34d2012-10-09 15:00:00 +03001494 protected function _prep_quoted_printable($str)
Derek Allard2067d1a2008-11-13 22:59:24 +00001495 {
Andrey Andreevb06b5c42015-11-16 13:37:58 +02001496 // ASCII code numbers for "safe" characters that can always be
1497 // used literally, without encoding, as described in RFC 2049.
1498 // http://www.ietf.org/rfc/rfc2049.txt
1499 static $ascii_safe_chars = array(
1500 // ' ( ) + , - . / : = ?
1501 39, 40, 41, 43, 44, 45, 46, 47, 58, 61, 63,
1502 // numbers
1503 48, 49, 50, 51, 52, 53, 54, 55, 56, 57,
1504 // upper-case letters
1505 65, 66, 67, 68, 69, 70, 71, 72, 73, 74, 75, 76, 77, 78, 79, 80, 81, 82, 83, 84, 85, 86, 87, 88, 89, 90,
1506 // lower-case letters
1507 97, 98, 99, 100, 101, 102, 103, 104, 105, 106, 107, 108, 109, 110, 111, 112, 113, 114, 115, 116, 117, 118, 119, 120, 121, 122
1508 );
1509
Andrey Andreev00ea2a92012-10-18 14:59:29 +03001510 // We are intentionally wrapping so mail servers will encode characters
1511 // properly and MUAs will behave, so {unwrap} must go!
1512 $str = str_replace(array('{unwrap}', '{/unwrap}'), '', $str);
1513
Andrey Andreev683b34d2012-10-09 15:00:00 +03001514 // RFC 2045 specifies CRLF as "\r\n".
1515 // However, many developers choose to override that and violate
1516 // the RFC rules due to (apparently) a bug in MS Exchange,
1517 // which only works with "\n".
Andrey Andreev26f0cf92012-10-11 13:52:39 +03001518 if ($this->crlf === "\r\n")
Andrey Andreev683b34d2012-10-09 15:00:00 +03001519 {
Andrey Andreeva8382792016-07-28 16:40:12 +03001520 return quoted_printable_encode($str);
Andrey Andreev683b34d2012-10-09 15:00:00 +03001521 }
1522
Andrey Andreev1bd3d882011-12-22 15:38:20 +02001523 // Reduce multiple spaces & remove nulls
Andrey Andreev56454792012-05-17 14:32:19 +03001524 $str = preg_replace(array('| +|', '/\x00+/'), array(' ', ''), $str);
Derek Allard2067d1a2008-11-13 22:59:24 +00001525
1526 // Standardize newlines
1527 if (strpos($str, "\r") !== FALSE)
1528 {
1529 $str = str_replace(array("\r\n", "\r"), "\n", $str);
1530 }
1531
Derek Allard2067d1a2008-11-13 22:59:24 +00001532 $escape = '=';
1533 $output = '';
1534
Andrey Andreev1bd3d882011-12-22 15:38:20 +02001535 foreach (explode("\n", $str) as $line)
Derek Allard2067d1a2008-11-13 22:59:24 +00001536 {
Andrey Andreev4e2cdec2016-10-28 14:19:08 +03001537 $length = self::strlen($line);
Derek Allard2067d1a2008-11-13 22:59:24 +00001538 $temp = '';
1539
1540 // Loop through each character in the line to add soft-wrap
1541 // characters at the end of a line " =\r\n" and add the newly
1542 // processed line(s) to the output (see comment on $crlf class property)
1543 for ($i = 0; $i < $length; $i++)
1544 {
1545 // Grab the next character
Andrey Andreev1bd3d882011-12-22 15:38:20 +02001546 $char = $line[$i];
Derek Allard2067d1a2008-11-13 22:59:24 +00001547 $ascii = ord($char);
1548
1549 // Convert spaces and tabs but only if it's the end of the line
Andrey Andreevb06b5c42015-11-16 13:37:58 +02001550 if ($ascii === 32 OR $ascii === 9)
Derek Allard2067d1a2008-11-13 22:59:24 +00001551 {
Andrey Andreevb06b5c42015-11-16 13:37:58 +02001552 if ($i === ($length - 1))
1553 {
1554 $char = $escape.sprintf('%02s', dechex($ascii));
1555 }
Derek Allard2067d1a2008-11-13 22:59:24 +00001556 }
Andrey Andreevb06b5c42015-11-16 13:37:58 +02001557 // DO NOT move this below the $ascii_safe_chars line!
1558 //
1559 // = (equals) signs are allowed by RFC2049, but must be encoded
1560 // as they are the encoding delimiter!
1561 elseif ($ascii === 61)
Derek Allard2067d1a2008-11-13 22:59:24 +00001562 {
Derek Jones37f4b9c2011-07-01 17:56:50 -05001563 $char = $escape.strtoupper(sprintf('%02s', dechex($ascii))); // =3D
Derek Allard2067d1a2008-11-13 22:59:24 +00001564 }
Andrey Andreevb06b5c42015-11-16 13:37:58 +02001565 elseif ( ! in_array($ascii, $ascii_safe_chars, TRUE))
1566 {
1567 $char = $escape.strtoupper(sprintf('%02s', dechex($ascii)));
1568 }
Derek Allard2067d1a2008-11-13 22:59:24 +00001569
1570 // If we're at the character limit, add the line to the output,
1571 // reset our temp variable, and keep on chuggin'
Andrey Andreev4e2cdec2016-10-28 14:19:08 +03001572 if ((self::strlen($temp) + self::strlen($char)) >= 76)
Derek Allard2067d1a2008-11-13 22:59:24 +00001573 {
1574 $output .= $temp.$escape.$this->crlf;
1575 $temp = '';
1576 }
1577
1578 // Add the character to our temporary line
1579 $temp .= $char;
1580 }
1581
1582 // Add our completed line to the output
1583 $output .= $temp.$this->crlf;
1584 }
1585
1586 // get rid of extra CRLF tacked onto the end
Andrey Andreev4e2cdec2016-10-28 14:19:08 +03001587 return self::substr($output, 0, self::strlen($this->crlf) * -1);
Derek Allard2067d1a2008-11-13 22:59:24 +00001588 }
1589
1590 // --------------------------------------------------------------------
Barry Mienydd671972010-10-04 16:33:58 +02001591
Derek Allard2067d1a2008-11-13 22:59:24 +00001592 /**
1593 * Prep Q Encoding
1594 *
Andrey Andreev925dd902012-10-19 11:06:31 +03001595 * Performs "Q Encoding" on a string for use in email headers.
1596 * It's related but not identical to quoted-printable, so it has its
1597 * own method.
Derek Allard2067d1a2008-11-13 22:59:24 +00001598 *
Andrey Andreev081c9462012-03-01 12:58:11 +02001599 * @param string
Andrey Andreev081c9462012-03-01 12:58:11 +02001600 * @return string
Derek Allard2067d1a2008-11-13 22:59:24 +00001601 */
Andrey Andreev925dd902012-10-19 11:06:31 +03001602 protected function _prep_q_encoding($str)
Derek Allard2067d1a2008-11-13 22:59:24 +00001603 {
Andrey Andreev925dd902012-10-19 11:06:31 +03001604 $str = str_replace(array("\r", "\n"), '', $str);
Derek Allard2067d1a2008-11-13 22:59:24 +00001605
Andrey Andreev925dd902012-10-19 11:06:31 +03001606 if ($this->charset === 'UTF-8')
Derek Allard2067d1a2008-11-13 22:59:24 +00001607 {
Andrey Andreev3368ceb2015-10-30 12:25:15 +02001608 // Note: We used to have mb_encode_mimeheader() as the first choice
1609 // here, but it turned out to be buggy and unreliable. DO NOT
1610 // re-add it! -- Narf
1611 if (ICONV_ENABLED === TRUE)
Andrey Andreev925dd902012-10-19 11:06:31 +03001612 {
1613 $output = @iconv_mime_encode('', $str,
1614 array(
1615 'scheme' => 'Q',
1616 'line-length' => 76,
1617 'input-charset' => $this->charset,
1618 'output-charset' => $this->charset,
1619 'line-break-chars' => $this->crlf
1620 )
1621 );
1622
1623 // There are reports that iconv_mime_encode() might fail and return FALSE
1624 if ($output !== FALSE)
1625 {
1626 // iconv_mime_encode() will always put a header field name.
1627 // We've passed it an empty one, but it still prepends our
1628 // encoded string with ': ', so we need to strip it.
Andrey Andreev4e2cdec2016-10-28 14:19:08 +03001629 return self::substr($output, 2);
Andrey Andreev925dd902012-10-19 11:06:31 +03001630 }
1631
1632 $chars = iconv_strlen($str, 'UTF-8');
1633 }
Andrey Andreev3368ceb2015-10-30 12:25:15 +02001634 elseif (MB_ENABLED === TRUE)
1635 {
1636 $chars = mb_strlen($str, 'UTF-8');
1637 }
Derek Allard2067d1a2008-11-13 22:59:24 +00001638 }
1639
Andrey Andreev925dd902012-10-19 11:06:31 +03001640 // We might already have this set for UTF-8
Andrey Andreev4e2cdec2016-10-28 14:19:08 +03001641 isset($chars) OR $chars = self::strlen($str);
Derek Allard2067d1a2008-11-13 22:59:24 +00001642
Andrey Andreev925dd902012-10-19 11:06:31 +03001643 $output = '=?'.$this->charset.'?Q?';
Andrey Andreev4e2cdec2016-10-28 14:19:08 +03001644 for ($i = 0, $length = self::strlen($output); $i < $chars; $i++)
Derek Allard2067d1a2008-11-13 22:59:24 +00001645 {
Andrey Andreevbe1496d2014-02-11 22:48:45 +02001646 $chr = ($this->charset === 'UTF-8' && ICONV_ENABLED === TRUE)
Andrey Andreev925dd902012-10-19 11:06:31 +03001647 ? '='.implode('=', str_split(strtoupper(bin2hex(iconv_substr($str, $i, 1, $this->charset))), 2))
1648 : '='.strtoupper(bin2hex($str[$i]));
Derek Allard2067d1a2008-11-13 22:59:24 +00001649
Andrey Andreev925dd902012-10-19 11:06:31 +03001650 // RFC 2045 sets a limit of 76 characters per line.
1651 // We'll append ?= to the end of each line though.
Andrey Andreev4e2cdec2016-10-28 14:19:08 +03001652 if ($length + ($l = self::strlen($chr)) > 74)
Derek Allard2067d1a2008-11-13 22:59:24 +00001653 {
Andrey Andreev925dd902012-10-19 11:06:31 +03001654 $output .= '?='.$this->crlf // EOL
1655 .' =?'.$this->charset.'?Q?'.$chr; // New line
Andrey Andreev4e2cdec2016-10-28 14:19:08 +03001656 $length = 6 + self::strlen($this->charset) + $l; // Reset the length for the new line
Derek Allard2067d1a2008-11-13 22:59:24 +00001657 }
Andrey Andreev925dd902012-10-19 11:06:31 +03001658 else
Derek Allard2067d1a2008-11-13 22:59:24 +00001659 {
Andrey Andreev925dd902012-10-19 11:06:31 +03001660 $output .= $chr;
1661 $length += $l;
Derek Allard2067d1a2008-11-13 22:59:24 +00001662 }
Derek Allard2067d1a2008-11-13 22:59:24 +00001663 }
1664
Andrey Andreev925dd902012-10-19 11:06:31 +03001665 // End the header
1666 return $output.'?=';
Derek Allard2067d1a2008-11-13 22:59:24 +00001667 }
1668
1669 // --------------------------------------------------------------------
Barry Mienydd671972010-10-04 16:33:58 +02001670
Derek Allard2067d1a2008-11-13 22:59:24 +00001671 /**
1672 * Send Email
1673 *
Andrey Andreev5fd3ae82012-10-24 14:55:35 +03001674 * @param bool $auto_clear = TRUE
Derek Allard2067d1a2008-11-13 22:59:24 +00001675 * @return bool
1676 */
Alex Bilbied7bc8d02012-07-30 09:46:20 +01001677 public function send($auto_clear = TRUE)
Derek Allard2067d1a2008-11-13 22:59:24 +00001678 {
Michael Granados92e2fd22014-11-11 21:49:26 -02001679 if ( ! isset($this->_headers['From']))
Michael Granados0b85d6b2014-11-09 02:43:48 -02001680 {
Michael Granados92e2fd22014-11-11 21:49:26 -02001681 $this->_set_error_message('lang:email_no_from');
Michael Granados0b85d6b2014-11-09 02:43:48 -02001682 return FALSE;
1683 }
1684
Alex Bilbied261b1e2012-06-02 11:12:16 +01001685 if ($this->_replyto_flag === FALSE)
Derek Allard2067d1a2008-11-13 22:59:24 +00001686 {
1687 $this->reply_to($this->_headers['From']);
1688 }
1689
Andrey Andreev76f15c92012-03-01 13:05:07 +02001690 if ( ! isset($this->_recipients) && ! isset($this->_headers['To'])
1691 && ! isset($this->_bcc_array) && ! isset($this->_headers['Bcc'])
1692 && ! isset($this->_headers['Cc']))
Derek Allard2067d1a2008-11-13 22:59:24 +00001693 {
patworkb0707982011-04-08 15:10:05 +02001694 $this->_set_error_message('lang:email_no_recipients');
Derek Allard2067d1a2008-11-13 22:59:24 +00001695 return FALSE;
1696 }
1697
1698 $this->_build_headers();
1699
Andrey Andreev76f15c92012-03-01 13:05:07 +02001700 if ($this->bcc_batch_mode && count($this->_bcc_array) > $this->bcc_batch_size)
Derek Allard2067d1a2008-11-13 22:59:24 +00001701 {
Alex Bilbieb901e732012-07-30 09:44:57 +01001702 $result = $this->batch_bcc_send();
Alex Bilbied7bc8d02012-07-30 09:46:20 +01001703
Alex Bilbiea87aab32012-07-30 09:50:37 +01001704 if ($result && $auto_clear)
Alex Bilbied7bc8d02012-07-30 09:46:20 +01001705 {
1706 $this->clear();
1707 }
1708
Alex Bilbieb901e732012-07-30 09:44:57 +01001709 return $result;
Derek Allard2067d1a2008-11-13 22:59:24 +00001710 }
1711
buhay466e8082013-04-17 14:25:34 -07001712 if ($this->_build_message() === FALSE)
1713 {
1714 return FALSE;
1715 }
buhayc6da1ef2013-04-18 09:06:49 -07001716
Alex Bilbieb901e732012-07-30 09:44:57 +01001717 $result = $this->_spool_email();
Andrey Andreevbdb99992012-07-30 17:38:05 +03001718
Alex Bilbiea87aab32012-07-30 09:50:37 +01001719 if ($result && $auto_clear)
Alex Bilbied7bc8d02012-07-30 09:46:20 +01001720 {
1721 $this->clear();
1722 }
Alex Bilbiea87aab32012-07-30 09:50:37 +01001723
Alex Bilbieb901e732012-07-30 09:44:57 +01001724 return $result;
Derek Allard2067d1a2008-11-13 22:59:24 +00001725 }
Barry Mienydd671972010-10-04 16:33:58 +02001726
Derek Allard2067d1a2008-11-13 22:59:24 +00001727 // --------------------------------------------------------------------
1728
1729 /**
Andrey Andreev081c9462012-03-01 12:58:11 +02001730 * Batch Bcc Send. Sends groups of BCCs in batches
Derek Allard2067d1a2008-11-13 22:59:24 +00001731 *
Andrey Andreev081c9462012-03-01 12:58:11 +02001732 * @return void
Derek Allard2067d1a2008-11-13 22:59:24 +00001733 */
Phil Sturgeona0f980e2011-01-13 10:59:12 +00001734 public function batch_bcc_send()
Derek Allard2067d1a2008-11-13 22:59:24 +00001735 {
Andrey Andreev59c5b532012-03-01 14:09:51 +02001736 $float = $this->bcc_batch_size - 1;
1737 $set = '';
Derek Allard2067d1a2008-11-13 22:59:24 +00001738 $chunk = array();
1739
Andrey Andreev1bd3d882011-12-22 15:38:20 +02001740 for ($i = 0, $c = count($this->_bcc_array); $i < $c; $i++)
Derek Allard2067d1a2008-11-13 22:59:24 +00001741 {
1742 if (isset($this->_bcc_array[$i]))
1743 {
Andrey Andreev59c5b532012-03-01 14:09:51 +02001744 $set .= ', '.$this->_bcc_array[$i];
Derek Allard2067d1a2008-11-13 22:59:24 +00001745 }
1746
Alex Bilbied261b1e2012-06-02 11:12:16 +01001747 if ($i === $float)
Derek Allard2067d1a2008-11-13 22:59:24 +00001748 {
Andrey Andreev4e2cdec2016-10-28 14:19:08 +03001749 $chunk[] = self::substr($set, 1);
Andrey Andreev1bd3d882011-12-22 15:38:20 +02001750 $float += $this->bcc_batch_size;
Andrey Andreev59c5b532012-03-01 14:09:51 +02001751 $set = '';
Derek Allard2067d1a2008-11-13 22:59:24 +00001752 }
1753
Andrey Andreev1bd3d882011-12-22 15:38:20 +02001754 if ($i === $c-1)
Derek Allard2067d1a2008-11-13 22:59:24 +00001755 {
Andrey Andreev4e2cdec2016-10-28 14:19:08 +03001756 $chunk[] = self::substr($set, 1);
Derek Allard2067d1a2008-11-13 22:59:24 +00001757 }
1758 }
1759
Andrey Andreev1bd3d882011-12-22 15:38:20 +02001760 for ($i = 0, $c = count($chunk); $i < $c; $i++)
Derek Allard2067d1a2008-11-13 22:59:24 +00001761 {
1762 unset($this->_headers['Bcc']);
Derek Allard2067d1a2008-11-13 22:59:24 +00001763
Andrey Andreev1bd3d882011-12-22 15:38:20 +02001764 $bcc = $this->clean_email($this->_str_to_array($chunk[$i]));
Derek Allard2067d1a2008-11-13 22:59:24 +00001765
Andrey Andreev1bd3d882011-12-22 15:38:20 +02001766 if ($this->protocol !== 'smtp')
Derek Allard2067d1a2008-11-13 22:59:24 +00001767 {
Mickey Wubfc1cad2012-05-31 22:28:40 -07001768 $this->set_header('Bcc', implode(', ', $bcc));
Derek Allard2067d1a2008-11-13 22:59:24 +00001769 }
1770 else
1771 {
1772 $this->_bcc_array = $bcc;
1773 }
1774
buhay466e8082013-04-17 14:25:34 -07001775 if ($this->_build_message() === FALSE)
1776 {
1777 return FALSE;
1778 }
buhayc6da1ef2013-04-18 09:06:49 -07001779
Derek Allard2067d1a2008-11-13 22:59:24 +00001780 $this->_spool_email();
1781 }
1782 }
Barry Mienydd671972010-10-04 16:33:58 +02001783
Derek Allard2067d1a2008-11-13 22:59:24 +00001784 // --------------------------------------------------------------------
1785
1786 /**
1787 * Unwrap special elements
1788 *
Derek Allard2067d1a2008-11-13 22:59:24 +00001789 * @return void
1790 */
Phil Sturgeon6d2f13a2011-07-20 10:04:52 -06001791 protected function _unwrap_specials()
Derek Allard2067d1a2008-11-13 22:59:24 +00001792 {
Andrey Andreev56454792012-05-17 14:32:19 +03001793 $this->_finalbody = preg_replace_callback('/\{unwrap\}(.*?)\{\/unwrap\}/si', array($this, '_remove_nl_callback'), $this->_finalbody);
Derek Allard2067d1a2008-11-13 22:59:24 +00001794 }
Barry Mienydd671972010-10-04 16:33:58 +02001795
Derek Allard2067d1a2008-11-13 22:59:24 +00001796 // --------------------------------------------------------------------
1797
1798 /**
1799 * Strip line-breaks via callback
1800 *
Andrey Andreev5fd3ae82012-10-24 14:55:35 +03001801 * @param string $matches
Derek Allard2067d1a2008-11-13 22:59:24 +00001802 * @return string
1803 */
Phil Sturgeon6d2f13a2011-07-20 10:04:52 -06001804 protected function _remove_nl_callback($matches)
Derek Allard2067d1a2008-11-13 22:59:24 +00001805 {
1806 if (strpos($matches[1], "\r") !== FALSE OR strpos($matches[1], "\n") !== FALSE)
1807 {
1808 $matches[1] = str_replace(array("\r\n", "\r", "\n"), '', $matches[1]);
1809 }
1810
1811 return $matches[1];
1812 }
Barry Mienydd671972010-10-04 16:33:58 +02001813
Derek Allard2067d1a2008-11-13 22:59:24 +00001814 // --------------------------------------------------------------------
1815
1816 /**
1817 * Spool mail to the mail server
1818 *
Derek Allard2067d1a2008-11-13 22:59:24 +00001819 * @return bool
1820 */
Phil Sturgeon6d2f13a2011-07-20 10:04:52 -06001821 protected function _spool_email()
Derek Allard2067d1a2008-11-13 22:59:24 +00001822 {
1823 $this->_unwrap_specials();
1824
Andrey Andreevb9e45e32017-04-06 14:36:19 +03001825 $protocol = $this->_get_protocol();
1826 $method = '_send_with_'.$protocol;
Andrey Andreev1bd3d882011-12-22 15:38:20 +02001827 if ( ! $this->$method())
Derek Allard2067d1a2008-11-13 22:59:24 +00001828 {
Andrey Andreevb9e45e32017-04-06 14:36:19 +03001829 $this->_set_error_message('lang:email_send_failure_'.($protocol === 'mail' ? 'phpmail' : $protocol));
Diogo Osório7fcc7bd2012-03-02 16:54:52 +00001830 return FALSE;
Derek Allard2067d1a2008-11-13 22:59:24 +00001831 }
1832
Andrey Andreevb9e45e32017-04-06 14:36:19 +03001833 $this->_set_error_message('lang:email_sent', $protocol);
Derek Allard2067d1a2008-11-13 22:59:24 +00001834 return TRUE;
1835 }
Barry Mienydd671972010-10-04 16:33:58 +02001836
Derek Allard2067d1a2008-11-13 22:59:24 +00001837 // --------------------------------------------------------------------
1838
1839 /**
Andrey Andreeva9600162017-01-05 17:35:28 +02001840 * Validate email for shell
1841 *
1842 * Applies stricter, shell-safe validation to email addresses.
1843 * Introduced to prevent RCE via sendmail's -f option.
1844 *
1845 * @see https://github.com/bcit-ci/CodeIgniter/issues/4963
1846 * @see https://gist.github.com/Zenexer/40d02da5e07f151adeaeeaa11af9ab36
1847 * @license https://creativecommons.org/publicdomain/zero/1.0/ CC0 1.0, Public Domain
1848 *
1849 * Credits for the base concept go to Paul Buonopane <paul@namepros.com>
1850 *
1851 * @param string $email
1852 * @return bool
1853 */
1854 protected function _validate_email_for_shell(&$email)
1855 {
Andrey Andreev4fd2d492017-11-09 20:38:33 +02001856 if (function_exists('idn_to_ascii') && strpos($email, '@'))
Andrey Andreeva9600162017-01-05 17:35:28 +02001857 {
Andrey Andreev4fd2d492017-11-09 20:38:33 +02001858 list($account, $domain) = explode('@', $email, 2);
1859 $domain = is_php('5.4')
1860 ? idn_to_ascii($domain, 0, INTL_IDNA_VARIANT_UTS46)
1861 : idn_to_ascii($domain);
1862 $email = $account.'@'.$domain;
Andrey Andreeva9600162017-01-05 17:35:28 +02001863 }
1864
1865 return (filter_var($email, FILTER_VALIDATE_EMAIL) === $email && preg_match('#\A[a-z0-9._+-]+@[a-z0-9.-]{1,253}\z#i', $email));
1866 }
1867
1868 // --------------------------------------------------------------------
1869
1870 /**
Derek Allard2067d1a2008-11-13 22:59:24 +00001871 * Send using mail()
1872 *
Derek Allard2067d1a2008-11-13 22:59:24 +00001873 * @return bool
1874 */
Phil Sturgeon6d2f13a2011-07-20 10:04:52 -06001875 protected function _send_with_mail()
Derek Allard2067d1a2008-11-13 22:59:24 +00001876 {
Andrey Andreev8a7078b2012-10-17 10:52:49 +03001877 if (is_array($this->_recipients))
1878 {
1879 $this->_recipients = implode(', ', $this->_recipients);
1880 }
1881
Andrey Andreeva9600162017-01-05 17:35:28 +02001882 // _validate_email_for_shell() below accepts by reference,
1883 // so this needs to be assigned to a variable
1884 $from = $this->clean_email($this->_headers['Return-Path']);
1885
1886 if ($this->_safe_mode === TRUE || ! $this->_validate_email_for_shell($from))
Derek Allard2067d1a2008-11-13 22:59:24 +00001887 {
Andrey Andreev1bd3d882011-12-22 15:38:20 +02001888 return mail($this->_recipients, $this->_subject, $this->_finalbody, $this->_header_str);
Derek Allard2067d1a2008-11-13 22:59:24 +00001889 }
1890 else
1891 {
1892 // most documentation of sendmail using the "-f" flag lacks a space after it, however
1893 // we've encountered servers that seem to require it to be in place.
Andrey Andreeva9600162017-01-05 17:35:28 +02001894 return mail($this->_recipients, $this->_subject, $this->_finalbody, $this->_header_str, '-f '.$from);
Derek Allard2067d1a2008-11-13 22:59:24 +00001895 }
1896 }
Barry Mienydd671972010-10-04 16:33:58 +02001897
Derek Allard2067d1a2008-11-13 22:59:24 +00001898 // --------------------------------------------------------------------
1899
1900 /**
1901 * Send using Sendmail
1902 *
Derek Allard2067d1a2008-11-13 22:59:24 +00001903 * @return bool
1904 */
Phil Sturgeon6d2f13a2011-07-20 10:04:52 -06001905 protected function _send_with_sendmail()
Derek Allard2067d1a2008-11-13 22:59:24 +00001906 {
Andrey Andreeva9600162017-01-05 17:35:28 +02001907 // _validate_email_for_shell() below accepts by reference,
1908 // so this needs to be assigned to a variable
1909 $from = $this->clean_email($this->_headers['From']);
1910 if ($this->_validate_email_for_shell($from))
Derek Jones4cefaa42009-04-29 19:13:56 +00001911 {
Andrey Andreeva9600162017-01-05 17:35:28 +02001912 $from = '-f '.$from;
1913 }
1914 else
1915 {
1916 $from = '';
1917 }
1918
1919 // is popen() enabled?
1920 if ( ! function_usable('popen') OR FALSE === ($fp = @popen($this->mailpath.' -oi '.$from.' -t', 'w')))
1921 {
1922 // server probably has popen disabled, so nothing we can do to get a verbose error.
Derek Jones4cefaa42009-04-29 19:13:56 +00001923 return FALSE;
1924 }
Derek Jones71141ce2010-03-02 16:41:20 -06001925
Derek Jonesc630bcf2008-11-17 21:09:45 +00001926 fputs($fp, $this->_header_str);
1927 fputs($fp, $this->_finalbody);
1928
Barry Mienydd671972010-10-04 16:33:58 +02001929 $status = pclose($fp);
Eric Barnes6113f542010-12-29 13:36:12 -05001930
Andrey Andreev1bd3d882011-12-22 15:38:20 +02001931 if ($status !== 0)
Derek Allard2067d1a2008-11-13 22:59:24 +00001932 {
patworkb0707982011-04-08 15:10:05 +02001933 $this->_set_error_message('lang:email_exit_status', $status);
1934 $this->_set_error_message('lang:email_no_socket');
Derek Allard2067d1a2008-11-13 22:59:24 +00001935 return FALSE;
1936 }
1937
Derek Allard2067d1a2008-11-13 22:59:24 +00001938 return TRUE;
1939 }
Barry Mienydd671972010-10-04 16:33:58 +02001940
Derek Allard2067d1a2008-11-13 22:59:24 +00001941 // --------------------------------------------------------------------
1942
1943 /**
1944 * Send using SMTP
1945 *
Derek Allard2067d1a2008-11-13 22:59:24 +00001946 * @return bool
1947 */
Phil Sturgeon6d2f13a2011-07-20 10:04:52 -06001948 protected function _send_with_smtp()
nisheeth-barthwal59209de2013-02-18 17:02:13 +05301949 {
1950 if ($this->smtp_host === '')
1951 {
1952 $this->_set_error_message('lang:email_no_hostname');
1953 return FALSE;
1954 }
Derek Allard2067d1a2008-11-13 22:59:24 +00001955
nisheeth-barthwal59209de2013-02-18 17:02:13 +05301956 if ( ! $this->_smtp_connect() OR ! $this->_smtp_authenticate())
1957 {
1958 return FALSE;
1959 }
Derek Allard2067d1a2008-11-13 22:59:24 +00001960
Andrey Andreev8e138ec2015-08-31 15:23:42 +03001961 if ( ! $this->_send_command('from', $this->clean_email($this->_headers['From'])))
1962 {
Andrey Andreev0c821bb2016-07-19 16:04:02 +03001963 $this->_smtp_end();
Andrey Andreev8e138ec2015-08-31 15:23:42 +03001964 return FALSE;
1965 }
Derek Allard2067d1a2008-11-13 22:59:24 +00001966
nisheeth-barthwal59209de2013-02-18 17:02:13 +05301967 foreach ($this->_recipients as $val)
1968 {
Andrey Andreev8e138ec2015-08-31 15:23:42 +03001969 if ( ! $this->_send_command('to', $val))
1970 {
Andrey Andreev0c821bb2016-07-19 16:04:02 +03001971 $this->_smtp_end();
Andrey Andreev8e138ec2015-08-31 15:23:42 +03001972 return FALSE;
1973 }
nisheeth-barthwal59209de2013-02-18 17:02:13 +05301974 }
Derek Allard2067d1a2008-11-13 22:59:24 +00001975
nisheeth-barthwal59209de2013-02-18 17:02:13 +05301976 if (count($this->_cc_array) > 0)
1977 {
1978 foreach ($this->_cc_array as $val)
1979 {
Andrey Andreev8e138ec2015-08-31 15:23:42 +03001980 if ($val !== '' && ! $this->_send_command('to', $val))
nisheeth-barthwal59209de2013-02-18 17:02:13 +05301981 {
Andrey Andreev0c821bb2016-07-19 16:04:02 +03001982 $this->_smtp_end();
Andrey Andreev8e138ec2015-08-31 15:23:42 +03001983 return FALSE;
nisheeth-barthwal59209de2013-02-18 17:02:13 +05301984 }
1985 }
1986 }
Derek Allard2067d1a2008-11-13 22:59:24 +00001987
nisheeth-barthwal59209de2013-02-18 17:02:13 +05301988 if (count($this->_bcc_array) > 0)
1989 {
1990 foreach ($this->_bcc_array as $val)
1991 {
Andrey Andreev8e138ec2015-08-31 15:23:42 +03001992 if ($val !== '' && ! $this->_send_command('to', $val))
nisheeth-barthwal59209de2013-02-18 17:02:13 +05301993 {
Andrey Andreev0c821bb2016-07-19 16:04:02 +03001994 $this->_smtp_end();
Andrey Andreev8e138ec2015-08-31 15:23:42 +03001995 return FALSE;
nisheeth-barthwal59209de2013-02-18 17:02:13 +05301996 }
1997 }
1998 }
Derek Allard2067d1a2008-11-13 22:59:24 +00001999
Andrey Andreev8e138ec2015-08-31 15:23:42 +03002000 if ( ! $this->_send_command('data'))
2001 {
Andrey Andreev0c821bb2016-07-19 16:04:02 +03002002 $this->_smtp_end();
Andrey Andreev8e138ec2015-08-31 15:23:42 +03002003 return FALSE;
2004 }
Derek Allard2067d1a2008-11-13 22:59:24 +00002005
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302006 // perform dot transformation on any lines that begin with a dot
2007 $this->_send_data($this->_header_str.preg_replace('/^\./m', '..$1', $this->_finalbody));
Derek Allard2067d1a2008-11-13 22:59:24 +00002008
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302009 $this->_send_data('.');
Derek Allard2067d1a2008-11-13 22:59:24 +00002010
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302011 $reply = $this->_get_smtp_data();
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302012 $this->_set_error_message($reply);
Derek Allard2067d1a2008-11-13 22:59:24 +00002013
Andrey Andreev0c821bb2016-07-19 16:04:02 +03002014 $this->_smtp_end();
2015
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302016 if (strpos($reply, '250') !== 0)
2017 {
2018 $this->_set_error_message('lang:email_smtp_error', $reply);
2019 return FALSE;
2020 }
Derek Allard2067d1a2008-11-13 22:59:24 +00002021
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302022 return TRUE;
2023 }
Derek Allard2067d1a2008-11-13 22:59:24 +00002024
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302025 // --------------------------------------------------------------------
Radu Potop4c589ae2011-09-29 10:19:55 +03002026
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302027 /**
Andrey Andreev0c821bb2016-07-19 16:04:02 +03002028 * SMTP End
2029 *
2030 * Shortcut to send RSET or QUIT depending on keep-alive
2031 *
2032 * @return void
2033 */
2034 protected function _smtp_end()
2035 {
2036 ($this->smtp_keepalive)
2037 ? $this->_send_command('reset')
2038 : $this->_send_command('quit');
2039 }
2040
2041 // --------------------------------------------------------------------
2042
2043 /**
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302044 * SMTP Connect
2045 *
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302046 * @return string
2047 */
nisheeth-barthwala44e6912013-02-18 18:07:03 +05302048 protected function _smtp_connect()
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302049 {
nisheeth-barthwala44e6912013-02-18 18:07:03 +05302050 if (is_resource($this->_smtp_connect))
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302051 {
nisheeth-barthwala44e6912013-02-18 18:07:03 +05302052 return TRUE;
2053 }
Derek Allard2067d1a2008-11-13 22:59:24 +00002054
Andrey Andreev89b67b42013-03-12 19:34:23 +02002055 $ssl = ($this->smtp_crypto === 'ssl') ? 'ssl://' : '';
Derek Allard2067d1a2008-11-13 22:59:24 +00002056
nisheeth-barthwala44e6912013-02-18 18:07:03 +05302057 $this->_smtp_connect = fsockopen($ssl.$this->smtp_host,
2058 $this->smtp_port,
2059 $errno,
2060 $errstr,
2061 $this->smtp_timeout);
2062
2063 if ( ! is_resource($this->_smtp_connect))
2064 {
2065 $this->_set_error_message('lang:email_smtp_error', $errno.' '.$errstr);
2066 return FALSE;
2067 }
2068
2069 stream_set_timeout($this->_smtp_connect, $this->smtp_timeout);
2070 $this->_set_error_message($this->_get_smtp_data());
2071
2072 if ($this->smtp_crypto === 'tls')
2073 {
2074 $this->_send_command('hello');
2075 $this->_send_command('starttls');
2076
Andrey Andreev329dd032018-01-22 10:54:10 +02002077 /**
2078 * STREAM_CRYPTO_METHOD_TLS_CLIENT is quite the mess ...
2079 *
2080 * - On PHP <5.6 it doesn't even mean TLS, but SSL 2.0, and there's no option to use actual TLS
2081 * - On PHP 5.6.0-5.6.6, >=7.2 it means negotiation with any of TLS 1.0, 1.1, 1.2
2082 * - On PHP 5.6.7-7.1.* it means only TLS 1.0
2083 *
2084 * We want the negotiation, so we'll force it below ...
2085 */
2086 $method = is_php('5.6')
2087 ? STREAM_CRYPTO_METHOD_TLSv1_0_CLIENT | STREAM_CRYPTO_METHOD_TLSv1_1_CLIENT | STREAM_CRYPTO_METHOD_TLSv1_2_CLIENT
2088 : STREAM_CRYPTO_METHOD_TLS_CLIENT;
2089 $crypto = stream_socket_enable_crypto($this->_smtp_connect, TRUE, $method);
nisheeth-barthwala44e6912013-02-18 18:07:03 +05302090
2091 if ($crypto !== TRUE)
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302092 {
nisheeth-barthwala44e6912013-02-18 18:07:03 +05302093 $this->_set_error_message('lang:email_smtp_error', $this->_get_smtp_data());
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302094 return FALSE;
2095 }
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302096 }
nisheeth-barthwala44e6912013-02-18 18:07:03 +05302097
2098 return $this->_send_command('hello');
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302099 }
Derek Allard2067d1a2008-11-13 22:59:24 +00002100
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302101 // --------------------------------------------------------------------
Derek Allard2067d1a2008-11-13 22:59:24 +00002102
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302103 /**
2104 * Send SMTP command
2105 *
2106 * @param string
2107 * @param string
Andrey Andreev19277092016-08-22 11:34:56 +03002108 * @return bool
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302109 */
2110 protected function _send_command($cmd, $data = '')
2111 {
2112 switch ($cmd)
2113 {
2114 case 'hello' :
Derek Allard2067d1a2008-11-13 22:59:24 +00002115
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302116 if ($this->_smtp_auth OR $this->_get_encoding() === '8bit')
2117 {
2118 $this->_send_data('EHLO '.$this->_get_hostname());
2119 }
2120 else
2121 {
2122 $this->_send_data('HELO '.$this->_get_hostname());
2123 }
Radu Potopbbf04b02011-09-28 13:57:51 +03002124
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302125 $resp = 250;
2126 break;
2127 case 'starttls' :
Derek Allard2067d1a2008-11-13 22:59:24 +00002128
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302129 $this->_send_data('STARTTLS');
2130 $resp = 220;
2131 break;
2132 case 'from' :
Andrey Andreev56454792012-05-17 14:32:19 +03002133
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302134 $this->_send_data('MAIL FROM:<'.$data.'>');
2135 $resp = 250;
2136 break;
2137 case 'to' :
Andrey Andreev56454792012-05-17 14:32:19 +03002138
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302139 if ($this->dsn)
2140 {
2141 $this->_send_data('RCPT TO:<'.$data.'> NOTIFY=SUCCESS,DELAY,FAILURE ORCPT=rfc822;'.$data);
2142 }
2143 else
2144 {
2145 $this->_send_data('RCPT TO:<'.$data.'>');
2146 }
Derek Allard2067d1a2008-11-13 22:59:24 +00002147
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302148 $resp = 250;
2149 break;
2150 case 'data' :
Derek Allard2067d1a2008-11-13 22:59:24 +00002151
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302152 $this->_send_data('DATA');
2153 $resp = 354;
2154 break;
2155 case 'reset':
Derek Allard2067d1a2008-11-13 22:59:24 +00002156
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302157 $this->_send_data('RSET');
2158 $resp = 250;
2159 break;
2160 case 'quit' :
Derek Allard2067d1a2008-11-13 22:59:24 +00002161
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302162 $this->_send_data('QUIT');
2163 $resp = 221;
2164 break;
2165 }
Derek Allard2067d1a2008-11-13 22:59:24 +00002166
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302167 $reply = $this->_get_smtp_data();
Derek Allard2067d1a2008-11-13 22:59:24 +00002168
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302169 $this->_debug_msg[] = '<pre>'.$cmd.': '.$reply.'</pre>';
Derek Allard2067d1a2008-11-13 22:59:24 +00002170
Andrey Andreev4e2cdec2016-10-28 14:19:08 +03002171 if ((int) self::substr($reply, 0, 3) !== $resp)
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302172 {
2173 $this->_set_error_message('lang:email_smtp_error', $reply);
2174 return FALSE;
2175 }
Barry Mienydd671972010-10-04 16:33:58 +02002176
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302177 if ($cmd === 'quit')
2178 {
2179 fclose($this->_smtp_connect);
2180 }
Derek Allard2067d1a2008-11-13 22:59:24 +00002181
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302182 return TRUE;
2183 }
Derek Allard2067d1a2008-11-13 22:59:24 +00002184
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302185 // --------------------------------------------------------------------
Derek Allard2067d1a2008-11-13 22:59:24 +00002186
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302187 /**
2188 * SMTP Authenticate
2189 *
2190 * @return bool
2191 */
2192 protected function _smtp_authenticate()
2193 {
2194 if ( ! $this->_smtp_auth)
2195 {
2196 return TRUE;
2197 }
Derek Allard2067d1a2008-11-13 22:59:24 +00002198
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302199 if ($this->smtp_user === '' && $this->smtp_pass === '')
2200 {
2201 $this->_set_error_message('lang:email_no_smtp_unpw');
2202 return FALSE;
2203 }
Derek Allard2067d1a2008-11-13 22:59:24 +00002204
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302205 $this->_send_data('AUTH LOGIN');
Derek Allard2067d1a2008-11-13 22:59:24 +00002206
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302207 $reply = $this->_get_smtp_data();
Derek Allard2067d1a2008-11-13 22:59:24 +00002208
Andrey Andreevfa01ae42013-03-04 14:40:18 +02002209 if (strpos($reply, '503') === 0) // Already authenticated
nisheeth-barthwal758f40c2013-02-18 17:18:51 +05302210 {
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302211 return TRUE;
nisheeth-barthwal758f40c2013-02-18 17:18:51 +05302212 }
Andrey Andreevfa01ae42013-03-04 14:40:18 +02002213 elseif (strpos($reply, '334') !== 0)
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302214 {
2215 $this->_set_error_message('lang:email_failed_smtp_login', $reply);
2216 return FALSE;
2217 }
Derek Allard2067d1a2008-11-13 22:59:24 +00002218
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302219 $this->_send_data(base64_encode($this->smtp_user));
Derek Allard2067d1a2008-11-13 22:59:24 +00002220
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302221 $reply = $this->_get_smtp_data();
Derek Allard2067d1a2008-11-13 22:59:24 +00002222
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302223 if (strpos($reply, '334') !== 0)
2224 {
2225 $this->_set_error_message('lang:email_smtp_auth_un', $reply);
2226 return FALSE;
2227 }
Derek Allard2067d1a2008-11-13 22:59:24 +00002228
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302229 $this->_send_data(base64_encode($this->smtp_pass));
Derek Allard2067d1a2008-11-13 22:59:24 +00002230
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302231 $reply = $this->_get_smtp_data();
nisheeth-barthwalcf225572013-02-18 01:08:04 +05302232
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302233 if (strpos($reply, '235') !== 0)
2234 {
2235 $this->_set_error_message('lang:email_smtp_auth_pw', $reply);
2236 return FALSE;
2237 }
nisheeth-barthwalcf225572013-02-18 01:08:04 +05302238
Andrey Andreev84253192016-05-09 12:24:52 +03002239 if ($this->smtp_keepalive)
2240 {
2241 $this->_smtp_auth = FALSE;
2242 }
2243
nisheeth-barthwal59209de2013-02-18 17:02:13 +05302244 return TRUE;
2245 }
Barry Mienydd671972010-10-04 16:33:58 +02002246
Derek Allard2067d1a2008-11-13 22:59:24 +00002247 // --------------------------------------------------------------------
2248
2249 /**
2250 * Send SMTP data
2251 *
Andrey Andreev5fd3ae82012-10-24 14:55:35 +03002252 * @param string $data
Derek Allard2067d1a2008-11-13 22:59:24 +00002253 * @return bool
2254 */
Phil Sturgeon6d2f13a2011-07-20 10:04:52 -06002255 protected function _send_data($data)
Derek Allard2067d1a2008-11-13 22:59:24 +00002256 {
Andrey Andreevd8b1ad32014-01-15 17:42:52 +02002257 $data .= $this->newline;
Andrey Andreev4e2cdec2016-10-28 14:19:08 +03002258 for ($written = $timestamp = 0, $length = self::strlen($data); $written < $length; $written += $result)
Andrey Andreevd8b1ad32014-01-15 17:42:52 +02002259 {
Andrey Andreev4e2cdec2016-10-28 14:19:08 +03002260 if (($result = fwrite($this->_smtp_connect, self::substr($data, $written))) === FALSE)
Andrey Andreevd8b1ad32014-01-15 17:42:52 +02002261 {
2262 break;
2263 }
Andrey Andreev4e0496e2015-06-22 12:34:38 +03002264 // See https://bugs.php.net/bug.php?id=39598 and http://php.net/manual/en/function.fwrite.php#96951
2265 elseif ($result === 0)
2266 {
2267 if ($timestamp === 0)
2268 {
2269 $timestamp = time();
2270 }
2271 elseif ($timestamp < (time() - $this->smtp_timeout))
2272 {
2273 $result = FALSE;
2274 break;
2275 }
2276
2277 usleep(250000);
2278 continue;
2279 }
Andrey Andreevfbe4d792017-12-27 19:49:03 +02002280
2281 $timestamp = 0;
Andrey Andreevd8b1ad32014-01-15 17:42:52 +02002282 }
2283
2284 if ($result === FALSE)
Derek Allard2067d1a2008-11-13 22:59:24 +00002285 {
patworkb0707982011-04-08 15:10:05 +02002286 $this->_set_error_message('lang:email_smtp_data_failure', $data);
Derek Allard2067d1a2008-11-13 22:59:24 +00002287 return FALSE;
2288 }
Andrey Andreev1bd3d882011-12-22 15:38:20 +02002289
2290 return TRUE;
Derek Allard2067d1a2008-11-13 22:59:24 +00002291 }
Barry Mienydd671972010-10-04 16:33:58 +02002292
Derek Allard2067d1a2008-11-13 22:59:24 +00002293 // --------------------------------------------------------------------
2294
2295 /**
2296 * Get SMTP data
2297 *
Derek Allard2067d1a2008-11-13 22:59:24 +00002298 * @return string
2299 */
Phil Sturgeon6d2f13a2011-07-20 10:04:52 -06002300 protected function _get_smtp_data()
Derek Allard2067d1a2008-11-13 22:59:24 +00002301 {
Andrey Andreev56454792012-05-17 14:32:19 +03002302 $data = '';
Derek Allard2067d1a2008-11-13 22:59:24 +00002303
2304 while ($str = fgets($this->_smtp_connect, 512))
2305 {
2306 $data .= $str;
2307
Alex Bilbied261b1e2012-06-02 11:12:16 +01002308 if ($str[3] === ' ')
Derek Allard2067d1a2008-11-13 22:59:24 +00002309 {
2310 break;
2311 }
2312 }
2313
2314 return $data;
2315 }
Barry Mienydd671972010-10-04 16:33:58 +02002316
Derek Allard2067d1a2008-11-13 22:59:24 +00002317 // --------------------------------------------------------------------
2318
2319 /**
2320 * Get Hostname
Andrey Andreev396eb892015-02-06 14:50:10 +02002321 *
2322 * There are only two legal types of hostname - either a fully
2323 * qualified domain name (eg: "mail.example.com") or an IP literal
2324 * (eg: "[1.2.3.4]").
2325 *
2326 * @link https://tools.ietf.org/html/rfc5321#section-2.3.5
James Wade3245af42015-02-06 11:48:51 +00002327 * @link http://cbl.abuseat.org/namingproblems.html
Derek Allard2067d1a2008-11-13 22:59:24 +00002328 * @return string
2329 */
Phil Sturgeon6d2f13a2011-07-20 10:04:52 -06002330 protected function _get_hostname()
Derek Allard2067d1a2008-11-13 22:59:24 +00002331 {
Andrey Andreev396eb892015-02-06 14:50:10 +02002332 if (isset($_SERVER['SERVER_NAME']))
2333 {
2334 return $_SERVER['SERVER_NAME'];
2335 }
2336
2337 return isset($_SERVER['SERVER_ADDR']) ? '['.$_SERVER['SERVER_ADDR'].']' : '[127.0.0.1]';
Derek Allard2067d1a2008-11-13 22:59:24 +00002338 }
Barry Mienydd671972010-10-04 16:33:58 +02002339
Derek Allard2067d1a2008-11-13 22:59:24 +00002340 // --------------------------------------------------------------------
2341
2342 /**
Derek Allard2067d1a2008-11-13 22:59:24 +00002343 * Get Debug Message
2344 *
Andrey Andreev61797f62012-11-26 16:15:12 +02002345 * @param array $include List of raw data chunks to include in the output
2346 * Valid options are: 'headers', 'subject', 'body'
Derek Allard2067d1a2008-11-13 22:59:24 +00002347 * @return string
2348 */
Andrey Andreev61797f62012-11-26 16:15:12 +02002349 public function print_debugger($include = array('headers', 'subject', 'body'))
Derek Allard2067d1a2008-11-13 22:59:24 +00002350 {
2351 $msg = '';
2352
2353 if (count($this->_debug_msg) > 0)
2354 {
2355 foreach ($this->_debug_msg as $val)
2356 {
2357 $msg .= $val;
2358 }
2359 }
2360
Andrey Andreev61797f62012-11-26 16:15:12 +02002361 // Determine which parts of our raw data needs to be printed
2362 $raw_data = '';
2363 is_array($include) OR $include = array($include);
2364
2365 if (in_array('headers', $include, TRUE))
2366 {
Andrey Andreev58f677f2013-07-16 11:01:37 +03002367 $raw_data = htmlspecialchars($this->_header_str)."\n";
Andrey Andreev61797f62012-11-26 16:15:12 +02002368 }
2369
2370 if (in_array('subject', $include, TRUE))
2371 {
2372 $raw_data .= htmlspecialchars($this->_subject)."\n";
2373 }
2374
2375 if (in_array('body', $include, TRUE))
2376 {
2377 $raw_data .= htmlspecialchars($this->_finalbody);
2378 }
2379
2380 return $msg.($raw_data === '' ? '' : '<pre>'.$raw_data.'</pre>');
Derek Allard2067d1a2008-11-13 22:59:24 +00002381 }
Barry Mienydd671972010-10-04 16:33:58 +02002382
Derek Allard2067d1a2008-11-13 22:59:24 +00002383 // --------------------------------------------------------------------
2384
2385 /**
2386 * Set Message
2387 *
Andrey Andreev5fd3ae82012-10-24 14:55:35 +03002388 * @param string $msg
2389 * @param string $val = ''
Andrey Andreev081c9462012-03-01 12:58:11 +02002390 * @return void
Derek Allard2067d1a2008-11-13 22:59:24 +00002391 */
Phil Sturgeon6d2f13a2011-07-20 10:04:52 -06002392 protected function _set_error_message($msg, $val = '')
Derek Allard2067d1a2008-11-13 22:59:24 +00002393 {
2394 $CI =& get_instance();
2395 $CI->lang->load('email');
2396
Andrey Andreev7a7ad782012-11-12 17:21:01 +02002397 if (sscanf($msg, 'lang:%s', $line) !== 1 OR FALSE === ($line = $CI->lang->line($line)))
Derek Allard2067d1a2008-11-13 22:59:24 +00002398 {
Andrey Andreev56454792012-05-17 14:32:19 +03002399 $this->_debug_msg[] = str_replace('%s', $val, $msg).'<br />';
Derek Allard2067d1a2008-11-13 22:59:24 +00002400 }
2401 else
2402 {
Andrey Andreev56454792012-05-17 14:32:19 +03002403 $this->_debug_msg[] = str_replace('%s', $val, $line).'<br />';
Derek Allard2067d1a2008-11-13 22:59:24 +00002404 }
2405 }
Barry Mienydd671972010-10-04 16:33:58 +02002406
Derek Allard2067d1a2008-11-13 22:59:24 +00002407 // --------------------------------------------------------------------
2408
2409 /**
2410 * Mime Types
2411 *
Derek Allard2067d1a2008-11-13 22:59:24 +00002412 * @param string
2413 * @return string
2414 */
Andrey Andreev59c5b532012-03-01 14:09:51 +02002415 protected function _mime_types($ext = '')
Derek Allard2067d1a2008-11-13 22:59:24 +00002416 {
Andrey Andreev6ef498b2012-06-05 22:01:58 +03002417 $ext = strtolower($ext);
2418
vlakoff66c7bb42014-05-19 13:45:12 +02002419 $mimes =& get_mimes();
vlakoff69550c52014-05-19 13:45:02 +02002420
Andrey Andreev6ef498b2012-06-05 22:01:58 +03002421 if (isset($mimes[$ext]))
2422 {
2423 return is_array($mimes[$ext])
2424 ? current($mimes[$ext])
2425 : $mimes[$ext];
2426 }
2427
2428 return 'application/x-unknown-content-type';
Derek Allard2067d1a2008-11-13 22:59:24 +00002429 }
2430
Andrey Andreev8c95c3d2016-05-09 12:35:41 +03002431 // --------------------------------------------------------------------
2432
2433 /**
2434 * Destructor
2435 *
2436 * @return void
2437 */
2438 public function __destruct()
2439 {
2440 is_resource($this->_smtp_connect) && $this->_send_command('quit');
2441 }
Andrey Andreev4e2cdec2016-10-28 14:19:08 +03002442
2443 // --------------------------------------------------------------------
2444
2445 /**
2446 * Byte-safe strlen()
2447 *
2448 * @param string $str
2449 * @return int
2450 */
2451 protected static function strlen($str)
2452 {
Andrey Andreevc0c74d52017-01-19 15:26:35 +02002453 return (self::$func_overload)
Andrey Andreev4e2cdec2016-10-28 14:19:08 +03002454 ? mb_strlen($str, '8bit')
2455 : strlen($str);
2456 }
2457
2458 // --------------------------------------------------------------------
2459
2460 /**
2461 * Byte-safe substr()
2462 *
2463 * @param string $str
2464 * @param int $start
2465 * @param int $length
2466 * @return string
2467 */
2468 protected static function substr($str, $start, $length = NULL)
2469 {
Andrey Andreevc0c74d52017-01-19 15:26:35 +02002470 if (self::$func_overload)
Andrey Andreev4e2cdec2016-10-28 14:19:08 +03002471 {
2472 // mb_substr($str, $start, null, '8bit') returns an empty
2473 // string on PHP 5.3
2474 isset($length) OR $length = ($start >= 0 ? self::strlen($str) - $start : -$start);
2475 return mb_substr($str, $start, $length, '8bit');
2476 }
2477
2478 return isset($length)
2479 ? substr($str, $start, $length)
2480 : substr($str, $start);
2481 }
Derek Allard2067d1a2008-11-13 22:59:24 +00002482}