def test_condstore_flags_refresh(db, default_account, all_mail_folder, mock_imapclient, monkeypatch): monkeypatch.setattr( 'inbox.mailsync.backends.imap.generic.CONDSTORE_FLAGS_REFRESH_BATCH_SIZE', 10) uid_dict = uids.example() mock_imapclient.add_folder_data(all_mail_folder.name, uid_dict) mock_imapclient.capabilities = lambda: ['CONDSTORE'] folder_sync_engine = FolderSyncEngine(default_account.id, default_account.namespace.id, all_mail_folder.name, default_account.email_address, 'gmail', BoundedSemaphore(1)) folder_sync_engine.initial_sync() # Change the labels provided by the mock IMAP server for k, v in mock_imapclient._data[all_mail_folder.name].items(): v['X-GM-LABELS'] = ('newlabel',) v['MODSEQ'] = (k,) folder_sync_engine.highestmodseq = 0 # Don't sleep at the end of poll_impl before returning. folder_sync_engine.poll_frequency = 0 folder_sync_engine.poll_impl() imapuids = db.session.query(ImapUid). \ filter_by(folder_id=all_mail_folder.id).all() for imapuid in imapuids: assert 'newlabel' in [l.name for l in imapuid.labels] assert folder_sync_engine.highestmodseq == mock_imapclient.folder_status( all_mail_folder.name, ['HIGHESTMODSEQ'])['HIGHESTMODSEQ']
def test_generic_flags_refresh_expunges_transient_uids( db, generic_account, inbox_folder, mock_imapclient, monkeypatch): # Check that we delete UIDs which are synced but quickly deleted, so never # show up in flags refresh. uid_dict = uids.example() mock_imapclient.add_folder_data(inbox_folder.name, uid_dict) inbox_folder.imapfolderinfo = ImapFolderInfo(account=generic_account, uidvalidity=1, uidnext=1) db.session.commit() folder_sync_engine = FolderSyncEngine(generic_account.id, generic_account.namespace.id, inbox_folder.name, generic_account.email_address, 'custom', BoundedSemaphore(1)) folder_sync_engine.initial_sync() # Don't sleep at the end of poll_impl before returning. folder_sync_engine.poll_frequency = 0 folder_sync_engine.poll_impl() msg = db.session.query(Message).filter_by( namespace_id=generic_account.namespace.id).first() transient_uid = ImapUid(folder=inbox_folder, account=generic_account, message=msg, msg_uid=max(uid_dict) + 1) db.session.add(transient_uid) db.session.commit() folder_sync_engine.last_slow_refresh = None folder_sync_engine.poll_impl() with pytest.raises(ObjectDeletedError): transient_uid.id
def test_generic_flags_refresh_expunges_transient_uids(db, generic_account, inbox_folder, mock_imapclient, monkeypatch): # Check that we delete UIDs which are synced but quickly deleted, so never # show up in flags refresh. uid_dict = uids.example() mock_imapclient.add_folder_data(inbox_folder.name, uid_dict) inbox_folder.imapfolderinfo = ImapFolderInfo(account=generic_account, uidvalidity=1, uidnext=1) db.session.commit() folder_sync_engine = FolderSyncEngine(generic_account.id, generic_account.namespace.id, inbox_folder.name, generic_account.email_address, 'custom', BoundedSemaphore(1)) folder_sync_engine.initial_sync() # Don't sleep at the end of poll_impl before returning. folder_sync_engine.poll_frequency = 0 folder_sync_engine.poll_impl() msg = db.session.query(Message).filter_by( namespace_id=generic_account.namespace.id).first() transient_uid = ImapUid(folder=inbox_folder, account=generic_account, message=msg, msg_uid=max(uid_dict) + 1) db.session.add(transient_uid) db.session.commit() folder_sync_engine.last_slow_refresh = None folder_sync_engine.poll_impl() with pytest.raises(ObjectDeletedError): transient_uid.id
def test_condstore_flags_refresh(db, default_account, all_mail_folder, mock_imapclient, monkeypatch): monkeypatch.setattr( 'inbox.mailsync.backends.imap.generic.CONDSTORE_FLAGS_REFRESH_BATCH_SIZE', 10) uid_dict = uids.example() mock_imapclient.add_folder_data(all_mail_folder.name, uid_dict) mock_imapclient.capabilities = lambda: ['CONDSTORE'] folder_sync_engine = FolderSyncEngine(default_account.id, default_account.namespace.id, all_mail_folder.name, default_account.email_address, 'gmail', BoundedSemaphore(1)) folder_sync_engine.initial_sync() # Change the labels provided by the mock IMAP server for k, v in mock_imapclient._data[all_mail_folder.name].items(): v['X-GM-LABELS'] = ('newlabel', ) v['MODSEQ'] = (k, ) folder_sync_engine.highestmodseq = 0 # Don't sleep at the end of poll_impl before returning. folder_sync_engine.poll_frequency = 0 folder_sync_engine.poll_impl() imapuids = db.session.query(ImapUid). \ filter_by(folder_id=all_mail_folder.id).all() for imapuid in imapuids: assert 'newlabel' in [l.name for l in imapuid.labels] assert folder_sync_engine.highestmodseq == mock_imapclient.folder_status( all_mail_folder.name, ['HIGHESTMODSEQ'])['HIGHESTMODSEQ']
def test_gmail_initial_sync(db, default_account, all_mail_folder, mock_imapclient): uid_dict = uids.example() mock_imapclient.add_folder_data(all_mail_folder.name, uid_dict) mock_imapclient.list_folders = lambda: [(('\\All', '\\HasNoChildren',), '/', u'[Gmail]/All Mail')] mock_imapclient.idle = lambda: None mock_imapclient.idle_check = raise_imap_error folder_sync_engine = GmailFolderSyncEngine(default_account.id, default_account.namespace.id, all_mail_folder.name, default_account.email_address, 'gmail', BoundedSemaphore(1)) folder_sync_engine.initial_sync() saved_uids = db.session.query(ImapUid).filter( ImapUid.folder_id == all_mail_folder.id) assert {u.msg_uid for u in saved_uids} == set(uid_dict)
def test_new_uids_synced_when_polling(db, generic_account, inbox_folder, mock_imapclient): uid_dict = uids.example() mock_imapclient.add_folder_data(inbox_folder.name, uid_dict) inbox_folder.imapfolderinfo = ImapFolderInfo(account=generic_account, uidvalidity=1, uidnext=1) db.session.commit() folder_sync_engine = FolderSyncEngine(generic_account.id, generic_account.namespace.id, inbox_folder.name, generic_account.email_address, 'custom', BoundedSemaphore(1)) # Don't sleep at the end of poll_impl before returning. folder_sync_engine.poll_frequency = 0 folder_sync_engine.poll_impl() saved_uids = db.session.query(ImapUid).filter( ImapUid.folder_id == inbox_folder.id) assert {u.msg_uid for u in saved_uids} == set(uid_dict)
def test_gmail_initial_sync(db, default_account, all_mail_folder, mock_imapclient): uid_dict = uids.example() mock_imapclient.add_folder_data(all_mail_folder.name, uid_dict) mock_imapclient.list_folders = lambda: [(( '\\All', '\\HasNoChildren', ), '/', u'[Gmail]/All Mail')] mock_imapclient.idle = lambda: None mock_imapclient.idle_check = raise_imap_error folder_sync_engine = GmailFolderSyncEngine(default_account.id, default_account.namespace.id, all_mail_folder.name, default_account.email_address, 'gmail', BoundedSemaphore(1)) folder_sync_engine.initial_sync() saved_uids = db.session.query(ImapUid).filter( ImapUid.folder_id == all_mail_folder.id) assert {u.msg_uid for u in saved_uids} == set(uid_dict)
def test_handle_uidinvalid(db, generic_account, inbox_folder, mock_imapclient): uid_dict = uids.example() mock_imapclient.add_folder_data(inbox_folder.name, uid_dict) inbox_folder.imapfolderinfo = ImapFolderInfo(account=generic_account, uidvalidity=1, uidnext=1) db.session.commit() folder_sync_engine = FolderSyncEngine(generic_account.id, generic_account.namespace.id, inbox_folder.name, generic_account.email_address, 'custom', BoundedSemaphore(1)) folder_sync_engine.initial_sync() mock_imapclient.uidvalidity = 2 with pytest.raises(UidInvalid): folder_sync_engine.poll_impl() new_state = folder_sync_engine.resync_uids() assert new_state == 'initial' assert db.session.query(ImapUid).filter( ImapUid.folder_id == inbox_folder.id).all() == []
def test_handle_uidinvalid_loops(db, generic_account, inbox_folder, mock_imapclient, monkeypatch): import inbox.mailsync.backends.imap.generic as generic_import mock_imapclient.uidvalidity = 1 # We're using a list here because of weird monkeypatching shenanigans. uidinvalid_count = [] def fake_poll_function(self): uidinvalid_count.append(1) raise UidInvalid monkeypatch.setattr("inbox.mailsync.backends.imap.generic.FolderSyncEngine.poll", fake_poll_function) uid_dict = uids.example() mock_imapclient.add_folder_data(inbox_folder.name, uid_dict) inbox_folder.imapfolderinfo = ImapFolderInfo(account=generic_account, uidvalidity=1, uidnext=1) db.session.commit() folder_sync_engine = generic_import.FolderSyncEngine(generic_account.id, generic_account.namespace.id, inbox_folder.name, generic_account.email_address, 'custom', BoundedSemaphore(1)) folder_sync_engine.state = 'poll' db.session.expunge(inbox_folder.imapsyncstatus) with pytest.raises(MailsyncDone): folder_sync_engine._run() assert len(uidinvalid_count) == MAX_UIDINVALID_RESYNCS + 1
def test_initial_sync(db, generic_account, inbox_folder, mock_imapclient): # We should really be using hypothesis.given() to generate lots of # different uid sets, but it's not trivial to ensure that no state is # carried over between runs. This will have to suffice for now as a way to # at least establish coverage. uid_dict = uids.example() mock_imapclient.add_folder_data(inbox_folder.name, uid_dict) folder_sync_engine = FolderSyncEngine(generic_account.id, generic_account.namespace.id, inbox_folder.name, generic_account.email_address, 'custom', BoundedSemaphore(1)) folder_sync_engine.initial_sync() saved_uids = db.session.query(ImapUid).filter( ImapUid.folder_id == inbox_folder.id) assert {u.msg_uid for u in saved_uids} == set(uid_dict) saved_message_hashes = {u.message.data_sha256 for u in saved_uids} assert saved_message_hashes == {sha256(v['BODY[]']).hexdigest() for v in uid_dict.values()}
def test_initial_sync(db, generic_account, inbox_folder, mock_imapclient): # We should really be using hypothesis.given() to generate lots of # different uid sets, but it's not trivial to ensure that no state is # carried over between runs. This will have to suffice for now as a way to # at least establish coverage. uid_dict = uids.example() mock_imapclient.add_folder_data(inbox_folder.name, uid_dict) folder_sync_engine = FolderSyncEngine(generic_account.id, generic_account.namespace.id, inbox_folder.name, generic_account.email_address, 'custom', BoundedSemaphore(1)) folder_sync_engine.initial_sync() saved_uids = db.session.query(ImapUid).filter( ImapUid.folder_id == inbox_folder.id) assert {u.msg_uid for u in saved_uids} == set(uid_dict) saved_message_hashes = {u.message.data_sha256 for u in saved_uids} assert saved_message_hashes == { sha256(v['BODY[]']).hexdigest() for v in uid_dict.values() }
def test_handle_uidinvalid_loops(db, generic_account, inbox_folder, mock_imapclient, monkeypatch): import inbox.mailsync.backends.imap.generic as generic_import mock_imapclient.uidvalidity = 1 # We're using a list here because of weird monkeypatching shenanigans. uidinvalid_count = [] def fake_poll_function(self): uidinvalid_count.append(1) raise UidInvalid monkeypatch.setattr( "inbox.mailsync.backends.imap.generic.FolderSyncEngine.poll", fake_poll_function) uid_dict = uids.example() mock_imapclient.add_folder_data(inbox_folder.name, uid_dict) inbox_folder.imapfolderinfo = ImapFolderInfo(account=generic_account, uidvalidity=1, uidnext=1) db.session.commit() folder_sync_engine = generic_import.FolderSyncEngine( generic_account.id, generic_account.namespace.id, inbox_folder.name, generic_account.email_address, 'custom', BoundedSemaphore(1)) folder_sync_engine.state = 'poll' db.session.expunge(inbox_folder.imapsyncstatus) with pytest.raises(MailsyncDone): folder_sync_engine._run() assert len(uidinvalid_count) == MAX_UIDINVALID_RESYNCS + 1