diff options
author | Patrick Maia and Victor Shyba <pixelated-team+pmaia+vshyba@thoughtworks.com> | 2014-09-05 22:45:55 +0000 |
---|---|---|
committer | Patrick Maia <pmaia@thoughtworks.com> | 2014-09-05 22:46:17 +0000 |
commit | 3c79a54ab332e15f31a4a57a4a9baabf4b62e26a (patch) | |
tree | ce6b592049227da18b0500f4695b0feb490a944d /service | |
parent | d2cf8b51904420917a5f86986ce7c02e89935998 (diff) |
#51 - persists new tags globally (in a local file) and shows on tag list
Diffstat (limited to 'service')
-rw-r--r-- | service/pixelated/adapter/mail_service.py | 12 | ||||
-rw-r--r-- | service/pixelated/adapter/pixelated_mail.py | 16 | ||||
-rw-r--r-- | service/pixelated/adapter/pixelated_mailbox.py | 37 | ||||
-rw-r--r-- | service/pixelated/adapter/tag.py | 61 | ||||
-rw-r--r-- | service/pixelated/adapter/tag_index.py | 44 | ||||
-rw-r--r-- | service/test/adapter/pixelated_mail_test.py | 8 | ||||
-rw-r--r-- | service/test/adapter/pixelated_mailbox_test.py | 42 | ||||
-rw-r--r-- | service/test/adapter/test_helper.py | 6 | ||||
-rw-r--r-- | service/test/adapter/test_tag.py | 76 |
9 files changed, 185 insertions, 117 deletions
diff --git a/service/pixelated/adapter/mail_service.py b/service/pixelated/adapter/mail_service.py index e3444b9f..2c67ae2d 100644 --- a/service/pixelated/adapter/mail_service.py +++ b/service/pixelated/adapter/mail_service.py @@ -14,8 +14,6 @@ # 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 pixelated.adapter.tag import Tag - class MailService: __slots__ = ['leap_session', 'account', 'mailbox_name'] @@ -41,13 +39,9 @@ class MailService: def update_tags(self, mail_id, new_tags): mail = self.mail(mail_id) - tags = set(Tag(str_tag) for str_tag in new_tags) - current_tags = mail.update_tags(tags) - self._update_mailbox_tags(tags) - return current_tags - - def _update_mailbox_tags(self, tags): - self.mailbox.update_tags(tags) + added, removed = mail.update_tags(new_tags) + self.mailbox.notify_tags_updated(added, removed, self.ident) + return new_tags def mail(self, mail_id): return self.mailbox.mail(mail_id) diff --git a/service/pixelated/adapter/pixelated_mail.py b/service/pixelated/adapter/pixelated_mail.py index cd34fe46..31e8ccc7 100644 --- a/service/pixelated/adapter/pixelated_mail.py +++ b/service/pixelated/adapter/pixelated_mail.py @@ -13,7 +13,6 @@ # # 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 pixelated.adapter.tag import Tag from pixelated.adapter.status import Status import dateutil.parser as dateparser from email.MIMEMultipart import MIMEMultipart @@ -68,34 +67,35 @@ class PixelatedMail: return temporary_headers def _extract_tags(self): - return set(Tag(tag_name) for tag_name in self.headers.get('x-tags', [])) + return set(self.headers.get('x-tags', [])) def update_tags(self, tags): + old_tags = self.tags self.tags = tags + removed = old_tags.difference(tags) + added = tags.difference(old_tags) self._persist_mail_tags(tags) - return self.tags + return added, removed def mark_as_read(self): self.status.add("read") def _persist_mail_tags(self, current_tags): - tags_headers = [tag.name for tag in current_tags] hdoc = self.leap_mail.hdoc - hdoc.content['headers']['X-Tags'] = tags_headers + hdoc.content['headers']['X-Tags'] = current_tags self.leap_mail._soledad.put_doc(hdoc) def has_tag(self, tag): - return Tag(tag) in self.tags + return tag in self.tags def as_dict(self): - tags = [tag.name for tag in self.tags] statuses = [status.name for status in self.status] _headers = self.headers.copy() _headers['date'] = self.date return { 'header': _headers, 'ident': self.ident, - 'tags': tags, + 'tags': self.tags, 'status': statuses, 'security_casing': self.security_casing, 'body': self.body diff --git a/service/pixelated/adapter/pixelated_mailbox.py b/service/pixelated/adapter/pixelated_mailbox.py index 06e0cccc..3424ddd7 100644 --- a/service/pixelated/adapter/pixelated_mailbox.py +++ b/service/pixelated/adapter/pixelated_mailbox.py @@ -17,14 +17,19 @@ from pixelated.adapter.pixelated_mail import PixelatedMail from pixelated.adapter.tag import Tag +from pixelated.adapter.tag_index import TagIndex class PixelatedMailbox: - SPECIAL_TAGS = ['inbox', 'sent', 'drafts', 'trash'] + SPECIAL_TAGS = set([Tag('inbox', True), Tag('sent', True), Tag('drafts', True), Tag('trash', True)]) - def __init__(self, leap_mailbox): + def __init__(self, leap_mailbox, index_file_path): self.leap_mailbox = leap_mailbox + self.tag_index = TagIndex(index_file_path) + for tag in self.SPECIAL_TAGS: + if tag not in self.tag_index.values(): + self.tag_index.set(tag) @property def messages(self): @@ -47,22 +52,18 @@ class PixelatedMailbox: return PixelatedMail.from_leap_mail(message) def all_tags(self): - return Tag.from_flags(self._getFlags()) - - def _getFlags(self): - # XXX Temporary workaround while getFlags from leap is disabled - mbox = self.leap_mailbox._get_mbox_doc() - if not mbox: - return self.leap_mailbox.getFlags() - return mbox.content.get(self.leap_mailbox.FLAGS_KEY, []) - - def update_tags(self, tags): - new_flags = set(tag.to_flag() for tag in tags) - current_flags = set(self._getFlags()) - - flags = tuple(current_flags.union(new_flags)) - self.leap_mailbox.setFlags(flags) + return self.tag_index.values().union(self.SPECIAL_TAGS) + + def notify_tags_updated(self, added_tags, removed_tags, mail_ident): + for removed_tag in removed_tags: + tag = self.tag_index.get(removed_tag) + tag.decrement(mail_ident) + self.tag_index.set(tag) + for added_tag in added_tags: + tag = self.tag_index.get(added_tag) or Tag(added_tag) + tag.increment(mail_ident) + self.tag_index.set(tag) @classmethod def create(cls, account, mailbox_name='INBOX'): - return PixelatedMailbox(account.getMailbox(mailbox_name)) + return PixelatedMailbox(account.getMailbox(mailbox_name), '~/.pixelated_index') diff --git a/service/pixelated/adapter/tag.py b/service/pixelated/adapter/tag.py index 386ccafc..45a972ab 100644 --- a/service/pixelated/adapter/tag.py +++ b/service/pixelated/adapter/tag.py @@ -14,41 +14,32 @@ # You should have received a copy of the GNU Affero General Public License # along with Pixelated. If not, see <http://www.gnu.org/licenses/>. +import json -class Tag: - LEAP_FLAGS_TAGS = { - '\\Deleted': 'trash', - '\\Draft': 'drafts', - '\\Recent': 'inbox' - } +class Tag: @classmethod - def from_flags(cls, flags): - return set(filter(None, (cls.from_flag(flag) for flag in flags))) + def from_dict(cls, tag_dict): + tag = Tag(tag_dict['name'], tag_dict['default']) + tag.mails = tag_dict['mails'] + return tag @classmethod - def from_flag(cls, flag): - if flag in cls.LEAP_FLAGS_TAGS.keys(): - return Tag(cls.LEAP_FLAGS_TAGS[flag]) - if flag.startswith('tag_'): - return Tag(cls._remove_prefix(flag)) - return None + def from_json_string(cls, json_string): + tag_dict = json.loads(json_string) + tag_dict['mails'] = set(tag_dict['mails']) + return Tag.from_dict(tag_dict) - def to_flag(self): - for flag, tag in self.LEAP_FLAGS_TAGS.items(): - if tag == str(self.name): - return flag - return 'tag_' + str(self.name) - - @classmethod - def _remove_prefix(cls, flag_name): - return flag_name.replace('tag_', '', 1) + @property + def total(self): + return len(self.mails) def __init__(self, name, default=False): self.name = name - self.default = default self.ident = name.__hash__() + self.default = default + self.mails = set() def __eq__(self, other): return self.name == other.name @@ -56,18 +47,28 @@ class Tag: def __hash__(self): return self.name.__hash__() + def increment(self, mail_ident): + self.mails.add(mail_ident) + + def decrement(self, mail_ident): + self.mails.discard(mail_ident) + def as_dict(self): return { 'name': self.name, 'default': self.default, 'ident': self.ident, - 'counts': { - 'total': 0, - 'read': 0, - 'starred': 0, - 'replied': 0 - } + 'counts': {'total': self.total, + 'read': 0, + 'starred': 0, + 'replied': 0}, + 'mails': self.mails } + def as_json_string(self): + tag_dict = self.as_dict() + tag_dict['mails'] = list(self.mails) + return json.dumps(tag_dict) + def __repr__(self): return self.name diff --git a/service/pixelated/adapter/tag_index.py b/service/pixelated/adapter/tag_index.py new file mode 100644 index 00000000..6cc2a68d --- /dev/null +++ b/service/pixelated/adapter/tag_index.py @@ -0,0 +1,44 @@ +# +# Copyright (c) 2014 ThoughtWorks, Inc. +# +# Pixelated is free software: you can redistribute it and/or modify +# it under the terms of the GNU Affero General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# Pixelated is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU Affero General Public License for more details. +# +# You should have received a copy of the GNU Affero General Public License +# along with Pixelated. If not, see <http://www.gnu.org/licenses/>. + +import dbm +import atexit +from pixelated.adapter.tag import Tag + + +class TagIndex: + """ + Manages an index for mail's tags using a file storage. + """ + + def __init__(self, filename): + self.db = dbm.open(filename, 'c') + atexit.register(self.close_db) + + def set(self, tag): + self.db[tag.name] = tag.as_json_string() + + def get(self, tag_name): + if tag_name in self.db: + return Tag.from_json_string(self.db.get(tag_name)) + else: + return None + + def values(self): + return set(self.get(key) for key in self.db.keys()) + + def close_db(self): + self.db.close() diff --git a/service/test/adapter/pixelated_mail_test.py b/service/test/adapter/pixelated_mail_test.py index 7de7c96a..557c3f6a 100644 --- a/service/test/adapter/pixelated_mail_test.py +++ b/service/test/adapter/pixelated_mail_test.py @@ -16,7 +16,6 @@ import unittest from pixelated.adapter.pixelated_mail import PixelatedMail -from pixelated.adapter.tag import Tag import test_helper @@ -65,10 +64,11 @@ class TestPixelatedMail(unittest.TestCase): self.assertEqual(mail.tags, ['sent']) self.assertEqual(mail.body, 'Este \xe9 o corpo') - def test_update_tags_return_a_set_for_current_tags(self): + def test_update_tags_return_a_set_for_added_tags_and_a_set_for_removed_ones(self): pixelated_mail = PixelatedMail.from_leap_mail(test_helper.leap_mail(extra_headers={'X-tags': ['custom_1', 'custom_2']})) - current_tags = pixelated_mail.update_tags(set([Tag('custom_1'), Tag('custom_3')])) - self.assertEquals(set([Tag('custom_3'), Tag('custom_1')]), current_tags) + added, removed = pixelated_mail.update_tags(set(['custom_1', 'custom_3'])) + self.assertEquals(set(['custom_3']), added) + self.assertEquals(set(['custom_2']), removed) def test_to_mime_multipart(self): mail = PixelatedMail.from_dict(self.mail_dict) diff --git a/service/test/adapter/pixelated_mailbox_test.py b/service/test/adapter/pixelated_mailbox_test.py index 1047eed4..2944275f 100644 --- a/service/test/adapter/pixelated_mailbox_test.py +++ b/service/test/adapter/pixelated_mailbox_test.py @@ -14,28 +14,42 @@ # You should have received a copy of the GNU Affero General Public License # along with Pixelated. If not, see <http://www.gnu.org/licenses/>. import unittest +import os import test_helper from pixelated.adapter.tag import Tag +from pixelated.adapter.tag_index import TagIndex from pixelated.adapter.pixelated_mailbox import PixelatedMailbox class TestPixelatedMailbox(unittest.TestCase): - def test_retrieve_all_tags_from_mailbox(self): - leap_flags = ['\\Deleted', '\\Draft', '\\Recent', 'tag_custom', 'should_ignore_all_from_here', 'List'] - mailbox = PixelatedMailbox(test_helper.leap_mailbox(leap_flags=leap_flags)) - - self.assertEquals(set([Tag('trash'), Tag('inbox'), Tag('drafts'), Tag('custom')]), mailbox.all_tags()) + def setUp(self): + self.db_file_path = '/tmp/test_tags' - def test_new_tags_are_added_to_mailbox(self): - leap_flags = ['\\Deleted', 'tag_custom_one', 'tag_custom_two'] - leap_mailbox_mock = test_helper.leap_mailbox(leap_flags=leap_flags) - mailbox = PixelatedMailbox(leap_mailbox_mock) - tags = [Tag('custom_one'), Tag('custom_three')] - mailbox.update_tags(tags) + def tearDown(self): + os.remove(self.db_file_path + '.db') - expected = set(('\\Deleted', 'tag_custom_one', 'tag_custom_two', 'tag_custom_three')) - actual_args = set(leap_mailbox_mock.setFlags.call_args[0][0]) + def test_special_tags_always_exists(self): + mailbox = PixelatedMailbox(test_helper.leap_mailbox(), self.db_file_path) + self.assertEquals(mailbox.SPECIAL_TAGS, mailbox.all_tags()) - self.assertEquals(expected, actual_args) + def test_retrieve_all_tags_from_mailbox(self): + tag_index = TagIndex(self.db_file_path) + tag_index.set(Tag('one_tag')) + tag_index.set(Tag('two_tag')) + tag_index.close_db() + mailbox = PixelatedMailbox(test_helper.leap_mailbox(), self.db_file_path) + expected_tags = set([Tag('one_tag'), Tag('two_tag')]).union(mailbox.SPECIAL_TAGS) + self.assertEquals(expected_tags, mailbox.all_tags()) + + def test_notify_tags_updated_method_properly_changes_tags_state(self): + tag_index = TagIndex(self.db_file_path) + tag = Tag('one_tag') + tag.increment(12) + tag_index.set(tag) + tag_index.close_db() + mailbox = PixelatedMailbox(test_helper.leap_mailbox(), self.db_file_path) + self.assertEquals(0, mailbox.tag_index.get('inbox').total) + mailbox.notify_tags_updated(set(['inbox']), set(['one_tag']), 12) + self.assertEquals(1, mailbox.tag_index.get('inbox').total) diff --git a/service/test/adapter/test_helper.py b/service/test/adapter/test_helper.py index 756013ab..70673481 100644 --- a/service/test/adapter/test_helper.py +++ b/service/test/adapter/test_helper.py @@ -49,8 +49,6 @@ def leap_mail(uid=0, flags=LEAP_FLAGS, headers=DEFAULT_HEADERS, extra_headers={} hdoc=Mock(content={'headers': headers})) -def leap_mailbox(leap_flags=LEAP_FLAGS, extra_flags=[]): - flags = leap_flags + extra_flags - return Mock(getFlags=Mock(return_value=flags), - _get_mbox_doc=Mock(return_value=None), +def leap_mailbox(): + return Mock(_get_mbox_doc=Mock(return_value=None), messages=[leap_mail(uid=6)]) diff --git a/service/test/adapter/test_tag.py b/service/test/adapter/test_tag.py index b9b502d4..1f0478d3 100644 --- a/service/test/adapter/test_tag.py +++ b/service/test/adapter/test_tag.py @@ -20,42 +20,58 @@ from pixelated.adapter.tag import Tag class TestTag(unittest.TestCase): - def test_leap_recent_flag_is_translated_to_inbox_tag(self): - tag = Tag.from_flag('\\Recent') - self.assertEquals(Tag('inbox'), tag) + def test_from_dict_sets_all_tag_attributes(self): + tag_dict = {'name': 'a_tag', + 'default': False, + 'counts': {'total': 3, + 'read': 1, + 'starred': 1, + 'replied': 1}, + 'mails': set([1, 2, 3])} - def test_leap_deleted_flag_is_translated_to_trash_tag(self): - tag = Tag.from_flag('\\Deleted') - self.assertEquals(Tag('trash'), tag) + tag = Tag.from_dict(tag_dict) - def test_leap_draft_flag_is_translated_to_draft_tag(self): - tag = Tag.from_flag('\\Draft') - self.assertEquals(Tag('drafts'), tag) + self.assertEquals(tag_dict['name'], tag.name) + self.assertEquals(tag_dict['default'], tag.default) + self.assertEquals(tag_dict['counts']['total'], tag.total) + self.assertEquals(tag_dict['mails'], tag.mails) - def test_leap_flags_that_are_custom_tags_are_handled(self): - tag = Tag.from_flag('tag_work') - self.assertEquals(Tag('work'), tag) + def test_as_dict_puts_all_tag_attributes_in_the_returning_dict(self): + tag = Tag('some_tag', default=True) + tag.counts = {'total': 0, 'read': 0, 'starred': 0, 'replied': 0} + tag.mails = set([1, 2, 3]) - def test_custom_tags_containing_our_prefix_are_handled(self): - tag = Tag.from_flag('tag_tag_work_tag_') - self.assertEquals(Tag('tag_work_tag_'), tag) + tag_dict = tag.as_dict() - def test_bulk_conversion(self): - tags = Tag.from_flags(['\\Answered', '\\Seen', '\\Recent', 'tag_a_custom', 'List']) - self.assertEquals(set([Tag('inbox'), Tag('a_custom')]), tags) + self.assertEquals(tag.name, tag_dict['name']) + self.assertEquals(tag.default, tag_dict['default']) + self.assertEquals(tag.total, tag_dict['counts']['total']) + self.assertEquals(tag.mails, tag_dict['mails']) - def test_inbox_tag_is_translated_to_leap_recent_flag(self): - flag = Tag('inbox').to_flag() - self.assertEquals('\\Recent', flag) + def test_increments_total_count_and_adds_mails_id_to_mails(self): + tag = Tag('another') + tag.increment(12) - def test_trash_tag_is_translated_to_leap_deleted_flag(self): - flag = Tag('trash').to_flag() - self.assertEquals('\\Deleted', flag) + self.assertIn(12, tag.mails) + self.assertEquals(1, tag.total) - def test_drafts_tag_is_translated_to_leap_draft_flag(self): - flag = Tag('drafts').to_flag() - self.assertEquals('\\Draft', flag) + def test_decrement_does_nothing_if_mail_has_not_the_tag(self): + tag = Tag('tag') + tag.decrement(2000) - def test_custom_tag_has_prefix_when_translated_to_flag(self): - flag = Tag('work').to_flag() - self.assertEquals('tag_work', flag) + self.assertEquals(0, tag.total) + + def test_increment_does_nothing_if_mail_already_has_the_tag(self): + tag = Tag('tag') + tag.mails = set([12]) + tag.increment(12) + + self.assertEquals(1, tag.total) + + def test_decrements_total_count_and_removes_mails_id_from_mails(self): + tag = Tag('one_more') + tag.mails = set([12]) + tag.decrement(12) + + self.assertNotIn(12, tag.mails) + self.assertEquals(0, tag.total) |