| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648 |
- <?php
- App::uses('CakeEmail', 'Network/Email');
- App::uses('CakeLog', 'Log');
- App::uses('MimeLib', 'Tools.Lib');
- if (!defined('BR')) {
- define('BR', '<br />');
- }
- /**
- * Convenience class for internal mailer.
- * Adds some useful features and fixes some bugs:
- * - enable easier attachment adding (and also from blob)
- * - enable embedded images in html mails
- * - extensive logging and error tracing
- * - create mails with blob attachments (embedded or attached)
- * - allow wrapLength to be adjusted
- * - Configure::read('Config.x-mailer') can modify the x-mailer
- * - basic validation supported
- * - allow priority to be set (1 to 5)
- *
- * @author Mark Scherer
- * @license MIT
- * @cakephp 2.x
- * 2012-03-30 ms
- */
- class EmailLib extends CakeEmail {
- protected $_log = null;
- protected $_debug = null;
- protected $_error = null;
- protected $_wrapLength = null;
- protected $_priority = null;
- public function __construct($config = null) {
- if ($config === null) {
- $config = 'default';
- }
- parent::__construct($config);
- $this->resetAndSet();
- }
- /**
- * Quick way to send emails to admin.
- * App::uses() + EmailLib::systemEmail()
- *
- * Note: always go out with default settings (e.g.: SMTP even if debug > 0)
- *
- * @param string $subject
- * @param string $message
- * @param string $transportConfig
- * @return boolean Success
- * 2011-10-31 ms
- */
- public static function systemEmail($subject, $message = 'System Email', $transportConfig = null) {
- $class = __CLASS__;
- $instance = new $class($transportConfig);
- $instance->from(Configure::read('Config.system_email'), Configure::read('Config.system_name'));
- $instance->to(Configure::read('Config.admin_email'), Configure::read('Config.admin_name'));
- if ($subject !== null) {
- $instance->subject($subject);
- }
- if (is_array($message)) {
- $instance->viewVars($message);
- $message = null;
- } elseif ($message === null && array_key_exists('message', $config = $instance->config())) {
- $message = $config['message'];
- }
- return $instance->send($message);
- }
- /**
- * Change the layout
- *
- * @param string $layout Layout to use (or false to use none)
- * @return resource EmailLib
- * 2011-11-02 ms
- */
- public function layout($layout = false) {
- if ($layout !== false) {
- $this->_layout = $layout;
- }
- return $this;
- }
- /**
- * Add an attachment from file
- *
- * @param string $file: absolute path
- * @param string $filename
- * @param array $fileInfo
- * @return resource EmailLib
- * 2011-11-02 ms
- */
- public function addAttachment($file, $name = null, $fileInfo = array()) {
- $fileInfo['file'] = $file;
- if (!empty($name)) {
- $fileInfo = array($name => $fileInfo);
- } else {
- $fileInfo = array($fileInfo);
- }
- return $this->addAttachments($fileInfo);
- }
- /**
- * Add an attachment as blob
- *
- * @param binary $content: blob data
- * @param string $filename to attach it
- * @param string $mimeType (leave it empty to get mimetype from $filename)
- * @param array $fileInfo
- * @return resource EmailLib
- * 2011-11-02 ms
- */
- public function addBlobAttachment($content, $name, $mimeType = null, $fileInfo = array()) {
- $fileInfo['content'] = $content;
- $fileInfo['mimetype'] = $mimeType;
- $file = array($name => $fileInfo);
- return $this->addAttachments($file);
- }
- /**
- * Add an inline attachment from file
- *
- * @param string $file: absolute path
- * @param string $filename (optional)
- * @param string $contentId (optional)
- * @param array $options
- * - mimetype
- * - contentDisposition
- * @return mixed resource $EmailLib or string $contentId
- * 2011-11-02 ms
- */
- public function addEmbeddedAttachment($file, $name = null, $contentId = null, $options = array()) {
- $path = realpath($file);
- if (empty($name)) {
- $name = basename($file);
- }
- if ($contentId === null && ($cid = $this->_isEmbeddedAttachment($path, $name))) {
- return $cid;
- }
- $options['file'] = $path;
- if (empty($options['mimetype'])) {
- $options['mimetype'] = $this->_getMime($file);
- }
- $options['contentId'] = $contentId ? $contentId : str_replace('-', '', String::uuid()) . '@' . $this->_domain;
- $file = array($name => $options);
- $res = $this->addAttachments($file);
- if ($contentId === null) {
- return $options['contentId'];
- }
- return $res;
- }
- /**
- * Add an inline attachment as blob
- *
- * @param binary $content: blob data
- * @param string $filename to attach it
- * @param string $mimeType (leave it empty to get mimetype from $filename)
- * @param string $contentId (optional)
- * @param array $options
- * - contentDisposition
- * @return mixed resource $EmailLib or string $contentId
- * 2011-11-02 ms
- */
- public function addEmbeddedBlobAttachment($content, $name, $mimeType = null, $contentId = null, $options = array()) {
- $options['content'] = $content;
- $options['mimetype'] = $mimeType;
- $options['contentId'] = $contentId ? $contentId : str_replace('-', '', String::uuid()) . '@' . $this->_domain;
- $file = array($name => $options);
- $res = $this->addAttachments($file);
- if ($contentId === null) {
- return $options['contentId'];
- }
- return $res;
- }
- /**
- * Returns if this particular file has already been attached as embedded file with this exact name
- * to prevent the same image to overwrite each other and also to only send this image once.
- * Allows multiple usage of the same embedded image (using the same cid)
- *
- * @return string cid of the found file or false if no such attachment can be found
- */
- protected function _isEmbeddedAttachment($file, $name) {
- foreach ($this->_attachments as $filename => $fileInfo) {
- if ($filename !== $name) {
- continue;
- }
- if ($fileInfo['file'] === $file) {
- return $fileInfo['contentId'];
- }
- }
- return false;
- }
- /**
- * Try to determine the mimetype by filename.
- * Uses finfo_open() if availble, otherwise guesses it via file extension.
- *
- * @param string $filename
- * @param string Mimetype
- */
- protected function _getMime($filename) {
- if (function_exists('finfo_open')) {
- $finfo = finfo_open(FILEINFO_MIME);
- $mimetype = finfo_file($finfo, $filename);
- finfo_close($finfo);
- } else {
- //TODO: improve
- $ext = pathinfo($filename, PATHINFO_EXTENSION);
- $mimetype = $this->_getMimeByExtension($ext);
- }
- return $mimetype;
- }
- /**
- * Try to find mimetype by file extension
- *
- * @param string $ext lowercase (jpg, png, pdf, ...)
- * @param string $defaultMimeType
- * @return string Mimetype (falls back to `application/octet-stream`)
- * 2012-04-17 ms
- */
- protected function _getMimeByExtension($ext, $default = 'application/octet-stream') {
- if (!isset($this->_Mime)) {
- $this->_Mime = new MimeLib();
- }
- $mime = $this->_Mime->getMimeType($ext);
- if (!$mime) {
- $mime = $default;
- }
- return $mime;
- }
- /**
- * Validate if the email has the required fields necessary to make send() work.
- * Assumes layouting (does not check on content to be present or if view/layout files are missing).
- *
- * @return boolean Success
- */
- public function validates() {
- if (!empty($this->_subject) && !empty($this->_to)) {
- return true;
- }
- return false;
- }
- /**
- * Attach inline/embedded files to the message.
- *
- * CUSTOM FIX: blob data support
- *
- * @override
- * @param string $boundary Boundary to use. If null, will default to $this->_boundary
- * @return array An array of lines to add to the message
- */
- protected function _attachInlineFiles($boundary = null) {
- if ($boundary === null) {
- $boundary = $this->_boundary;
- }
- $msg = array();
- foreach ($this->_attachments as $filename => $fileInfo) {
- if (empty($fileInfo['contentId'])) {
- continue;
- }
- if (!empty($fileInfo['content'])) {
- $data = $fileInfo['content'];
- $data = chunk_split(base64_encode($data));
- } elseif (!empty($fileInfo['file'])) {
- $data = $this->_readFile($fileInfo['file']);
- } else {
- continue;
- }
- $msg[] = '--' . $boundary;
- $msg[] = 'Content-Type: ' . $fileInfo['mimetype'];
- $msg[] = 'Content-Transfer-Encoding: base64';
- $msg[] = 'Content-ID: <' . $fileInfo['contentId'] . '>';
- $msg[] = 'Content-Disposition: inline; filename="' . $filename . '"';
- $msg[] = '';
- $msg[] = $data;
- $msg[] = '';
- }
- return $msg;
- }
- /**
- * Attach non-embedded files by adding file contents inside boundaries.
- *
- * CUSTOM FIX: blob data support
- *
- * @override
- * @param string $boundary Boundary to use. If null, will default to $this->_boundary
- * @return array An array of lines to add to the message
- */
- protected function _attachFiles($boundary = null) {
- if ($boundary === null) {
- $boundary = $this->_boundary;
- }
- $msg = array();
- foreach ($this->_attachments as $filename => $fileInfo) {
- if (!empty($fileInfo['contentId'])) {
- continue;
- }
- if (!empty($fileInfo['content'])) {
- $data = $fileInfo['content'];
- $data = chunk_split(base64_encode($data));
- } elseif (!empty($fileInfo['file'])) {
- $data = $this->_readFile($fileInfo['file']);
- } else {
- continue;
- }
- $msg[] = '--' . $boundary;
- $msg[] = 'Content-Type: ' . $fileInfo['mimetype'];
- $msg[] = 'Content-Transfer-Encoding: base64';
- if (
- !isset($fileInfo['contentDisposition']) ||
- $fileInfo['contentDisposition']
- ) {
- $msg[] = 'Content-Disposition: attachment; filename="' . $filename . '"';
- }
- $msg[] = '';
- $msg[] = $data;
- $msg[] = '';
- }
- return $msg;
- }
- /**
- * Add attachments to the email message
- *
- * CUSTOM FIX: blob data support
- *
- * Attachments can be defined in a few forms depending on how much control you need:
- *
- * Attach a single file:
- *
- * {{{
- * $email->attachments('path/to/file');
- * }}}
- *
- * Attach a file with a different filename:
- *
- * {{{
- * $email->attachments(array('custom_name.txt' => 'path/to/file.txt'));
- * }}}
- *
- * Attach a file and specify additional properties:
- *
- * {{{
- * $email->attachments(array('custom_name.png' => array(
- * 'file' => 'path/to/file',
- * 'mimetype' => 'image/png',
- * 'contentId' => 'abc123'
- * ));
- * }}}
- *
- * The `contentId` key allows you to specify an inline attachment. In your email text, you
- * can use `<img src="cid:abc123" />` to display the image inline.
- *
- * @override
- * @param mixed $attachments String with the filename or array with filenames
- * @return mixed Either the array of attachments when getting or $this when setting.
- * @throws SocketException
- */
- public function attachments($attachments = null) {
- if ($attachments === null) {
- return $this->_attachments;
- }
- $attach = array();
- foreach ((array)$attachments as $name => $fileInfo) {
- if (!is_array($fileInfo)) {
- $fileInfo = array('file' => $fileInfo);
- }
- if (empty($fileInfo['content'])) {
- if (!isset($fileInfo['file'])) {
- throw new SocketException(__d('cake_dev', 'File not specified.'));
- }
- $fileInfo['file'] = realpath($fileInfo['file']);
- if ($fileInfo['file'] === false || !file_exists($fileInfo['file'])) {
- throw new SocketException(__d('cake_dev', 'File not found: "%s"', $fileInfo['file']));
- }
- if (is_int($name)) {
- $name = basename($fileInfo['file']);
- }
- }
- if (empty($fileInfo['mimetype'])) {
- $ext = pathinfo($name, PATHINFO_EXTENSION);
- $fileInfo['mimetype'] = $this->_getMimeByExtension($ext);
- }
- $attach[$name] = $fileInfo;
- }
- $this->_attachments = $attach;
- return $this;
- }
- /**
- * Apply the config to an instance
- *
- * @overwrite
- * @param CakeEmail $obj CakeEmail
- * @param array $config
- * @return void
- * @throws ConfigureException When configuration file cannot be found, or is missing
- * the named config.
- */
- protected function _applyConfig($config) {
- if (is_string($config)) {
- if (!class_exists('EmailConfig') && !config('email')) {
- throw new ConfigureException(__d('cake_dev', '%s not found.', APP . 'Config' . DS . 'email.php'));
- }
- $configs = new EmailConfig();
- if (!isset($configs->{$config})) {
- throw new ConfigureException(__d('cake_dev', 'Unknown email configuration "%s".', $config));
- }
- $config = $configs->{$config};
- }
- $this->_config += $config;
- if (!empty($config['charset'])) {
- $this->charset = $config['charset'];
- }
- if (!empty($config['headerCharset'])) {
- $this->headerCharset = $config['headerCharset'];
- }
- if (empty($this->headerCharset)) {
- $this->headerCharset = $this->charset;
- }
- $simpleMethods = array(
- 'from', 'sender', 'to', 'replyTo', 'readReceipt', 'returnPath', 'cc', 'bcc',
- 'messageId', 'domain', 'subject', 'viewRender', 'viewVars', 'attachments',
- 'transport', 'emailFormat', 'theme', 'helpers'
- );
- foreach ($simpleMethods as $method) {
- if (isset($config[$method])) {
- $this->$method($config[$method]);
- unset($config[$method]);
- }
- }
- if (isset($config['headers'])) {
- $this->setHeaders($config['headers']);
- unset($config['headers']);
- }
- if (array_key_exists('template', $config)) {
- $layout = false;
- if (array_key_exists('layout', $config)) {
- $layout = $config['layout'];
- unset($config['layout']);
- }
- $this->template($config['template'], $layout);
- unset($config['template']);
- }
- $this->transportClass()->config($config);
- }
- /**
- * Set the body of the mail as we send it.
- * Note: the text can be an array, each element will appear as a seperate line in the message body.
- *
- * Do NOT pass a message if you use $this->set() in combination with templates
- *
- * @overwrite
- * @param string/array: message
- * @return boolean Success
- */
- public function send($message = null) {
- $this->_log = array(
- 'to' => $this->_to,
- 'from' => $this->_from,
- 'sender' => $this->_sender,
- 'replyTo' => $this->_replyTo,
- 'cc' => $this->_cc,
- 'subject' => $this->_subject,
- 'bcc' => $this->_bcc,
- 'transport' => $this->_transportName
- );
- if ($this->_priority) {
- $this->_headers['X-Priority'] = $this->_priority;
- //$this->_headers['X-MSMail-Priority'] = 'High';
- //$this->_headers['Importance'] = 'High';
- }
- try {
- $this->_debug = parent::send($message);
- } catch (Exception $e) {
- $this->_error = $e->getMessage();
- $this->_error .= ' (line '.$e->getLine().' in '.$e->getFile().')'.PHP_EOL.$e->getTraceAsString();
- if (!empty($this->_config['report'])) {
- $this->_logEmail();
- }
- return false;
- }
- if (!empty($this->_config['report'])) {
- $this->_logEmail();
- }
- return true;
- }
- /**
- * Allow modifications of the message
- *
- * @param string $text
- * @return string Text
- */
- protected function _prepMessage($text) {
- return $text;
- }
- /**
- * Returns the error if existent
- *
- * @return string
- */
- public function getError() {
- return $this->_error;
- }
- /**
- * Returns the debug content returned by send()
- *
- * @return string
- */
- public function getDebug() {
- return $this->_debug;
- }
- /**
- * Set/Get wrapLength
- *
- * @param integer $length Must not be more than CakeEmail::LINE_LENGTH_MUST
- * @return integer|CakeEmail
- */
- public function wrapLength($length = null) {
- if ($length === null) {
- return $this->_wrapLength;
- }
- $this->_wrapLength = $length;
- return $this;
- }
- /**
- * Set/Get priority
- *
- * @param integer $priority 1 (highest) to 5 (lowest)
- * @return integer|CakeEmail
- */
- public function priority($priority = null) {
- if ($priority === null) {
- return $this->_priority;
- }
- $this->_priority = $priority;
- return $this;
- }
- /**
- * Fix line length
- *
- * @overwrite
- * @param string $message Message to wrap
- * @return array Wrapped message
- */
- protected function _wrap($message, $wrapLength = CakeEmail::LINE_LENGTH_MUST) {
- if ($this->_wrapLength !== null) {
- $wrapLength = $this->_wrapLength;
- }
- return parent::_wrap($message, $wrapLength);
- }
- /**
- * Logs Email to type email
- *
- * @return void
- */
- protected function _logEmail($append = null) {
- $res = $this->_log['transport'].
- ' - '.'TO:'.implode(',', array_keys($this->_log['to'])).
- '||FROM:'.implode(',', array_keys($this->_log['from'])).
- '||REPLY:'.implode(',', array_keys($this->_log['replyTo'])).
- '||S:'.$this->_log['subject'];
- $type = 'email';
- if (!empty($this->_error)) {
- $type = 'email_error';
- $res .= '||ERROR:' . $this->_error;
- }
- if ($append) {
- $res .= '||'.$append;
- }
- CakeLog::write($type, $res);
- }
- /**
- * EmailLib::resetAndSet()
- *
- * @return void
- */
- public function resetAndSet() {
- //$this->reset();
- $this->_to = array();
- $this->_cc = array();
- $this->_bcc = array();
- $this->_messageId = true;
- $this->_subject = '';
- $this->_headers = array();
- $this->_viewVars = array();
- $this->_textMessage = '';
- $this->_htmlMessage = '';
- $this->_message = '';
- $this->_attachments = array();
- $this->_error = null;
- $this->_debug = null;
- if ($fromEmail = Configure::read('Config.system_email')) {
- $fromName = Configure::read('Config.system_name');
- } else {
- $fromEmail = Configure::read('Config.admin_email');
- $fromName = Configure::read('Config.admin_name');
- }
- $this->from($fromEmail, $fromName);
- if ($xMailer = Configure::read('Config.x-mailer')) {
- $this->addHeaders(array('X-Mailer' => $xMailer));
- }
- //$this->_errors = array();
- //$this->charset($this->charset);
- //$this->sendAs($this->sendAs);
- //$this->layout($this->_layout);
- //$this->delivery($this->deliveryMethod);
- }
- }
|