Ejemplo n.º 1
0
 def setUp(self):
     self.user = User.objects.create_user('testuser', '*****@*****.**',
                                          'testPass')
     self.client.login(username='******', password='******')
     store = kittystore.get_store(SettingsModule(), debug=False)
     ml = FakeList("*****@*****.**")
     ml.subject_prefix = u"[example] "
     # Create 3 threads
     messages = []
     for msgnum in range(3):
         msg = Message()
         msg["From"] = "*****@*****.**"
         msg["Message-ID"] = "<id%d>" % (msgnum + 1)
         msg["Subject"] = "Dummy message"
         msg.set_payload("Dummy message")
         store.add_to_list(ml, msg)
         messages.append(msg)
     # 1st is unread, 2nd is read, 3rd is updated
     LastView.objects.create(list_address="*****@*****.**",
                             user=self.user,
                             threadid=get_message_id_hash("<id2>"))
     LastView.objects.create(list_address="*****@*****.**",
                             user=self.user,
                             threadid=get_message_id_hash("<id3>"))
     msg4 = Message()
     msg4["From"] = "*****@*****.**"
     msg4["Message-ID"] = "<id4>"
     msg4["Subject"] = "Dummy message"
     msg4["In-Reply-To"] = "<id3>"
     msg4.set_payload("Dummy message")
     store.add_to_list(ml, msg4)
     # Factory
     defaults = {"kittystore.store": store, "HTTP_USER_AGENT": "testbot"}
     self.factory = RequestFactory(**defaults)
 def test_list_complex_rule_deletion(self):
     # Test that the mailing-list header-match complex rules are read
     # properly after deletion.
     chain = config.chains['header-match']
     header_matches = IHeaderMatchList(self._mlist)
     header_matches.append('Foo', 'a+', 'reject')
     header_matches.append('Bar', 'b+', 'discard')
     header_matches.append('Baz', 'z+', 'accept')
     links = [
         link for link in chain.get_links(self._mlist, Message(), {})
         if link.rule.name != 'any'
     ]
     self.assertEqual(len(links), 3)
     self.assertEqual([
         (link.rule.header, link.rule.pattern, link.action, link.chain.name)
         for link in links
     ], [
         ('foo', 'a+', LinkAction.jump, 'reject'),
         ('bar', 'b+', LinkAction.jump, 'discard'),
         ('baz', 'z+', LinkAction.jump, 'accept'),
     ])  # noqa: E124
     del header_matches[0]
     links = [
         link for link in chain.get_links(self._mlist, Message(), {})
         if link.rule.name != 'any'
     ]
     self.assertEqual(len(links), 2)
     self.assertEqual([
         (link.rule.header, link.rule.pattern, link.action, link.chain.name)
         for link in links
     ], [
         ('bar', 'b+', LinkAction.jump, 'discard'),
         ('baz', 'z+', LinkAction.jump, 'accept'),
     ])  # noqa: E124
