diff options
| author | Barry Warsaw | 2012-03-15 17:34:34 -0700 |
|---|---|---|
| committer | Barry Warsaw | 2012-03-15 17:34:34 -0700 |
| commit | 148bd63fcca2613c4d10d234fe47a173d307b3e5 (patch) | |
| tree | 8f4034945d35b752055f64d9d3e547676d362ead /src | |
| parent | 854acf2f858950d6c926c82c5ee642014d7d973f (diff) | |
| download | mailman-148bd63fcca2613c4d10d234fe47a173d307b3e5.tar.gz mailman-148bd63fcca2613c4d10d234fe47a173d307b3e5.tar.zst mailman-148bd63fcca2613c4d10d234fe47a173d307b3e5.zip | |
Diffstat (limited to 'src')
| -rw-r--r-- | src/mailman/docs/NEWS.rst | 1 | ||||
| -rw-r--r-- | src/mailman/interfaces/roster.py | 3 | ||||
| -rw-r--r-- | src/mailman/model/roster.py | 69 | ||||
| -rw-r--r-- | src/mailman/model/tests/test_roster.py | 156 | ||||
| -rw-r--r-- | src/mailman/rest/lists.py | 2 |
5 files changed, 203 insertions, 28 deletions
diff --git a/src/mailman/docs/NEWS.rst b/src/mailman/docs/NEWS.rst index 22e988360..2906be1cd 100644 --- a/src/mailman/docs/NEWS.rst +++ b/src/mailman/docs/NEWS.rst @@ -92,6 +92,7 @@ Interfaces * `IMailingList.real_name` -> `IMailingList.display_name` * `IUser.real_name` -> `IUser.display_name` * `IAddress.real_name` -> `IAddress.display_name` + * Add property `IRoster.member_count`. Commands -------- diff --git a/src/mailman/interfaces/roster.py b/src/mailman/interfaces/roster.py index 4ec3c611c..ebe057d21 100644 --- a/src/mailman/interfaces/roster.py +++ b/src/mailman/interfaces/roster.py @@ -40,6 +40,9 @@ class IRoster(Interface): members = Attribute( """An iterator over all the IMembers managed by this roster.""") + member_count = Attribute( + """The number of members managed by this roster.""") + users = Attribute( """An iterator over all the IUsers reachable by this roster. diff --git a/src/mailman/model/roster.py b/src/mailman/model/roster.py index 35ddcf438..48d434ab1 100644 --- a/src/mailman/model/roster.py +++ b/src/mailman/model/roster.py @@ -64,16 +64,24 @@ class AbstractRoster: def __init__(self, mlist): self._mlist = mlist + def _query(self): + return config.db.store.find( + Member, + mailing_list=self._mlist.fqdn_listname, + role=self.role) + @property def members(self): """See `IRoster`.""" - for member in config.db.store.find( - Member, - mailing_list=self._mlist.fqdn_listname, - role=self.role): + for member in self._query(): yield member @property + def member_count(self): + """See `IRoster`.""" + return self._query().count() + + @property def users(self): """See `IRoster`.""" # Members are linked to addresses, which in turn are linked to users. @@ -149,18 +157,12 @@ class AdministratorRoster(AbstractRoster): name = 'administrator' - @property - def members(self): - """See `IRoster`.""" - # Administrators are defined as the union of the owners and the - # moderators. - members = config.db.store.find( - Member, - Member.mailing_list == self._mlist.fqdn_listname, - Or(Member.role == MemberRole.owner, - Member.role == MemberRole.moderator)) - for member in members: - yield member + def _query(self): + return config.db.store.find( + Member, + Member.mailing_list == self._mlist.fqdn_listname, + Or(Member.role == MemberRole.owner, + Member.role == MemberRole.moderator)) def get_member(self, address): """See `IRoster`.""" @@ -184,6 +186,14 @@ class AdministratorRoster(AbstractRoster): class DeliveryMemberRoster(AbstractRoster): """Return all the members having a particular kind of delivery.""" + @property + def member_count(self): + """See `IRoster`.""" + # XXX 2012-03-15 BAW: It would be nice to make this more efficient. + # The problem is that you'd have to change the loop in _get_members() + # checking the delivery mode to a query parameter. + return len(tuple(self.members)) + def _get_members(self, *delivery_modes): """The set of members for a mailing list, filter by delivery mode. @@ -234,13 +244,10 @@ class Subscribers(AbstractRoster): name = 'subscribers' - @property - def members(self): - """See `IRoster`.""" - for member in config.db.store.find( - Member, - mailing_list=self._mlist.fqdn_listname): - yield member + def _query(self): + return config.db.store.find( + Member, + mailing_list=self._mlist.fqdn_listname) @@ -254,15 +261,23 @@ class Memberships: def __init__(self, user): self._user = user - @property - def members(self): - """See `IRoster`.""" + def _query(self): results = config.db.store.find( Member, Or(Member.user_id == self._user.id, And(Address.user_id == self._user.id, Member.address_id == Address.id))) - for member in results.config(distinct=True): + return results.config(distinct=True) + + @property + def member_count(self): + """See `IRoster`.""" + return self._query().count() + + @property + def members(self): + """See `IRoster`.""" + for member in self._query(): yield member @property diff --git a/src/mailman/model/tests/test_roster.py b/src/mailman/model/tests/test_roster.py new file mode 100644 index 000000000..8d5a7b81b --- /dev/null +++ b/src/mailman/model/tests/test_roster.py @@ -0,0 +1,156 @@ +# Copyright (C) 2012 by the Free Software Foundation, Inc. +# +# This file is part of GNU Mailman. +# +# GNU Mailman is free software: you can redistribute it and/or modify it under +# the terms of the GNU General Public License as published by the Free +# Software Foundation, either version 3 of the License, or (at your option) +# any later version. +# +# GNU Mailman 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 General Public License for +# more details. +# +# You should have received a copy of the GNU General Public License along with +# GNU Mailman. If not, see <http://www.gnu.org/licenses/>. + +"""Test rosters.""" + +from __future__ import absolute_import, print_function, unicode_literals + +__metaclass__ = type +__all__ = [ + 'TestMailingListRoster', + 'TestMembershipsRoster', + ] + + +import unittest + +from zope.component import getUtility + +from mailman.app.lifecycle import create_list +from mailman.interfaces.member import DeliveryMode, MemberRole +from mailman.interfaces.usermanager import IUserManager +from mailman.testing.layers import ConfigLayer +from mailman.utilities.datetime import now + + + +class TestMailingListRoster(unittest.TestCase): + """Test various aspects of a mailing list's roster.""" + + layer = ConfigLayer + + def setUp(self): + self._mlist = create_list('test@example.com') + user_manager = getUtility(IUserManager) + self._anne = user_manager.create_address('anne@example.com') + self._bart = user_manager.create_address('bart@example.com') + self._cris = user_manager.create_address('cris@example.com') + + def test_no_members(self): + # Nobody with any role is subscribed to the mailing list. + self.assertEqual(self._mlist.owners.member_count, 0) + self.assertEqual(self._mlist.moderators.member_count, 0) + self.assertEqual(self._mlist.administrators.member_count, 0) + self.assertEqual(self._mlist.members.member_count, 0) + self.assertEqual(self._mlist.regular_members.member_count, 0) + self.assertEqual(self._mlist.digest_members.member_count, 0) + self.assertEqual(self._mlist.subscribers.member_count, 0) + + def test_one_regular_member(self): + # One person getting regular delivery is subscribed to the mailing + # list as a member. + self._mlist.subscribe(self._anne, role=MemberRole.member) + self.assertEqual(self._mlist.owners.member_count, 0) + self.assertEqual(self._mlist.moderators.member_count, 0) + self.assertEqual(self._mlist.administrators.member_count, 0) + self.assertEqual(self._mlist.members.member_count, 1) + self.assertEqual(self._mlist.regular_members.member_count, 1) + self.assertEqual(self._mlist.digest_members.member_count, 0) + self.assertEqual(self._mlist.subscribers.member_count, 1) + + def test_two_regular_members(self): + # Two people getting regular delivery are subscribed to the mailing + # list as members. + self._mlist.subscribe(self._anne, role=MemberRole.member) + self._mlist.subscribe(self._bart, role=MemberRole.member) + self.assertEqual(self._mlist.owners.member_count, 0) + self.assertEqual(self._mlist.moderators.member_count, 0) + self.assertEqual(self._mlist.administrators.member_count, 0) + self.assertEqual(self._mlist.members.member_count, 2) + self.assertEqual(self._mlist.regular_members.member_count, 2) + self.assertEqual(self._mlist.digest_members.member_count, 0) + self.assertEqual(self._mlist.subscribers.member_count, 2) + + def test_one_regular_members_one_digest_member(self): + # Two people are subscribed to the mailing list as members. One gets + # regular delivery and one gets digest delivery. + self._mlist.subscribe(self._anne, role=MemberRole.member) + member = self._mlist.subscribe(self._bart, role=MemberRole.member) + member.preferences.delivery_mode = DeliveryMode.mime_digests + self.assertEqual(self._mlist.owners.member_count, 0) + self.assertEqual(self._mlist.moderators.member_count, 0) + self.assertEqual(self._mlist.administrators.member_count, 0) + self.assertEqual(self._mlist.members.member_count, 2) + self.assertEqual(self._mlist.regular_members.member_count, 1) + self.assertEqual(self._mlist.digest_members.member_count, 1) + self.assertEqual(self._mlist.subscribers.member_count, 2) + + def test_a_person_is_both_a_member_and_an_owner(self): + # Anne is the owner of a mailing list and she gets subscribed as a + # member of the mailing list, receiving regular deliveries. + self._mlist.subscribe(self._anne, role=MemberRole.member) + self._mlist.subscribe(self._anne, role=MemberRole.owner) + self.assertEqual(self._mlist.owners.member_count, 1) + self.assertEqual(self._mlist.moderators.member_count, 0) + self.assertEqual(self._mlist.administrators.member_count, 1) + self.assertEqual(self._mlist.members.member_count, 1) + self.assertEqual(self._mlist.regular_members.member_count, 1) + self.assertEqual(self._mlist.digest_members.member_count, 0) + self.assertEqual(self._mlist.subscribers.member_count, 2) + + def test_a_bunch_of_members_and_administrators(self): + # Anne is the owner of a mailing list, and Bart is a moderator. Anne + # gets subscribed as a member of the mailing list, receiving regular + # deliveries. Cris subscribes to the mailing list as a digest member. + self._mlist.subscribe(self._anne, role=MemberRole.owner) + self._mlist.subscribe(self._bart, role=MemberRole.moderator) + self._mlist.subscribe(self._anne, role=MemberRole.member) + member = self._mlist.subscribe(self._cris, role=MemberRole.member) + member.preferences.delivery_mode = DeliveryMode.mime_digests + self.assertEqual(self._mlist.owners.member_count, 1) + self.assertEqual(self._mlist.moderators.member_count, 1) + self.assertEqual(self._mlist.administrators.member_count, 2) + self.assertEqual(self._mlist.members.member_count, 2) + self.assertEqual(self._mlist.regular_members.member_count, 1) + self.assertEqual(self._mlist.digest_members.member_count, 1) + self.assertEqual(self._mlist.subscribers.member_count, 4) + + + +class TestMembershipsRoster(unittest.TestCase): + """Test the memberships roster.""" + + layer = ConfigLayer + + def setUp(self): + self._ant = create_list('ant@example.com') + self._bee = create_list('bee@example.com') + user_manager = getUtility(IUserManager) + self._anne = user_manager.create_user('anne@example.com') + preferred = list(self._anne.addresses)[0] + preferred.verified_on = now() + self._anne.preferred_address = preferred + + def test_no_memberships(self): + # An unsubscribed user has no memberships. + self.assertEqual(self._anne.memberships.member_count, 0) + + def test_subscriptions(self): + # Anne subscribes to a couple of mailing lists. + self._ant.subscribe(self._anne) + self._bee.subscribe(self._anne) + self.assertEqual(self._anne.memberships.member_count, 2) diff --git a/src/mailman/rest/lists.py b/src/mailman/rest/lists.py index 38e2d9841..c95c9a88a 100644 --- a/src/mailman/rest/lists.py +++ b/src/mailman/rest/lists.py @@ -109,7 +109,7 @@ class _ListBase(resource.Resource, CollectionMixin): fqdn_listname=mlist.fqdn_listname, list_name=mlist.list_name, mail_host=mlist.mail_host, - member_count=len(tuple(mlist.members.members)), + member_count=mlist.members.member_count, volume=mlist.volume, self_link=path_to('lists/{0}'.format(mlist.fqdn_listname)), ) |
