Merge pull request #13358 from yashodha/campaign_reports
[civicrm-core.git] / tests / phpunit / CRM / Utils / Mail / EmailProcessorTest.php
CommitLineData
cdc5c450 1<?php
2
3/**
4 * Class CRM_Utils_EmailProcessorTest
5 * @group headless
6 */
7
8class CRM_Utils_EmailProcessorTest extends CiviUnitTestCase {
9
10 /**
11 * Event queue record.
12 *
13 * @var array
14 */
15 protected $eventQueue = array();
16
17 /**
18 * ID of our sample contact.
19 *
20 * @var int
21 */
22 protected $contactID;
23
24 public function setUp() {
25 parent::setUp();
e7693ddd 26 CRM_Utils_File::cleanDir(__DIR__ . '/data/mail');
27 mkdir(__DIR__ . '/data/mail');
cdc5c450 28 $this->callAPISuccess('MailSettings', 'get', array(
29 'api.MailSettings.create' => array(
30 'name' => 'local',
31 'protocol' => 'Localdir',
32 'source' => __DIR__ . '/data/mail',
33 'domain' => 'example.com',
34 ),
35 ));
36 }
37
e7693ddd 38 public function tearDown() {
39 CRM_Utils_File::cleanDir(__DIR__ . '/data/mail');
40 parent::tearDown();
41 $this->quickCleanup(array('civicrm_group', 'civicrm_group_contact', 'civicrm_mailing', 'civicrm_mailing_job', 'civicrm_mailing_event_bounce', 'civicrm_mailing_event_queue', 'civicrm_mailing_group', 'civicrm_mailing_recipients', 'civicrm_contact', 'civicrm_email'));
42 }
43
cdc5c450 44 /**
45 * Test the job processing function works and processes a bounce.
46 */
47 public function testBounceProcessing() {
48 $this->setUpMailing();
49
50 copy(__DIR__ . '/data/bounces/bounce_no_verp.txt', __DIR__ . '/data/mail/bounce_no_verp.txt');
51 $this->assertTrue(file_exists(__DIR__ . '/data/mail/bounce_no_verp.txt'));
52 $this->callAPISuccess('job', 'fetch_bounces', array());
53 $this->assertFalse(file_exists(__DIR__ . '/data/mail/bounce_no_verp.txt'));
54 $this->checkMailingBounces(1);
55 }
56
e7693ddd 57 /**
58 * Tests that a multipart related email does not cause pain & misery & fatal errors.
59 *
60 * Sample taken from https://www.phpclasses.org/browse/file/14672.html
61 */
62 public function testProcessingMultipartRelatedEmail() {
63 $this->setUpMailing();
64 $mail = 'test_sample_message.eml';
65
66 copy(__DIR__ . '/data/bounces/' . $mail, __DIR__ . '/data/mail/' . $mail);
67 $this->callAPISuccess('job', 'fetch_bounces', array());
68 $this->assertFalse(file_exists(__DIR__ . '/data/mail/' . $mail));
69 $this->checkMailingBounces(1);
70 }
71
01467aec
EE
72 /**
73 * Tests that a nested multipart email does not cause pain & misery & fatal errors.
74 *
75 * Sample anonymized from an email that broke bounce processing at Wikimedia
76 */
77 public function testProcessingNestedMultipartEmail() {
78 $this->setUpMailing();
79 $mail = 'test_nested_message.eml';
80
81 copy(__DIR__ . '/data/bounces/' . $mail, __DIR__ . '/data/mail/' . $mail);
82 $this->callAPISuccess('job', 'fetch_bounces', array());
83 $this->assertFalse(file_exists(__DIR__ . '/data/mail/' . $mail));
84 $this->checkMailingBounces(1);
85 }
86
cdc5c450 87 /**
88 * Test that a deleted email does not cause a hard fail.
89 *
90 * The civicrm_mailing_event_queue table tracks email ids to represent an
91 * email address. The id may not represent the same email by the time the bounce may
92 * come in - a weakness of storing the id not the email. Relevant here
93 * is that it might have been deleted altogether, in which case the bounce should be
94 * silently ignored. This ignoring is also at the expense of the contact
95 * having the same email address with a different id.
96 *
97 * Longer term it would make sense to track the email address & track bounces back to that
98 * rather than an id that may not reflect the email used. Issue logged CRM-20021.
99 *
100 * For not however, we are testing absence of mysql error in conjunction with CRM-20016.
101 */
102 public function testBounceProcessingDeletedEmail() {
103 $this->setUpMailing();
104 $this->callAPISuccess('Email', 'get', array(
105 'contact_id' => $this->contactID,
106 'api.email.delete' => 1,
107 ));
108
109 copy(__DIR__ . '/data/bounces/bounce_no_verp.txt', __DIR__ . '/data/mail/bounce_no_verp.txt');
110 $this->assertTrue(file_exists(__DIR__ . '/data/mail/bounce_no_verp.txt'));
111 $this->callAPISuccess('job', 'fetch_bounces', array());
112 $this->assertFalse(file_exists(__DIR__ . '/data/mail/bounce_no_verp.txt'));
113 $this->checkMailingBounces(1);
114 }
115
116 /**
e7693ddd 117 *
cdc5c450 118 * Wrapper to check for mailing bounces.
119 *
120 * Normally we would call $this->callAPISuccessGetCount but there is not one & there is resistance to
121 * adding apis for 'convenience' so just adding a hacky function to get past the impasse.
122 *
123 * @param int $expectedCount
124 */
125 public function checkMailingBounces($expectedCount) {
126 $this->assertEquals($expectedCount, CRM_Core_DAO::singleValueQuery(
e7693ddd 127 "SELECT count(*) FROM civicrm_mailing_event_bounce"
cdc5c450 128 ));
129 }
130
131 /**
132 * Set up a mailing.
133 */
134 public function setUpMailing() {
135 $this->contactID = $this->individualCreate(array('email' => 'undeliverable@example.com'));
d15a97f4 136 $groupID = $this->callAPISuccess('Group', 'create', array(
cdc5c450 137 'title' => 'Mailing group',
138 'api.GroupContact.create' => array(
139 'contact_id' => $this->contactID,
d15a97f4 140 ),
141 ));
cdc5c450 142 $this->createMailing(array('scheduled_date' => 'now', 'groups' => array('include' => array($groupID))));
143 $this->callAPISuccess('job', 'process_mailing', array());
144 $this->eventQueue = $this->callAPISuccess('MailingEventQueue', 'get', array('api.MailingEventQueue.create' => array('hash' => 'aaaaaaaaaaaaaaaa')));
145 }
146
147}