Ejemplo n.º 3
0
    def test_thread_neighbors(self):
        ml = FakeList("example-list")
        # Create 3 threads
        msg_t1_1 = Message()
        msg_t1_1["From"] = "*****@*****.**"
        msg_t1_1["Message-ID"] = "<id1_1>"
        msg_t1_1.set_payload("Dummy message")
        self.store.add_to_list(ml, msg_t1_1)
        msg_t2_1 = Message()
        msg_t2_1["From"] = "*****@*****.**"
        msg_t2_1["Message-ID"] = "<id2_1>"
        msg_t2_1.set_payload("Dummy message")
        self.store.add_to_list(ml, msg_t2_1)
        msg_t3_1 = Message()
        msg_t3_1["From"] = "*****@*****.**"
        msg_t3_1["Message-ID"] = "<id3_1>"
        msg_t3_1.set_payload("Dummy message")
        self.store.add_to_list(ml, msg_t3_1)

        # Check the neighbors
        def check_neighbors(thread, expected_prev, expected_next):
            thread_id = get_message_id_hash("<id%s_1>" % thread)
            prev_th, next_th = self.store.get_thread_neighbors(
                "example-list", thread_id)
            # convert to something I can compare
            prev_th = prev_th and prev_th.thread_id
            expected_prev = expected_prev and \
                    get_message_id_hash("<id%s_1>" % expected_prev)
            next_th = next_th and next_th.thread_id
            expected_next = expected_next and \
                    get_message_id_hash("<id%s_1>" % expected_next)
            # compare
            self.assertEqual(prev_th, expected_prev)
            self.assertEqual(next_th, expected_next)

        # Order should be: 1, 2, 3
        check_neighbors(1, None, 2)
        check_neighbors(2, 1, 3)
        check_neighbors(3, 2, None)
        # now add a new message in thread 1, which becomes the most recently
        # active
        msg_t1_2 = Message()
        msg_t1_2["From"] = "*****@*****.**"
        msg_t1_2["Message-ID"] = "<id1_2>"
        msg_t1_2["In-Reply-To"] = "<id1_1>"
        msg_t1_2.set_payload("Dummy message")
        self.store.add_to_list(ml, msg_t1_2)
        # Order should be: 2, 3, 1
        check_neighbors(2, None, 3)
        check_neighbors(3, 2, 1)
        check_neighbors(1, 3, None)
Ejemplo n.º 4
0
 def _get_msg(self):
     msg = Message()
     msg["From"] = "*****@*****.**"
     msg["Message-ID"] = "<dummy>"
     msg["Message-ID-Hash"] = "QKODQBCADMDSP5YPOPKECXQWEQAMXZL3"
     msg.set_payload("Dummy message")
     return msg
Ejemplo n.º 5
0
 def setUp(self):
     self.user = User.objects.create_user('testuser', '*****@*****.**',
                                          'testPass')
     self.user.is_staff = True
     self.user.save()
     self.client.login(username='******', password='******')
     self.store = kittystore.get_store(SettingsModule(),
                                       debug=False,
                                       auto_create=True)
     self.client.defaults = {
         "kittystore.store": self.store,
         "HTTP_USER_AGENT": "testbot",
     }
     ml = FakeList("*****@*****.**")
     ml.subject_prefix = u"[example] "
     # Create 2 threads
     self.messages = []
     for msgnum in range(2):
         msg = Message()
         msg["From"] = "*****@*****.**"
         msg["Message-ID"] = "<id%d>" % (msgnum + 1)
         msg["Subject"] = "Dummy message"
         msg.set_payload("Dummy message")
         msg["Message-ID-Hash"] = self.store.add_to_list(ml, msg)
         self.messages.append(msg)
Ejemplo n.º 6
0
 def test_vote_cancel(self):
     ml = FakeList("*****@*****.**")
     msg = Message()
     msg["From"] = "*****@*****.**"
     msg["Message-ID"] = "<msg1>"
     msg.set_payload("Dummy message")
     self.store.add_to_list(ml, msg)
     msg.replace_header("Message-ID", "<msg2>")
     self.store.add_to_list(ml, msg)
     msg1 = self.store.get_message_by_id_from_list("*****@*****.**",
                                                   "msg1")
     msg1.vote(1, u"testuser")
     msg2 = self.store.get_message_by_id_from_list("*****@*****.**",
                                                   "msg2")
     msg2.vote(-1, u"testuser")
     self.assertEqual(msg1.likes, 1)
     self.assertEqual(msg2.dislikes, 1)
     for msg in (msg1, msg2):
         url = reverse('message_vote',
                       args=("*****@*****.**", msg.message_id_hash))
         resp = self.client.post(url, {"vote": "0"})
         self.assertEqual(resp.status_code, 200)
         self.assertEqual(msg.likes, 0)
         self.assertEqual(msg.dislikes, 0)
         result = json.loads(resp.content)
         self.assertEqual(result["like"], 0)
         self.assertEqual(result["dislike"], 0)
