Revert "deactivate Felamimail_Frontend_JsonTest::testChangeDeleteAccount"
[tine20] / tests / tine20 / Felamimail / Frontend / JsonTest.php
1 <?php
2
3 use Sabre\DAV;
4
5 /**
6  * Tine 2.0 - http://www.tine20.org
7  *
8  * @package     Felamimail
9  * @license     http://www.gnu.org/licenses/agpl.html
10  * @copyright   Copyright (c) 2009-2014 Metaways Infosystems GmbH (http://www.metaways.de)
11  * @author      Philipp Schüle <p.schuele@metaways.de>
12  */
13
14 /**
15  * Test helper
16  */
17 require_once dirname(dirname(dirname(__FILE__))) . DIRECTORY_SEPARATOR . 'TestHelper.php';
18
19 /**
20  * Test class for Tinebase_Group
21  */
22 class Felamimail_Frontend_JsonTest extends TestCase
23 {
24     /**
25      * @var Felamimail_Frontend_Json
26      */
27     protected $_json = array();
28
29     /**
30      * message ids to delete
31      *
32      * @var array
33      */
34     protected $_messageIds = array();
35     
36     /**
37      * @var Felamimail_Model_Account
38      */
39     protected $_account = NULL;
40     
41     /**
42      * imap backend
43
44      * @var Felamimail_Backend_ImapProxy
45      */
46     protected $_imap = NULL;
47     
48     /**
49      * name of the folder to use for tests
50      * @var string
51      */
52     protected $_testFolderName = 'Junk';
53     
54     /**
55      * folders to delete in tearDown()
56      * 
57      * @var array
58      */
59     protected $_createdFolders = array();
60
61     /**
62      * are there messages to delete?
63      * 
64      * @var array
65      */
66     protected $_foldersToClear = array();
67
68     /**
69      * active sieve script name to be restored
70      *
71      * @var string
72      */
73     protected $_oldActiveSieveScriptName = NULL;
74
75     /**
76      * was sieve_vacation_active ?
77      *
78      * @var boolean
79      */
80     protected $_oldSieveVacationActiveState = FALSE;
81     
82     /**
83      * old sieve data
84      *
85      * @var Felamimail_Sieve_Backend_Sql
86      */
87     protected $_oldSieveData = NULL;
88
89     /**
90      * sieve script name to delete
91      *
92      * @var string
93      */
94     protected $_testSieveScriptName = NULL;
95
96     /**
97      * sieve vacation template file name
98      *
99      * @var string
100      */
101     protected $_sieveVacationTemplateFile = 'vacation_template.tpl';
102
103     /**
104      * test email domain
105      *
106      * @var string
107      */
108     protected $_mailDomain = 'tine20.org';
109
110     /**
111      * @var Felamimail_Model_Folder
112      */
113     protected $_folder = NULL;
114
115     /**
116      * paths in the vfs to delete
117      *
118      * @var array
119      */
120     protected $_pathsToDelete = array();
121
122     /**
123      *
124      * @var Tinebase_Frontend_Json
125      */
126     protected $_frontend = NULL;
127     
128     /**
129      * Sets up the fixture.
130      * This method is called before a test is executed.
131      *
132      * @access protected
133      */
134     protected function setUp()
135     {
136         Tinebase_TransactionManager::getInstance()->startTransaction(Tinebase_Core::getDb());
137         
138         // get (or create) test accout
139         $this->_account = Felamimail_Controller_Account::getInstance()->search()->getFirstRecord();
140         $this->_oldSieveVacationActiveState = $this->_account->sieve_vacation_active;
141         try {
142             $this->_oldSieveData = new Felamimail_Sieve_Backend_Sql($this->_account);
143         } catch (Tinebase_Exception_NotFound $tenf) {
144             // do nothing
145         }
146         
147         $this->_json = new Felamimail_Frontend_Json();
148         $this->_imap = Felamimail_Backend_ImapFactory::factory($this->_account);
149         
150         foreach (array($this->_testFolderName, $this->_account->sent_folder, $this->_account->trash_folder) as $folderToCreate) {
151             // create folder if it does not exist
152             $this->_getFolder($folderToCreate);
153         }
154         
155         $this->_mailDomain = TestServer::getPrimaryMailDomain();
156
157         $this->_frontend = new Tinebase_Frontend_Json();
158     }
159
160     /**
161      * Tears down the fixture
162      * This method is called after a test is executed.
163      *
164      * @access protected
165      */
166     protected function tearDown()
167     {
168         if (Tinebase_Core::isLogLevel(Zend_Log::DEBUG)) Tinebase_Core::getLogger()->debug(__METHOD__ . '::' . __LINE__
169             . ' Tearing down ...');
170         
171         if (count($this->_createdFolders) > 0) {
172             foreach ($this->_createdFolders as $folderName) {
173                 //echo "delete $folderName\n";
174                 try {
175                     $this->_imap->removeFolder(Felamimail_Model_Folder::encodeFolderName($folderName));
176                 } catch (Zend_Mail_Storage_Exception $zmse) {
177                     // already deleted
178                 }
179             }
180             Felamimail_Controller_Cache_Folder::getInstance()->clear($this->_account);
181         }
182         
183         if (! empty($this->_foldersToClear)) {
184             foreach ($this->_foldersToClear as $folderName) {
185                 // delete test messages from given folders on imap server (search by special header)
186                 $this->_imap->selectFolder($folderName);
187                 $result = $this->_imap->search(array(
188                     'HEADER X-Tine20TestMessage jsontest'
189                 ));
190                 //print_r($result);
191                 foreach ($result as $messageUid) {
192                     $this->_imap->removeMessage($messageUid);
193                 }
194                 
195                 // clear message cache
196                 $folder = Felamimail_Controller_Folder::getInstance()->getByBackendAndGlobalName($this->_account->getId(), $folderName);
197                 Felamimail_Controller_Cache_Message::getInstance()->clear($folder);
198             }
199         }
200         
201         // sieve cleanup
202         if ($this->_testSieveScriptName !== NULL) {
203             Felamimail_Controller_Sieve::getInstance()->setScriptName($this->_testSieveScriptName);
204             try {
205                 Felamimail_Controller_Sieve::getInstance()->deleteScript($this->_account->getId());
206             } catch (Zend_Mail_Protocol_Exception $zmpe) {
207                 // do not delete script if active
208             }
209             Felamimail_Controller_Account::getInstance()->setVacationActive($this->_account, $this->_oldSieveVacationActiveState);
210             
211             if ($this->_oldSieveData !== NULL) {
212                 $this->_oldSieveData->save();
213             }
214         }
215         if ($this->_oldActiveSieveScriptName !== NULL) {
216             Felamimail_Controller_Sieve::getInstance()->setScriptName($this->_oldActiveSieveScriptName);
217             Felamimail_Controller_Sieve::getInstance()->activateScript($this->_account->getId());
218         }
219         
220         // vfs cleanup
221         foreach ($this->_pathsToDelete as $path) {
222             $webdavRoot = new DAV\ObjectTree(new Tinebase_WebDav_Root());
223             //echo "delete $path";
224             $webdavRoot->delete($path);
225         }
226         
227         Tinebase_TransactionManager::getInstance()->rollBack();
228     }
229
230     /************************ test functions *********************************/
231     
232     /*********************** folder tests ****************************/
233     
234     /**
235      * test search folders (check order of folders as well)
236      */
237     public function testSearchFolders()
238     {
239         $filter = $this->_getFolderFilter();
240         $result = $this->_json->searchFolders($filter);
241         
242         $this->assertGreaterThan(1, $result['totalcount']);
243         $expectedFolders = array('INBOX', $this->_testFolderName, $this->_account->trash_folder, $this->_account->sent_folder);
244         
245         $foundCount = 0;
246         foreach ($result['results'] as $index => $folder) {
247             if (in_array($folder['localname'], $expectedFolders)) {
248                 $foundCount++;
249             }
250         }
251         $this->assertEquals(count($expectedFolders), $foundCount);
252     }
253     
254     /**
255      * clear test folder
256      */
257     public function testClearFolder()
258     {
259         $folderName = $this->_testFolderName;
260         $folder = $this->_getFolder($this->_testFolderName);
261         $folder = Felamimail_Controller_Folder::getInstance()->emptyFolder($folder->getId());
262
263         $filter = $this->_getMessageFilter($folder->getId());
264         $result = $this->_json->searchMessages($filter, '');
265         
266         $this->assertEquals(0, $result['totalcount'], 'Found too many messages in folder ' . $this->_testFolderName);
267         $this->assertEquals(0, $folder->cache_totalcount);
268     }
269
270     /**
271      * try to create some folders
272      */
273     public function testCreateFolders()
274     {
275         $filter = $this->_getFolderFilter();
276         $result = $this->_json->searchFolders($filter);
277         
278         $foldernames = array('test' => 'test', 'Schlüssel' => 'Schlüssel', 'test//1' => 'test1', 'test\2' => 'test2');
279         
280         foreach ($foldernames as $foldername => $expected) {
281             $result = $this->_json->addFolder($foldername, $this->_testFolderName, $this->_account->getId());
282             $globalname = $this->_testFolderName . $this->_account->delimiter . $expected;
283             $this->_createdFolders[] = $globalname;
284             $this->assertEquals($expected, $result['localname']);
285             $this->assertEquals($globalname, $result['globalname']);
286             $this->assertEquals(Felamimail_Model_Folder::CACHE_STATUS_EMPTY, $result['cache_status']);
287         }
288     }
289     
290     /**
291      * test emtpy folder (with subfolder)
292      */
293     public function testEmptyFolderWithSubfolder()
294     {
295         $folderName = $this->_testFolderName;
296         $folder = $this->_getFolder($this->_testFolderName);
297         $this->testCreateFolders();
298         
299         $folderArray = $this->_json->emptyFolder($folder->getId());
300         $this->assertEquals(0, $folderArray['has_children']);
301         
302         $result = $this->_json->updateFolderCache($this->_account->getId(), $this->_testFolderName);
303         $this->assertEquals(0, count($result));
304     }
305     
306     /**
307      * testUpdateFolderCache
308      */
309     public function testUpdateFolderCache()
310     {
311         $result = $this->_json->updateFolderCache($this->_account->getId(), '');
312         
313         // create folders directly on imap server
314         $this->_imap->createFolder('test', $this->_testFolderName, $this->_account->delimiter);
315         $this->_imap->createFolder('testsub', $this->_testFolderName . $this->_account->delimiter . 'test', $this->_account->delimiter);
316         // if something goes wrong, we need to delete these folders in tearDown
317         $this->_createdFolders[] = $this->_testFolderName . $this->_account->delimiter . 'test' . $this->_account->delimiter . 'testsub';
318         $this->_createdFolders[] = $this->_testFolderName . $this->_account->delimiter . 'test';
319         
320         if (Tinebase_Core::isLogLevel(Zend_Log::DEBUG)) Tinebase_Core::getLogger()->debug(__METHOD__ . '::' . __LINE__
321             . ' Update cache and check if folder is found');
322         
323         $result = $this->_json->updateFolderCache($this->_account->getId(), $this->_testFolderName);
324         $testfolder = $result[0];
325         $this->assertGreaterThan(0, count($result));
326         $this->assertEquals($this->_testFolderName . $this->_account->delimiter . 'test', $testfolder['globalname']);
327         $this->assertEquals(TRUE, (bool)$testfolder['has_children'], 'should have children');
328         
329         if (Tinebase_Core::isLogLevel(Zend_Log::DEBUG)) Tinebase_Core::getLogger()->debug(__METHOD__ . '::' . __LINE__
330             . ' Delete subfolder directly on imap server');
331         
332         $this->_imap->removeFolder($this->_testFolderName . $this->_account->delimiter . 'test' . $this->_account->delimiter . 'testsub');
333         array_shift($this->_createdFolders);
334         
335         if (Tinebase_Core::isLogLevel(Zend_Log::DEBUG)) Tinebase_Core::getLogger()->debug(__METHOD__ . '::' . __LINE__
336             . ' Check if has_children got updated and folder is removed from cache');
337         
338         $this->_json->updateFolderCache($this->_account->getId(), '');
339         $testfolder = $this->_getFolder($this->_testFolderName . $this->_account->delimiter . 'test');
340         $this->assertEquals(FALSE, (bool)$testfolder['has_children'], 'should have no children');
341
342         return $testfolder;
343     }
344     
345     /**
346      * testUpdateFolderCacheOfNonexistantFolder
347      *
348      * @see 0009800: unselectable folder with subfolders disappears
349      */
350     public function testUpdateFolderCacheOfNonexistantFolder()
351     {
352         $testfolder = $this->testUpdateFolderCache();
353         
354         try {
355             $folderName = $this->_testFolderName . $this->_account->delimiter . 'test' . $this->_account->delimiter . 'testsub';
356             if (Tinebase_Core::isLogLevel(Zend_Log::DEBUG)) Tinebase_Core::getLogger()->debug(__METHOD__ . '::' . __LINE__
357                 . ' Trying to fetch deleted folder ' . $folderName);
358             
359             $testfoldersub = Felamimail_Controller_Folder::getInstance()->getByBackendAndGlobalName($this->_account->getId(), $folderName);
360             $this->fail('Tinebase_Exception_NotFound expected when looking for folder ' . $folderName);
361         } catch (Tinebase_Exception_NotFound $tenf) {
362         }
363         
364         $this->_imap->removeFolder($this->_testFolderName . $this->_account->delimiter . 'test');
365         array_shift($this->_createdFolders);
366         
367         // try to update message cache of nonexistant folder
368         $removedTestfolder = $this->_json->updateMessageCache($testfolder['id'], 1);
369         $this->assertEquals(0, $removedTestfolder['is_selectable'], 'Folder should not be selectable');
370         
371         // update cache and check if folder is deleted
372         $result = $this->_json->updateFolderCache($this->_account->getId(), $this->_testFolderName);
373         $this->assertEquals(0, count($result));
374     }
375     
376     /*********************** accounts tests **************************/
377     
378     /**
379      * test search for accounts and check default account from config
380      */
381     public function testSearchAccounts()
382     {
383         $system = $this->_getSystemAccount();
384         
385         $this->assertTrue(! empty($system), 'no accounts found');
386         if (TestServer::getInstance()->getConfig()->mailserver) {
387             $this->assertEquals(TestServer::getInstance()->getConfig()->mailserver, $system['host']);
388             $this->assertEquals(TestServer::getInstance()->getConfig()->mailserver, $system['sieve_hostname']);
389         }
390     }
391     
392     /**
393      * get system account
394      *
395      * @return array
396      */
397     protected function _getSystemAccount()
398     {
399         $results = $this->_json->searchAccounts(array());
400         
401         $this->assertGreaterThan(0, $results['totalcount']);
402         $system = array();
403         foreach ($results['results'] as $result) {
404             if ($result['name'] == Tinebase_Core::getUser()->accountLoginName . '@' . $this->_mailDomain) {
405                 $system = $result;
406             }
407         }
408         
409         return $system;
410     }
411     
412     /**
413      * test change / delete of account
414      */
415     public function testChangeDeleteAccount()
416     {
417         $system = $this->_getSystemAccount();
418         unset($system['id']);
419         $system['type'] = Felamimail_Model_Account::TYPE_USER;
420         $account = $this->_json->saveAccount($system);
421         
422         $accountRecord = new Felamimail_Model_Account($account, TRUE);
423         $accountRecord->resolveCredentials(FALSE);
424         if (TestServer::getInstance()->getConfig()->mailserver) {
425             $this->assertEquals(TestServer::getInstance()->getConfig()->mailserver, $account['host']);
426         }
427         
428         $this->_json->changeCredentials($account['id'], $accountRecord->user, 'neuespasswort');
429         $account = $this->_json->getAccount($account['id']);
430         
431         $accountRecord = new Felamimail_Model_Account($account, TRUE);
432         $accountRecord->resolveCredentials(FALSE);
433         $this->assertEquals('neuespasswort', $accountRecord->password);
434         
435         $this->_json->deleteAccounts($account['id']);
436     }
437     
438     /*********************** message tests ****************************/
439     
440     /**
441      * test update message cache
442      */
443     public function testUpdateMessageCache()
444     {
445         $message = $this->_sendMessage();
446         $inbox = $this->_getFolder('INBOX');
447         // update message cache and check result
448         $result = $this->_json->updateMessageCache($inbox['id'], 30);
449         
450         if ($result['cache_status'] == Felamimail_Model_Folder::CACHE_STATUS_COMPLETE) {
451             $this->assertEquals($result['imap_totalcount'], $result['cache_totalcount'], 'totalcounts should be equal');
452         } else if ($result['cache_status'] == Felamimail_Model_Folder::CACHE_STATUS_INCOMPLETE) {
453             $this->assertNotEquals(0, $result['cache_job_actions_est']);
454         }
455     }
456     
457     /**
458      * test folder status
459      */
460     public function testGetFolderStatus()
461     {
462         $filter = $this->_getFolderFilter();
463         $result = $this->_json->searchFolders($filter);
464         $this->assertGreaterThan(1, $result['totalcount']);
465         $expectedFolders = array('INBOX', $this->_testFolderName, $this->_account->trash_folder, $this->_account->sent_folder);
466         
467         foreach ($result['results'] as $folder) {
468             $this->_json->updateMessageCache($folder['id'], 30);
469         }
470         
471         $message = $this->_sendMessage();
472         
473         $status = $this->_json->getFolderStatus(array(array('field' => 'account_id', 'operator' => 'equals', 'value' => $this->_account->getId())));
474         $this->assertEquals(1, count($status));
475         $this->assertEquals($this->_account->sent_folder, $status[0]['localname']);
476     }
477
478     /**
479      * test folder status of deleted folder
480      *
481      * @see 0007134: getFolderStatus should ignore non-existent folders
482      */
483     public function testGetFolderStatusOfDeletedFolder()
484     {
485         $this->testCreateFolders();
486         // remove one of the created folders
487         $removedFolder = $this->_createdFolders[0];
488         $this->_imap->removeFolder(Felamimail_Model_Folder::encodeFolderName($removedFolder));
489         
490         $status = $this->_json->getFolderStatus(array(array('field' => 'account_id', 'operator' => 'equals', 'value' => $this->_account->getId())));
491         $this->assertGreaterThan(2, count($status), 'Expected more than 2 folders that need an update: ' . print_r($status, TRUE));
492         foreach ($status as $folder) {
493             if ($folder['globalname'] == $removedFolder) {
494                 $this->fail('removed folder should not appear in status array!');
495             }
496         }
497     }
498     
499     /**
500      * test send message
501      */
502     public function testSendMessage()
503     {
504         // set email to unittest@tine20.org
505         $contactFilter = new Addressbook_Model_ContactFilter(array(
506             array('field' => 'n_family', 'operator' => 'equals', 'value' => 'Clever')
507         ));
508         $contactIds = Addressbook_Controller_Contact::getInstance()->search($contactFilter, NULL, FALSE, TRUE);
509         $this->assertTrue(count($contactIds) > 0, 'sclever not found in addressbook');
510
511         $contact = Addressbook_Controller_Contact::getInstance()->get($contactIds[0]);
512         $originalEmail =  $contact->email;
513         $contact->email = $this->_account->email;
514         $contact = Addressbook_Controller_Contact::getInstance()->update($contact, FALSE);
515
516         // send email
517         $messageToSend = $this->_getMessageData('unittestalias@' . $this->_mailDomain);
518         $messageToSend['note'] = 1;
519         $messageToSend['bcc']  = array(Tinebase_Core::getUser()->accountEmailAddress);
520         //print_r($messageToSend);
521         $returned = $this->_json->saveMessage($messageToSend);
522         $this->_foldersToClear = array('INBOX', $this->_account->sent_folder);
523         
524         // check if message is in sent folder
525         $message = $this->_searchForMessageBySubject($messageToSend['subject'], $this->_account->sent_folder);
526         $this->assertEquals($message['from_email'], $messageToSend['from_email']);
527         $this->assertTrue(isset($message['to'][0]));
528         $this->assertEquals($message['to'][0],      $messageToSend['to'][0], 'recipient not found');
529         $this->assertEquals($message['bcc'][0],     $messageToSend['bcc'][0], 'bcc recipient not found');
530         $this->assertEquals($message['subject'],    $messageToSend['subject']);
531         
532         $this->_checkEmailNote($contact, $messageToSend['subject']);
533         
534         // reset sclevers original email address
535         $contact->email = $originalEmail;
536         Addressbook_Controller_Contact::getInstance()->update($contact, FALSE);
537     }
538     
539     /**
540      * check email note
541      *
542      * @param Addressbook_Model_Contact $contact
543      * @param string $subject
544      */
545     protected function _checkEmailNote($contact, $subject)
546     {
547         // check if email note has been added to contact(s)
548         $contact = Addressbook_Controller_Contact::getInstance()->get($contact->getId());
549         $emailNoteType = Tinebase_Notes::getInstance()->getNoteTypeByName('email');
550         
551         // check / delete notes
552         $emailNoteIds = array();
553         foreach ($contact->notes as $note) {
554             if ($note->note_type_id == $emailNoteType->getId()) {
555                 $this->assertContains($subject, $note->note, 'did not find note subject');
556                 $this->assertEquals(Tinebase_Core::getUser()->getId(), $note->created_by);
557                 $this->assertContains('aaaaaä', $note->note);
558                 $emailNoteIds[] = $note->getId();
559             }
560         }
561         $this->assertGreaterThan(0, count($emailNoteIds), 'no email notes found');
562         Tinebase_Notes::getInstance()->deleteNotes($emailNoteIds);
563     }
564
565     /**
566      * test send message to invalid recipient
567      */
568     public function testSendMessageToInvalidRecipient()
569     {
570         $messageToSend = $this->_getMessageData($this->_account->email);
571         $invalidEmail = 'invaliduser@' . $this->_mailDomain;
572         $messageToSend['to'] = array($invalidEmail);
573         
574         try {
575             $returned = $this->_json->saveMessage($messageToSend);
576             $this->fail('Tinebase_Exception_SystemGeneric expected');
577         } catch (Tinebase_Exception_SystemGeneric $tesg) {
578             $this->assertContains('550 5.1.1 <' . $invalidEmail . '>: Recipient address rejected', $tesg->getMessage(),
579                 'exception message did not match: ' . $tesg->getMessage());
580         }
581     }
582     
583     /**
584      * try to get a message from imap server (with complete body, attachments, etc)
585      *
586      * @see 0006300: add unique message-id header to new messages (for message-id check)
587      */
588     public function testGetMessage()
589     {
590         $message = $this->_sendMessage();
591         
592         // get complete message
593         $message = $this->_json->getMessage($message['id']);
594         
595         // check
596         $this->assertTrue(isset($message['headers']) && $message['headers']['message-id']);
597         $this->assertContains('@' . $this->_mailDomain, $message['headers']['message-id']);
598         $this->assertGreaterThan(0, preg_match('/aaaaaä/', $message['body']));
599         
600         // delete message on imap server and check if correct exception is thrown when trying to get it
601         $this->_imap->selectFolder('INBOX');
602         $this->_imap->removeMessage($message['messageuid']);
603         Tinebase_Core::getCache()->clean();
604         $this->setExpectedException('Felamimail_Exception_IMAPMessageNotFound');
605         $message = $this->_json->getMessage($message['id']);
606     }
607     
608     /**
609      * try to get a message as plain/text
610      */
611     public function testGetPlainTextMessage()
612     {
613         $accountBackend = new Felamimail_Backend_Account();
614         $message = $this->_sendMessage();
615         
616         // get complete message
617         $this->_account->display_format = Felamimail_Model_Account::DISPLAY_PLAIN;
618         $accountBackend->update($this->_account);
619         $message = $this->_json->getMessage($message['id']);
620         $this->_account->display_format = Felamimail_Model_Account::DISPLAY_HTML;
621         $accountBackend->update($this->_account);
622         
623         // check
624         $this->assertEquals("aaaaaä \n\r\n", $message['body']);
625     }
626     
627     /**
628      * try search for a message with path filter
629      */
630     public function testSearchMessageWithPathFilter()
631     {
632         $sentMessage = $this->_sendMessage();
633         $filter = array(array(
634             'field' => 'path', 'operator' => 'in', 'value' => '/' . $this->_account->getId()
635         ));
636         $result = $this->_json->searchMessages($filter, '');
637         $message = $this->_getMessageFromSearchResult($result, $sentMessage['subject']);
638         $this->assertTrue(! empty($message), 'Sent message not found with account path filter');
639
640         $inbox = $this->_getFolder('INBOX');
641         $filter = array(array(
642             'field' => 'path', 'operator' => 'in', 'value' => '/' . $this->_account->getId() . '/' . $inbox->getId()
643         ));
644         $result = $this->_json->searchMessages($filter, '');
645         $message = $this->_getMessageFromSearchResult($result, $sentMessage['subject']);
646         $this->assertTrue(! empty($message), 'Sent message not found with path filter');
647         foreach ($result['results'] as $mail) {
648             $this->assertEquals($inbox->getId(), $mail['folder_id'], 'message is in wrong folder: ' . print_r($mail, TRUE));
649         }
650     }
651     
652     /**
653      * try search for a message with all inboxes and flags filter
654      */
655     public function testSearchMessageWithAllInboxesFilter()
656     {
657         $sentMessage = $this->_sendMessage();
658         $filter = array(
659             array('field' => 'path',  'operator' => 'in',       'value' => Felamimail_Model_MessageFilter::PATH_ALLINBOXES),
660             array('field' => 'flags', 'operator' => 'notin',    'value' => Zend_Mail_Storage::FLAG_FLAGGED),
661         );
662         $result = $this->_json->searchMessages($filter, '');
663         $this->assertGreaterThan(0, $result['totalcount']);
664         $this->assertEquals($result['totalcount'], count($result['results']));
665         
666         $message = $this->_getMessageFromSearchResult($result, $sentMessage['subject']);
667         $this->assertTrue(! empty($message), 'Sent message not found with all inboxes filter');
668     }
669     
670     /**
671      * try search for a message with three cache filters to force a foreign relation join with at least 2 tables
672      */
673     public function testSearchMessageWithThreeCacheFilter()
674     {
675         $filter = array(
676             array('field' => 'flags',   'operator' => 'in',       'value' => Zend_Mail_Storage::FLAG_ANSWERED),
677             array('field' => 'to',      'operator' => 'contains', 'value' => 'testDOESNOTEXIST'),
678             array('field' => 'subject', 'operator' => 'contains', 'value' => 'testDOESNOTEXIST'),
679         );
680         $result = $this->_json->searchMessages($filter, '');
681         $this->assertEquals(0, $result['totalcount']);
682     }
683     
684     /**
685      * try search for a message with empty path filter
686      */
687     public function testSearchMessageEmptyPath()
688     {
689         $sentMessage = $this->_sendMessage();
690         
691         $filter = array(
692             array('field' => 'path',  'operator' => 'equals',   'value' => ''),
693         );
694         $result = $this->_json->searchMessages($filter, '');
695         
696         $this->assertEquals(0, $result['totalcount']);
697         $accountFilterFound = FALSE;
698         
699         foreach ($result['filter'] as $filter) {
700             if ($filter['field'] === 'account_id' && empty($filter['value'])) {
701                 $accountFilterFound = TRUE;
702                 break;
703             }
704         }
705         $this->assertTrue($accountFilterFound);
706     }
707     
708     /**
709      * test flags (add + clear + deleted)
710      */
711     public function testAddAndClearFlags()
712     {
713         $message = $this->_sendMessage();
714         $inboxBefore = $this->_getFolder('INBOX');
715         
716         $this->_json->addFlags($message['id'], Zend_Mail_Storage::FLAG_SEEN);
717         
718         // check if unread count got decreased
719         $inboxAfter = $this->_getFolder('INBOX');
720         $this->assertTrue($inboxBefore->cache_unreadcount - 1 == $inboxAfter->cache_unreadcount, 'wrong cache unreadcount');
721         
722         $message = $this->_json->getMessage($message['id']);
723         $this->assertTrue(in_array(Zend_Mail_Storage::FLAG_SEEN, $message['flags']), 'seen flag not set');
724         
725         // try with a filter
726         $filter = array(
727             array('field' => 'id', 'operator' => 'in', array($message['id']))
728         );
729         $this->_json->clearFlags($filter, Zend_Mail_Storage::FLAG_SEEN);
730         
731         $message = $this->_json->getMessage($message['id']);
732         $this->assertFalse(in_array(Zend_Mail_Storage::FLAG_SEEN, $message['flags']), 'seen flag should not be set');
733
734         $this->setExpectedException('Tinebase_Exception_NotFound');
735         $this->_json->addFlags(array($message['id']), Zend_Mail_Storage::FLAG_DELETED);
736         $this->_json->getMessage($message['id']);
737     }
738     
739     /**
740      * testMarkFolderRead
741      *
742      * @see 0009812: mark folder as read does not work with pgsql
743      */
744     public function testMarkFolderRead()
745     {
746         $inboxBefore = $this->_getFolder('INBOX');
747         $filter = array(array(
748             'field' => 'folder_id', 'operator' => 'equals', 'value' => $inboxBefore->getId()
749         ), array(
750             'field' => 'flags', 'operator' => 'notin', 'value' => array(Zend_Mail_Storage::FLAG_SEEN)
751         ));
752         $this->_json->addFlags($filter, Zend_Mail_Storage::FLAG_SEEN);
753         
754         $inboxAfter = $this->_getFolder('INBOX');
755         $this->assertEquals(0, $inboxAfter->cache_unreadcount);
756     }
757     
758     /**
759      * test delete from trash
760      */
761     public function testDeleteFromTrashWithFilter()
762     {
763         $message = $this->_sendMessage();
764         $this->_foldersToClear = array('INBOX', $this->_account->sent_folder, $this->_account->trash_folder);
765         
766         $trash = $this->_getFolder($this->_account->trash_folder);
767         $result = $this->_json->moveMessages(array(array(
768             'field' => 'id', 'operator' => 'in', 'value' => array($message['id'])
769         )), $trash->getId());
770
771         $messageInTrash = $this->_searchForMessageBySubject($message['subject'], $this->_account->trash_folder);
772         
773         // delete messages in trash with filter
774         $this->_json->addFlags(array(array(
775             'field' => 'folder_id', 'operator' => 'equals', 'value' => $trash->getId()
776         ), array(
777             'field' => 'id', 'operator' => 'in', 'value' => array($messageInTrash['id'])
778         )), Zend_Mail_Storage::FLAG_DELETED);
779         
780         $this->setExpectedException('Tinebase_Exception_NotFound');
781         $this->_json->getMessage($messageInTrash['id']);
782     }
783     
784     /**
785      * move message to trash with trash folder constant (Felamimail_Model_Folder::FOLDER_TRASH)
786      */
787     public function testMoveMessagesToTrash()
788     {
789         $message = $this->_sendMessage();
790         $this->_foldersToClear = array('INBOX', $this->_account->sent_folder, $this->_account->trash_folder);
791         
792         $result = $this->_json->moveMessages(array(array(
793             'field' => 'id', 'operator' => 'in', 'value' => array($message['id'])
794         )), Felamimail_Model_Folder::FOLDER_TRASH);
795
796         $messageInTrash = $this->_searchForMessageBySubject($message['subject'], $this->_account->trash_folder);
797     }
798     
799     /**
800      * test reply mail and check some headers
801      *
802      * @see 0006106: Add References header / https://forge.tine20.org/mantisbt/view.php?id=6106
803      */
804     public function testReplyMessage()
805     {
806         $message = $this->_sendMessage();
807         
808         $replyMessage = $this->_getReply($message);
809         $returned = $this->_json->saveMessage($replyMessage);
810         
811         $result = $this->_getMessages();
812         
813         $replyMessageFound = array();
814         $originalMessage = array();
815         foreach ($result['results'] as $mail) {
816             if ($mail['subject'] == $replyMessage['subject']) {
817                 $replyMessageFound = $mail;
818             }
819             if ($mail['subject'] == $message['subject']) {
820                 $originalMessage = $mail;
821             }
822         }
823         $replyMessageFound = $this->_json->getMessage($replyMessageFound['id']);
824         $originalMessage = $this->_json->getMessage($originalMessage['id']);
825         
826         $this->assertTrue(! empty($replyMessageFound), 'replied message not found');
827         $this->assertTrue(! empty($originalMessage), 'original message not found');
828         
829         // check headers
830         $this->assertTrue(isset($replyMessageFound['headers']['in-reply-to']));
831         $this->assertEquals($originalMessage['headers']['message-id'], $replyMessageFound['headers']['in-reply-to']);
832         $this->assertTrue(isset($replyMessageFound['headers']['references']));
833         $this->assertEquals($originalMessage['headers']['message-id'], $replyMessageFound['headers']['references']);
834         
835         // check answered flag
836         $this->assertTrue(in_array(Zend_Mail_Storage::FLAG_ANSWERED, $originalMessage['flags'], 'could not find flag'));
837     }
838     
839     /**
840      * get reply message data
841      *
842      * @param array $_original
843      * @return array
844      */
845     protected function _getReply($_original)
846     {
847         $replyMessage               = $this->_getMessageData();
848         $replyMessage['subject']    = 'Re: ' . $_original['subject'];
849         $replyMessage['original_id']= $_original['id'];
850         $replyMessage['flags']      = Zend_Mail_Storage::FLAG_ANSWERED;
851         
852         return $replyMessage;
853     }
854
855     /**
856      * test reply mail in sent folder
857      */
858     public function testReplyMessageInSentFolder()
859     {
860         $messageInSent = $this->_sendMessage($this->_account->sent_folder);
861         $replyMessage = $this->_getReply($messageInSent);
862         $returned = $this->_json->saveMessage($replyMessage);
863         
864         $result = $this->_getMessages();
865         $sentMessage = $this->_getMessageFromSearchResult($result, $replyMessage['subject']);
866         $this->assertTrue(! empty($sentMessage));
867     }
868
869     /**
870      * test reply mail with long references header
871      *
872      * @see 0006644: "At least one mail header line is too long"
873      */
874     public function testReplyMessageWithLongHeader()
875     {
876         $messageInSent = $this->_sendMessage($this->_account->sent_folder, array(
877             'references' => '<c95d8187-2c71-437e-adb8-5e1dcdbdc507@email.test.org>
878    <2601bbfa-566e-4490-a3db-aad005733d32@email.test.org>
879    <20120530154350.1854610131@ganymed.de>
880    <7e393ce1-d193-44fc-bf5f-30c61a271fe6@email.test.org>
881    <4FC8B49C.8040704@funk.de>
882    <dba2ad5c-6726-4171-8710-984847c010a1@email.test.org>
883    <20120601123551.5E98610131@ganymed.de>
884    <f1cc3195-8641-46e3-8f20-f60f3e16b107@email.test.org>
885    <20120619093658.37E4210131@ganymed.de>
886    <CA+6Rn2PX2Q3tOk2tCQfCjcaC8zYS5XZX327OoyJfUb+w87vCLQ@mail.net.com>
887    <20120619130652.03DD310131@ganymed.de>
888    <37616c6a-4c47-4b54-9ca6-56875bc9205d@email.test.org>
889    <20120620074843.42E2010131@ganymed.de>
890    <CA+6Rn2MAb2x0qeSfcaW6F=0S7LEQL442Sx2ha9RtwMs4B0esBg@mail.net.com>
891    <20120620092902.88C8C10131@ganymed.de>
892    <c95d8187-2c71-437e-adb8-5e1dcdbdc507@email.test.org>
893    <2601bbfa-566e-4490-a3db-aad005733d32@email.test.org>
894    <20120530154350.1854610131@ganymed.de>
895    <7e393ce1-d193-44fc-bf5f-30c61a271fe6@email.test.org>
896    <4FC8B49C.8040704@funk.de>
897    <dba2ad5c-6726-4171-8710-984847c010a1@email.test.org>
898    <20120601123551.5E98610131@ganymed.de>
899    <f1cc3195-8641-46e3-8f20-f60f3e16b107@email.test.org>
900    <20120619093658.37E4210131@ganymed.de>
901    <CA+6Rn2PX2Q3tOk2tCQfCjcaC8zYS5XZX327OoyJfUb+w87vCLQ@mail.net.com>
902    <20120619130652.03DD310131@ganymed.de>
903    <37616c6a-4c47-4b54-9ca6-56875bc9205d@email.test.org>
904    <20120620074843.42E2010131@ganymed.de>
905    <CA+6Rn2MAb2x0qeSfcaW6F=0S7LEQL442Sx2ha9RtwMs4B0esBg@mail.net.com>
906    <20120620092902.88C8C10131@ganymed.de>'
907         ));
908         $replyMessage = $this->_getReply($messageInSent);
909         $returned = $this->_json->saveMessage($replyMessage);
910         
911         $result = $this->_getMessages();
912         $sentMessage = $this->_getMessageFromSearchResult($result, $replyMessage['subject']);
913         $this->assertTrue(! empty($sentMessage));
914     }
915     
916     /**
917      * test move
918      */
919     public function testMoveMessage()
920     {
921         $message = $this->_sendMessage();
922         $this->_foldersToClear = array('INBOX', $this->_account->sent_folder, $this->_testFolderName);
923         
924         $inbox = $this->_getFolder('INBOX');
925         $inboxBefore = $this->_json->updateMessageCache($inbox['id'], 30);
926         
927         // move
928         $testFolder = $this->_getFolder($this->_testFolderName);
929         $result = $this->_json->moveMessages(array(array(
930             'field' => 'id', 'operator' => 'in', 'value' => array($message['id'])
931         )), $testFolder->getId());
932
933         $inboxAfter = $this->_getFolder('INBOX');
934         
935         // check if count was decreased correctly
936         $this->assertEquals($inboxBefore['cache_unreadcount'] - 1, $inboxAfter['cache_unreadcount']);
937         $this->assertEquals($inboxBefore['cache_totalcount'] - 1, $inboxAfter['cache_totalcount']);
938         
939         $result = $this->_getMessages($this->_testFolderName);
940         $movedMessage = array();
941         foreach ($result['results'] as $mail) {
942             if ($mail['subject'] == $message['subject']) {
943                 $movedMessage = $mail;
944             }
945         }
946         $this->assertTrue(! empty($movedMessage), 'moved message not found');
947     }
948     
949     /**
950      * forward message test
951      *
952      * @see 0007624: losing umlauts in attached filenames
953      */
954     public function testForwardMessageWithAttachment()
955     {
956         $testFolder = $this->_getFolder($this->_testFolderName);
957         $message = fopen(dirname(__FILE__) . '/../files/multipart_related.eml', 'r');
958         Felamimail_Controller_Message::getInstance()->appendMessage($testFolder, $message);
959         
960         $subject = 'Tine 2.0 bei Metaways - Verbessurngsvorschlag';
961         $message = $this->_searchForMessageBySubject($subject, $this->_testFolderName);
962         
963         $fwdSubject = 'Fwd: ' . $subject;
964         $forwardMessageData = array(
965             'account_id'    => $this->_account->getId(),
966             'subject'       => $fwdSubject,
967             'to'            => array($this->_getEmailAddress()),
968             'body'          => "aaaaaä <br>",
969             'headers'       => array('X-Tine20TestMessage' => 'jsontest'),
970             'original_id'   => $message['id'],
971             'attachments'   => array(new Tinebase_Model_TempFile(array(
972                 'type'  => Felamimail_Model_Message::CONTENT_TYPE_MESSAGE_RFC822,
973                 'name'  => 'Verbessurüngsvorschlag',
974             ), TRUE)),
975             'flags'         => Zend_Mail_Storage::FLAG_PASSED,
976         );
977         
978         $this->_foldersToClear[] = 'INBOX';
979         $this->_json->saveMessage($forwardMessageData);
980         $forwardMessage = $this->_searchForMessageBySubject($fwdSubject);
981         
982         // check attachment name
983         $forwardMessageComplete = $this->_json->getMessage($forwardMessage['id']);
984         $this->assertEquals(1, count($forwardMessageComplete['attachments']));
985         $this->assertEquals('Verbessurüngsvorschlag.eml', $forwardMessageComplete['attachments'][0]['filename'], 'umlaut missing from attachment filename');
986         
987         $forwardMessage = $this->_json->getMessage($forwardMessage['id']);
988         $this->assertTrue((isset($forwardMessage['structure']) || array_key_exists('structure', $forwardMessage)), 'structure should be set when fetching complete message: ' . print_r($forwardMessage, TRUE));
989         $this->assertEquals(Felamimail_Model_Message::CONTENT_TYPE_MESSAGE_RFC822, $forwardMessage['structure']['parts'][2]['contentType']);
990         
991         $message = $this->_json->getMessage($message['id']);
992         $this->assertTrue(in_array(Zend_Mail_Storage::FLAG_PASSED, $message['flags']), 'forwarded flag missing in flags: ' . print_r($message, TRUE));
993     }
994     
995     /**
996      * testSendMessageWithAttachmentWithoutExtension
997      *
998      * @see 0008328: email attachment without file extension is not sent properly
999      */
1000     public function testSendMessageWithAttachmentWithoutExtension()
1001     {
1002         $subject = 'attachment test';
1003         $messageToSend = $this->_getMessageData('unittestalias@' . $this->_mailDomain, $subject);
1004         $tempfileName = 'jsontest' . Tinebase_Record_Abstract::generateUID(10);
1005         $tempfilePath = Tinebase_Core::getTempDir() . DIRECTORY_SEPARATOR . $tempfileName;
1006         file_put_contents($tempfilePath, 'some content');
1007         $tempFile = Tinebase_TempFile::getInstance()->createTempFile($tempfilePath, $tempfileName);
1008         $messageToSend['attachments'] = array(array('tempFile' => array('id' => $tempFile->getId())));
1009         $this->_json->saveMessage($messageToSend);
1010         $forwardMessage = $this->_searchForMessageBySubject($subject);
1011         $this->_foldersToClear = array('INBOX', $this->_account->sent_folder);
1012         
1013         $fullMessage = $this->_json->getMessage($forwardMessage['id']);
1014         $this->assertTrue(count($fullMessage['attachments']) === 1);
1015         $attachment = $fullMessage['attachments'][0];
1016         $this->assertContains($tempfileName, $attachment['filename'], 'wrong attachment filename: ' . print_r($attachment, TRUE));
1017         $this->assertEquals(16, $attachment['size'], 'wrong attachment size: ' . print_r($attachment, TRUE));
1018     }
1019     
1020     /**
1021      * save message in folder (draft) test
1022      *
1023      * @see 0007178: BCC does not save the draft message
1024      */
1025     public function testSaveMessageInFolder()
1026     {
1027         $messageToSave = $this->_getMessageData();
1028         $messageToSave['bcc'] = array('bccaddress@email.org', 'bccaddress2@email.org');
1029         
1030         $draftsFolder = $this->_getFolder($this->_account->drafts_folder);
1031         $returned = $this->_json->saveMessageInFolder($this->_account->drafts_folder, $messageToSave);
1032         $this->_foldersToClear = array($this->_account->drafts_folder);
1033         
1034         // check if message is in drafts folder and recipients are present
1035         $message = $this->_searchForMessageBySubject($messageToSave['subject'], $this->_account->drafts_folder);
1036         $this->assertEquals($messageToSave['subject'],  $message['subject']);
1037         $this->assertEquals($messageToSave['to'][0],    $message['to'][0], 'recipient not found');
1038         $this->assertEquals(2, count($message['bcc']), 'bcc recipient not found: ' . print_r($message, TRUE));
1039         $this->assertEquals($messageToSave['bcc'][0],   $message['bcc'][0], '1st bcc recipient not found');
1040         $this->assertEquals($messageToSave['bcc'][1],   $message['bcc'][1], '2nd bcc recipient not found');
1041     }
1042     
1043     /**
1044      * testSendReadingConfirmation
1045      *
1046      * @see 0007736: ask user before sending reading confirmation
1047      * @see 0008402: Wrong recipient with read confirmation
1048      */
1049     public function testSendReadingConfirmation()
1050     {
1051         $messageToSave = $this->_getMessageData();
1052         $messageToSave['headers']['disposition-notification-to'] = '"' . Tinebase_Core::getUser()->accountFullName . '" <' . $this->_account->email . '>';
1053         $returned = $this->_json->saveMessageInFolder($this->_testFolderName, $messageToSave);
1054         $messageWithReadingConfirmationHeader = $this->_searchForMessageBySubject($messageToSave['subject'], $this->_testFolderName);
1055         $this->_messageIds[] = $messageWithReadingConfirmationHeader['id'];
1056         $this->_json->sendReadingConfirmation($messageWithReadingConfirmationHeader['id']);
1057         
1058         $translate = Tinebase_Translation::getTranslation('Felamimail');
1059         $subject = $translate->_('Reading Confirmation:') . ' '. $messageToSave['subject'];
1060         $message = $this->_searchForMessageBySubject($subject);
1061         $this->_messageIds[] = $message['id'];
1062         
1063         $complete = $this->_json->getMessage($message['id']);
1064         $this->assertContains($translate->_('Was read by:') . ' ' . $this->_account->from, $complete['body']);
1065     }
1066
1067     /**
1068      * save message in non-existant folder (templates) test
1069      *
1070      * @see 0008476: Drafts are not working
1071      */
1072     public function testSaveMessageInNonExistantTemplatesFolder()
1073     {
1074         $messageToSave = $this->_getMessageData();
1075         
1076         $templatesFolder = $this->_getFolder($this->_account->templates_folder, FALSE);
1077         if ($templatesFolder) {
1078             $this->_json->deleteFolder($templatesFolder['id'], $this->_account->getId());
1079         }
1080         $returned = $this->_json->saveMessageInFolder($this->_account->templates_folder, $messageToSave);
1081         $this->_foldersToClear = array($this->_account->templates_folder);
1082         
1083         // check if message is in templates folder
1084         $message = $this->_searchForMessageBySubject($messageToSave['subject'], $this->_account->templates_folder);
1085         $this->assertEquals($messageToSave['subject'],  $message['subject']);
1086         $this->assertEquals($messageToSave['to'][0],    $message['to'][0], 'recipient not found');
1087     }
1088     
1089     /**
1090      * testSaveMessageNoteWithInvalidChar
1091      *
1092      * @see 0008644: error when sending mail with note (wrong charset)
1093      */
1094     public function testSaveMessageNoteWithInvalidChar()
1095     {
1096         $subject = Tinebase_Core::filterInputForDatabase("\xF0\x9F\x98\x8A\xC2"); // :-) emoji &nbsp;
1097         $messageData = $this->_getMessageData('', $subject);
1098         $messageData['note'] = true;
1099         $messageData['body'] .= "&nbsp;";
1100         
1101         $this->_foldersToClear[] = 'INBOX';
1102         $this->_json->saveMessage($messageData);
1103         $message = $this->_searchForMessageBySubject($subject);
1104         
1105         $contact = Addressbook_Controller_Contact::getInstance()->getContactByUserId(Tinebase_Core::getUser()->getId());
1106         $this->_checkEmailNote($contact, $subject);
1107     }
1108     
1109     /**
1110      * testSaveMessageNoteWithInvalidChar
1111      *
1112      * @see 0008644: error when sending mail with note (wrong charset)
1113      */
1114     public function testSaveMessageWithInvalidChar()
1115     {
1116         $subject = "\xF0\x9F\x98\x8A"; // :-) emoji
1117         $messageData = $this->_getMessageData('', $subject);
1118         $this->_foldersToClear[] = 'INBOX';
1119         $this->_json->saveMessage($messageData);
1120         $message = $this->_searchForMessageBySubject(Tinebase_Core::filterInputForDatabase($subject));
1121     }
1122     
1123     /**
1124      * testMessageWithInvalidICS
1125      *
1126      * @see 0008786: broken ics causes js error when showing details
1127      */
1128     public function testMessageWithInvalidICS()
1129     {
1130         $inbox = $this->_getFolder('INBOX');
1131         $mailAsString = file_get_contents(dirname(__FILE__) . '/../files/invalidimip.eml');
1132         Felamimail_Controller_Message::getInstance()->appendMessage($inbox, $mailAsString);
1133         
1134         $this->_foldersToClear = array('INBOX');
1135         $message = $this->_searchForMessageBySubject('test invalid imip');
1136         
1137         $fullMessage = $this->_json->getMessage($message['id']);
1138         $this->assertTrue(empty($fullMessage->preparedParts));
1139     }
1140     
1141     /*********************** sieve tests ****************************/
1142     
1143     /**
1144      * set and get vacation sieve script
1145      *
1146      * @see 0007768: Sieve - Vacation notify frequency not being set (Cyrus)
1147      */
1148     public function testGetSetVacation()
1149     {
1150         $vacationData = $this->_getVacationData();
1151         $this->_sieveTestHelper($vacationData);
1152         
1153         // check if script was activated
1154         $activeScriptName = Felamimail_Controller_Sieve::getInstance()->getActiveScriptName($this->_account->getId());
1155         $this->assertEquals($this->_testSieveScriptName, $activeScriptName);
1156         $updatedAccount = Felamimail_Controller_Account::getInstance()->get($this->_account->getId());
1157         $this->assertTrue((bool) $updatedAccount->sieve_vacation_active);
1158         
1159         $result = $this->_json->getVacation($this->_account->getId());
1160
1161         $this->assertEquals($this->_account->email, $result['addresses'][0]);
1162         
1163         $sieveBackend = Felamimail_Backend_SieveFactory::factory($this->_account->getId());
1164         if (preg_match('/dbmail/i', $sieveBackend->getImplementation())) {
1165             $translate = Tinebase_Translation::getTranslation('Felamimail');
1166             $vacationData['subject'] = sprintf($translate->_('Out of Office reply from %1$s'), Tinebase_Core::getUser()->accountFullName);
1167         }
1168         
1169         foreach (array('reason', 'enabled', 'subject', 'from', 'days') as $field) {
1170             $this->assertEquals($vacationData[$field], $result[$field], 'vacation data mismatch: ' . $field);
1171         }
1172     }
1173     
1174     /**
1175      * get vacation data
1176      *
1177      * @return array
1178      */
1179     protected function _getVacationData()
1180     {
1181         return array(
1182             'id'                    => $this->_account->getId(),
1183             'subject'               => 'unittest vacation subject',
1184             'from'                  => $this->_account->from . ' <' . $this->_account->email . '>',
1185             'days'                  => 3,
1186             'enabled'               => TRUE,
1187             'reason'                => 'unittest vacation message<br /><br />signature',
1188             'mime'                  => NULL,
1189         );
1190     }
1191     
1192     /**
1193      * test mime vacation sieve script
1194      */
1195     public function testMimeVacation()
1196     {
1197         $vacationData = $this->_getVacationData();
1198         $vacationData['reason'] = "\n<html><body><h1>unittest vacation&nbsp;message</h1></body></html>";
1199         
1200         $_sieveBackend = Felamimail_Backend_SieveFactory::factory($this->_account->getId());
1201         if (! in_array('mime', $_sieveBackend->capability())) {
1202             $vacationData['mime'] = 'text/html';
1203         }
1204         
1205         $this->_sieveTestHelper($vacationData, TRUE);
1206     }
1207     
1208     /**
1209      * test get/set of rules sieve script
1210      */
1211     public function testGetSetRules()
1212     {
1213         $ruleData = $this->_getRuleData();
1214         
1215         $this->_sieveTestHelper($ruleData);
1216         
1217         // check getRules
1218         $result = $this->_json->getRules($this->_account->getId());
1219         $this->assertEquals($result['totalcount'], count($ruleData));
1220         
1221         // check by sending mail
1222         $messageData = $this->_getMessageData('', 'viagra');
1223         $returned = $this->_json->saveMessage($messageData);
1224         $this->_foldersToClear = array('INBOX', $this->_testFolderName);
1225         // check if message is in test folder
1226         $message = $this->_searchForMessageBySubject($messageData['subject'], $this->_testFolderName);
1227     }
1228     
1229     /**
1230      * testRemoveRules
1231      *
1232      * @see 0006490: can not delete single filter rule
1233      */
1234     public function testRemoveRules()
1235     {
1236         $this->testGetSetRules();
1237         $this->_json->saveRules($this->_account->getId(), array());
1238         
1239         $result = $this->_json->getRules($this->_account->getId());
1240         $this->assertEquals(0, $result['totalcount'], 'found rules: ' . print_r($result, TRUE));
1241     }
1242     
1243     /**
1244      * get sieve rule data
1245      *
1246      * @return array
1247      */
1248     protected function _getRuleData()
1249     {
1250         return array(array(
1251             'id'            => 1,
1252             'action_type'   => Felamimail_Sieve_Rule_Action::FILEINTO,
1253             'action_argument' => $this->_testFolderName,
1254             'conjunction'  => 'allof',
1255             'conditions'    => array(array(
1256                 'test'          => Felamimail_Sieve_Rule_Condition::TEST_ADDRESS,
1257                 'comperator'    => Felamimail_Sieve_Rule_Condition::COMPERATOR_CONTAINS,
1258                 'header'        => 'From',
1259                 'key'           => '"abcd" <info@example.org>',
1260             )),
1261             'enabled'       => 1,
1262         ), array(
1263             'id'            => 2,
1264             'action_type'   => Felamimail_Sieve_Rule_Action::FILEINTO,
1265             'action_argument' => $this->_testFolderName,
1266             'conjunction'  => 'allof',
1267             'conditions'    => array(array(
1268                 'test'          => Felamimail_Sieve_Rule_Condition::TEST_ADDRESS,
1269                 'comperator'    => Felamimail_Sieve_Rule_Condition::COMPERATOR_CONTAINS,
1270                 'header'        => 'From',
1271                 'key'           => 'info@example.org',
1272             )),
1273             'enabled'       => 0,
1274         ), array(
1275             'id'            => 3,
1276             'action_type'   => Felamimail_Sieve_Rule_Action::FILEINTO,
1277             'action_argument' => $this->_testFolderName,
1278             'conjunction'  => 'allof',
1279             'conditions'    => array(array(
1280                 'test'          => Felamimail_Sieve_Rule_Condition::TEST_HEADER,
1281                 'comperator'    => Felamimail_Sieve_Rule_Condition::COMPERATOR_REGEX,
1282                 'header'        => 'subject',
1283                 'key'           => '[vV]iagra|cyalis',
1284             )),
1285             'enabled'       => 1,
1286         ));
1287     }
1288     
1289     /**
1290      * test to set a forward rule to this accounts email address
1291      * -> should throw exception to prevent mail cycling
1292      */
1293     public function testSetForwardRuleToSelf()
1294     {
1295         $ruleData = array(array(
1296             'id'            => 1,
1297             'action_type'   => Felamimail_Sieve_Rule_Action::REDIRECT,
1298             'action_argument' => $this->_account->email,
1299             'conjunction'     => 'allof',
1300             'conditions'    => array(array(
1301                 'test'          => Felamimail_Sieve_Rule_Condition::TEST_ADDRESS,
1302                 'comperator'    => Felamimail_Sieve_Rule_Condition::COMPERATOR_CONTAINS,
1303                 'header'        => 'From',
1304                 'key'           => 'info@example.org',
1305             )),
1306             'enabled'       => 1,
1307         ));
1308         
1309         try {
1310             $this->_sieveTestHelper($ruleData);
1311             $this->assertTrue(FALSE, 'it is not allowed to set own email address for redirect!');
1312         } catch (Felamimail_Exception_Sieve $e) {
1313             $this->assertTrue(TRUE);
1314         }
1315
1316         // this should work
1317         $ruleData[0]['enabled'] = 0;
1318         $this->_sieveTestHelper($ruleData);
1319     }
1320     
1321     /**
1322      * testGetVacationTemplates
1323      *
1324      * @return array
1325      */
1326     public function testGetVacationTemplates()
1327     {
1328         $this->markTestSkipped('0010194: fix felamimail webdav tests');
1329         
1330         $this->_addVacationTemplateFile();
1331         $result = $this->_json->getVacationMessageTemplates();
1332         
1333         $this->assertTrue($result['totalcount'] > 0, 'no templates found');
1334         $found = FALSE;
1335         foreach ($result['results'] as $template) {
1336             if ($template['name'] === $this->_sieveVacationTemplateFile) {
1337                 $found = TRUE;
1338                 break;
1339             }
1340         }
1341         
1342         $this->assertTrue($found, 'wrong templates: ' . print_r($result['results'], TRUE));
1343         
1344         return $template;
1345     }
1346     
1347     /**
1348      * add vacation template file to vfs
1349      */
1350     protected function _addVacationTemplateFile()
1351     {
1352         $webdavRoot = new DAV\ObjectTree(new Tinebase_WebDav_Root());
1353         $path = '/webdav/Felamimail/shared/Vacation Templates';
1354         $node = $webdavRoot->getNodeForPath($path);
1355         $this->_pathsToDelete[] = $path . '/' . $this->_sieveVacationTemplateFile;
1356         $node->createFile($this->_sieveVacationTemplateFile, fopen(dirname(__FILE__) . '/../files/' . $this->_sieveVacationTemplateFile, 'r'));
1357     }
1358     
1359     /**
1360      * testGetVacationMessage
1361      */
1362     public function testGetVacationMessage()
1363     {
1364         $this->markTestSkipped('0010194: fix felamimail webdav tests');
1365         
1366         $result = $this->_getVacationMessageWithTemplate();
1367         $sclever = Tinebase_User::getInstance()->getFullUserByLoginName('sclever');
1368         $pwulf = Tinebase_User::getInstance()->getFullUserByLoginName('pwulf');
1369         $this->assertEquals("Ich bin vom 18.04.2012 bis zum 20.04.2012 im Urlaub. Bitte kontaktieren Sie<br /> Paul Wulf (" .
1370             $pwulf->accountEmailAddress . ") oder Susan Clever (" .
1371             $sclever->accountEmailAddress . ").<br /><br />I am on vacation until Apr 20, 2012. Please contact Paul Wulf<br />(" .
1372             $pwulf->accountEmailAddress . ") or Susan Clever (" .
1373             $sclever->accountEmailAddress . ") instead.<br /><br />" .
1374             Addressbook_Controller_Contact::getInstance()->getContactByUserId(Tinebase_Core::getUser()->getId())->n_fn, $result['message']);
1375     }
1376     
1377     /**
1378      * get vacation message with template
1379      *
1380      * @return array
1381      */
1382     protected function _getVacationMessageWithTemplate()
1383     {
1384         $template = $this->testGetVacationTemplates();
1385         $sclever = Tinebase_User::getInstance()->getFullUserByLoginName('sclever');
1386         $result = $this->_json->getVacationMessage(array(
1387             'start_date' => '2012-04-18',
1388             'end_date'   => '2012-04-20',
1389             'contact_ids' => array(
1390                 Tinebase_User::getInstance()->getFullUserByLoginName('pwulf')->contact_id,
1391                 $sclever->contact_id,
1392             ),
1393             'template_id' => $template['id'],
1394             'signature' => $this->_account->signature
1395         ));
1396         
1397         return $result;
1398     }
1399     
1400     /**
1401      * testGetVacationWithSignature
1402      *
1403      * @see 0006866: check signature linebreaks in vacation message from template
1404      */
1405     public function testGetVacationWithSignature()
1406     {
1407         $this->markTestSkipped('0010194: fix felamimail webdav tests');
1408         
1409         $this->_sieveVacationTemplateFile = 'vacation_template_sig.tpl';
1410         
1411         // set signature with <br> + linebreaks
1412         $this->_account->signature = "llalala<br>\nxyz<br>\nblubb<br>";
1413         
1414         $result = $this->_getVacationMessageWithTemplate();
1415         $this->assertContains('-- <br />llalala<br />xyz<br />blubb<br />', $result['message'], 'wrong linebreaks or missing signature');
1416     }
1417     
1418     /**
1419     * testSetVacationWithStartAndEndDate
1420     *
1421     * @see 0006266: automatic deactivation of vacation message
1422     */
1423     public function testSetVacationWithStartAndEndDate()
1424     {
1425         $vacationData = $this->_getVacationData();
1426         $vacationData['start_date'] = '2012-04-18';
1427         $vacationData['end_date'] = '2012-04-20';
1428         $result = $this->_sieveTestHelper($vacationData);
1429         
1430         $this->assertContains($vacationData['start_date'], $result['start_date']);
1431         $this->assertContains($vacationData['end_date'], $result['end_date']);
1432     }
1433     
1434     /**
1435      * testSieveRulesOrder
1436      *
1437      * @see 0007240: order of sieve rules changes when vacation message is saved
1438      */
1439     public function testSieveRulesOrder()
1440     {
1441         $this->_setTestScriptname();
1442         
1443         // disable vacation first
1444         $this->_setDisabledVacation();
1445         
1446         $sieveBackend = Felamimail_Backend_SieveFactory::factory($this->_account->getId());
1447         
1448         $ruleData = $this->_getRuleData();
1449         $ruleData[0]['id'] = $ruleData[2]['id'];
1450         $ruleData[2]['id'] = 11;
1451         $resultSet = $this->_json->saveRules($this->_account->getId(), $ruleData);
1452         $sieveScriptRules = $sieveBackend->getScript($this->_testSieveScriptName);
1453         
1454         $this->_setDisabledVacation();
1455         $sieveScriptVacation = $sieveBackend->getScript($this->_testSieveScriptName);
1456         
1457         // compare sieve scripts
1458         $this->assertContains($sieveScriptRules, $sieveScriptVacation, 'rule order changed');
1459     }
1460     
1461     /**
1462      * use another name for test sieve script
1463      */
1464     protected function _setTestScriptname()
1465     {
1466         $this->_oldActiveSieveScriptName = Felamimail_Controller_Sieve::getInstance()->getActiveScriptName($this->_account->getId());
1467         $this->_testSieveScriptName = 'Felamimail_Unittest';
1468         Felamimail_Controller_Sieve::getInstance()->setScriptName($this->_testSieveScriptName);
1469     }
1470     
1471     /**
1472      * set disabled vacation message
1473      */
1474     protected function _setDisabledVacation()
1475     {
1476         $vacationData = $this->_getVacationData();
1477         $vacationData['enabled'] = FALSE;
1478         $resultSet = $this->_json->saveVacation($vacationData);
1479     }
1480     
1481     /**
1482      * get folder filter
1483      *
1484      * @return array
1485      */
1486     protected function _getFolderFilter()
1487     {
1488         return array(array(
1489             'field' => 'globalname', 'operator' => 'equals', 'value' => ''
1490         ));
1491     }
1492
1493     /**
1494      * get message filter
1495      *
1496      * @param string $_folderId
1497      * @return array
1498      */
1499     protected function _getMessageFilter($_folderId)
1500     {
1501         $result = array(array(
1502             'field' => 'folder_id', 'operator' => 'equals', 'value' => $_folderId
1503         ));
1504         
1505         return $result;
1506     }
1507     
1508     /**
1509      * get mailbox
1510      *
1511      * @param string $name
1512      * @param boolean $createFolder
1513      * @return Felamimail_Model_Folder|NULL
1514      */
1515     protected function _getFolder($name, $createFolder = TRUE)
1516     {
1517         Felamimail_Controller_Cache_Folder::getInstance()->update($this->_account->getId());
1518         try {
1519             $folder = Felamimail_Controller_Folder::getInstance()->getByBackendAndGlobalName($this->_account->getId(), $name);
1520         } catch (Tinebase_Exception_NotFound $tenf) {
1521             $folder = ($createFolder) ? Felamimail_Controller_Folder::getInstance()->create($this->_account, $name) : NULL;
1522         }
1523         
1524         return $folder;
1525     }
1526
1527     /**
1528      * get message data
1529      *
1530      * @return array
1531      */
1532     protected function _getMessageData($_emailFrom = '', $_subject = 'test')
1533     {
1534         return array(
1535             'account_id'    => $this->_account->getId(),
1536             'subject'       => $_subject,
1537             'to'            => array(Tinebase_Core::getUser()->accountEmailAddress),
1538             'body'          => 'aaaaaä <br>',
1539             'headers'       => array('X-Tine20TestMessage' => 'jsontest'),
1540             'from_email'    => $_emailFrom,
1541             'content_type'  => Felamimail_Model_Message::CONTENT_TYPE_HTML,
1542         );
1543     }
1544
1545     /**
1546      * send message and return message array
1547      *
1548      * @param string $folderName
1549      * @param array $addtionalHeaders
1550      * @return array
1551      */
1552     protected function _sendMessage($folderName = 'INBOX', $addtionalHeaders = array())
1553     {
1554         $messageToSend = $this->_getMessageData();
1555         $messageToSend['headers'] = array_merge($messageToSend['headers'], $addtionalHeaders);
1556         $returned = $this->_json->saveMessage($messageToSend);
1557         $this->_foldersToClear = array('INBOX', $this->_account->sent_folder);
1558         
1559         $result = $this->_getMessages($folderName);
1560         $message = $this->_getMessageFromSearchResult($result, $messageToSend['subject']);
1561         
1562         $this->assertTrue(! empty($message), 'Sent message not found.');
1563         
1564         return $message;
1565     }
1566     
1567     /**
1568      * returns message array from result
1569      *
1570      * @param array $_result
1571      * @param string $_subject
1572      * @return array
1573      */
1574     protected function _getMessageFromSearchResult($_result, $_subject)
1575     {
1576         $message = array();
1577         foreach ($_result['results'] as $mail) {
1578             if ($mail['subject'] == $_subject) {
1579                 $message = $mail;
1580             }
1581         }
1582         
1583         return $message;
1584     }
1585     
1586     /**
1587      * get messages from folder
1588      *
1589      * @param string $_folderName
1590      * @return array
1591      */
1592     protected function _getMessages($_folderName = 'INBOX')
1593     {
1594         $folder = $this->_getFolder($_folderName);
1595         $filter = $this->_getMessageFilter($folder->getId());
1596         // update cache
1597         $folder = Felamimail_Controller_Cache_Message::getInstance()->updateCache($folder, 10, 1);
1598         $i = 0;
1599         while ($folder->cache_status != Felamimail_Model_Folder::CACHE_STATUS_COMPLETE && $i < 10) {
1600             $folder = Felamimail_Controller_Cache_Message::getInstance()->updateCache($folder, 10);
1601             $i++;
1602         }
1603         $result = $this->_json->searchMessages($filter, '');
1604         //print_r($result);
1605         
1606         return $result;
1607     }
1608     
1609     /**
1610      * search for message defined by subject in folder
1611      *
1612      * @param string $_subject
1613      * @param string $_folderName
1614      * @return string message data
1615      */
1616     protected function _searchForMessageBySubject($_subject, $_folderName = 'INBOX')
1617     {
1618         $result = $this->_getMessages($_folderName);
1619         
1620         $message = array();
1621         foreach ($result['results'] as $mail) {
1622             if ($mail['subject'] == $_subject) {
1623                 $message = $mail;
1624             }
1625         }
1626         $this->assertGreaterThan(0, $result['totalcount'], 'folder is empty');
1627         $this->assertTrue(! empty($message), 'Message not found');
1628         
1629         return $message;
1630     }
1631     
1632     /**
1633      * sieve test helper
1634      *
1635      * @param array $_sieveData
1636      * @return array
1637      */
1638     protected function _sieveTestHelper($_sieveData, $_isMime = FALSE)
1639     {
1640         $this->_setTestScriptname();
1641         
1642         // check which save fn to use
1643         if ((isset($_sieveData['reason']) || array_key_exists('reason', $_sieveData))) {
1644             $resultSet = $this->_json->saveVacation($_sieveData);
1645             $this->assertEquals($this->_account->email, $resultSet['addresses'][0]);
1646             
1647             $_sieveBackend = Felamimail_Backend_SieveFactory::factory($this->_account->getId());
1648             
1649             if (preg_match('/dbmail/i', $_sieveBackend->getImplementation())) {
1650                 $translate = Tinebase_Translation::getTranslation('Felamimail');
1651                 $this->assertEquals(sprintf(
1652                     $translate->_('Out of Office reply from %1$s'), Tinebase_Core::getUser()->accountFullName),
1653                     $resultSet['subject']
1654                 );
1655             } else {
1656                 $this->assertEquals($_sieveData['subject'], $resultSet['subject']);
1657             }
1658             
1659             if ($_isMime) {
1660                 $this->assertEquals(html_entity_decode('unittest vacation&nbsp;message', ENT_NOQUOTES, 'UTF-8'), $resultSet['reason']);
1661             } else {
1662                 $this->assertEquals($_sieveData['reason'], $resultSet['reason']);
1663             }
1664             
1665         } else if ((isset($_sieveData[0]['action_type']) || array_key_exists('action_type', $_sieveData[0]))) {
1666             $resultSet = $this->_json->saveRules($this->_account->getId(), $_sieveData);
1667             $this->assertEquals($_sieveData, $resultSet);
1668         }
1669         
1670         return $resultSet;
1671     }
1672
1673     /**
1674      * search preferences by application felamimail
1675      *
1676      */
1677     public function testSearchFelamimailPreferences()
1678     {
1679         // search prefs
1680         $result = $this->_frontend->searchPreferencesForApplication('Felamimail', '');
1681         
1682         // check results
1683         $this->assertTrue(isset($result['results']));
1684         $this->assertGreaterThan(0, $result['totalcount']);
1685     }
1686     
1687     /**
1688      * testGetRegistryData
1689      *
1690      * @see 0010251: do not send unused config data to client
1691      */
1692     public function testGetRegistryData()
1693     {
1694         $regData = $this->_json->getRegistryData();
1695
1696         $this->assertFalse(isset($regData['defaults']));
1697         $supportedFlags = Felamimail_Config::getInstance()->featureEnabled(Felamimail_Config::FEATURE_TINE20_FLAG)
1698             ? 6
1699             : 5;
1700         $this->assertEquals($supportedFlags, $regData['supportedFlags']['totalcount']);
1701     }
1702 }