email.class.inc.php 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399
  1. <?php
  2. // Copyright (C) 2010-2012 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-2012 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. $oEncoder = new Swift_Mime_ContentEncoder_PlainContentEncoder('8bit');
  50. $this->m_oMessage->setEncoder($oEncoder);
  51. }
  52. /**
  53. * Custom serialization method
  54. * No longer use the brute force "serialize" method since
  55. * 1) It does not work with binary attachments (since they cannot be stored in a UTF-8 text field)
  56. * 2) The size tends to be quite big (sometimes ten times the size of the email)
  57. */
  58. public function SerializeV2()
  59. {
  60. return serialize($this->m_aData);
  61. }
  62. /**
  63. * Custom de-serialization method
  64. * @param string $sSerializedMessage The serialized representation of the message
  65. */
  66. static public function UnSerializeV2($sSerializedMessage)
  67. {
  68. $aData = unserialize($sSerializedMessage);
  69. $oMessage = new Email();
  70. if (array_key_exists('body', $aData))
  71. {
  72. $oMessage->SetBody($aData['body']['body'], $aData['body']['mimeType']);
  73. }
  74. if (array_key_exists('message_id', $aData))
  75. {
  76. $oMessage->SetMessageId($aData['message_id']);
  77. }
  78. if (array_key_exists('bcc', $aData))
  79. {
  80. $oMessage->SetRecipientBCC($aData['bcc']);
  81. }
  82. if (array_key_exists('cc', $aData))
  83. {
  84. $oMessage->SetRecipientCC($aData['cc']);
  85. }
  86. if (array_key_exists('from', $aData))
  87. {
  88. $oMessage->SetRecipientFrom($aData['from']['address'], $aData['from']['label']);
  89. }
  90. if (array_key_exists('reply_to', $aData))
  91. {
  92. $oMessage->SetRecipientReplyTo($aData['reply_to']);
  93. }
  94. if (array_key_exists('to', $aData))
  95. {
  96. $oMessage->SetRecipientTO($aData['to']);
  97. }
  98. if (array_key_exists('subject', $aData))
  99. {
  100. $oMessage->SetSubject($aData['subject']);
  101. }
  102. if (array_key_exists('headers', $aData))
  103. {
  104. foreach($aData['headers'] as $sKey => $sValue)
  105. {
  106. $oMessage->AddToHeader($sKey, $sValue);
  107. }
  108. }
  109. if (array_key_exists('parts', $aData))
  110. {
  111. foreach($aData['parts'] as $aPart)
  112. {
  113. $oMessage->AddPart($aPart['text'], $aPart['mimeType']);
  114. }
  115. }
  116. if (array_key_exists('attachments', $aData))
  117. {
  118. foreach($aData['attachments'] as $aAttachment)
  119. {
  120. $oMessage->AddAttachment(base64_decode($aAttachment['data']), $aAttachment['filename'], $aAttachment['mimeType']);
  121. }
  122. }
  123. return $oMessage;
  124. }
  125. protected function SendAsynchronous(&$aIssues, $oLog = null)
  126. {
  127. try
  128. {
  129. AsyncSendEmail::AddToQueue($this, $oLog);
  130. }
  131. catch(Exception $e)
  132. {
  133. $aIssues = array($e->GetMessage());
  134. return EMAIL_SEND_ERROR;
  135. }
  136. $aIssues = array();
  137. return EMAIL_SEND_PENDING;
  138. }
  139. protected function SendSynchronous(&$aIssues, $oLog = null)
  140. {
  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 'PHPMail':
  159. default:
  160. $oTransport = Swift_MailTransport::newInstance();
  161. }
  162. $oMailer = Swift_Mailer::newInstance($oTransport);
  163. $aFailedRecipients = array();
  164. $iSent = $oMailer->send($this->m_oMessage, $aFailedRecipients);
  165. if ($iSent === 0)
  166. {
  167. // Beware: it seems that $aFailedRecipients sometimes contains the recipients that actually received the message !!!
  168. IssueLog::Warning('Email sending failed: Some recipients were invalid, aFailedRecipients contains: '.implode(', ', $aFailedRecipients));
  169. $aIssues = array('Some recipients were invalid.');
  170. return EMAIL_SEND_ERROR;
  171. }
  172. else
  173. {
  174. $aIssues = array();
  175. return EMAIL_SEND_OK;
  176. }
  177. }
  178. public function Send(&$aIssues, $bForceSynchronous = false, $oLog = null)
  179. {
  180. if ($bForceSynchronous)
  181. {
  182. return $this->SendSynchronous($aIssues, $oLog);
  183. }
  184. else
  185. {
  186. $bConfigASYNC = MetaModel::GetConfig()->Get('email_asynchronous');
  187. if ($bConfigASYNC)
  188. {
  189. return $this->SendAsynchronous($aIssues, $oLog);
  190. }
  191. else
  192. {
  193. return $this->SendSynchronous($aIssues, $oLog);
  194. }
  195. }
  196. }
  197. public function AddToHeader($sKey, $sValue)
  198. {
  199. if (!array_key_exists('headers', $this->m_aData))
  200. {
  201. $this->m_aData['headers'] = array();
  202. }
  203. $this->m_aData['headers'][$sKey] = $sValue;
  204. if (strlen($sValue) > 0)
  205. {
  206. $oHeaders = $this->m_oMessage->getHeaders();
  207. switch(strtolower($sKey))
  208. {
  209. default:
  210. $oHeaders->addTextHeader($sKey, $sValue);
  211. }
  212. }
  213. }
  214. public function SetMessageId($sId)
  215. {
  216. $this->m_aData['message_id'] = $sId;
  217. // Note: Swift will add the angle brackets for you
  218. // so let's remove the angle brackets if present, for historical reasons
  219. $sId = str_replace(array('<', '>'), '', $sId);
  220. $oMsgId = $this->m_oMessage->getHeaders()->get('Message-ID');
  221. $oMsgId->SetId($sId);
  222. }
  223. public function SetReferences($sReferences)
  224. {
  225. $this->AddToHeader('References', $sReferences);
  226. }
  227. public function SetBody($sBody, $sMimeType = 'text/html')
  228. {
  229. $this->m_aData['body'] = array('body' => $sBody, 'mimeType' => $sMimeType);
  230. $this->m_oMessage->setBody($sBody, $sMimeType);
  231. }
  232. public function AddPart($sText, $sMimeType = 'text/html')
  233. {
  234. if (!array_key_exists('parts', $this->m_aData))
  235. {
  236. $this->m_aData['parts'] = array();
  237. }
  238. $this->m_aData['parts'][] = array('text' => $sText, 'mimeType' => $sMimeType);
  239. $this->m_oMessage->addPart($sText, $sMimeType);
  240. }
  241. public function AddAttachment($data, $sFileName, $sMimeType)
  242. {
  243. if (!array_key_exists('attachments', $this->m_aData))
  244. {
  245. $this->m_aData['attachments'] = array();
  246. }
  247. $this->m_aData['attachments'][] = array('data' => base64_encode($data), 'filename' => $sFileName, 'mimeType' => $sMimeType);
  248. $this->m_oMessage->attach(Swift_Attachment::newInstance($data, $sFileName, $sMimeType));
  249. }
  250. public function SetSubject($sSubject)
  251. {
  252. $this->m_aData['subject'] = $sSubject;
  253. $this->m_oMessage->setSubject($sSubject);
  254. }
  255. public function GetSubject()
  256. {
  257. return $this->m_oMessage->getSubject();
  258. }
  259. /**
  260. * Helper to transform and sanitize addresses
  261. * - get rid of empty addresses
  262. */
  263. protected function AddressStringToArray($sAddressCSVList)
  264. {
  265. $aAddresses = array();
  266. foreach(explode(',', $sAddressCSVList) as $sAddress)
  267. {
  268. $sAddress = trim($sAddress);
  269. if (strlen($sAddress) > 0)
  270. {
  271. $aAddresses[] = $sAddress;
  272. }
  273. }
  274. return $aAddresses;
  275. }
  276. public function SetRecipientTO($sAddress)
  277. {
  278. $this->m_aData['to'] = $sAddress;
  279. if (!empty($sAddress))
  280. {
  281. $aAddresses = $this->AddressStringToArray($sAddress);
  282. $this->m_oMessage->setTo($aAddresses);
  283. }
  284. }
  285. public function GetRecipientTO($bAsString = false)
  286. {
  287. $aRes = $this->m_oMessage->getTo();
  288. if ($aRes === null)
  289. {
  290. // There is no "To" header field
  291. $aRes = array();
  292. }
  293. if ($bAsString)
  294. {
  295. $aStrings = array();
  296. foreach ($aRes as $sEmail => $sName)
  297. {
  298. if (is_null($sName))
  299. {
  300. $aStrings[] = $sEmail;
  301. }
  302. else
  303. {
  304. $sName = str_replace(array('<', '>'), '', $sName);
  305. $aStrings[] = "$sName <$sEmail>";
  306. }
  307. }
  308. return implode(', ', $aStrings);
  309. }
  310. else
  311. {
  312. return $aRes;
  313. }
  314. }
  315. public function SetRecipientCC($sAddress)
  316. {
  317. $this->m_aData['cc'] = $sAddress;
  318. if (!empty($sAddress))
  319. {
  320. $aAddresses = $this->AddressStringToArray($sAddress);
  321. $this->m_oMessage->setCc($aAddresses);
  322. }
  323. }
  324. public function SetRecipientBCC($sAddress)
  325. {
  326. $this->m_aData['bcc'] = $sAddress;
  327. if (!empty($sAddress))
  328. {
  329. $aAddresses = $this->AddressStringToArray($sAddress);
  330. $this->m_oMessage->setBcc($aAddresses);
  331. }
  332. }
  333. public function SetRecipientFrom($sAddress, $sLabel = '')
  334. {
  335. $this->m_aData['from'] = array('address' => $sAddress, 'label' => $sLabel);
  336. if ($sLabel != '')
  337. {
  338. $this->m_oMessage->setFrom(array($sAddress => $sLabel));
  339. }
  340. else if (!empty($sAddress))
  341. {
  342. $this->m_oMessage->setFrom($sAddress);
  343. }
  344. }
  345. public function SetRecipientReplyTo($sAddress)
  346. {
  347. $this->m_aData['reply_to'] = $sAddress;
  348. if (!empty($sAddress))
  349. {
  350. $this->m_oMessage->setReplyTo($sAddress);
  351. }
  352. }
  353. }
  354. ?>