Ejemplo n.º 7
0
 def process(self, mlist, msg, msgdata):
     """See `IHandler`."""
     # Short circuit for non-digestable messages.
     if not mlist.digestable or msgdata.get('isdigest'):
         return
     # Open the mailbox that will be used to collect the current digest.
     mailbox_path = os.path.join(mlist.data_path, 'digest.mmdf')
     # Lock the mailbox and append the message.
     with Mailbox(mailbox_path, create=True) as mbox:
         mbox.add(msg)
     # Calculate the current size of the mailbox file.  This will not tell
     # us exactly how big the resulting MIME and rfc1153 digest will
     # actually be, but it's the most easily available metric to decide
     # whether the size threshold has been reached.
     size = os.path.getsize(mailbox_path)
     if size >= mlist.digest_size_threshold * 1024.0:
         # The digest is ready to send.  Because we don't want to hold up
         # this process with crafting the digest, we're going to move the
         # digest file to a safe place, then craft a fake message for the
         # DigestRunner as a trigger for it to build and send the digest.
         mailbox_dest = os.path.join(
             mlist.data_path,
             'digest.{0.volume}.{0.next_digest_number}.mmdf'.format(mlist))
         volume = mlist.volume
         digest_number = mlist.next_digest_number
         bump_digest_number_and_volume(mlist)
         os.rename(mailbox_path, mailbox_dest)
         config.switchboards['digest'].enqueue(Message(),
                                               listid=mlist.list_id,
                                               digest_path=mailbox_dest,
                                               volume=volume,
                                               digest_number=digest_number)
Ejemplo n.º 8
0
 def test_message_has_sender(self):
     msg = Message()
     msg['From'] = '*****@*****.**'
     msgdata = {}
     result = self._rule.check(self._mlist, msg, msgdata)
     self.assertFalse(result)
     self.assertEqual(msgdata, {})
Ejemplo n.º 9
0
 def test_on_new_message_userid(self):
     # Check that the user_id is set on a new message
     msg = Message()
     msg["From"] = "*****@*****.**"
     msg["Message-ID"] = "<dummy>"
     msg.set_payload("Dummy message")
     # setup Mailman's reply
     new_user_id = FakeMMUser()
     uid = uuid.uuid1()
     new_user_id.user_id = uid.int
     self.mm_client.get_user.side_effect = lambda addr: new_user_id
     # check the User does not exist yet
     self.assertEqual(0,
             self.store.get_message_count_by_user_id(uid))
     # do the test and check
     self.store.add_to_list("example-list", msg)
     dbmsg = self.store.get_message_by_id_from_list(
             "example-list", "dummy")
     self.assertEqual(dbmsg.sender.user_id, uid)
     self.assertTrue(dbmsg.sender.user is not None,
             "A 'User' instance was not created")
     self.assertEqual(dbmsg.sender.user.id, uid)
     self.assertEqual(1,
             self.store.get_message_count_by_user_id(uid))
     self.assertEqual(self.store.get_users_count(), 1)
 def test_join_bad_argument_no_equal(self):
     # Try to subscribe a member with a bad argument via join.
     msg = Message()
     msg['From'] = '*****@*****.**'
     results = Results()
     self._command.process(self._mlist, msg, {}, ('digest', ), results)
     self.assertIn('bad argument: digest', str(results))
Ejemplo n.º 11
0
 def test_add_in_classical_thread(self):
     # msg1
     # |-msg2
     # | `-msg4
     # `-msg3
     ml = FakeList("example-list")
     msgs = []
     for num in range(1, 5):
         msg = Message()
         msg["From"] = "*****@*****.**" % num
         msg["Message-ID"] = "<msg%d>" % num
         msg.set_payload("message %d" % num)
         msgs.append(msg)
     msgs[1]["In-Reply-To"] = "<msg1>"
     msgs[2]["In-Reply-To"] = "<msg1>"
     msgs[3]["In-Reply-To"] = "<msg2>"
     for msg in msgs:
         self.store.add_to_list(ml, msg)
     msgs = []
     for num in range(1, 5):
         msg = self.store.get_message_by_id_from_list(
             "example-list", "msg%d" % num)
         msgs.append(msg)
     msg1, msg2, msg3, msg4 = msgs
     self.assertEqual(msg1.thread_order, 0)
     self.assertEqual(msg1.thread_depth, 0)
     self.assertEqual(msg2.thread_order, 1)
     self.assertEqual(msg2.thread_depth, 1)
     self.assertEqual(msg3.thread_order, 3)
     self.assertEqual(msg3.thread_depth, 1)
     self.assertEqual(msg4.thread_order, 2)
     self.assertEqual(msg4.thread_depth, 2)
