Index: appengine/monorail/tracker/fielddetail.py |
diff --git a/appengine/monorail/tracker/fielddetail.py b/appengine/monorail/tracker/fielddetail.py |
new file mode 100644 |
index 0000000000000000000000000000000000000000..e3e6e5d58a434354ce2feb33fae33d83e68c9268 |
--- /dev/null |
+++ b/appengine/monorail/tracker/fielddetail.py |
@@ -0,0 +1,154 @@ |
+# Copyright 2016 The Chromium Authors. All rights reserved. |
+# Use of this source code is govered by a BSD-style |
+# license that can be found in the LICENSE file or at |
+# https://developers.google.com/open-source/licenses/bsd |
+ |
+"""A servlet for project and component owners to view and edit field defs.""" |
+ |
+import logging |
+import time |
+ |
+from third_party import ezt |
+ |
+from framework import framework_helpers |
+from framework import framework_views |
+from framework import permissions |
+from framework import servlet |
+from framework import urls |
+from proto import tracker_pb2 |
+from tracker import field_helpers |
+from tracker import tracker_bizobj |
+from tracker import tracker_helpers |
+from tracker import tracker_views |
+ |
+ |
+class FieldDetail(servlet.Servlet): |
+ """Servlet allowing project owners to view and edit a custom field.""" |
+ |
+ _MAIN_TAB_MODE = servlet.Servlet.MAIN_TAB_PROCESS |
+ _PAGE_TEMPLATE = 'tracker/field-detail-page.ezt' |
+ |
+ def _GetFieldDef(self, mr): |
+ """Get the config and field definition to be viewed or edited.""" |
+ # TODO(jrobbins): since so many requests get the config object, and |
+ # it is usually cached in RAM, just always get it and include it |
+ # in the MonorailRequest, mr. |
+ config = self.services.config.GetProjectConfig( |
+ mr.cnxn, mr.project_id) |
+ field_def = tracker_bizobj.FindFieldDef(mr.field_name, config) |
+ if not field_def: |
+ self.abort(404, 'custom field not found') |
+ return config, field_def |
+ |
+ def AssertBasePermission(self, mr): |
+ """Check whether the user has any permission to visit this page. |
+ |
+ Args: |
+ mr: commonly used info parsed from the request. |
+ """ |
+ super(FieldDetail, self).AssertBasePermission(mr) |
+ _config, field_def = self._GetFieldDef(mr) |
+ |
+ allow_view = permissions.CanViewFieldDef( |
+ mr.auth.effective_ids, mr.perms, mr.project, field_def) |
+ if not allow_view: |
+ raise permissions.PermissionException( |
+ 'User is not allowed to view this field definition') |
+ |
+ def GatherPageData(self, mr): |
+ """Build up a dictionary of data values to use when rendering the page. |
+ |
+ Args: |
+ mr: commonly used info parsed from the request. |
+ |
+ Returns: |
+ Dict of values used by EZT for rendering the page. |
+ """ |
+ config, field_def = self._GetFieldDef(mr) |
+ user_views = framework_views.MakeAllUserViews( |
+ mr.cnxn, self.services.user, field_def.admin_ids) |
+ field_def_view = tracker_views.FieldDefView( |
+ field_def, config, user_views=user_views) |
+ |
+ well_known_issue_types = tracker_helpers.FilterIssueTypes(config) |
+ |
+ allow_edit = permissions.CanEditFieldDef( |
+ mr.auth.effective_ids, mr.perms, mr.project, field_def) |
+ |
+ # Right now we do not allow renaming of enum fields. |
+ uneditable_name = field_def.field_type == tracker_pb2.FieldTypes.ENUM_TYPE |
+ |
+ initial_admins = ', '.join(sorted([ |
+ uv.email for uv in field_def_view.admins])) |
+ |
+ return { |
+ 'admin_tab_mode': servlet.Servlet.PROCESS_TAB_LABELS, |
+ 'field_def': field_def_view, |
+ 'allow_edit': ezt.boolean(allow_edit), |
+ 'uneditable_name': ezt.boolean(uneditable_name), |
+ 'initial_admins': initial_admins, |
+ 'initial_applicable_type': field_def.applicable_type, |
+ 'initial_applicable_predicate': field_def.applicable_predicate, |
+ 'well_known_issue_types': well_known_issue_types, |
+ } |
+ |
+ def ProcessFormData(self, mr, post_data): |
+ """Validate and store the contents of the issues tracker admin page. |
+ |
+ Args: |
+ mr: commonly used info parsed from the request. |
+ post_data: HTML form data from the request. |
+ |
+ Returns: |
+ String URL to redirect the user to, or None if response was already sent. |
+ """ |
+ config, field_def = self._GetFieldDef(mr) |
+ allow_edit = permissions.CanEditFieldDef( |
+ mr.auth.effective_ids, mr.perms, mr.project, field_def) |
+ if not allow_edit: |
+ raise permissions.PermissionException( |
+ 'User is not allowed to delete this field') |
+ |
+ if 'deletefield' in post_data: |
+ self._ProcessDeleteField(mr, field_def) |
+ return framework_helpers.FormatAbsoluteURL( |
+ mr, urls.ADMIN_LABELS, deleted=1, ts=int(time.time())) |
+ |
+ else: |
+ self._ProcessEditField(mr, post_data, config, field_def) |
+ return framework_helpers.FormatAbsoluteURL( |
+ mr, urls.FIELD_DETAIL, field=field_def.field_name, |
+ saved=1, ts=int(time.time())) |
+ |
+ def _ProcessDeleteField(self, mr, field_def): |
+ """The user wants to delete the specified custom field definition.""" |
+ self.services.config.SoftDeleteFieldDef( |
+ mr.cnxn, mr.project_id, field_def.field_id) |
+ |
+ # TODO(jrobbins): add logic to reaper cron task to look for |
+ # soft deleted field definitions that have no issues with |
+ # any value and hard deleted them. |
+ |
+ def _ProcessEditField(self, mr, post_data, config, field_def): |
+ """The user wants to edit this field definition.""" |
+ # TODO(jrobbins): future feature: editable field names |
+ |
+ parsed = field_helpers.ParseFieldDefRequest(post_data, config) |
+ |
+ admin_ids, _admin_str = tracker_helpers.ParseAdminUsers( |
+ mr.cnxn, post_data['admin_names'], self.services.user) |
+ |
+ # TODO(jrobbins): bounce on validation errors |
+ |
+ self.services.config.UpdateFieldDef( |
+ mr.cnxn, mr.project_id, field_def.field_id, |
+ applicable_type=parsed.applicable_type, |
+ applicable_predicate=parsed.applicable_predicate, |
+ is_required=parsed.is_required, |
+ min_value=parsed.min_value, max_value=parsed.max_value, |
+ regex=parsed.regex, needs_member=parsed.needs_member, |
+ needs_perm=parsed.needs_perm, grants_perm=parsed.grants_perm, |
+ notify_on=parsed.notify_on, is_multivalued=parsed.is_multivalued, |
+ docstring=parsed.field_docstring, admin_ids=admin_ids) |
+ self.services.config.UpdateConfig( |
+ mr.cnxn, mr.project, well_known_labels=parsed.revised_labels) |