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

浙公网安备 33010602011771号