Ejemplo n.º 12
0
 def test_re_only_mixed(self):
     # Incoming subject is only Re:.
     msg = Message()
     msg['Subject'] = '=?utf-8?Q?Re:?='
     self._process(self._mlist, msg, {})
     subject = msg['subject']
     self.assertEqual(str(subject), '[Test] Re: ')
Ejemplo n.º 13
0
 def test_sync_mailman_user(self):
     # Check that the user_id is set when sync_mailman_user is run
     msg = Message()
     msg["From"] = "*****@*****.**"
     msg["Message-ID"] = "<dummy>"
     msg.set_payload("Dummy message")
     self.store.add_to_list("example-list", msg)
     dbmsg = self.store.get_message_by_id_from_list(
             "example-list", "dummy")
     self.assertEqual(dbmsg.sender.user_id, None)
     # setup Mailman's reply
     uid = uuid.uuid1()
     new_user_id = FakeMMUser()
     new_user_id.user_id = uid.int
     self.mm_client.get_user.side_effect = lambda addr: new_user_id
     # do the test and check
     mailman_user.sync_mailman_user(self.store)
     #dbmsg = self.store.get_message_by_id_from_list(
     #        "example-list", "dummy")
     self.assertEqual(dbmsg.sender.user_id, uid)
     self.assertTrue(dbmsg.sender.user is not None,
             "A 'User' instance was not created")
     self.assertEqual(dbmsg.sender.user.id, uid)
     self.assertEqual(1,
             self.store.get_message_count_by_user_id(uid))
Ejemplo n.º 14
0
 def test_re_prefix(self):
     # The subject has a Re: prefix.  Make sure that gets preserved, but
     # after the list prefix.
     msg = Message()
     msg['Subject'] = 'Re: [Test] A test message'
     self._process(self._mlist, msg, {})
     self.assertEqual(str(msg['subject']), '[Test] Re: A test message')
 def test_properties_on_new_message(self):
     ml = FakeList("example-list")
     ml.display_name = "name 1"
     ml.subject_prefix = "[prefix 1]"
     ml.description = "desc 1"
     kittystore.utils.MM_CLIENT.get_list.side_effect = lambda n: ml
     msg = Message()
     msg["From"] = "*****@*****.**"
     msg["Message-ID"] = "<dummy>"
     msg.set_payload("Dummy message")
     self.store.add_to_list("example-list", msg)
     ml_db = self.store.get_lists()[0]
     self.assertEqual(ml_db.display_name, "name 1")
     self.assertEqual(ml_db.subject_prefix, "[prefix 1]")
     ml.display_name = "name 2"
     ml.subject_prefix = "[prefix 2]"
     ml.description = "desc 2"
     ml.archive_policy = "private"
     msg.replace_header("Message-ID", "<dummy2>")
     self.store.add_to_list("example-list", msg)
     ml_db = self.store.get_lists()[0]
     #ml_db = self.store.db.find(List).one()
     self.assertEqual(ml_db.display_name, "name 2")
     self.assertEqual(ml_db.subject_prefix, "[prefix 2]")
     self.assertEqual(ml_db.description, "desc 2")
     self.assertEqual(ml_db.archive_policy, ArchivePolicy.private)
Ejemplo n.º 16
0
 def test_fasttrack(self):
     # Messages internally crafted are 'fast tracked' and don't get their
     # Subjects prefixed either.
     msg = Message()
     msg['Subject'] = 'A test message'
     self._process(self._mlist, msg, dict(_fasttrack=True))
     self.assertEqual(str(msg['subject']), 'A test message')
