email.class.inc.php 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523
  1. <?php
  2. // Copyright (C) 2010-2016 Combodo SARL
  3. //
  4. // This file is part of iTop.
  5. //
  6. // iTop is free software; you can redistribute it and/or modify
  7. // it under the terms of the GNU Affero General Public License as published by
  8. // the Free Software Foundation, either version 3 of the License, or
  9. // (at your option) any later version.
  10. //
  11. // iTop is distributed in the hope that it will be useful,
  12. // but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  14. // GNU Affero General Public License for more details.
  15. //
  16. // You should have received a copy of the GNU Affero General Public License
  17. // along with iTop. If not, see <http://www.gnu.org/licenses/>
  18. /**
  19. * Send an email (abstraction for synchronous/asynchronous modes)
  20. *
  21. * @copyright Copyright (C) 2010-2016 Combodo SARL
  22. * @license http://opensource.org/licenses/AGPL-3.0
  23. */
  24. require_once(APPROOT.'/lib/swiftmailer/lib/swift_required.php');
  25. Swift_Preferences::getInstance()->setCharset('UTF-8');
  26. define ('EMAIL_SEND_OK', 0);
  27. define ('EMAIL_SEND_PENDING', 1);
  28. define ('EMAIL_SEND_ERROR', 2);
  29. class EMail
  30. {
  31. // Serialization formats
  32. const ORIGINAL_FORMAT = 1; // Original format, consisting in serializing the whole object, inculding the Swift Mailer's object.
  33. // Did not work with attachements since their binary representation cannot be stored as a valid UTF-8 string
  34. const FORMAT_V2 = 2; // New format, only the raw data are serialized (base64 encoded if needed)
  35. protected static $m_oConfig = null;
  36. protected $m_aData; // For storing data to serialize
  37. public function LoadConfig($sConfigFile = ITOP_DEFAULT_CONFIG_FILE)
  38. {
  39. if (is_null(self::$m_oConfig))
  40. {
  41. self::$m_oConfig = new Config($sConfigFile);
  42. }
  43. }
  44. protected $m_oMessage;
  45. public function __construct()
  46. {
  47. $this->m_aData = array();
  48. $this->m_oMessage = Swift_Message::newInstance();
  49. }
  50. /**
  51. * Custom serialization method
  52. * No longer use the brute force "serialize" method since
  53. * 1) It does not work with binary attachments (since they cannot be stored in a UTF-8 text field)
  54. * 2) The size tends to be quite big (sometimes ten times the size of the email)
  55. */
  56. public function SerializeV2()
  57. {
  58. return serialize($this->m_aData);
  59. }
  60. /**
  61. * Custom de-serialization method
  62. * @param string $sSerializedMessage The serialized representation of the message
  63. */
  64. static public function UnSerializeV2($sSerializedMessage)
  65. {
  66. $aData = unserialize($sSerializedMessage);
  67. $oMessage = new Email();
  68. if (array_key_exists('body', $aData))
  69. {
  70. $oMessage->SetBody($aData['body']['body'], $aData['body']['mimeType']);
  71. }
  72. if (array_key_exists('message_id', $aData))
  73. {
  74. $oMessage->SetMessageId($aData['message_id']);
  75. }
  76. if (array_key_exists('bcc', $aData))
  77. {
  78. $oMessage->SetRecipientBCC($aData['bcc']);
  79. }
  80. if (array_key_exists('cc', $aData))
  81. {
  82. $oMessage->SetRecipientCC($aData['cc']);
  83. }
  84. if (array_key_exists('from', $aData))
  85. {
  86. $oMessage->SetRecipientFrom($aData['from']['address'], $aData['from']['label']);
  87. }
  88. if (array_key_exists('reply_to', $aData))
  89. {
  90. $oMessage->SetRecipientReplyTo($aData['reply_to']);
  91. }
  92. if (array_key_exists('to', $aData))
  93. {
  94. $oMessage->SetRecipientTO($aData['to']);
  95. }
  96. if (array_key_exists('subject', $aData))
  97. {
  98. $oMessage->SetSubject($aData['subject']);
  99. }
  100. if (array_key_exists('headers', $aData))
  101. {
  102. foreach($aData['headers'] as $sKey => $sValue)
  103. {
  104. $oMessage->AddToHeader($sKey, $sValue);
  105. }
  106. }
  107. if (array_key_exists('parts', $aData))
  108. {
  109. foreach($aData['parts'] as $aPart)
  110. {
  111. $oMessage->AddPart($aPart['text'], $aPart['mimeType']);
  112. }
  113. }
  114. if (array_key_exists('attachments', $aData))
  115. {
  116. foreach($aData['attachments'] as $aAttachment)
  117. {
  118. $oMessage->AddAttachment(base64_decode($aAttachment['data']), $aAttachment['filename'], $aAttachment['mimeType']);
  119. }
  120. }
  121. return $oMessage;
  122. }
  123. protected function SendAsynchronous(&$aIssues, $oLog = null)
  124. {
  125. try
  126. {
  127. AsyncSendEmail::AddToQueue($this, $oLog);
  128. }
  129. catch(Exception $e)
  130. {
  131. $aIssues = array($e->GetMessage());
  132. return EMAIL_SEND_ERROR;
  133. }
  134. $aIssues = array();
  135. return EMAIL_SEND_PENDING;
  136. }
  137. protected function SendSynchronous(&$aIssues, $oLog = null)
  138. {
  139. // If the body of the message is in HTML, embed all images based on attachments
  140. $this->EmbedInlineImages();
  141. $this->LoadConfig();
  142. $sTransport = self::$m_oConfig->Get('email_transport');
  143. switch ($sTransport)
  144. {
  145. case 'SMTP':
  146. $sHost = self::$m_oConfig->Get('email_transport_smtp.host');
  147. $sPort = self::$m_oConfig->Get('email_transport_smtp.port');
  148. $sEncryption = self::$m_oConfig->Get('email_transport_smtp.encryption');
  149. $sUserName = self::$m_oConfig->Get('email_transport_smtp.username');
  150. $sPassword = self::$m_oConfig->Get('email_transport_smtp.password');
  151. $oTransport = Swift_SmtpTransport::newInstance($sHost, $sPort, $sEncryption);
  152. if (strlen($sUserName) > 0)
  153. {
  154. $oTransport->setUsername($sUserName);
  155. $oTransport->setPassword($sPassword);
  156. }
  157. break;
  158. case 'Null':
  159. $oTransport = Swift_NullTransport::newInstance();
  160. break;
  161. case 'LogFile':
  162. $oTransport = Swift_LogFileTransport::newInstance();
  163. $oTransport->setLogFile(APPROOT.'log/mail.log');
  164. break;
  165. case 'PHPMail':
  166. default:
  167. $oTransport = Swift_MailTransport::newInstance();
  168. }
  169. $oMailer = Swift_Mailer::newInstance($oTransport);
  170. $aFailedRecipients = array();
  171. $this->m_oMessage->setMaxLineLength(0);
  172. $iSent = $oMailer->send($this->m_oMessage, $aFailedRecipients);
  173. if ($iSent === 0)
  174. {
  175. // Beware: it seems that $aFailedRecipients sometimes contains the recipients that actually received the message !!!
  176. IssueLog::Warning('Email sending failed: Some recipients were invalid, aFailedRecipients contains: '.implode(', ', $aFailedRecipients));
  177. $aIssues = array('Some recipients were invalid.');
  178. return EMAIL_SEND_ERROR;
  179. }
  180. else
  181. {
  182. $aIssues = array();
  183. return EMAIL_SEND_OK;
  184. }
  185. }
  186. /**
  187. * Reprocess the body of the message (if it is an HTML message)
  188. * to replace the URL of images based on attachments by a link
  189. * to an embedded image (i.e. cid:....)
  190. */
  191. protected function EmbedInlineImages()
  192. {
  193. if ($this->m_aData['body']['mimeType'] == 'text/html')
  194. {
  195. $oDOMDoc = new DOMDocument();
  196. $oDOMDoc->preserveWhitespace = true;
  197. @$oDOMDoc->loadHTML('<?xml encoding="UTF-8"?>'.$this->m_aData['body']['body']); // For loading HTML chunks where the character set is not specified
  198. $oXPath = new DOMXPath($oDOMDoc);
  199. $sXPath = "//img[@data-img-id]";
  200. $oImagesList = $oXPath->query($sXPath);
  201. if ($oImagesList->length != 0)
  202. {
  203. foreach($oImagesList as $oImg)
  204. {
  205. $iAttId = $oImg->getAttribute('data-img-id');
  206. $oAttachment = MetaModel::GetObject('InlineImage', $iAttId, false, true /* Allow All Data */);
  207. if ($oAttachment)
  208. {
  209. $oDoc = $oAttachment->Get('contents');
  210. $oSwiftImage = new Swift_Image($oDoc->GetData(), $oDoc->GetFileName(), $oDoc->GetMimeType());
  211. $sCid = $this->m_oMessage->embed($oSwiftImage);
  212. $oImg->setAttribute('src', $sCid);
  213. }
  214. }
  215. }
  216. $sHtmlBody = $oDOMDoc->saveHTML();
  217. $this->m_oMessage->setBody($sHtmlBody, 'text/html', 'UTF-8');
  218. }
  219. }
  220. public function Send(&$aIssues, $bForceSynchronous = false, $oLog = null)
  221. {
  222. if ($bForceSynchronous)
  223. {
  224. return $this->SendSynchronous($aIssues, $oLog);
  225. }
  226. else
  227. {
  228. $bConfigASYNC = MetaModel::GetConfig()->Get('email_asynchronous');
  229. if ($bConfigASYNC)
  230. {
  231. return $this->SendAsynchronous($aIssues, $oLog);
  232. }
  233. else
  234. {
  235. return $this->SendSynchronous($aIssues, $oLog);
  236. }
  237. }
  238. }
  239. public function AddToHeader($sKey, $sValue)
  240. {
  241. if (!array_key_exists('headers', $this->m_aData))
  242. {
  243. $this->m_aData['headers'] = array();
  244. }
  245. $this->m_aData['headers'][$sKey] = $sValue;
  246. if (strlen($sValue) > 0)
  247. {
  248. $oHeaders = $this->m_oMessage->getHeaders();
  249. switch(strtolower($sKey))
  250. {
  251. default:
  252. $oHeaders->addTextHeader($sKey, $sValue);
  253. }
  254. }
  255. }
  256. public function SetMessageId($sId)
  257. {
  258. $this->m_aData['message_id'] = $sId;
  259. // Note: Swift will add the angle brackets for you
  260. // so let's remove the angle brackets if present, for historical reasons
  261. $sId = str_replace(array('<', '>'), '', $sId);
  262. $oMsgId = $this->m_oMessage->getHeaders()->get('Message-ID');
  263. $oMsgId->SetId($sId);
  264. }
  265. public function SetReferences($sReferences)
  266. {
  267. $this->AddToHeader('References', $sReferences);
  268. }
  269. public function SetBody($sBody, $sMimeType = 'text/html', $sCustomStyles = null)
  270. {
  271. if (($sMimeType === 'text/html') && ($sCustomStyles !== null))
  272. {
  273. require_once(APPROOT.'lib/emogrifier/Classes/Emogrifier.php');
  274. $emogrifier = new \Pelago\Emogrifier($sBody, $sCustomStyles);
  275. $sBody = $emogrifier->emogrify(); // Adds html/body tags if not already present
  276. }
  277. $this->m_aData['body'] = array('body' => $sBody, 'mimeType' => $sMimeType);
  278. $this->m_oMessage->setBody($sBody, $sMimeType);
  279. }
  280. public function AddPart($sText, $sMimeType = 'text/html')
  281. {
  282. if (!array_key_exists('parts', $this->m_aData))
  283. {
  284. $this->m_aData['parts'] = array();
  285. }
  286. $this->m_aData['parts'][] = array('text' => $sText, 'mimeType' => $sMimeType);
  287. $this->m_oMessage->addPart($sText, $sMimeType);
  288. }
  289. public function AddAttachment($data, $sFileName, $sMimeType)
  290. {
  291. if (!array_key_exists('attachments', $this->m_aData))
  292. {
  293. $this->m_aData['attachments'] = array();
  294. }
  295. $this->m_aData['attachments'][] = array('data' => base64_encode($data), 'filename' => $sFileName, 'mimeType' => $sMimeType);
  296. $this->m_oMessage->attach(Swift_Attachment::newInstance($data, $sFileName, $sMimeType));
  297. }
  298. public function SetSubject($sSubject)
  299. {
  300. $this->m_aData['subject'] = $sSubject;
  301. $this->m_oMessage->setSubject($sSubject);
  302. }
  303. public function GetSubject()
  304. {
  305. return $this->m_oMessage->getSubject();
  306. }
  307. /**
  308. * Helper to transform and sanitize addresses
  309. * - get rid of empty addresses
  310. */
  311. protected function AddressStringToArray($sAddressCSVList)
  312. {
  313. $aAddresses = array();
  314. foreach(explode(',', $sAddressCSVList) as $sAddress)
  315. {
  316. $sAddress = trim($sAddress);
  317. if (strlen($sAddress) > 0)
  318. {
  319. $aAddresses[] = $sAddress;
  320. }
  321. }
  322. return $aAddresses;
  323. }
  324. public function SetRecipientTO($sAddress)
  325. {
  326. $this->m_aData['to'] = $sAddress;
  327. if (!empty($sAddress))
  328. {
  329. $aAddresses = $this->AddressStringToArray($sAddress);
  330. $this->m_oMessage->setTo($aAddresses);
  331. }
  332. }
  333. public function GetRecipientTO($bAsString = false)
  334. {
  335. $aRes = $this->m_oMessage->getTo();
  336. if ($aRes === null)
  337. {
  338. // There is no "To" header field
  339. $aRes = array();
  340. }
  341. if ($bAsString)
  342. {
  343. $aStrings = array();
  344. foreach ($aRes as $sEmail => $sName)
  345. {
  346. if (is_null($sName))
  347. {
  348. $aStrings[] = $sEmail;
  349. }
  350. else
  351. {
  352. $sName = str_replace(array('<', '>'), '', $sName);
  353. $aStrings[] = "$sName <$sEmail>";
  354. }
  355. }
  356. return implode(', ', $aStrings);
  357. }
  358. else
  359. {
  360. return $aRes;
  361. }
  362. }
  363. public function SetRecipientCC($sAddress)
  364. {
  365. $this->m_aData['cc'] = $sAddress;
  366. if (!empty($sAddress))
  367. {
  368. $aAddresses = $this->AddressStringToArray($sAddress);
  369. $this->m_oMessage->setCc($aAddresses);
  370. }
  371. }
  372. public function SetRecipientBCC($sAddress)
  373. {
  374. $this->m_aData['bcc'] = $sAddress;
  375. if (!empty($sAddress))
  376. {
  377. $aAddresses = $this->AddressStringToArray($sAddress);
  378. $this->m_oMessage->setBcc($aAddresses);
  379. }
  380. }
  381. public function SetRecipientFrom($sAddress, $sLabel = '')
  382. {
  383. $this->m_aData['from'] = array('address' => $sAddress, 'label' => $sLabel);
  384. if ($sLabel != '')
  385. {
  386. $this->m_oMessage->setFrom(array($sAddress => $sLabel));
  387. }
  388. else if (!empty($sAddress))
  389. {
  390. $this->m_oMessage->setFrom($sAddress);
  391. }
  392. }
  393. public function SetRecipientReplyTo($sAddress)
  394. {
  395. $this->m_aData['reply_to'] = $sAddress;
  396. if (!empty($sAddress))
  397. {
  398. $this->m_oMessage->setReplyTo($sAddress);
  399. }
  400. }
  401. }
  402. /////////////////////////////////////////////////////////////////////////////////////
  403. /**
  404. * Extension to SwiftMailer: "debug" transport that pretends messages have been sent,
  405. * but just log them to a file.
  406. *
  407. * @package Swift
  408. * @author Denis Flaven
  409. */
  410. class Swift_Transport_LogFileTransport extends Swift_Transport_NullTransport
  411. {
  412. protected $sLogFile;
  413. /**
  414. * Sends the given message.
  415. *
  416. * @param Swift_Mime_Message $message
  417. * @param string[] $failedRecipients An array of failures by-reference
  418. *
  419. * @return int The number of sent emails
  420. */
  421. public function send(Swift_Mime_Message $message, &$failedRecipients = null)
  422. {
  423. $hFile = @fopen($this->sLogFile, 'a');
  424. if ($hFile)
  425. {
  426. $sTxt = "================== ".date('Y-m-d H:i:s')." ==================\n";
  427. $sTxt .= $message->toString()."\n";
  428. @fwrite($hFile, $sTxt);
  429. @fclose($hFile);
  430. }
  431. return parent::send($message, $failedRecipients);
  432. }
  433. public function setLogFile($sFilename)
  434. {
  435. $this->sLogFile = $sFilename;
  436. }
  437. }
  438. /**
  439. * Pretends messages have been sent, but just log them to a file.
  440. *
  441. * @package Swift
  442. * @author Denis Flaven
  443. */
  444. class Swift_LogFileTransport extends Swift_Transport_LogFileTransport
  445. {
  446. /**
  447. * Create a new LogFileTransport.
  448. */
  449. public function __construct()
  450. {
  451. call_user_func_array(
  452. array($this, 'Swift_Transport_LogFileTransport::__construct'),
  453. Swift_DependencyContainer::getInstance()
  454. ->createDependenciesFor('transport.null')
  455. );
  456. }
  457. /**
  458. * Create a new LogFileTransport instance.
  459. *
  460. * @return Swift_LogFileTransport
  461. */
  462. public static function newInstance()
  463. {
  464. return new self();
  465. }
  466. }