diff options
-rw-r--r-- | service/pixelated/adapter/model/mail.py | 2 | ||||
-rw-r--r-- | service/pixelated/adapter/services/draft_service.py | 13 | ||||
-rw-r--r-- | service/pixelated/adapter/services/mail_service.py | 17 | ||||
-rw-r--r-- | service/pixelated/adapter/services/mailbox.py | 7 | ||||
-rw-r--r-- | service/pixelated/adapter/soledad/soledad_writer_mixin.py | 7 | ||||
-rw-r--r-- | service/pixelated/resources/mails_resource.py | 18 | ||||
-rw-r--r-- | service/test/integration/test_drafts.py | 68 |
7 files changed, 72 insertions, 60 deletions
diff --git a/service/pixelated/adapter/model/mail.py b/service/pixelated/adapter/model/mail.py index 552c2011..3cf97111 100644 --- a/service/pixelated/adapter/model/mail.py +++ b/service/pixelated/adapter/model/mail.py @@ -419,7 +419,7 @@ class PixelatedMail(Mail): self.fdoc.content['mbox'] = mailbox_name def remove_all_tags(self): - self.update_tags(set([])) + return self.update_tags(set([])) @defer.inlineCallbacks def update_tags(self, tags): diff --git a/service/pixelated/adapter/services/draft_service.py b/service/pixelated/adapter/services/draft_service.py index c8df0a05..5a0ee5f3 100644 --- a/service/pixelated/adapter/services/draft_service.py +++ b/service/pixelated/adapter/services/draft_service.py @@ -13,6 +13,7 @@ # # You should have received a copy of the GNU Affero General Public License # along with Pixelated. If not, see <http://www.gnu.org/licenses/>. +from twisted.internet import defer class DraftService(object): @@ -21,11 +22,13 @@ class DraftService(object): def __init__(self, mailboxes): self._mailboxes = mailboxes + @defer.inlineCallbacks def create_draft(self, input_mail): - pixelated_mail = self._mailboxes.drafts.add(input_mail) - return pixelated_mail + pixelated_mail = yield (yield self._mailboxes.drafts).add(input_mail) + defer.returnValue(pixelated_mail) + @defer.inlineCallbacks def update_draft(self, ident, input_mail): - pixelated_mail = self.create_draft(input_mail) - self._mailboxes.drafts.remove(ident) - return pixelated_mail + pixelated_mail = yield self.create_draft(input_mail) + yield (yield self._mailboxes.drafts).remove(ident) + defer.returnValue(pixelated_mail) diff --git a/service/pixelated/adapter/services/mail_service.py b/service/pixelated/adapter/services/mail_service.py index f98020a2..3dfbc9a7 100644 --- a/service/pixelated/adapter/services/mail_service.py +++ b/service/pixelated/adapter/services/mail_service.py @@ -69,24 +69,23 @@ class MailService(object): def attachment(self, attachment_id, encoding): return self.querier.attachment(attachment_id, encoding) + @defer.inlineCallbacks def mail_exists(self, mail_id): - return not(not(self.querier.get_header_by_chash(mail_id))) + defer.returnValue(not(not((yield self.querier.get_header_by_chash(mail_id))))) + @defer.inlineCallbacks def send_mail(self, content_dict): mail = InputMail.from_dict(content_dict) draft_id = content_dict.get('ident') - def move_to_sent(_): - return self.move_to_sent(draft_id, mail) - - deferred = self.mail_sender.sendmail(mail) - deferred.addCallback(move_to_sent) - return deferred + yield self.mail_sender.sendmail(mail) + yield self.move_to_sent(draft_id, mail) + @defer.inlineCallbacks def move_to_sent(self, last_draft_ident, mail): if last_draft_ident: - self.mailboxes.drafts.remove(last_draft_ident) - return self.mailboxes.sent.add(mail) + yield (yield self.mailboxes.drafts).remove(last_draft_ident) + defer.returnValue((yield (yield self.mailboxes.sent).add(mail))) def mark_as_read(self, mail_id): mail = self.mail(mail_id) diff --git a/service/pixelated/adapter/services/mailbox.py b/service/pixelated/adapter/services/mailbox.py index 479d0ee7..3ed311b2 100644 --- a/service/pixelated/adapter/services/mailbox.py +++ b/service/pixelated/adapter/services/mailbox.py @@ -39,11 +39,12 @@ class Mailbox(object): self.search_engine.index_mail(added_mail) defer.returnValue(added_mail) + @defer.inlineCallbacks def remove(self, ident): - mail = self.querier.mail(ident) + mail = yield self.querier.mail(ident) self.search_engine.remove_from_index(mail.ident) - mail.remove_all_tags() - self.querier.remove_mail(mail) + yield mail.remove_all_tags() + yield self.querier.remove_mail(mail) @classmethod def create(cls, mailbox_name, soledad_querier, search_engine): diff --git a/service/pixelated/adapter/soledad/soledad_writer_mixin.py b/service/pixelated/adapter/soledad/soledad_writer_mixin.py index bce7d083..2e45b7d1 100644 --- a/service/pixelated/adapter/soledad/soledad_writer_mixin.py +++ b/service/pixelated/adapter/soledad/soledad_writer_mixin.py @@ -46,8 +46,9 @@ class SoledadWriterMixin(SoledadDbFacadeMixin, object): defer.returnValue((yield self.mail(mail.ident))) + @defer.inlineCallbacks def remove_mail(self, mail): # FIX-ME: Must go through all the part_map phash to delete all the cdocs - self.delete_doc(mail.fdoc) - self.delete_doc(mail.hdoc) - self.delete_doc(mail.bdoc) + yield self.delete_doc(mail.fdoc) + yield self.delete_doc(mail.hdoc) + yield self.delete_doc(mail.bdoc) diff --git a/service/pixelated/resources/mails_resource.py b/service/pixelated/resources/mails_resource.py index 7c9f8cc5..ef112900 100644 --- a/service/pixelated/resources/mails_resource.py +++ b/service/pixelated/resources/mails_resource.py @@ -128,11 +128,19 @@ class MailsResource(Resource): _mail = InputMail.from_dict(content_dict) draft_id = content_dict.get('ident') + def defer_response(deferred): + deferred.addCallback(lambda pixelated_mail: respond_json_deferred({'ident': pixelated_mail.ident}, request)) + if draft_id: - if not self._mail_service.mail_exists(draft_id): - return respond_json("", request, status_code=422) - pixelated_mail = self._draft_service.update_draft(draft_id, _mail) + deferred_check = self._mail_service.mail_exists(draft_id) + + def return422otherwise(mail_exists): + if not mail_exists: + respond_json_deferred("", request, status_code=422) + else: + defer_response(self._draft_service.update_draft(draft_id, _mail)) + deferred_check.addCallback(return422otherwise) else: - pixelated_mail = self._draft_service.create_draft(_mail) + defer_response(self._draft_service.create_draft(_mail)) - return respond_json({'ident': pixelated_mail.ident}, request) + return server.NOT_DONE_YET diff --git a/service/test/integration/test_drafts.py b/service/test/integration/test_drafts.py index 3a0f120b..d0505d75 100644 --- a/service/test/integration/test_drafts.py +++ b/service/test/integration/test_drafts.py @@ -16,7 +16,7 @@ from test.support.integration import SoledadTestBase, MailBuilder from mockito import unstub, when, any -from twisted.internet.defer import Deferred +from twisted.internet import defer class DraftsTest(SoledadTestBase): @@ -24,14 +24,15 @@ class DraftsTest(SoledadTestBase): def tearDown(self): unstub() + @defer.inlineCallbacks def test_post_sends_mail_and_deletes_previous_draft_if_it_exists(self): - # act is if sending the mail by SMTP succeeded - sendmail_deferred = Deferred() + # act as if sending the mail by SMTP succeeded + sendmail_deferred = defer.Deferred() when(self.mail_sender).sendmail(any()).thenReturn(sendmail_deferred) # creates one draft first_draft = MailBuilder().with_subject('First draft').build_json() - first_draft_ident = self.put_mail(first_draft)[0]['ident'] + first_draft_ident = (yield self.put_mail(first_draft)[0])['ident'] # sends an updated version of the draft second_draft = MailBuilder().with_subject('Second draft').with_ident(first_draft_ident).build_json() @@ -39,69 +40,68 @@ class DraftsTest(SoledadTestBase): sendmail_deferred.callback(None) # SMTP succeeded - def onSuccess(mail): - sent_mails = self.get_mails_by_tag('sent') - drafts = self.get_mails_by_tag('drafts') + yield deferred_res - # make sure there is one email in the sent mailbox and it is the second draft - self.assertEquals(1, len(sent_mails)) - self.assertEquals('Second draft', sent_mails[0].subject) + sent_mails = yield self.get_mails_by_tag('sent') + drafts = yield self.get_mails_by_tag('drafts') - # make sure that there are no drafts in the draft mailbox - self.assertEquals(0, len(drafts)) + # make sure there is one email in the sent mailbox and it is the second draft + self.assertEquals(1, len(sent_mails)) + self.assertEquals('Second draft', sent_mails[0].subject) - deferred_res.addCallback(onSuccess) - return deferred_res + # make sure that there are no drafts in the draft mailbox + self.assertEquals(0, len(drafts)) + @defer.inlineCallbacks def test_post_sends_mail_even_when_draft_does_not_exist(self): - # act is if sending the mail by SMTP succeeded - sendmail_deferred = Deferred() + # act as if sending the mail by SMTP succeeded + sendmail_deferred = defer.Deferred() when(self.mail_sender).sendmail(any()).thenReturn(sendmail_deferred) first_draft = MailBuilder().with_subject('First draft').build_json() - deferred_res = self.post_mail(first_draft) + res = self.post_mail(first_draft) sendmail_deferred.callback(True) + yield res - def onSuccess(result): - sent_mails = self.get_mails_by_tag('sent') - drafts = self.get_mails_by_tag('drafts') - - self.assertEquals(1, len(sent_mails)) - self.assertEquals('First draft', sent_mails[0].subject) - self.assertEquals(0, len(drafts)) + sent_mails = yield self.get_mails_by_tag('sent') + drafts = yield self.get_mails_by_tag('drafts') - deferred_res.addCallback(onSuccess) - return deferred_res + self.assertEquals(1, len(sent_mails)) + self.assertEquals('First draft', sent_mails[0].subject) + self.assertEquals(0, len(drafts)) def post_mail(self, data): deferred_res, req = self.post('/mails', data) - deferred_res.callback(None) return deferred_res + @defer.inlineCallbacks def test_put_creates_a_draft_if_it_does_not_exist(self): mail = MailBuilder().with_subject('A new draft').build_json() - self.put_mail(mail) - mails = self.get_mails_by_tag('drafts') + yield self.put_mail(mail)[0] + mails = yield self.get_mails_by_tag('drafts') self.assertEquals('A new draft', mails[0].subject) + @defer.inlineCallbacks def test_put_updates_draft_if_it_already_exists(self): draft = MailBuilder().with_subject('First draft').build_json() - draft_ident = self.put_mail(draft)[0]['ident'] + draft_ident = (yield self.put_mail(draft)[0])['ident'] updated_draft = MailBuilder().with_subject('First draft edited').with_ident(draft_ident).build_json() - self.put_mail(updated_draft) + yield self.put_mail(updated_draft)[0] - drafts = self.get_mails_by_tag('drafts') + drafts = yield self.get_mails_by_tag('drafts') self.assertEquals(1, len(drafts)) self.assertEquals('First draft edited', drafts[0].subject) + @defer.inlineCallbacks def test_respond_unprocessable_entity_if_draft_to_remove_doesnt_exist(self): draft = MailBuilder().with_subject('First draft').build_json() - self.put_mail(draft) + yield self.put_mail(draft)[0] updated_draft = MailBuilder().with_subject('First draft edited').with_ident('NOTFOUND').build_json() - _, request = self.put_mail(updated_draft) + response, request = self.put_mail(updated_draft) + yield response self.assertEquals(422, request.code) |