cakephp2-php8/lib/Cake/Network/CakeEmail.php

1181 lines
27 KiB
PHP
Raw Normal View History

2011-02-27 15:25:23 -03:00
<?php
/**
* Cake E-Mail
*
* PHP 5
*
* CakePHP(tm) : Rapid Development Framework (http://cakephp.org)
* Copyright 2005-2011, Cake Software Foundation, Inc. (http://cakefoundation.org)
*
* Licensed under The MIT License
* Redistributions of files must retain the above copyright notice.
*
* @copyright Copyright 2005-2011, Cake Software Foundation, Inc. (http://cakefoundation.org)
* @link http://cakephp.org CakePHP(tm) Project
* @package cake.libs
* @since CakePHP(tm) v 2.0.0
* @license MIT License (http://www.opensource.org/licenses/mit-license.php)
*/
2011-04-12 23:29:34 -04:00
App::uses('Validation', 'Utility');
App::uses('Multibyte', 'I18n');
App::uses('AbstractTransport', 'Network/Email');
App::uses('String', 'Utility');
App::uses('View', 'View');
2011-02-27 15:25:23 -03:00
/**
* Cake e-mail class.
*
* This class is used for handling Internet Message Format based
* based on the standard outlined in http://www.rfc-editor.org/rfc/rfc2822.txt
*
* @package cake.libs
*/
class CakeEmail {
/**
* What mailer should EmailComponent identify itself as
*
* @constant EMAIL_CLIENT
*/
const EMAIL_CLIENT = 'CakePHP Email Component';
2011-03-02 18:17:49 -03:00
/**
* Line length - no should more - RFC 2822 - 2.1.1
*
* @constant LINE_LENGTH_SHOULD
*/
const LINE_LENGTH_SHOULD = 78;
/**
* Line length - no must more - RFC 2822 - 2.1.1
*
* @constant LINE_LENGTH_MUST
*/
const LINE_LENGTH_MUST = 998;
2011-02-27 15:25:23 -03:00
/**
* Recipient of the email
*
* @var string
*/
protected $_to = array();
2011-02-27 15:25:23 -03:00
/**
* The mail which the email is sent from
*
* @var string
*/
protected $_from = array();
2011-02-27 15:25:23 -03:00
/**
* The email the recipient will reply to
*
* @var string
*/
protected $_replyTo = array();
2011-02-27 15:25:23 -03:00
/**
* The read receipt email
*
* @var string
*/
protected $_readReceipt = array();
2011-02-27 15:25:23 -03:00
/**
* The mail that will be used in case of any errors like
* - Remote mailserver down
* - Remote user has exceeded his quota
* - Unknown user
*
* @var string
*/
protected $_returnPath = array();
2011-02-27 15:25:23 -03:00
/**
* Carbon Copy
*
* List of email's that should receive a copy of the email.
* The Recipient WILL be able to see this list
*
* @var array
*/
protected $_cc = array();
2011-02-27 15:25:23 -03:00
/**
* Blind Carbon Copy
*
* List of email's that should receive a copy of the email.
* The Recipient WILL NOT be able to see this list
*
* @var array
*/
protected $_bcc = array();
2011-02-27 15:25:23 -03:00
/**
* Message ID
*
* @var mixed True to generate, False to ignore, String with value
*/
protected $_messageId = true;
2011-02-27 15:25:23 -03:00
/**
* The subject of the email
*
* @var string
*/
2011-03-01 14:20:13 -03:00
protected $_subject = '';
2011-02-27 15:25:23 -03:00
/**
* Associative array of a user defined headers
* Keys will be prefixed 'X-' as per RFC2822 Section 4.7.5
*
* @var array
*/
protected $_headers = array();
2011-02-27 15:25:23 -03:00
/**
* Layout for the View
*
* @var string
*/
protected $_layout = 'default';
2011-02-27 15:25:23 -03:00
/**
* Template for the view
*
* @var string
*/
protected $_template = '';
2011-02-27 15:25:23 -03:00
2011-03-04 08:29:38 -03:00
/**
* View for render
*
* @var string
*/
protected $_viewRender = 'View';
/**
* Vars to sent to render
*
* @var array
*/
protected $_viewVars = array();
2011-03-02 18:17:49 -03:00
/**
* Text message
*
* @var string
*/
protected $_textMessage = '';
/**
* Html message
*
* @var string
*/
protected $_htmlMessage = '';
/**
* Final message to send
*
* @var array
*/
protected $_message = array();
2011-02-27 15:25:23 -03:00
/**
* Available formats to be sent.
2011-02-27 15:25:23 -03:00
*
* @var array
*/
protected $_emailFormatAvailable = array('text', 'html', 'both');
/**
* What format should the email be sent in
2011-02-27 15:25:23 -03:00
*
* @var string
*/
protected $_emailFormat = 'text';
2011-02-27 15:25:23 -03:00
/**
* What method should the email be sent
*
* @var string
*/
2011-03-01 18:29:46 -03:00
protected $_transportName = 'mail';
2011-02-27 15:25:23 -03:00
/**
* Instance of tranport class
*
* @var object
*/
protected $_transportClass = null;
2011-02-27 15:25:23 -03:00
/**
* charset the email is sent in
*
* @var string
*/
public $charset = 'utf-8';
/**
* List of files that should be attached to the email.
*
2011-03-01 18:51:12 -03:00
* Only absolute paths
2011-02-27 15:25:23 -03:00
*
* @var array
*/
2011-03-01 18:51:12 -03:00
protected $_attachments = array();
2011-02-27 15:25:23 -03:00
/**
* If set, boundary to use for multipart mime messages
*
* @var string
*/
protected $_boundary = null;
/**
* Configuration to transport
*
* @var mixed
*/
protected $_config = 'default';
2011-02-27 15:25:23 -03:00
/**
* Constructor
*
*/
public function __construct() {
$charset = Configure::read('App.encoding');
if ($charset !== null) {
$this->charset = $charset;
}
}
/**
* From
*
* @param mixed $email
* @param string $name
* @return mixed
* @thrown SocketException
*/
public function from($email = null, $name = null) {
if ($email === null) {
return $this->_from;
}
2011-04-12 23:45:02 -04:00
return $this->_setEmailSingle('_from', $email, $name, __d('cake', 'From requires only 1 email address.'));
}
/**
* Reply-To
*
* @param mixed $email
* @param string $name
* @return mixed
* @thrown SocketException
*/
public function replyTo($email = null, $name = null) {
if ($email === null) {
return $this->_replyTo;
}
2011-04-12 23:45:02 -04:00
return $this->_setEmailSingle('_replyTo', $email, $name, __d('cake', 'Reply-To requires only 1 email address.'));
}
/**
* Read Receipt (Disposition-Notification-To header)
*
* @param mixed $email
* @param string $name
* @return mixed
* @thrown SocketException
*/
public function readReceipt($email = null, $name = null) {
if ($email === null) {
return $this->_readReceipt;
}
2011-04-12 23:45:02 -04:00
return $this->_setEmailSingle('_readReceipt', $email, $name, __d('cake', 'Disposition-Notification-To requires only 1 email address.'));
}
/**
* Return Path
*
* @param mixed $email
* @param string $name
* @return mixed
* @thrown SocketException
*/
public function returnPath($email = null, $name = null) {
if ($email === null) {
return $this->_returnPath;
}
2011-04-12 23:45:02 -04:00
return $this->_setEmailSingle('_returnPath', $email, $name, __d('cake', 'Return-Path requires only 1 email address.'));
}
2011-03-01 02:09:05 -03:00
/**
* To
2011-03-01 02:09:05 -03:00
*
* @param mixed $email Null to get, String with email, Array with email as key, name as value or email as value (without name)
2011-03-01 02:09:05 -03:00
* @param string $name
* @return mixed
2011-03-01 02:09:05 -03:00
*/
public function to($email = null, $name = null) {
if ($email === null) {
return $this->_to;
}
2011-04-12 23:45:02 -04:00
return $this->_setEmail('_to', $email, $name);
2011-03-01 02:09:05 -03:00
}
/**
* Add To
*
* @param mixed $email String with email, Array with email as key, name as value or email as value (without name)
* @param string $name
2011-04-12 23:45:02 -04:00
* @return object $this
2011-03-01 02:09:05 -03:00
*/
public function addTo($email, $name = null) {
2011-04-12 23:45:02 -04:00
return $this->_addEmail('_to', $email, $name);
2011-03-01 02:09:05 -03:00
}
/**
* Cc
*
* @param mixed $email String with email, Array with email as key, name as value or email as value (without name)
* @param string $name
* @return mixed
*/
public function cc($email = null, $name = null) {
if ($email === null) {
return $this->_cc;
}
2011-04-12 23:45:02 -04:00
return $this->_setEmail('_cc', $email, $name);
}
/**
* Add Cc
*
* @param mixed $email String with email, Array with email as key, name as value or email as value (without name)
* @param string $name
2011-04-12 23:45:02 -04:00
* @return object $this
*/
public function addCc($email, $name = null) {
2011-04-12 23:45:02 -04:00
return $this->_addEmail('_cc', $email, $name);
}
/**
* Bcc
*
* @param mixed $email String with email, Array with email as key, name as value or email as value (without name)
* @param string $name
2011-04-12 23:45:02 -04:00
* @return mixed
*/
public function bcc($email = null, $name = null) {
if ($email === null) {
return $this->_bcc;
}
2011-04-12 23:45:02 -04:00
return $this->_setEmail('_bcc', $email, $name);
}
/**
* Add Bcc
*
* @param mixed $email String with email, Array with email as key, name as value or email as value (without name)
* @param string $name
2011-04-12 23:45:02 -04:00
* @return object $this
*/
public function addBcc($email, $name = null) {
2011-04-12 23:45:02 -04:00
return $this->_addEmail('_bcc', $email, $name);
}
2011-03-01 02:09:05 -03:00
/**
* Set email
*
* @param string $varName
* @param mixed $email
* @param mixed $name
2011-04-12 23:45:02 -04:00
* @return object $this
* @thrown SocketException
2011-03-01 02:09:05 -03:00
*/
protected function _setEmail($varName, $email, $name) {
if (!is_array($email)) {
if (!Validation::email($email)) {
throw new SocketException(__d('cake', 'Invalid email: "%s"', $email));
}
2011-03-01 02:09:05 -03:00
if ($name === null) {
2011-03-01 19:58:04 -03:00
$name = $email;
2011-03-01 02:09:05 -03:00
}
2011-03-01 19:58:04 -03:00
$this->{$varName} = array($email => $name);
2011-04-12 23:45:02 -04:00
return $this;
2011-03-01 02:09:05 -03:00
}
$list = array();
foreach ($email as $key => $value) {
if (is_int($key)) {
2011-03-01 19:58:04 -03:00
$key = $value;
2011-03-01 02:09:05 -03:00
}
2011-03-01 19:58:04 -03:00
if (!Validation::email($key)) {
throw new SocketException(__d('cake', 'Invalid email: "%s"', $key));
2011-03-01 19:58:04 -03:00
}
$list[$key] = $value;
2011-03-01 02:09:05 -03:00
}
$this->{$varName} = $list;
2011-04-12 23:45:02 -04:00
return $this;
2011-03-01 02:09:05 -03:00
}
/**
* Set only 1 email
*
* @param string $varName
* @param mixed $email
* @param string $name
* @param string $throwMessage
2011-04-12 23:45:02 -04:00
* @return object $this
* @thrown SocketExpceiton
*/
protected function _setEmailSingle($varName, $email, $name, $throwMessage) {
$current = $this->{$varName};
$this->_setEmail($varName, $email, $name);
if (count($this->{$varName}) !== 1) {
$this->{$varName} = $current;
throw new SocketException($throwMessage);
}
2011-04-12 23:45:02 -04:00
return $this;
}
2011-03-01 02:09:05 -03:00
/**
* Add email
*
* @param string $varName
* @param mixed $email
* @param mixed $name
2011-04-12 23:45:02 -04:00
* @return object $this
2011-03-01 02:09:05 -03:00
*/
protected function _addEmail($varName, $email, $name) {
if (!is_array($email)) {
if (!Validation::email($email)) {
throw new SocketException(__d('cake', 'Invalid email: "%s"', $email));
}
2011-03-01 02:09:05 -03:00
if ($name === null) {
2011-03-01 19:58:04 -03:00
$name = $email;
2011-03-01 02:09:05 -03:00
}
2011-03-01 19:58:04 -03:00
$this->{$varName}[$email] = $name;
2011-04-12 23:45:02 -04:00
return $this;
2011-03-01 02:09:05 -03:00
}
$list = array();
foreach ($email as $key => $value) {
if (is_int($key)) {
2011-03-01 19:58:04 -03:00
$key = $value;
}
if (!Validation::email($key)) {
throw new SocketException(__d('cake', 'Invalid email: "%s"', $key));
2011-03-01 02:09:05 -03:00
}
2011-03-01 19:58:04 -03:00
$list[$key] = $value;
2011-03-01 02:09:05 -03:00
}
$this->{$varName} = array_merge($this->{$varName}, $list);
2011-04-12 23:45:02 -04:00
return $this;
2011-03-01 02:09:05 -03:00
}
2011-03-01 14:20:13 -03:00
/**
* Set Subject
*
* @param string $subject
* @return mixed
2011-03-01 14:20:13 -03:00
*/
public function subject($subject = null) {
if ($subject === null) {
return $this->_subject;
}
2011-04-17 14:00:19 -04:00
$this->_subject = $this->_encode((string)$subject);
2011-04-12 23:45:02 -04:00
return $this;
2011-03-01 14:20:13 -03:00
}
2011-02-27 15:25:23 -03:00
/**
* Sets eaders for the message
2011-02-27 15:25:23 -03:00
*
* @param array Associative array containing headers to be set.
2011-04-12 23:45:02 -04:00
* @return object $this
* @thrown SocketException
2011-02-27 15:25:23 -03:00
*/
public function setHeaders($headers) {
if (!is_array($headers)) {
throw new SocketException(__d('cake', '$headers should be an array.'));
}
$this->_headers = $headers;
2011-04-12 23:45:02 -04:00
return $this;
}
/**
* Add header for the message
*
* @param array $headers
2011-04-12 23:45:02 -04:00
* @return mixed $this
* @thrown SocketException
*/
public function addHeaders($headers) {
if (!is_array($headers)) {
throw new SocketException(__d('cake', '$headers should be an array.'));
}
$this->_headers = array_merge($this->_headers, $headers);
2011-04-12 23:45:02 -04:00
return $this;
}
/**
* Get list of headers
*
2011-03-01 15:16:50 -03:00
* ### Includes:
*
* - `from`
* - `replyTo`
* - `readReceipt`
* - `returnPath`
* - `to`
* - `cc`
* - `bcc`
* - `subject`
*
* @param array $include
* @return array
*/
2011-03-01 15:16:50 -03:00
public function getHeaders($include = array()) {
$defaults = array(
'from' => false,
'replyTo' => false,
'readReceipt' => false,
'returnPath' => false,
'to' => false,
'cc' => false,
'bcc' => false,
'subject' => false
);
$include += $defaults;
$headers = array();
2011-03-01 15:16:50 -03:00
$relation = array(
'from' => 'From',
'replyTo' => 'Reply-To',
'readReceipt' => 'Disposition-Notification-To',
'returnPath' => 'Return-Path'
);
foreach ($relation as $var => $header) {
if ($include[$var]) {
$var = '_' . $var;
$headers[$header] = current($this->_formatAddress($this->{$var}));
}
}
foreach (array('to', 'cc', 'bcc') as $var) {
if ($include[$var]) {
$classVar = '_' . $var;
$headers[ucfirst($var)] = implode(', ', $this->_formatAddress($this->{$classVar}));
}
}
$headers += $this->_headers;
if (!isset($headers['X-Mailer'])) {
$headers['X-Mailer'] = self::EMAIL_CLIENT;
}
if (!isset($headers['Date'])) {
$headers['Date'] = date(DATE_RFC2822);
}
if ($this->_messageId !== false) {
if ($this->_messageId === true) {
$headers['Message-ID'] = '<' . String::UUID() . '@' . env('HTTP_HOST') . '>';
} else {
$headers['Message-ID'] = $this->_messageId;
}
}
2011-03-01 15:16:50 -03:00
if ($include['subject']) {
$headers['Subject'] = $this->_subject;
}
2011-03-03 23:43:55 -03:00
if (!empty($this->_attachments)) {
2011-03-03 23:05:30 -03:00
$this->_createBoundary();
$headers['MIME-Version'] = '1.0';
$headers['Content-Type'] = 'multipart/mixed; boundary="' . $this->_boundary . '"';
$headers[] = 'This part of the E-mail should never be seen. If';
$headers[] = 'you are reading this, consider upgrading your e-mail';
$headers[] = 'client to a MIME-compatible client.';
} elseif ($this->_emailFormat === 'text') {
$headers['Content-Type'] = 'text/plain; charset=' . $this->charset;
} elseif ($this->_emailFormat === 'html') {
$headers['Content-Type'] = 'text/html; charset=' . $this->charset;
} elseif ($this->_emailFormat === 'both') {
$headers['Content-Type'] = 'multipart/alternative; boundary="alt-' . $this->_boundary . '"';
}
$headers['Content-Transfer-Encoding'] = '7bit';
2011-03-01 15:16:50 -03:00
return $headers;
}
/**
* Format addresses
*
* @param array $address
* @return array
*/
protected function _formatAddress($address) {
$return = array();
foreach ($address as $email => $alias) {
if ($email === $alias) {
$return[] = $email;
} else {
$return[] = sprintf('%s <%s>', $this->_encode($alias), $email);
2011-03-01 15:16:50 -03:00
}
2011-02-27 15:25:23 -03:00
}
2011-03-01 15:16:50 -03:00
return $return;
2011-02-27 15:25:23 -03:00
}
/**
* Layout and template
*
* @param string $layout
* @param string $template
* @return mixed
*/
public function layout($layout = null, $template = null) {
if ($layout === null) {
return array(
'layout' => $this->_layout,
'template' => $this->_template
);
}
$this->_layout = (string)$layout;
if ($template !== null) {
$this->_template = (string)$template;
}
2011-04-12 23:45:02 -04:00
return $this;
}
2011-03-04 08:29:38 -03:00
/**
* View class for render
2011-03-04 08:29:38 -03:00
*
* @param string $viewClass
* @return mixed
2011-03-04 08:29:38 -03:00
*/
public function viewRender($viewClass = null) {
if ($viewClass === null) {
return $this->_viewRender;
}
2011-03-04 08:29:38 -03:00
$this->_viewRender = $viewClass;
2011-04-12 23:45:02 -04:00
return $this;
2011-03-04 08:29:38 -03:00
}
/**
* Variables to be set on render
*
* @param array
* @return mixed
*/
public function viewVars($viewVars = null) {
if ($viewVars === null) {
return $this->_viewVars;
}
$this->_viewVars = $viewVars;
return $this;
}
/**
* Email format
*
* @param string $format
* @return mixed
* @thrown SocketException
*/
public function emailFormat($format = null) {
if ($format === null) {
return $this->_emailFormat;
}
if (!in_array($format, $this->_emailFormatAvailable)) {
throw new SocketException(__d('cake', 'Format not available.'));
}
$this->_emailFormat = $format;
2011-04-12 23:45:02 -04:00
return $this;
}
2011-03-01 18:29:46 -03:00
/**
* Transport name
2011-03-01 18:29:46 -03:00
*
* @param string $name
* @return mixed
2011-03-01 18:29:46 -03:00
*/
public function transport($name = null) {
if ($name === null) {
return $this->_transportName;
}
2011-03-01 18:29:46 -03:00
$this->_transportName = (string)$name;
$this->_transportClass = null;
2011-04-12 23:45:02 -04:00
return $this;
2011-03-01 18:29:46 -03:00
}
/**
* Return the transport class
*
* @return object
* @thrown SocketException
*/
public function transportClass() {
if ($this->_transportClass) {
return $this->_transportClass;
}
list($plugin, $transportClassname) = pluginSplit($this->_transportName, true);
$transportClassname .= 'Transport';
App::uses($transportClassname, $plugin . 'Network/Email');
if (!class_exists($transportClassname)) {
throw new SocketException(__d('cake', 'Class "%s" not found.', $transportClassname));
} elseif (!method_exists($transportClassname, 'send')) {
throw new SocketException(__d('cake', 'The "%s" do not have send method.', $transportClassname));
}
return $this->_transportClass = new $transportClassname();
}
/**
* Message-ID
*
* @param mixed $message True to generate a new Message-ID, False to ignore (not send in email), String to set as Message-ID
* @return mixed
* @thrown SocketException
*/
public function messageID($message = null) {
if ($message === null) {
return $this->_messageId;
}
if (is_bool($message)) {
$this->_messageId = $message;
} else {
if (!preg_match('/^\<.+@.+\>$/', $message)) {
throw new SocketException(__d('cake', 'Invalid format to Message-ID. The text should be something like "<uuid@server.com>"'));
}
$this->_messageId = $message;
}
2011-04-12 23:45:02 -04:00
return $this;
}
2011-03-01 18:51:12 -03:00
/**
* Attachments
2011-03-01 18:51:12 -03:00
*
* @param mixed $attachments String with the filename or array with filenames
* @return mixed
2011-03-01 18:51:12 -03:00
* @thrown SocketException
*/
public function attachments($attachments = null) {
if ($attachments === null) {
return $this->_attachments;
}
2011-03-04 00:20:55 -03:00
$attach = array();
foreach ((array)$attachments as $name => $file) {
$path = realpath($file);
2011-03-01 18:51:12 -03:00
if ($path === false) {
throw new SocketException(__d('cake', 'File not found: "%s"', $attach));
2011-03-01 18:51:12 -03:00
}
2011-03-04 00:20:55 -03:00
if (is_int($name)) {
$name = basename($path);
}
$attach[$name] = $path;
2011-03-01 18:51:12 -03:00
}
2011-03-04 00:20:55 -03:00
$this->_attachments = $attach;
2011-04-12 23:45:02 -04:00
return $this;
2011-03-01 18:51:12 -03:00
}
/**
* Add attachments
*
* @param mixed $attachments String with the filename or array with filenames
2011-04-12 23:45:02 -04:00
* @return object $this
2011-03-01 18:51:12 -03:00
* @thrown SocketException
*/
public function addAttachments($attachments) {
$current = $this->_attachments;
$this->attachments($attachments);
2011-03-04 00:20:55 -03:00
$this->_attachments = array_merge($current, $this->_attachments);
2011-04-12 23:45:02 -04:00
return $this;
2011-03-01 18:51:12 -03:00
}
2011-03-02 18:17:49 -03:00
/**
* Get generated message (used by transport classes)
*
* @return array
*/
public function message() {
2011-03-02 18:17:49 -03:00
return $this->_message;
}
2011-03-01 18:51:12 -03:00
/**
* Configuration to use when send email
*
* @param mixed $config String with configuration name (from email.php), array with config or null to return current config
2011-04-12 23:45:02 -04:00
* @return mixed
*/
public function config($config = null) {
if ($config === null) {
2011-04-12 23:45:02 -04:00
return $this->_config;
}
2011-04-12 23:45:02 -04:00
if (is_array($config)) {
$this->_config = $config;
} else {
$this->_config = (string)$config;
}
if ($this->_transportClass) {
$this->_transportClass->config($this->_config);
}
2011-04-12 23:45:02 -04:00
return $this;
}
2011-02-27 15:25:23 -03:00
/**
* Send an email using the specified content, template and layout
*
* @return boolean Success
2011-03-02 18:17:49 -03:00
* @thrown SocketExpcetion
2011-02-27 15:25:23 -03:00
*/
2011-03-02 18:17:49 -03:00
public function send($content = null) {
if (is_string($this->_config)) {
if (!config('email')) {
2011-04-13 00:20:35 -04:00
throw new SocketException(__d('cake', '%s not found.', APP . 'config' . DS . 'email.php'));
}
$configs = new EMAIL_CONFIG();
if (!isset($configs->{$this->_config})) {
throw new SocketException(__d('cake', 'Unknown email configuration "%s".', $this->_config));
}
$config = $configs->{$this->_config};
} else {
$config = $this->_config;
}
2011-03-02 18:17:49 -03:00
if (empty($this->_from)) {
if (!empty($config['from'])) {
$this->to($config['from']);
} else {
throw new SocketException(__d('cake', 'From is not specified.'));
}
2011-03-02 18:17:49 -03:00
}
if (empty($this->_to) && empty($this->_cc) && empty($this->_bcc)) {
throw new SocketExpcetion(__d('cake', 'You need specify one destination on to, cc or bcc.'));
2011-03-02 18:17:49 -03:00
}
if (is_array($content)) {
$content = implode("\n", $content) . "\n";
}
$this->_textMessage = $this->_htmlMessage = '';
if ($content !== null) {
if ($this->_emailFormat === 'text') {
$this->_textMessage = $content;
} elseif ($this->_emailFormat === 'html') {
$this->_htmlMessage = $content;
} elseif ($this->_emailFormat === 'both') {
$this->_textMessage = $this->_htmlMessage = $content;
}
}
$message = $this->_wrap($content);
2011-03-03 23:43:55 -03:00
if (empty($this->_template)) {
2011-03-03 23:05:30 -03:00
$message = $this->_formatMessage($message);
2011-03-02 18:17:49 -03:00
} else {
2011-03-03 23:43:55 -03:00
$message = $this->_render($message);
2011-03-02 18:17:49 -03:00
}
$message[] = '';
$this->_message = $message;
2011-03-03 23:43:55 -03:00
if (!empty($this->_attachments)) {
2011-03-04 01:27:18 -03:00
$this->_attachFiles();
2011-03-02 18:17:49 -03:00
}
if (!is_null($this->_boundary)) {
$this->_message[] = '';
$this->_message[] = '--' . $this->_boundary . '--';
$this->_message[] = '';
}
$transport = $this->transportClass();
$transport->config($config);
2011-03-02 18:17:49 -03:00
return $transport->send($this);
2011-02-27 15:25:23 -03:00
}
/**
* Reset all EmailComponent internal variables to be able to send out a new email.
*
2011-04-12 23:45:02 -04:00
* @return object $this
2011-02-27 15:25:23 -03:00
*/
public function reset() {
2011-03-01 14:25:25 -03:00
$this->_to = array();
$this->_from = array();
$this->_replyTo = array();
$this->_readReceipt = array();
$this->_returnPath = array();
$this->_cc = array();
$this->_bcc = array();
$this->_messageId = true;
2011-03-01 14:25:25 -03:00
$this->_subject = '';
$this->_headers = array();
$this->_layout = 'default';
$this->_template = '';
2011-03-04 08:29:38 -03:00
$this->_viewRender = 'View';
$this->_viewVars = array();
2011-03-02 18:17:49 -03:00
$this->_textMessage = '';
$this->_htmlMessage = '';
$this->_message = '';
$this->_emailFormat = 'text';
2011-03-01 18:29:46 -03:00
$this->_transportName = 'mail';
$this->_transportClass = null;
2011-03-01 18:51:12 -03:00
$this->_attachments = array();
2011-04-09 23:34:34 -04:00
$this->_config = 'default';
2011-04-12 23:45:02 -04:00
return $this;
2011-02-27 15:25:23 -03:00
}
/**
* Encode the specified string using the current charset
*
* @param string $text String to encode
* @return string Encoded string
*/
protected function _encode($text) {
$internalEncoding = function_exists('mb_internal_encoding');
if ($internalEncoding) {
$restore = mb_internal_encoding();
mb_internal_encoding($this->charset);
}
$return = mb_encode_mimeheader($text, $this->charset, 'B');
if ($internalEncoding) {
mb_internal_encoding($restore);
}
return $return;
}
2011-03-02 18:17:49 -03:00
/**
* Wrap the message to follow the RFC 2822 - 2.1.1
*
* @param string $message Message to wrap
* @return array Wrapped message
*/
protected function _wrap($message) {
2011-03-02 18:17:49 -03:00
$message = str_replace(array("\r\n", "\r"), "\n", $message);
$lines = explode("\n", $message);
$formatted = array();
foreach ($lines as $line) {
2011-03-03 13:30:23 -03:00
if (empty($line)) {
$formatted[] = '';
continue;
}
2011-03-02 18:17:49 -03:00
if ($line[0] === '.') {
$line = '.' . $line;
}
2011-03-03 13:30:23 -03:00
if (!preg_match('/\<[a-z]/i', $line)) {
$formatted = array_merge($formatted, explode("\n", wordwrap($line, self::LINE_LENGTH_SHOULD, "\n")));
2011-03-03 13:30:23 -03:00
continue;
}
$tagOpen = false;
$tmpLine = $tag = '';
$tmpLineLength = 0;
for ($i = 0, $count = strlen($line); $i < $count; $i++) {
$char = $line[$i];
if ($tagOpen) {
$tag .= $char;
if ($char === '>') {
$tagLength = strlen($tag);
if ($tagLength + $tmpLineLength < self::LINE_LENGTH_SHOULD) {
$tmpLine .= $tag;
$tmpLineLength += $tagLength;
} else {
if ($tmpLineLength > 0) {
$formatted[] = trim($tmpLine);
$tmpLine = '';
$tmpLineLength = 0;
}
if ($tagLength > self::LINE_LENGTH_SHOULD) {
$formatted[] = $tag;
} else {
$tmpLine = $tag;
$tmpLineLength = $tagLength;
}
}
$tag = '';
$tagOpen = false;
}
continue;
}
if ($char === '<') {
$tagOpen = true;
$tag = '<';
continue;
}
if ($char === ' ' && $tmpLineLength >= self::LINE_LENGTH_SHOULD) {
$formatted[] = $tmpLine;
$tmpLineLength = 0;
continue;
}
$tmpLine .= $char;
$tmpLineLength++;
if ($tmpLineLength === self::LINE_LENGTH_SHOULD) {
$nextChar = $line[$i + 1];
if ($nextChar === ' ' || $nextChar === '<') {
$formatted[] = trim($tmpLine);
$tmpLine = '';
$tmpLineLength = 0;
if ($nextChar === ' ') {
$i++;
}
} else {
$lastSpace = strrpos($tmpLine, ' ');
if ($lastSpace === false) {
continue;
}
$formatted[] = trim(substr($tmpLine, 0, $lastSpace));
$tmpLine = substr($tmpLine, $lastSpace + 1);
$tmpLineLength = strlen($tmpLine);
}
}
}
if (!empty($tmpLine)) {
$formatted[] = $tmpLine;
}
2011-03-02 18:17:49 -03:00
}
$formatted[] = '';
return $formatted;
}
2011-03-03 23:05:30 -03:00
/**
* Create unique boundary identifier
*
* @return void
*/
protected function _createboundary() {
2011-03-03 23:05:30 -03:00
$this->_boundary = md5(uniqid(time()));
}
2011-03-04 01:27:18 -03:00
/**
* Attach files by adding file contents inside boundaries.
*
* @return void
*/
protected function _attachFiles() {
2011-03-04 01:27:18 -03:00
foreach ($this->_attachments as $filename => $file) {
$handle = fopen($file, 'rb');
$data = fread($handle, filesize($file));
$data = chunk_split(base64_encode($data)) ;
fclose($handle);
$this->_message[] = '--' . $this->_boundary;
$this->_message[] = 'Content-Type: application/octet-stream';
$this->_message[] = 'Content-Transfer-Encoding: base64';
$this->_message[] = 'Content-Disposition: attachment; filename="' . $filename . '"';
$this->_message[] = '';
$this->_message[] = $data;
$this->_message[] = '';
}
}
2011-03-03 23:05:30 -03:00
/**
* Format the message by seeing if it has attachments.
*
* @param array $message Message to format
* @return array
*/
protected function _formatMessage($message) {
2011-03-03 23:05:30 -03:00
if (!empty($this->_attachments)) {
$prefix = array('--' . $this->_boundary);
if ($this->_emailFormat === 'text') {
$prefix[] = 'Content-Type: text/plain; charset=' . $this->charset;
} elseif ($this->_emailFormat === 'html') {
$prefix[] = 'Content-Type: text/html; charset=' . $this->charset;
} elseif ($this->_emailFormat === 'both') {
$prefix[] = 'Content-Type: multipart/alternative; boundary="alt-' . $this->_boundary . '"';
}
$prefix[] = 'Content-Transfer-Encoding: 7bit';
$prefix[] = '';
$message = array_merge($prefix, $message);
}
return $message;
}
2011-03-03 23:43:55 -03:00
/**
* Render the contents using the current layout and template.
*
* @param string $content Content to render
* @return array Email ready to be sent
* @access private
*/
protected function _render($content) {
2011-03-04 08:29:38 -03:00
$viewClass = $this->_viewRender;
2011-03-03 23:43:55 -03:00
if ($viewClass !== 'View') {
2011-04-12 23:29:34 -04:00
list($plugin, $viewClass) = pluginSplit($viewClass, true);
$viewClass .= 'View';
App::uses($viewClass, $plugin . 'View');
2011-03-03 23:43:55 -03:00
}
2011-03-04 08:29:38 -03:00
$View = new $viewClass(null);
$View->layout = $this->_layout;
$View->viewVars = $this->_viewVars;
2011-03-03 23:43:55 -03:00
$msg = array();
$content = implode("\n", $content);
if ($this->_emailFormat === 'both') {
$htmlContent = $content;
if (!empty($this->_attachments)) {
$msg[] = '--' . $this->_boundary;
$msg[] = 'Content-Type: multipart/alternative; boundary="alt-' . $this->_boundary . '"';
$msg[] = '';
}
$msg[] = '--alt-' . $this->_boundary;
$msg[] = 'Content-Type: text/plain; charset=' . $this->charset;
$msg[] = 'Content-Transfer-Encoding: 7bit';
$msg[] = '';
$content = $View->element('email' . DS . 'text' . DS . $this->_template, array('content' => $content), true);
$View->layoutPath = 'email' . DS . 'text';
$content = explode("\n", $this->_textMessage = str_replace(array("\r\n", "\r"), "\n", $View->renderLayout($content)));
$msg = array_merge($msg, $content);
$msg[] = '';
$msg[] = '--alt-' . $this->_boundary;
$msg[] = 'Content-Type: text/html; charset=' . $this->charset;
$msg[] = 'Content-Transfer-Encoding: 7bit';
$msg[] = '';
$htmlContent = $View->element('email' . DS . 'html' . DS . $this->_template, array('content' => $htmlContent), true);
$View->layoutPath = 'email' . DS . 'html';
$htmlContent = explode("\n", $this->_htmlMessage = str_replace(array("\r\n", "\r"), "\n", $View->renderLayout($htmlContent)));
$msg = array_merge($msg, $htmlContent);
$msg[] = '';
$msg[] = '--alt-' . $this->_boundary . '--';
$msg[] = '';
return $msg;
}
if (!empty($this->_attachments)) {
if ($this->_emailFormat === 'html') {
$msg[] = '';
$msg[] = '--' . $this->_boundary;
$msg[] = 'Content-Type: text/html; charset=' . $this->charset;
$msg[] = 'Content-Transfer-Encoding: 7bit';
$msg[] = '';
} else {
$msg[] = '--' . $this->_boundary;
$msg[] = 'Content-Type: text/plain; charset=' . $this->charset;
$msg[] = 'Content-Transfer-Encoding: 7bit';
$msg[] = '';
}
}
$content = $View->element('email' . DS . $this->_emailFormat . DS . $this->_template, array('content' => $content), true);
$View->layoutPath = 'email' . DS . $this->_emailFormat;
$content = explode("\n", $rendered = str_replace(array("\r\n", "\r"), "\n", $View->renderLayout($content)));
if ($this->_emailFormat === 'html') {
$this->_htmlMessage = $rendered;
} else {
$this->_textMessage = $rendered;
}
return array_merge($msg, $content);
2011-03-03 23:43:55 -03:00
}
2011-02-27 15:25:23 -03:00
}