2
0

MailTest.php 36 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993
  1. <?php
  2. /**
  3. * Zend Framework
  4. *
  5. * LICENSE
  6. *
  7. * This source file is subject to the new BSD license that is bundled
  8. * with this package in the file LICENSE.txt.
  9. * It is also available through the world-wide-web at this URL:
  10. * http://framework.zend.com/license/new-bsd
  11. * If you did not receive a copy of the license and are unable to
  12. * obtain it through the world-wide-web, please send an email
  13. * to license@zend.com so we can send you a copy immediately.
  14. *
  15. * @category Zend
  16. * @package Zend_Mail
  17. * @subpackage UnitTests
  18. * @copyright Copyright (c) 2005-2010 Zend Technologies USA Inc. (http://www.zend.com)
  19. * @license http://framework.zend.com/license/new-bsd New BSD License
  20. * @version $Id $
  21. */
  22. /**
  23. * Test helper
  24. */
  25. require_once dirname(dirname(__FILE__)) . DIRECTORY_SEPARATOR . '..' . DIRECTORY_SEPARATOR . 'TestHelper.php';
  26. /**
  27. * Zend_Mail
  28. */
  29. require_once 'Zend/Mail.php';
  30. /**
  31. * Zend_Mail_Transport_Abstract
  32. */
  33. require_once 'Zend/Mail/Transport/Abstract.php';
  34. /**
  35. * Zend_Mail_Transport_Sendmail
  36. */
  37. require_once 'Zend/Mail/Transport/Sendmail.php';
  38. /**
  39. * Zend_Mail_Transport_Smtp
  40. */
  41. require_once 'Zend/Mail/Transport/Smtp.php';
  42. /**
  43. * Zend_Date
  44. */
  45. require_once 'Zend/Date.php';
  46. /**
  47. * Mock mail transport class for testing purposes
  48. *
  49. * @category Zend
  50. * @package Zend_Mail
  51. * @subpackage UnitTests
  52. * @copyright Copyright (c) 2005-2010 Zend Technologies USA Inc. (http://www.zend.com)
  53. * @license http://framework.zend.com/license/new-bsd New BSD License
  54. */
  55. class Zend_Mail_Transport_Mock extends Zend_Mail_Transport_Abstract
  56. {
  57. /**
  58. * @var Zend_Mail
  59. */
  60. public $mail = null;
  61. public $returnPath = null;
  62. public $subject = null;
  63. public $from = null;
  64. public $headers = null;
  65. public $called = false;
  66. public function _sendMail()
  67. {
  68. $this->mail = $this->_mail;
  69. $this->subject = $this->_mail->getSubject();
  70. $this->from = $this->_mail->getFrom();
  71. $this->returnPath = $this->_mail->getReturnPath();
  72. $this->headers = $this->_headers;
  73. $this->called = true;
  74. }
  75. }
  76. /**
  77. * Mock mail transport class for testing Sendmail transport
  78. *
  79. * @category Zend
  80. * @package Zend_Mail
  81. * @subpackage UnitTests
  82. * @copyright Copyright (c) 2005-2010 Zend Technologies USA Inc. (http://www.zend.com)
  83. * @license http://framework.zend.com/license/new-bsd New BSD License
  84. */
  85. class Zend_Mail_Transport_Sendmail_Mock extends Zend_Mail_Transport_Sendmail
  86. {
  87. /**
  88. * @var Zend_Mail
  89. */
  90. public $mail = null;
  91. public $from = null;
  92. public $subject = null;
  93. public $called = false;
  94. public function _sendMail()
  95. {
  96. $this->mail = $this->_mail;
  97. $this->from = $this->_mail->getFrom();
  98. $this->subject = $this->_mail->getSubject();
  99. $this->called = true;
  100. }
  101. }
  102. /**
  103. * @category Zend
  104. * @package Zend_Mail
  105. * @subpackage UnitTests
  106. * @copyright Copyright (c) 2005-2010 Zend Technologies USA Inc. (http://www.zend.com)
  107. * @license http://framework.zend.com/license/new-bsd New BSD License
  108. * @group Zend_Mail
  109. */
  110. class Zend_Mail_MailTest extends PHPUnit_Framework_TestCase
  111. {
  112. public function tearDown() {
  113. Zend_Mail::clearDefaultFrom();
  114. Zend_Mail::clearDefaultReplyTo();
  115. }
  116. /**
  117. * Test case for a simple email text message with
  118. * multiple recipients.
  119. *
  120. */
  121. public function testOnlyText()
  122. {
  123. $mail = new Zend_Mail();
  124. $res = $mail->setBodyText('This is a test.');
  125. $mail->setFrom('testmail@example.com', 'test Mail User');
  126. $mail->setSubject('My Subject');
  127. $mail->addTo('recipient1@example.com');
  128. $mail->addTo('recipient2@example.com');
  129. $mail->addBcc('recipient1_bcc@example.com');
  130. $mail->addBcc('recipient2_bcc@example.com');
  131. $mail->addCc('recipient1_cc@example.com', 'Example no. 1 for cc');
  132. $mail->addCc('recipient2_cc@example.com', 'Example no. 2 for cc');
  133. $mock = new Zend_Mail_Transport_Mock();
  134. $mail->send($mock);
  135. $this->assertTrue($mock->called);
  136. $this->assertEquals('My Subject', $mock->subject);
  137. $this->assertEquals('testmail@example.com', $mock->from);
  138. $this->assertContains('recipient1@example.com', $mock->recipients);
  139. $this->assertContains('recipient2@example.com', $mock->recipients);
  140. $this->assertContains('recipient1_bcc@example.com', $mock->recipients);
  141. $this->assertContains('recipient2_bcc@example.com', $mock->recipients);
  142. $this->assertContains('recipient1_cc@example.com', $mock->recipients);
  143. $this->assertContains('recipient2_cc@example.com', $mock->recipients);
  144. $this->assertContains('This is a test.', $mock->body);
  145. $this->assertContains('Content-Transfer-Encoding: quoted-printable', $mock->header);
  146. $this->assertContains('Content-Type: text/plain', $mock->header);
  147. $this->assertContains('From: test Mail User <testmail@example.com>', $mock->header);
  148. $this->assertContains('Subject: My Subject', $mock->header);
  149. $this->assertContains('To: recipient1@example.com', $mock->header);
  150. $this->assertContains('Cc: Example no. 1 for cc <recipient1_cc@example.com>', $mock->header);
  151. }
  152. /**
  153. * Test sending in arrays of recipients
  154. */
  155. public function testArrayRecipients()
  156. {
  157. $mail = new Zend_Mail();
  158. $res = $mail->setBodyText('Test #2');
  159. $mail->setFrom('eli@example.com', 'test Mail User');
  160. $mail->setSubject('Subject #2');
  161. $mail->addTo(array('heather@example.com', 'Ramsey White' => 'ramsey@example.com'));
  162. $mail->addCc(array('keith@example.com', 'Cal Evans' => 'cal@example.com'));
  163. $mail->addBcc(array('ralph@example.com', 'matthew@example.com'));
  164. $mock = new Zend_Mail_Transport_Mock();
  165. $mail->send($mock);
  166. $this->assertTrue($mock->called);
  167. $this->assertEquals('eli@example.com', $mock->from);
  168. $this->assertContains('heather@example.com', $mock->recipients);
  169. $this->assertContains('ramsey@example.com', $mock->recipients);
  170. $this->assertContains('ralph@example.com', $mock->recipients);
  171. $this->assertContains('matthew@example.com', $mock->recipients);
  172. $this->assertContains('keith@example.com', $mock->recipients);
  173. $this->assertContains('cal@example.com', $mock->recipients);
  174. $this->assertContains('Test #2', $mock->body);
  175. $this->assertContains('From: test Mail User <eli@example.com>', $mock->header);
  176. $this->assertContains('Subject: Subject #2', $mock->header);
  177. $this->assertContains('To: heather@example.com', $mock->header);
  178. $this->assertContains('Ramsey White <ramsey@example.com>', $mock->header);
  179. $this->assertContains('Cal Evans <cal@example.com>', $mock->header);
  180. }
  181. /**
  182. * @group ZF-8503 Test recipients Header format.
  183. */
  184. public function testRecipientsHeaderFormat()
  185. {
  186. $mail = new Zend_Mail();
  187. $res = $mail->setBodyText('Test recipients Header format.');
  188. $mail->setFrom('yoshida@example.com', 'test Mail User');
  189. $mail->setSubject('Test recipients Header format.');
  190. $mail->addTo('address_to1@example.com', 'name_to@example.com');
  191. $mail->addTo('address_to2@example.com', 'noinclude comma nor at mark');
  192. $mail->addCc('address_cc@example.com', 'include, name_cc');
  193. $mock = new Zend_Mail_Transport_Mock();
  194. $mail->send($mock);
  195. $this->assertTrue($mock->called);
  196. $this->assertEquals('yoshida@example.com', $mock->from);
  197. $this->assertContains('Test recipients Header format.', $mock->body);
  198. $this->assertContains('To: "name_to@example.com" <address_to1@example.com>', $mock->header);
  199. $this->assertContains('noinclude comma nor at mark <address_to2@example.com>', $mock->header);
  200. $this->assertContains('Cc: "include, name_cc" <address_cc@example.com>', $mock->header);
  201. }
  202. /**
  203. * Check if Header Fields are encoded correctly and if
  204. * header injection is prevented.
  205. */
  206. public function testHeaderEncoding()
  207. {
  208. $mail = new Zend_Mail("UTF-8");
  209. $mail->setBodyText('My Nice Test Text');
  210. // try header injection:
  211. $mail->addTo("testmail@example.com\nCc:foobar@example.com");
  212. $mail->addHeader('X-MyTest', "Test\nCc:foobar2@example.com", true);
  213. // try special Chars in Header Fields:
  214. $mail->setFrom('mymail@example.com', "\xC6\x98\xC6\x90\xC3\xA4\xC4\xB8");
  215. $mail->addTo('testmail2@example.com', "\xC4\xA7\xC4\xAF\xC7\xAB");
  216. $mail->addCc('testmail3@example.com', "\xC7\xB6\xC7\xB7");
  217. $mail->setSubject("\xC7\xB1\xC7\xAE");
  218. $mail->addHeader('X-MyTest', "Test-\xC7\xB1", true);
  219. $mock = new Zend_Mail_Transport_Mock();
  220. $mail->send($mock);
  221. $this->assertTrue($mock->called);
  222. $this->assertContains(
  223. 'From: =?UTF-8?Q?=C6=98=C6=90=C3=A4=C4=B8?=',
  224. $mock->header,
  225. "From: Header was encoded unexpectedly."
  226. );
  227. $this->assertContains(
  228. "Cc:foobar@example.com",
  229. $mock->header
  230. );
  231. $this->assertNotContains(
  232. "\nCc:foobar@example.com",
  233. $mock->header,
  234. "Injection into From: header is possible."
  235. );
  236. $this->assertContains(
  237. '=?UTF-8?Q?=C4=A7=C4=AF=C7=AB?= <testmail2@example.com>',
  238. $mock->header
  239. );
  240. $this->assertContains(
  241. 'Cc: =?UTF-8?Q?=C7=B6=C7=B7?= <testmail3@example.com>',
  242. $mock->header
  243. );
  244. $this->assertContains(
  245. 'Subject: =?UTF-8?Q?=C7=B1=C7=AE?=',
  246. $mock->header
  247. );
  248. $this->assertContains(
  249. 'X-MyTest:',
  250. $mock->header
  251. );
  252. $this->assertNotContains(
  253. "\nCc:foobar2@example.com",
  254. $mock->header
  255. );
  256. $this->assertContains(
  257. '=?UTF-8?Q?Test-=C7=B1?=',
  258. $mock->header
  259. );
  260. }
  261. /**
  262. * @group ZF-7799
  263. */
  264. public function testHeaderSendMailTransportHaveNoRightTrim()
  265. {
  266. $mail = new Zend_Mail("UTF-8");
  267. $mail->setBodyText('My Nice Test Text');
  268. $mail->addTo("foobar@example.com");
  269. $mail->setSubject("hello world!");
  270. $transportMock = new Zend_Mail_Transport_Sendmail_Mock();
  271. $mail->send($transportMock);
  272. $this->assertEquals($transportMock->header, rtrim($transportMock->header));
  273. }
  274. /**
  275. * Check if Header Fields are stripped accordingly in sendmail transport;
  276. * also check for header injection
  277. * @todo Determine why this fails in Windows (testmail3@example.com example)
  278. */
  279. public function testHeaderEncoding2()
  280. {
  281. $mail = new Zend_Mail("UTF-8");
  282. $mail->setBodyText('My Nice Test Text');
  283. // try header injection:
  284. $mail->addTo("testmail@example.com\nCc:foobar@example.com");
  285. $mail->addHeader('X-MyTest', "Test\nCc:foobar2@example.com", true);
  286. // try special Chars in Header Fields:
  287. $mail->setFrom('mymail@example.com', "\xC6\x98\xC6\x90\xC3\xA4\xC4\xB8");
  288. $mail->addTo('testmail2@example.com', "\xC4\xA7\xC4\xAF\xC7\xAB");
  289. $mail->addCc('testmail3@example.com', "\xC7\xB6\xC7\xB7");
  290. $mail->setSubject("\xC7\xB1\xC7\xAE");
  291. $mail->addHeader('X-MyTest', "Test-\xC7\xB1", true);
  292. $mock = new Zend_Mail_Transport_Sendmail_Mock();
  293. $mail->send($mock);
  294. $this->assertTrue($mock->called);
  295. $this->assertContains(
  296. 'From: =?UTF-8?Q?=C6=98=C6=90=C3=A4=C4=B8?=',
  297. $mock->header,
  298. "From: Header was encoded unexpectedly."
  299. );
  300. $this->assertNotContains(
  301. "\nCc:foobar@example.com",
  302. $mock->header,
  303. "Injection into From: header is possible."
  304. );
  305. // To is done by mail() not in headers
  306. $this->assertNotContains(
  307. 'To: =?UTF-8?Q?=C4=A7=C4=AF=C7=AB?= <testmail2@example.com>',
  308. $mock->header
  309. );
  310. $this->assertContains(
  311. 'Cc: =?UTF-8?Q?=C7=B6=C7=B7?= <testmail3@example.com>',
  312. $mock->header
  313. );
  314. // Subject is done by mail() not in headers
  315. $this->assertNotContains(
  316. 'Subject: =?UTF-8?Q?=C7=B1=C7=AE?=',
  317. $mock->header
  318. );
  319. $this->assertContains(
  320. 'X-MyTest:',
  321. $mock->header
  322. );
  323. $this->assertNotContains(
  324. "\nCc:foobar2@example.com",
  325. $mock->header
  326. );
  327. $this->assertContains(
  328. '=?UTF-8?Q?Test-=C7=B1?=',
  329. $mock->header
  330. );
  331. }
  332. /**
  333. * Check if Mails with HTML and Text Body are generated correctly.
  334. *
  335. */
  336. public function testMultipartAlternative()
  337. {
  338. $mail = new Zend_Mail();
  339. $mail->setBodyText('My Nice Test Text');
  340. $mail->setBodyHtml('My Nice <b>Test</b> Text');
  341. $mail->addTo('testmail@example.com', 'Test Recipient');
  342. $mail->setFrom('mymail@example.com', 'Test Sender');
  343. $mail->setSubject('Test: Alternate Mail with Zend_Mail');
  344. $mock = new Zend_Mail_Transport_Mock();
  345. $mail->send($mock);
  346. // check headers
  347. $this->assertTrue($mock->called);
  348. $this->assertContains('multipart/alternative', $mock->header);
  349. $boundary = $mock->boundary;
  350. $this->assertContains('boundary="' . $boundary . '"', $mock->header);
  351. $this->assertContains('MIME-Version: 1.0', $mock->header);
  352. // check body
  353. // search for first boundary
  354. $p1 = strpos($mock->body, "--$boundary\n");
  355. $this->assertNotNull($p1, $boundary . ': ' . $mock->body);
  356. // cut out first (Text) part
  357. $start1 = $p1 + 3 + strlen($boundary);
  358. $p2 = strpos($mock->body, "--$boundary\n", $start1);
  359. $this->assertNotNull($p2);
  360. $partBody1 = substr($mock->body, $start1, ($p2 - $start1));
  361. $this->assertContains('Content-Type: text/plain', $partBody1);
  362. $this->assertContains('My Nice Test Text', $partBody1);
  363. // check second (HTML) part
  364. // search for end boundary
  365. $start2 = $p2 + 3 + strlen($boundary);
  366. $p3 = strpos($mock->body, "--$boundary--");
  367. $this->assertNotNull($p3);
  368. $partBody2 = substr($mock->body, $start2, ($p3 - $start2));
  369. $this->assertContains('Content-Type: text/html', $partBody2);
  370. $this->assertContains('My Nice <b>Test</b> Text', $partBody2);
  371. }
  372. /**
  373. * check if attachment handling works
  374. *
  375. */
  376. public function testAttachment()
  377. {
  378. $mail = new Zend_Mail();
  379. $mail->setBodyText('My Nice Test Text');
  380. $mail->addTo('testmail@example.com', 'Test Recipient');
  381. $mail->setFrom('mymail@example.com', 'Test Sender');
  382. $mail->setSubject('Test: Attachment Test with Zend_Mail');
  383. $at = $mail->createAttachment('abcdefghijklmnopqrstuvexyz');
  384. $at->type = 'image/gif';
  385. $at->id = 12;
  386. $at->filename = 'test.gif';
  387. $mock = new Zend_Mail_Transport_Mock();
  388. $mail->send($mock);
  389. // now check what was generated by Zend_Mail.
  390. // first the mail headers:
  391. $this->assertContains('Content-Type: multipart/mixed', $mock->header, $mock->header);
  392. $boundary = $mock->boundary;
  393. $this->assertContains('boundary="' . $boundary . '"', $mock->header);
  394. $this->assertContains('MIME-Version: 1.0', $mock->header);
  395. // check body
  396. // search for first boundary
  397. $p1 = strpos($mock->body, "--$boundary\n");
  398. $this->assertNotNull($p1);
  399. // cut out first (Text) part
  400. $start1 = $p1 + 3 + strlen($boundary);
  401. $p2 = strpos($mock->body, "--$boundary\n", $start1);
  402. $this->assertNotNull($p2);
  403. $partBody1 = substr($mock->body, $start1, ($p2 - $start1));
  404. $this->assertContains('Content-Type: text/plain', $partBody1);
  405. $this->assertContains('My Nice Test Text', $partBody1);
  406. // check second (HTML) part
  407. // search for end boundary
  408. $start2 = $p2 + 3 + strlen($boundary);
  409. $p3 = strpos($mock->body, "--$boundary--");
  410. $this->assertNotNull($p3);
  411. $partBody2 = substr($mock->body, $start2, ($p3 - $start2));
  412. $this->assertContains('Content-Type: image/gif', $partBody2);
  413. $this->assertContains('Content-Transfer-Encoding: base64', $partBody2);
  414. $this->assertContains('Content-ID: <12>', $partBody2);
  415. }
  416. /**
  417. * Check if Mails with HTML and Text Body are generated correctly.
  418. *
  419. */
  420. public function testMultipartAlternativePlusAttachment()
  421. {
  422. $mail = new Zend_Mail();
  423. $mail->setBodyText('My Nice Test Text');
  424. $mail->setBodyHtml('My Nice <b>Test</b> Text');
  425. $mail->addTo('testmail@example.com', 'Test Recipient');
  426. $mail->setFrom('mymail@example.com', 'Test Sender');
  427. $mail->setSubject('Test: Alternate Mail with Zend_Mail');
  428. $at = $mail->createAttachment('abcdefghijklmnopqrstuvexyz');
  429. $at->type = 'image/gif';
  430. $at->id = 12;
  431. $at->filename = 'test.gif';
  432. $mock = new Zend_Mail_Transport_Mock();
  433. $mail->send($mock);
  434. // check headers
  435. $this->assertTrue($mock->called);
  436. $this->assertContains('multipart/mixed', $mock->header);
  437. $boundary = $mock->boundary;
  438. $this->assertContains('boundary="' . $boundary . '"', $mock->header);
  439. $this->assertContains('MIME-Version: 1.0', $mock->header);
  440. // check body
  441. // search for first boundary
  442. $p1 = strpos($mock->body, "--$boundary\n");
  443. $this->assertNotNull($p1);
  444. // cut out first (multipart/alternative) part
  445. $start1 = $p1 + 3 + strlen($boundary);
  446. $p2 = strpos($mock->body, "--$boundary\n", $start1);
  447. $this->assertNotNull($p2);
  448. $partBody1 = substr($mock->body, $start1, ($p2 - $start1));
  449. $this->assertContains('Content-Type: multipart/alternative', $partBody1);
  450. $this->assertContains('Content-Type: text/plain', $partBody1);
  451. $this->assertContains('Content-Type: text/html', $partBody1);
  452. $this->assertContains('My Nice Test Text', $partBody1);
  453. $this->assertContains('My Nice <b>Test</b> Text', $partBody1);
  454. // check second (image) part
  455. // search for end boundary
  456. $start2 = $p2 + 3 + strlen($boundary);
  457. $p3 = strpos($mock->body, "--$boundary--");
  458. $this->assertNotNull($p3);
  459. $partBody2 = substr($mock->body, $start2, ($p3 - $start2));
  460. $this->assertContains('Content-Type: image/gif', $partBody2);
  461. $this->assertContains('Content-Transfer-Encoding: base64', $partBody2);
  462. $this->assertContains('Content-ID: <12>', $partBody2);
  463. }
  464. public function testReturnPath()
  465. {
  466. $mail = new Zend_Mail();
  467. $res = $mail->setBodyText('This is a test.');
  468. $mail->setFrom('testmail@example.com', 'test Mail User');
  469. $mail->setSubject('My Subject');
  470. $mail->addTo('recipient1@example.com');
  471. $mail->addTo('recipient2@example.com');
  472. $mail->addBcc('recipient1_bcc@example.com');
  473. $mail->addBcc('recipient2_bcc@example.com');
  474. $mail->addCc('recipient1_cc@example.com', 'Example no. 1 for cc');
  475. $mail->addCc('recipient2_cc@example.com', 'Example no. 2 for cc');
  476. // First example: from and return-path should be equal
  477. $mock = new Zend_Mail_Transport_Mock();
  478. $mail->send($mock);
  479. $this->assertTrue($mock->called);
  480. $this->assertEquals($mail->getFrom(), $mock->returnPath);
  481. // Second example: from and return-path should not be equal
  482. $mail->setReturnPath('sender2@example.com');
  483. $mock = new Zend_Mail_Transport_Mock();
  484. $mail->send($mock);
  485. $this->assertTrue($mock->called);
  486. $this->assertNotEquals($mail->getFrom(), $mock->returnPath);
  487. $this->assertEquals($mail->getReturnPath(), $mock->returnPath);
  488. $this->assertNotEquals($mock->returnPath, $mock->from);
  489. }
  490. public function testNoBody()
  491. {
  492. $mail = new Zend_Mail();
  493. $mail->setFrom('testmail@example.com', 'test Mail User');
  494. $mail->setSubject('My Subject');
  495. $mail->addTo('recipient1@example.com');
  496. // First example: from and return-path should be equal
  497. $mock = new Zend_Mail_Transport_Mock();
  498. try {
  499. $mail->send($mock);
  500. $this->assertTrue($mock->called);
  501. } catch (Exception $e) {
  502. // success
  503. $this->assertContains('No body specified', $e->getMessage());
  504. }
  505. }
  506. /**
  507. * Helper method for {@link testZf928ToAndBccHeadersShouldNotMix()}; extracts individual header lines
  508. *
  509. * @param Zend_Mail_Transport_Abstract $mock
  510. * @param string $type
  511. * @return string
  512. */
  513. protected function _getHeader(Zend_Mail_Transport_Abstract $mock, $type = 'To')
  514. {
  515. $headers = str_replace("\r\n", "\n", $mock->header);
  516. $headers = explode("\n", $mock->header);
  517. $return = '';
  518. foreach ($headers as $header) {
  519. if (!empty($return)) {
  520. // Check for header continuation
  521. if (!preg_match('/^[a-z-]+:/i', $header)) {
  522. $return .= "\r\n" . $header;
  523. continue;
  524. } else {
  525. break;
  526. }
  527. }
  528. if (preg_match('/^' . $type . ': /', $header)) {
  529. $return = $header;
  530. }
  531. }
  532. return $return;
  533. }
  534. public function testZf928ToAndBccHeadersShouldNotMix()
  535. {
  536. $mail = new Zend_Mail();
  537. $mail->setSubject('my subject');
  538. $mail->setBodyText('my body');
  539. $mail->setFrom('info@onlime.ch');
  540. $mail->addTo('to.address@email.com');
  541. $mail->addBcc('first.bcc@email.com');
  542. $mail->addBcc('second.bcc@email.com');
  543. // test with generic transport
  544. $mock = new Zend_Mail_Transport_Mock();
  545. $mail->send($mock);
  546. $to = $this->_getHeader($mock);
  547. $bcc = $this->_getHeader($mock, 'Bcc');
  548. $this->assertContains('to.address@email.com', $to, $to);
  549. $this->assertNotContains('second.bcc@email.com', $to, $bcc);
  550. // test with sendmail-like transport
  551. $mock = new Zend_Mail_Transport_Sendmail_Mock();
  552. $mail->send($mock);
  553. $to = $this->_getHeader($mock);
  554. $bcc = $this->_getHeader($mock, 'Bcc');
  555. // Remove the following line due to fixes by Simon
  556. // $this->assertNotContains('to.address@email.com', $to, $mock->header);
  557. $this->assertNotContains('second.bcc@email.com', $to, $bcc);
  558. }
  559. public function testZf927BlankLinesShouldPersist()
  560. {
  561. $mail = new Zend_Mail();
  562. $mail->setSubject('my subject');
  563. $mail->setBodyText("my body\r\n\r\n...after two newlines");
  564. $mail->setFrom('test@email.com');
  565. $mail->addTo('test@email.com');
  566. // test with generic transport
  567. $mock = new Zend_Mail_Transport_Sendmail_Mock();
  568. $mail->send($mock);
  569. $body = quoted_printable_decode($mock->body);
  570. $this->assertContains("\r\n\r\n...after", $body, $body);
  571. }
  572. public function testGetJustBodyText()
  573. {
  574. $text = "my body\r\n\r\n...after two newlines";
  575. $mail = new Zend_Mail();
  576. $mail->setBodyText($text);
  577. $this->assertContains('my body', $mail->getBodyText(true));
  578. $this->assertContains('after two newlines', $mail->getBodyText(true));
  579. }
  580. public function testGetJustBodyHtml()
  581. {
  582. $text = "<html><head></head><body><p>Some body text</p></body></html>";
  583. $mail = new Zend_Mail();
  584. $mail->setBodyHtml($text);
  585. $this->assertContains('Some body text', $mail->getBodyHtml(true));
  586. }
  587. public function testTypeAccessor()
  588. {
  589. $mail = new Zend_Mail();
  590. $this->assertNull($mail->getType());
  591. $mail->setType(Zend_Mime::MULTIPART_ALTERNATIVE);
  592. $this->assertEquals(Zend_Mime::MULTIPART_ALTERNATIVE, $mail->getType());
  593. $mail->setType(Zend_Mime::MULTIPART_RELATED);
  594. $this->assertEquals(Zend_Mime::MULTIPART_RELATED, $mail->getType());
  595. try {
  596. $mail->setType('text/plain');
  597. $this->fail('Invalid Zend_Mime type should throw an exception');
  598. } catch (Exception $e) {
  599. }
  600. }
  601. public function testDateSet()
  602. {
  603. $mail = new Zend_Mail();
  604. $res = $mail->setBodyText('Date Test');
  605. $mail->setFrom('testmail@example.com', 'test Mail User');
  606. $mail->setSubject('Date Test');
  607. $mail->addTo('recipient@example.com');
  608. $mock = new Zend_Mail_Transport_Mock();
  609. $mail->send($mock);
  610. $this->assertTrue($mock->called);
  611. $this->assertTrue(isset($mock->headers['Date']));
  612. $this->assertTrue(isset($mock->headers['Date'][0]));
  613. $this->assertTrue(strlen($mock->headers['Date'][0]) > 0);
  614. }
  615. public function testSetDateInt()
  616. {
  617. $mail = new Zend_Mail();
  618. $res = $mail->setBodyText('Date Test');
  619. $mail->setFrom('testmail@example.com', 'test Mail User');
  620. $mail->setSubject('Date Test');
  621. $mail->addTo('recipient@example.com');
  622. $mail->setDate(362656800);
  623. $mock = new Zend_Mail_Transport_Mock();
  624. $mail->send($mock);
  625. $this->assertTrue($mock->called);
  626. $this->assertTrue(strpos(implode('', $mock->headers['Date']), 'Mon, 29 Jun 1981') === 0);
  627. }
  628. public function testSetDateString()
  629. {
  630. $mail = new Zend_Mail();
  631. $res = $mail->setBodyText('Date Test');
  632. $mail->setFrom('testmail@example.com', 'test Mail User');
  633. $mail->setSubject('Date Test');
  634. $mail->addTo('recipient@example.com');
  635. $mail->setDate('1981-06-29T12:00:00');
  636. $mock = new Zend_Mail_Transport_Mock();
  637. $mail->send($mock);
  638. $this->assertTrue($mock->called);
  639. $this->assertTrue(strpos(implode('', $mock->headers['Date']), 'Mon, 29 Jun 1981') === 0);
  640. }
  641. public function testSetDateObject()
  642. {
  643. $mail = new Zend_Mail();
  644. $res = $mail->setBodyText('Date Test');
  645. $mail->setFrom('testmail@example.com', 'test Mail User');
  646. $mail->setSubject('Date Test');
  647. $mail->addTo('recipient@example.com');
  648. $mail->setDate(new Zend_Date('1981-06-29T12:00:00', Zend_Date::ISO_8601));
  649. $mock = new Zend_Mail_Transport_Mock();
  650. $mail->send($mock);
  651. $this->assertTrue($mock->called);
  652. $this->assertTrue(strpos(implode('', $mock->headers['Date']), 'Mon, 29 Jun 1981') === 0);
  653. }
  654. public function testSetDateInvalidString()
  655. {
  656. $mail = new Zend_Mail();
  657. try {
  658. $mail->setDate('invalid date');
  659. $this->fail('Invalid date should throw an exception');
  660. } catch (Exception $e) {
  661. }
  662. }
  663. public function testSetDateInvalidType()
  664. {
  665. $mail = new Zend_Mail();
  666. try {
  667. $mail->setDate(true);
  668. $this->fail('Invalid date should throw an exception');
  669. } catch (Exception $e) {
  670. }
  671. }
  672. public function testSetDateInvalidObject()
  673. {
  674. $mail = new Zend_Mail();
  675. try {
  676. $mail->setDate($mail);
  677. $this->fail('Invalid date should throw an exception');
  678. } catch (Exception $e) {
  679. }
  680. }
  681. public function testSetDateTwice()
  682. {
  683. $mail = new Zend_Mail();
  684. $mail->setDate();
  685. try {
  686. $mail->setDate(123456789);
  687. $this->fail('setting date twice should throw an exception');
  688. } catch (Exception $e) {
  689. }
  690. }
  691. /**
  692. * @group ZF-6872
  693. */
  694. public function testSetReplyTo()
  695. {
  696. $mail = new Zend_Mail('UTF-8');
  697. $mail->setReplyTo("foo@zend.com", "\xe2\x82\xa0!");
  698. $headers = $mail->getHeaders();
  699. $this->assertEquals("=?UTF-8?Q?=E2=82=A0!?= <foo@zend.com>", $headers["Reply-To"][0]);
  700. }
  701. /**
  702. * @group ZF-1688
  703. * @group ZF-2559
  704. */
  705. public function testSetHeaderEncoding()
  706. {
  707. $mail = new Zend_Mail();
  708. $this->assertEquals(Zend_Mime::ENCODING_QUOTEDPRINTABLE, $mail->getHeaderEncoding());
  709. $mail->setHeaderEncoding(Zend_Mime::ENCODING_BASE64);
  710. $this->assertEquals(Zend_Mime::ENCODING_BASE64, $mail->getHeaderEncoding());
  711. }
  712. /**
  713. * @group ZF-1688
  714. * @dataProvider dataSubjects
  715. */
  716. public function testIfLongSubjectsHaveCorrectLineBreaksAndEncodingMarks($subject)
  717. {
  718. $mail = new Zend_Mail("UTF-8");
  719. $mail->setSubject($subject);
  720. $headers = $mail->getHeaders();
  721. $this->assertMailHeaderConformsToRfc($headers['Subject'][0]);
  722. }
  723. /**
  724. * @group ZF-7702
  725. */
  726. public function testReplyToIsNoRecipient() {
  727. $mail = new Zend_Mail();
  728. $mail->setReplyTo('foo@example.com','foobar');
  729. $this->assertEquals(0, count($mail->getRecipients()));
  730. }
  731. public function testGetReplyToReturnsReplyTo() {
  732. $mail = new Zend_Mail();
  733. $mail->setReplyTo('foo@example.com');
  734. $this->assertEquals('foo@example.com',$mail->getReplyTo());
  735. }
  736. /**
  737. * @expectedException Zend_Mail_Exception
  738. */
  739. public function testReplyToCantBeSetTwice() {
  740. $mail = new Zend_Mail();
  741. $mail->setReplyTo('user@example.com');
  742. $mail->setReplyTo('user2@example.com');
  743. }
  744. public function testDefaultFrom() {
  745. Zend_Mail::setDefaultFrom('john@example.com','John Doe');
  746. $this->assertEquals(array('email' => 'john@example.com','name' =>'John Doe'), Zend_Mail::getDefaultFrom());
  747. Zend_Mail::clearDefaultFrom();
  748. $this->assertEquals(null, Zend_Mail::getDefaultFrom());
  749. Zend_Mail::setDefaultFrom('john@example.com');
  750. $this->assertEquals(array('email' => 'john@example.com','name' => null), Zend_Mail::getDefaultFrom());
  751. }
  752. public function testDefaultReplyTo() {
  753. Zend_Mail::setDefaultReplyTo('john@example.com','John Doe');
  754. $this->assertEquals(array('email' => 'john@example.com','name' =>'John Doe'), Zend_Mail::getDefaultReplyTo());
  755. Zend_Mail::clearDefaultReplyTo();
  756. $this->assertEquals(null, Zend_Mail::getDefaultReplyTo());
  757. Zend_Mail::setDefaultReplyTo('john@example.com');
  758. $this->assertEquals(array('email' => 'john@example.com','name' => null), Zend_Mail::getDefaultReplyTo());
  759. }
  760. public function testSettingFromDefaults() {
  761. Zend_Mail::setDefaultFrom('john@example.com', 'John Doe');
  762. Zend_Mail::setDefaultReplyTo('foo@example.com','Foo Bar');
  763. $mail = new Zend_Mail();
  764. $headers = $mail->setFromToDefaultFrom() // test fluent interface
  765. ->setReplyToFromDefault()
  766. ->getHeaders();
  767. $this->assertEquals('john@example.com', $mail->getFrom());
  768. $this->assertEquals('foo@example.com', $mail->getReplyTo());
  769. $this->assertEquals('John Doe <john@example.com>', $headers['From'][0]);
  770. $this->assertEquals('Foo Bar <foo@example.com>', $headers['Reply-To'][0]);
  771. }
  772. public function testMethodSendUsesDefaults()
  773. {
  774. Zend_Mail::setDefaultFrom('john@example.com', 'John Doe');
  775. Zend_Mail::setDefaultReplyTo('foo@example.com','Foo Bar');
  776. $mail = new Zend_Mail();
  777. $mail->setBodyText('Defaults Test');
  778. $mock = new Zend_Mail_Transport_Mock();
  779. $mail->send($mock);
  780. $headers = $mock->headers;
  781. $this->assertTrue($mock->called);
  782. $this->assertEquals($mock->from, 'john@example.com');
  783. $this->assertEquals($headers['From'][0], 'John Doe <john@example.com>');
  784. $this->assertEquals($headers['Reply-To'][0], 'Foo Bar <foo@example.com>');
  785. }
  786. public static function dataSubjects()
  787. {
  788. return array(
  789. array("Simple Ascii Subject"),
  790. array("Subject with US Specialchars: &%$/()"),
  791. array("Gimme more \xe2\x82\xa0!"),
  792. array("This is \xc3\xa4n germ\xc3\xa4n multiline s\xc3\xbcbject with rand\xc3\xb6m \xc3\xbcml\xc3\xa4uts."),
  793. array("Alle meine Entchen schwimmen in dem See, schwimmen in dem See, K\xc3\xb6pfchen in das Wasser, Schw\xc3\xa4nzchen in die H\xc3\xb6h!"),
  794. array("\xc3\xa4\xc3\xa4xxxxx\xc3\xa4\xc3\xa4\xc3\xa4\xc3\xa4\xc3\xa4\xc3\xa4\xc3\xa4"),
  795. array("\xd0\x90\xd0\x91\xd0\x92\xd0\x93\xd0\x94\xd0\x95 \xd0\x96\xd0\x97\xd0\x98\xd0\x99 \xd0\x9a\xd0\x9b\xd0\x9c\xd0\x9d"),
  796. array("Ich. Denke. Also. Bin. Ich! (Ein \xc3\xbcml\xc3\xa4\xc3\xbctautomat!)"),
  797. );
  798. }
  799. /**
  800. * Assertion that checks if a given mailing header string is RFC conform.
  801. *
  802. * @param string $header
  803. * @return void
  804. */
  805. protected function assertMailHeaderConformsToRfc($header)
  806. {
  807. $this->numAssertions++;
  808. $parts = explode(Zend_Mime::LINEEND, $header);
  809. if(count($parts) > 0) {
  810. for($i = 0; $i < count($parts); $i++) {
  811. if(preg_match('/(=?[a-z0-9-_]+\?[q|b]{1}\?)/i', $parts[$i], $matches)) {
  812. $dce = sprintf("=?%s", $matches[0]);
  813. // Check that Delimiter, Charset, Encoding are at the front of the string
  814. if(substr(trim($parts[$i]), 0, strlen($dce)) != $dce) {
  815. $this->fail(sprintf(
  816. "Header-Part '%s' in line '%d' has missing or malformated delimiter, charset, encoding information.",
  817. $parts[$i],
  818. $i+1
  819. ));
  820. }
  821. // check that the encoded word is not too long.);
  822. // this is only some kind of suggestion by the standard, in PHP its hard to hold it, so we do not enforce it here.
  823. /*if(strlen($parts[$i]) > 75) {
  824. $this->fail(sprintf(
  825. "Each encoded-word is only allowed to be 75 chars long, but line %d is %s chars long: %s",
  826. $i+1,
  827. strlen($parts[$i]),
  828. $parts[$i]
  829. ));
  830. }*/
  831. // Check that the end-delmiter ?= is correctly placed
  832. if(substr(trim($parts[$i]), -2, 2) != "?=") {
  833. $this->fail(sprintf(
  834. "Lines with an encoded-word have to end in ?=, but line %d does not: %s",
  835. $i+1,
  836. substr(trim($parts[$i]), -2, 2)
  837. ));
  838. }
  839. // Check that only one encoded-word can be found per line.
  840. if(substr_count($parts[$i], "=?") != 1) {
  841. $this->fail(sprintf(
  842. "Only one encoded-word is allowed per line in the header. It seems line %d contains more: %s",
  843. $i+1,
  844. $parts[$i]
  845. ));
  846. }
  847. // Check that the encoded-text only contains US-ASCII chars, and no space
  848. $encodedText = substr(trim($parts[$i]), strlen($dce), -2);
  849. if(preg_match('/([\s]+)/', $encodedText)) {
  850. $this->fail(sprintf(
  851. "No whitespace characters allowed in encoded-text of line %d: %s",
  852. $i+1,
  853. $parts[$i]
  854. ));
  855. }
  856. for($i = 0; $i < strlen($encodedText); $i++) {
  857. if(ord($encodedText[$i]) > 127) {
  858. $this->fail(sprintf(
  859. "No non US-ASCII characters allowed, but line %d has them: %s",
  860. $i+1,
  861. $parts[$i]
  862. ));
  863. }
  864. }
  865. } else if(Zend_Mime::isPrintable($parts[$i]) == false) {
  866. $this->fail(sprintf(
  867. "Encoded-word in line %d contains non printable characters.",
  868. $i+1
  869. ));
  870. }
  871. }
  872. }
  873. }
  874. }