summaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
authorBarry Warsaw2012-03-15 17:34:34 -0700
committerBarry Warsaw2012-03-15 17:34:34 -0700
commit148bd63fcca2613c4d10d234fe47a173d307b3e5 (patch)
tree8f4034945d35b752055f64d9d3e547676d362ead /src
parent854acf2f858950d6c926c82c5ee642014d7d973f (diff)
downloadmailman-148bd63fcca2613c4d10d234fe47a173d307b3e5.tar.gz
mailman-148bd63fcca2613c4d10d234fe47a173d307b3e5.tar.zst
mailman-148bd63fcca2613c4d10d234fe47a173d307b3e5.zip
Diffstat (limited to 'src')
-rw-r--r--src/mailman/docs/NEWS.rst1
-rw-r--r--src/mailman/interfaces/roster.py3
-rw-r--r--src/mailman/model/roster.py69
-rw-r--r--src/mailman/model/tests/test_roster.py156
-rw-r--r--src/mailman/rest/lists.py2
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)),
)