Ejemplo n.º 17
0
 def test_prefix(self):
     # The Subject gets prefixed.  The prefix gets automatically set by the
     # list style when the list gets created.
     msg = Message()
     msg['Subject'] = 'A test message'
     self._process(self._mlist, msg, {})
     self.assertEqual(str(msg['subject']), '[Test] A test message')
Ejemplo n.º 18
0
 def test_whitespace_only_prefix(self):
     # If the Subject prefix only contains whitespace, ignore it.
     self._mlist.subject_prefix = '    '
     msg = Message()
     msg['Subject'] = 'A test message'
     self._process(self._mlist, msg, {})
     self.assertEqual(str(msg['subject']), 'A test message')
Ejemplo n.º 19
0
 def test_starting_message_1(self):
     # A basic thread: msg2 replies to msg1
     ml = FakeList("example-list")
     msg1 = Message()
     msg1["From"] = "*****@*****.**"
     msg1["Message-ID"] = "<msg1>"
     msg1.set_payload("message 1")
     self.store.add_to_list(ml, msg1)
     msg2 = Message()
     msg2["From"] = "*****@*****.**"
     msg2["Message-ID"] = "<msg2>"
     msg2.set_payload("message 2")
     msg2["In-Reply-To"] = msg1["Message-ID"]
     self.store.add_to_list(ml, msg2)
     thread = self.store.db.find(Thread).one()
     self.assertEqual(thread.starting_email.message_id, "msg1")
Ejemplo n.º 20
0
 def test_on_new_message_invalidate(self):
     # Check that the cache is invalidated on new message
     msg = Message()
     msg["From"] = "*****@*****.**"
     msg["Message-ID"] = "<dummy>"
     msg.set_payload("Dummy message")
     today = datetime.datetime.utcnow().date() # don't use datetime.date.today(), we need UTC
     self.store.add_to_list("example-list", msg)
     # calls to cache.delete() -- invalidation
     delete_args = [ call[0][0] for call in
                     self.store.db.cache.delete.call_args_list ]
     #from pprint import pprint; pprint(delete_args)
     self.assertEqual(set(delete_args), set([
         u'list:example-list:recent_participants_count',
         u'list:example-list:recent_threads_count',
         u'list:example-list:participants_count:%d:%d' % (today.year, today.month),
         u'list:example-list:thread:QKODQBCADMDSP5YPOPKECXQWEQAMXZL3:emails_count',
         u'list:example-list:thread:QKODQBCADMDSP5YPOPKECXQWEQAMXZL3:participants_count'
         ]))
     # calls to cache.get_or_create() -- repopulation
     goc_args = [ call[0][0] for call in
                  self.store.db.cache.get_or_create.call_args_list ]
     #from pprint import pprint; pprint(goc_args)
     self.assertEqual(set(goc_args), set([
         u'list:example-list:recent_participants_count',
         u'list:example-list:recent_threads_count',
         u'list:example-list:participants_count:%d:%d' % (today.year, today.month),
         u'list:example-list:threads_count:%d:%d' % (today.year, today.month),
         u'list:example-list:thread:QKODQBCADMDSP5YPOPKECXQWEQAMXZL3:emails_count',
         u'list:example-list:thread:QKODQBCADMDSP5YPOPKECXQWEQAMXZL3:participants_count',
         u'list:example-list:thread:QKODQBCADMDSP5YPOPKECXQWEQAMXZL3:starting_email_id',
         ]))
Ejemplo n.º 21
0
 def test_prefix_only_mixed(self):
     # Incoming subject is only the prefix.
     msg = Message()
     msg['Subject'] = '=?utf-8?Q?[Test]_?='
     self._process(self._mlist, msg, {})
     subject = msg['subject']
     self.assertEqual(str(subject), '[Test] ')
