← Back to team overview

launchpad-reviewers team mailing list archive

Re: [Merge] lp:~cjwatson/launchpad/participation-query-count into lp:launchpad

 


Diff comments:

> === modified file 'lib/lp/registry/browser/person.py'
> --- lib/lp/registry/browser/person.py	2015-03-05 11:39:06 +0000
> +++ lib/lp/registry/browser/person.py	2015-03-09 11:42:57 +0000
> @@ -2028,7 +2028,8 @@
>      def label(self):
>          return 'Team participation for ' + self.context.displayname
>  
> -    def _asParticipation(self, membership=None, team=None, via=None):
> +    def _asParticipation(self, membership=None, team=None, via=None,
> +                         mailing_list=None, subscription=None):
>          """Return a dict of participation information for the membership.
>  
>          Method requires membership or team, not both.
> @@ -2057,15 +2058,14 @@
>              # The member is a direct member; use the membership data.
>              team = membership.team
>              datejoined = membership.datejoined
> -            if membership.person == team.teamowner:
> +            if membership.personID == team.teamownerID:
>                  role = 'Owner'
>              elif membership.status == TeamMembershipStatus.ADMIN:
>                  role = 'Admin'
>              else:
>                  role = 'Member'
>  
> -        if team.mailing_list is not None and team.mailing_list.is_usable:
> -            subscription = team.mailing_list.getSubscription(self.context)
> +        if mailing_list is not None:
>              if subscription is None:
>                  subscribed = 'Not subscribed'
>              else:
> @@ -2081,28 +2081,37 @@
>      def active_participations(self):
>          """Return the participation information for active memberships."""
>          paths, memberships = self.context.getPathsToTeams()
> +        memberships = [
> +            membership for membership in memberships
> +            if check_permission('launchpad.View', membership.team)]
>          direct_teams = [membership.team for membership in memberships]
>          indirect_teams = [
>              team for team in paths.keys()
> -            if team not in direct_teams]
> +            if team not in direct_teams and
> +               check_permission('launchpad.View', team)]
>          participations = []
>  
> +        # Bulk-load mailing list subscriptions.
> +        subscriptions = getUtility(IMailingListSet).getSubscriptionsForTeams(
> +            self.context, direct_teams + indirect_teams)
> +
>          # First, create a participation for all direct memberships.
>          for membership in memberships:
> -            # Add a participation record for the membership if allowed.
> -            if check_permission('launchpad.View', membership.team):
> -                participations.append(
> -                    self._asParticipation(membership=membership))
> +            mailing_list, subscription = subscriptions.get(
> +                membership.team.id, (None, None))
> +            participations.append(self._asParticipation(
> +                membership=membership, mailing_list=mailing_list,
> +                subscription=subscription))
>  
>          # Second, create a participation for all indirect memberships,
>          # using the remaining paths.
>          for indirect_team in indirect_teams:
> -            if not check_permission('launchpad.View', indirect_team):
> -                continue
> +            mailing_list, subscription = subscriptions.get(
> +                indirect_team.id, (None, None))
>              participations.append(
>                  self._asParticipation(
> -                    via=paths[indirect_team],
> -                    team=indirect_team))
> +                    via=paths[indirect_team], team=indirect_team,
> +                    mailing_list=mailing_list, subscription=subscription))
>          return sorted(participations, key=itemgetter('displayname'))
>  
>      @cachedproperty
> 
> === modified file 'lib/lp/registry/browser/tests/test_person.py'
> --- lib/lp/registry/browser/tests/test_person.py	2015-03-05 15:28:11 +0000
> +++ lib/lp/registry/browser/tests/test_person.py	2015-03-09 11:42:57 +0000
> @@ -47,6 +47,7 @@
>  from lp.services.identity.interfaces.emailaddress import IEmailAddressSet
>  from lp.services.log.logger import FakeLogger
>  from lp.services.mail import stub
> +from lp.services.propertycache import clear_property_cache
>  from lp.services.verification.interfaces.authtoken import LoginTokenType
>  from lp.services.verification.interfaces.logintoken import ILoginTokenSet
>  from lp.services.verification.tests.logintoken import get_token_url_from_email
> @@ -68,6 +69,7 @@
>      login_person,
>      monkey_patch,
>      person_logged_in,
> +    record_two_runs,
>      StormStatementRecorder,
>      TestCaseWithFactory,
>      )
> @@ -878,6 +880,26 @@
>          self.assertEqual(1, len(participations))
>          self.assertEqual(True, self.view.has_participations)
>  
> +    def test_mailing_list_subscriptions_query_count(self):
> +        # Additional mailing list subscriptions do not add additional queries.
> +        def create_subscriptions():
> +            direct_team = self.factory.makeTeam(members=[self.user])
> +            direct_list = self.factory.makeMailingList(
> +                direct_team, direct_team.teamowner)
> +            direct_list.subscribe(self.user)
> +            indirect_team = self.factory.makeTeam(members=[direct_team])
> +            indirect_list = self.factory.makeMailingList(
> +                indirect_team, indirect_team.teamowner)
> +            indirect_list.subscribe(self.user)
> +
> +        def get_participations():
> +            clear_property_cache(self.view)
> +            return list(self.view.active_participations)
> +
> +        recorder1, recorder2 = record_two_runs(
> +            get_participations, create_subscriptions, 5)
> +        self.assertThat(recorder2, HasQueryCount(Equals(recorder1.count)))
> +
>  
>  class TestPersonRelatedPackagesView(TestCaseWithFactory):
>      """Test the related software view."""
> 
> === modified file 'lib/lp/registry/configure.zcml'
> --- lib/lp/registry/configure.zcml	2015-02-26 11:34:47 +0000
> +++ lib/lp/registry/configure.zcml	2015-03-09 11:42:57 +0000
> @@ -37,6 +37,7 @@
>                  id
>                  team
>                  person
> +                personID
>                  status
>                  proposed_by
>                  acknowledged_by
> 
> === modified file 'lib/lp/registry/interfaces/mailinglist.py'
> --- lib/lp/registry/interfaces/mailinglist.py	2013-01-07 02:40:55 +0000
> +++ lib/lp/registry/interfaces/mailinglist.py	2015-03-09 11:42:57 +0000
> @@ -1,4 +1,4 @@
> -# Copyright 2009-2011 Canonical Ltd.  This software is licensed under the
> +# Copyright 2009-2015 Canonical Ltd.  This software is licensed under the
>  # GNU Affero General Public License version 3 (see the file LICENSE).
>  
>  """Mailing list interfaces."""
> @@ -464,6 +464,17 @@
>          :raises AssertionError: When `team_name` is not a string.
>          """
>  
> +    def getSubscriptionsForTeams(person, teams):
> +        """Return a person's subscriptions to usable lists for a set of teams.
> +
> +        :param person: An `IPerson`.
> +        :param teams: A list of `ITeam`s.
> +        :return: A dictionary mapping team IDs to tuples of `IMailingList`
> +            IDs and `IMailingListSubscription` IDs; the second element will
> +            be None if a team has a list to which the person is not
> +            subscribed.
> +        """
> +
>      def getSubscribedAddresses(team_names):
>          """Return the set of subscribed email addresses for members.
>  
> 
> === modified file 'lib/lp/registry/interfaces/teammembership.py'
> --- lib/lp/registry/interfaces/teammembership.py	2013-01-07 02:40:55 +0000
> +++ lib/lp/registry/interfaces/teammembership.py	2015-03-09 11:42:57 +0000
> @@ -1,4 +1,4 @@
> -# Copyright 2009 Canonical Ltd.  This software is licensed under the
> +# Copyright 2009-2015 Canonical Ltd.  This software is licensed under the
>  # GNU Affero General Public License version 3 (see the file LICENSE).
>  
>  """Team membership interfaces."""
> @@ -131,6 +131,7 @@
>          Reference(title=_("Member"), required=True, readonly=True,
>                    schema=Interface),  # Specified in interfaces/person.py.
>          exported_as='member')
> +    personID = Int(title=_("Person ID"), required=True, readonly=True)
>      proposed_by = Attribute(_('Proponent'))
>      reviewed_by = Attribute(
>          _("The team admin who approved/rejected the member."))
> 
> === modified file 'lib/lp/registry/model/mailinglist.py'
> --- lib/lp/registry/model/mailinglist.py	2013-06-20 05:50:00 +0000
> +++ lib/lp/registry/model/mailinglist.py	2015-03-09 11:42:57 +0000
> @@ -1,4 +1,4 @@
> -# Copyright 2009-2012 Canonical Ltd.  This software is licensed under the
> +# Copyright 2009-2015 Canonical Ltd.  This software is licensed under the
>  # GNU Affero General Public License version 3 (see the file LICENSE).
>  
>  __metaclass__ = type
> @@ -100,6 +100,13 @@
>      PostedMessageStatus.APPROVAL_PENDING)
>  
>  
> +USABLE_STATUSES = (
> +    MailingListStatus.ACTIVE,
> +    MailingListStatus.MODIFIED,
> +    MailingListStatus.UPDATING,
> +    MailingListStatus.MOD_FAILED)
> +
> +
>  class MessageApproval(SQLBase):
>      """A held message."""
>  
> @@ -344,10 +351,7 @@
>      @property
>      def is_usable(self):
>          """See `IMailingList`."""
> -        return self.status in [MailingListStatus.ACTIVE,
> -                               MailingListStatus.MODIFIED,
> -                               MailingListStatus.UPDATING,
> -                               MailingListStatus.MOD_FAILED]
> +        return self.status in USABLE_STATUSES
>  
>      def _get_welcome_message(self):
>          return self.welcome_message_
> @@ -547,6 +551,24 @@
>              """ % sqlvalues(team_name),
>              clauseTables=['Person'])
>  
> +    def getSubscriptionsForTeams(self, person, teams):
> +        """See `IMailingListSet`."""
> +        store = IStore(MailingList)
> +        team_ids = set(map(operator.attrgetter("id"), teams))
> +        lists = dict(store.find(
> +            (MailingList.teamID, MailingList.id),
> +            MailingList.teamID.is_in(team_ids),
> +            MailingList.status.is_in(USABLE_STATUSES)))
> +        subscriptions = dict(store.find(
> +            (MailingListSubscription.mailing_listID,
> +             MailingListSubscription.id),
> +            MailingListSubscription.person == person,
> +            MailingListSubscription.mailing_listID.is_in(lists.values())))
> +        by_team = {}
> +        for team, mailing_list in lists.items():
> +            by_team[team] = (mailing_list, subscriptions.get(mailing_list))
> +        return by_team

Yes: Person:+participation cares about mailing lists for the requested teams that don't have any relevant subscriptions as well as those that do, so a MailingListSubscription object on its own can't express the necessary information and we need the MailingList as well.  Given that, it doesn't seem worth materialising either whole object, although I realise that they aren't all that wide.

> +
>      def _getTeamIdsAndMailingListIds(self, team_names):
>          """Return a tuple of team and mailing list Ids for the team names."""
>          store = IStore(MailingList)
> 


-- 
https://code.launchpad.net/~cjwatson/launchpad/participation-query-count/+merge/252267
Your team Launchpad code reviewers is subscribed to branch lp:launchpad.


References