launchpad-reviewers team mailing list archive
-
launchpad-reviewers team
-
Mailing list archive
-
Message #23207
Re: [Merge] lp:~cjwatson/launchpad/git-grantee-widgets into lp:launchpad
Review: Approve code
Diff comments:
> === added file 'lib/lp/code/browser/widgets/gitgrantee.py'
> --- lib/lp/code/browser/widgets/gitgrantee.py 1970-01-01 00:00:00 +0000
> +++ lib/lp/code/browser/widgets/gitgrantee.py 2018-11-09 22:44:37 +0000
> @@ -0,0 +1,253 @@
> +# Copyright 2018 Canonical Ltd. This software is licensed under the
> +# GNU Affero General Public License version 3 (see the file LICENSE).
> +
> +from __future__ import absolute_import, print_function, unicode_literals
> +
> +__metaclass__ = type
> +__all__ = [
> + 'GitGranteeDisplayWidget',
> + 'GitGranteeField',
> + 'GitGranteeWidget',
> + ]
> +
> +from lazr.enum import DBItem
> +from lazr.restful.fields import Reference
> +from z3c.ptcompat import ViewPageTemplateFile
> +from zope.formlib.interfaces import (
> + ConversionError,
> + IDisplayWidget,
> + IInputWidget,
> + InputErrors,
> + MissingInputError,
> + WidgetInputError,
> + )
> +from zope.formlib.utility import setUpWidget
> +from zope.formlib.widget import (
> + BrowserWidget,
> + CustomWidgetFactory,
> + DisplayWidget,
> + InputWidget,
> + renderElement,
> + )
> +from zope.interface import implementer
> +from zope.schema import (
> + Choice,
> + Field,
> + )
> +from zope.schema.interfaces import IField
> +from zope.schema.vocabulary import getVocabularyRegistry
> +from zope.security.proxy import isinstance as zope_isinstance
> +
> +from lp import _
> +from lp.app.errors import UnexpectedFormData
> +from lp.app.validators import LaunchpadValidationError
> +from lp.app.widgets.popup import PersonPickerWidget
> +from lp.code.enums import GitGranteeType
> +from lp.code.interfaces.gitrule import IGitRule
> +from lp.registry.interfaces.person import IPerson
> +from lp.services.webapp.escaping import structured
> +from lp.services.webapp.interfaces import (
> + IAlwaysSubmittedWidget,
> + IMultiLineWidgetLayout,
> + )
> +from lp.services.webapp.publisher import canonical_url
> +
> +
> +class IGitGranteeField(IField):
> + """An interface for a Git access grantee field."""
> +
> + rule = Reference(
> + title=_("Rule"), required=True, readonly=True, schema=IGitRule,
> + description=_("The rule that this grantee is for."))
> +
> +
> +@implementer(IGitGranteeField)
> +class GitGranteeField(Field):
> + """A field that holds a Git access grantee."""
> +
> + def __init__(self, rule, *args, **kwargs):
> + super(GitGranteeField, self).__init__(*args, **kwargs)
> + self.rule = rule
> +
> + def constraint(self, value):
> + """See `IField`."""
> + if zope_isinstance(value, DBItem) and value.enum == GitGranteeType:
> + return value != GitGranteeType.PERSON
> + else:
> + return value in getVocabularyRegistry().get(
> + None, "ValidPersonOrTeam")
> +
> +
> +@implementer(IDisplayWidget)
> +class GitGranteePersonDisplayWidget(BrowserWidget):
> +
> + def __init__(self, context, vocabulary, request):
> + super(GitGranteePersonDisplayWidget, self).__init__(context, request)
> +
> + def __call__(self):
> + if self._renderedValueSet():
> + grantee = self._data
> + person_img = renderElement(
> + "img", style="padding-bottom: 2px", src="/@@/person", alt="")
> + return renderElement(
> + "a", href=canonical_url(grantee),
> + contents="%s %s" % (
> + person_img,
> + structured("%s", grantee.display_name).escapedtext))
Is this meaningfully distinct from PersonFormatterAPI.link apart from ignoring the team icon?
> + else:
> + return ""
> +
> +
> +@implementer(IMultiLineWidgetLayout)
> +class GitGranteeWidgetBase(BrowserWidget):
> +
> + template = ViewPageTemplateFile("templates/gitgrantee.pt")
> + default_option = "person"
> + _widgets_set_up = False
> +
> + def setUpSubWidgets(self):
> + if self._widgets_set_up:
> + return
> + fields = [
> + Choice(
> + __name__="person", title=u"Person",
> + required=False, vocabulary="ValidPersonOrTeam"),
> + ]
> + if self._read_only:
> + self.person_widget = CustomWidgetFactory(
> + GitGranteePersonDisplayWidget)
> + else:
> + self.person_widget = CustomWidgetFactory(
> + PersonPickerWidget,
> + # XXX cjwatson 2018-10-18: This is a little unfortunate, but
> + # otherwise there's no spacing at all between the
> + # (deliberately unlabelled) radio button and the text box.
> + style="margin-left: 4px;")
> + for field in fields:
> + setUpWidget(
> + self, field.__name__, field, self._sub_widget_interface,
> + prefix=self.name)
> + self._widgets_set_up = True
> +
> + def setUpOptions(self):
> + """Set up options to be rendered."""
> + self.options = {}
> + for option in ("repository_owner", "person"):
> + attributes = {
> + "type": "radio", "name": self.name, "value": option,
> + "id": "%s.option.%s" % (self.name, option),
> + # XXX cjwatson 2018-10-18: Ugly, but it's worse without
> + # this, especially in a permissions table where this widget
> + # is normally used.
> + "style": "margin-left: 0;",
> + }
> + if self.request.form_ng.getOne(
> + self.name, self.default_option) == option:
> + attributes["checked"] = "checked"
> + if self._read_only:
> + attributes["disabled"] = "disabled"
> + self.options[option] = renderElement("input", **attributes)
> +
> + @property
> + def show_options(self):
> + return {
> + option: not self._read_only or self.default_option == option
> + for option in ("repository_owner", "person")}
> +
> + def setRenderedValue(self, value):
> + """See `IWidget`."""
> + self.setUpSubWidgets()
> + if value == GitGranteeType.REPOSITORY_OWNER:
> + self.default_option = "repository_owner"
> + return
> + elif value is None or IPerson.providedBy(value):
> + self.default_option = "person"
> + self.person_widget.setRenderedValue(value)
> + return
> + else:
> + raise AssertionError("Not a valid value: %r" % value)
> +
> + def __call__(self):
> + """See `zope.formlib.interfaces.IBrowserWidget`."""
> + self.setUpSubWidgets()
> + self.setUpOptions()
> + return self.template()
> +
> +
> +@implementer(IDisplayWidget)
> +class GitGranteeDisplayWidget(GitGranteeWidgetBase, DisplayWidget):
> + """Widget for displaying a Git access grantee."""
> +
> + _sub_widget_interface = IDisplayWidget
> + _read_only = True
> +
> +
> +@implementer(IAlwaysSubmittedWidget, IInputWidget)
> +class GitGranteeWidget(GitGranteeWidgetBase, InputWidget):
> + """Widget for selecting a Git access grantee."""
> +
> + _sub_widget_interface = IInputWidget
> + _read_only = False
> + _widgets_set_up = False
> +
> + @property
> + def show_options(self):
> + show_options = super(GitGranteeWidget, self).show_options
> + # Hide options that indicate unique grantee_types (e.g.
> + # repository_owner) if they already exist for the context rule.
> + if (show_options["repository_owner"] and
> + not self.context.rule.repository.findRuleGrantsByGrantee(
> + GitGranteeType.REPOSITORY_OWNER,
> + ref_pattern=self.context.rule.ref_pattern,
> + exact_grantee=True).is_empty()):
> + show_options["repository_owner"] = False
> + return show_options
> +
> + def hasInput(self):
> + self.setUpSubWidgets()
> + form_value = self.request.form_ng.getOne(self.name)
> + if form_value is None:
> + return False
> + return form_value != "person" or self.person_widget.hasInput()
> +
> + def hasValidInput(self):
> + """See `zope.formlib.interfaces.IInputWidget`."""
> + try:
> + self.getInputValue()
> + return True
> + except (InputErrors, UnexpectedFormData):
> + return False
> +
> + def getInputValue(self):
> + """See `zope.formlib.interfaces.IInputWidget`."""
> + self.setUpSubWidgets()
> + form_value = self.request.form_ng.getOne(self.name)
> + if form_value == "repository_owner":
> + return GitGranteeType.REPOSITORY_OWNER
> + elif form_value == "person":
> + try:
> + return self.person_widget.getInputValue()
> + except MissingInputError:
> + raise WidgetInputError(
> + self.name, self.label,
> + LaunchpadValidationError(
> + "Please enter a person or team name"))
> + except ConversionError:
> + entered_name = self.request.form_ng.getOne(
> + "%s.person" % self.name)
> + raise WidgetInputError(
> + self.name, self.label,
> + LaunchpadValidationError(
> + "There is no person or team named '%s' registered in "
> + "Launchpad" % entered_name))
> + else:
> + raise UnexpectedFormData("No valid option was selected.")
> +
> + def error(self):
> + """See `zope.formlib.interfaces.IBrowserWidget`."""
> + try:
> + if self.hasInput():
> + self.getInputValue()
> + except InputErrors as error:
> + self._error = error
> + return super(GitGranteeWidget, self).error()
>
> === modified file 'lib/lp/code/interfaces/gitrepository.py'
> --- lib/lp/code/interfaces/gitrepository.py 2018-11-09 22:06:43 +0000
> +++ lib/lp/code/interfaces/gitrepository.py 2018-11-09 22:44:37 +0000
> @@ -766,12 +766,17 @@
> :param user: The `IPerson` who is moving the rule.
> """
>
> - def findRuleGrantsByGrantee(grantee):
> + def findRuleGrantsByGrantee(grantee, exact_grantee=False,
> + ref_pattern=None):
> """Find the grants for a grantee applied to this repository.
>
> :param grantee: The `IPerson` to search for, or an item of
> `GitGranteeType` other than `GitGranteeType.PERSON` to search
> for some other kind of entity.
> + :param exact_grantee: If True, match `grantee` exactly; if False
> + (the default), also accept teams of which `grantee` is a member.
include_transitive, maybe?
> + :param ref_pattern: If not None, only return grants for rules with
> + this ref_pattern.
> """
>
> @export_read_operation()
--
https://code.launchpad.net/~cjwatson/launchpad/git-grantee-widgets/+merge/357600
Your team Launchpad code reviewers is subscribed to branch lp:launchpad.
References