Ejemplo n.º 22
0
 def test_bad_configuration_line(self):
     # Take a mark on the error log file.
     mark = LogFileMark('mailman.error')
     # A bad value in [antispam]header_checks should just get ignored, but
     # with an error message logged.
     chain = config.chains['header-match']
     # The links are created dynamically; the rule names will all start
     # with the same prefix, but have a variable suffix.  The actions will
     # all be to jump to the named chain.  Do these checks now, while we
     # collect other useful information.
     post_checks = []
     saw_any_rule = False
     for link in chain.get_links(self._mlist, Message(), {}):
         if link.rule.name == 'any':
             saw_any_rule = True
             self.assertEqual(link.action, LinkAction.jump)
         elif saw_any_rule:
             raise AssertionError("'any' rule was not last")
         else:
             self.assertEqual(link.rule.name[:13], 'header-match-')
             self.assertEqual(link.action, LinkAction.defer)
             post_checks.append((link.rule.header, link.rule.pattern))
     self.assertListEqual(post_checks, [
         ('Foo', 'foo'),
         ('Bar', 'bar'),
         ])
     # Check the error log.
     self.assertEqual(mark.readline()[-77:-1],
                      'Configuration error: [antispam]header_checks '
                      'contains bogus line: A-bad-line')
Ejemplo n.º 23
0
 def test_isdigest(self):
     # If the message is destined for the digest, the Subject header does
     # not get touched.
     msg = Message()
     msg['Subject'] = 'A test message'
     self._process(self._mlist, msg, dict(isdigest=True))
     self.assertEqual(str(msg['subject']), 'A test message')
Ejemplo n.º 24
0
 def test_save_original_subject(self):
     # When the Subject gets prefixed, the original is saved in the message
     # metadata.
     msgdata = {}
     msg = Message()
     msg['Subject'] = 'A test message'
     self._process(self._mlist, msg, msgdata)
     self.assertEqual(msgdata['original_subject'], 'A test message')
 def test_join_other_bogus(self):
     # Try to subscribe a bogus different address via join.
     msg = Message()
     msg['From'] = '*****@*****.**'
     results = Results()
     self._command.process(self._mlist, msg, {}, ('address=bogus', ),
                           results)
     self.assertIn('Invalid email address: bogus', str(results))
Ejemplo n.º 26
0
 def test_no_welcome_message(self):
     # When configured not to send a welcome message, none is sent.
     self._mlist.send_welcome_message = False
     status = self._command.process(self._mlist, Message(), {},
                                    (self._token, ), Results())
     self.assertEqual(status, ContinueProcessing.yes)
     # There will be no messages in the queue.
     get_queue_messages('virgin', expected_count=0)
 def test_join_digest(self):
     # Subscribe a member to digest via join.
     msg = Message()
     msg['From'] = '*****@*****.**'
     results = Results()
     self._command.process(self._mlist, msg, {}, ('digest=mime', ), results)
     self.assertIn('Confirmation email sent to [email protected]',
                   str(results))
 def test_join_posting_address(self):
     # Try to subscribe the list posting address.
     msg = Message()
     msg['From'] = self._mlist.posting_address
     results = Results()
     self._command.process(self._mlist, msg, {}, (), results)
     self.assertEqual(
         str(results).splitlines()[-1], 'List posting address not allowed')
Ejemplo n.º 29
0
 def test_no_subject_returns_reason(self):
     msg = Message()
     msg['Subject'] = Header('')
     msgdata = {}
     result = self._rule.check(self._mlist, msg, msgdata)
     self.assertTrue(result)
     self.assertEqual(msgdata['moderation_reasons'],
                      ['Message has no subject'])
Ejemplo n.º 30
0
 def test_starting_message_2(self):
     # A partially-imported thread: msg1 replies to something we don't have
     ml = FakeList("example-list")
     msg1 = Message()
     msg1["From"] = "*****@*****.**"
     msg1["Message-ID"] = "<msg1>"
     msg1["In-Reply-To"] = "<msg0>"
     msg1.set_payload("message 1")
     self.store.add_to_list(ml, msg1)
     msg2 = Message()
     msg2["From"] = "*****@*****.**"
     msg2["Message-ID"] = "<msg2>"
     msg2["In-Reply-To"] = msg1["Message-ID"]
     msg2.set_payload("message 2")
     self.store.add_to_list(ml, msg2)
     thread = self.store.db.find(Thread).one()
     self.assertEqual(thread.starting_email.message_id, "msg1")