1
0
Fork 0
mirror of https://we.phorge.it/source/phorge.git synced 2025-01-18 10:41:08 +01:00

Clean up some EditEngine meta-policies

Summary:
Ref T9908. Simplify some of the policies here:

  - If you can edit an application (currently, always "Administrators"), you can view and edit all of its forms.
  - You must be able to edit an application to create new forms.
  - Improve some error messages.
  - Get about halfway through letting users reorder forms in the "Create" menu if they want to sort by something weird since it'll need schema changes and I can do them all in one go here.

Test Plan:
  - Tried to create and edit forms as an unprivileged user.
  - Created and edited forms as an administrator.

Reviewers: chad

Reviewed By: chad

Maniphest Tasks: T9908

Differential Revision: https://secure.phabricator.com/D14700
This commit is contained in:
epriestley 2015-12-07 14:55:05 -08:00
parent afacdbd814
commit 82e67e6bb9
8 changed files with 214 additions and 37 deletions

View file

@ -0,0 +1,11 @@
ALTER TABLE {$NAMESPACE}_search.search_editengineconfiguration
DROP editPolicy;
ALTER TABLE {$NAMESPACE}_search.search_editengineconfiguration
ADD isEdit BOOL NOT NULL;
ALTER TABLE {$NAMESPACE}_search.search_editengineconfiguration
ADD createOrder INT UNSIGNED NOT NULL;
ALTER TABLE {$NAMESPACE}_search.search_editengineconfiguration
ADD editOrder INT UNSIGNED NOT NULL;

View file

@ -624,7 +624,7 @@ abstract class PhabricatorApplication
'(?P<id>[0-9]\d*)/)?'. '(?P<id>[0-9]\d*)/)?'.
'(?:'. '(?:'.
'(?:'. '(?:'.
'(?P<editAction>parameters|nodefault|comment)'. '(?P<editAction>parameters|nodefault|nocreate|nomanage|comment)'.
'|'. '|'.
'(?:form/(?P<formKey>[^/]+))'. '(?:form/(?P<formKey>[^/]+))'.
')'. ')'.

View file

@ -10,7 +10,7 @@ final class PhabricatorEditEngineConfigurationViewController
public function handleRequest(AphrontRequest $request) { public function handleRequest(AphrontRequest $request) {
$viewer = $this->getViewer(); $viewer = $this->getViewer();
$config = $this->loadConfigForEdit(); $config = $this->loadConfigForView();
if (!$config) { if (!$config) {
return id(new Aphront404Response()); return id(new Aphront404Response());
} }

View file

@ -35,6 +35,14 @@ abstract class PhabricatorEditEngineController
} }
protected function loadConfigForEdit() { protected function loadConfigForEdit() {
return $this->loadConfig($need_edit = true);
}
protected function loadConfigForView() {
return $this->loadConfig($need_edit = false);
}
private function loadConfig($need_edit) {
$request = $this->getRequest(); $request = $this->getRequest();
$viewer = $this->getViewer(); $viewer = $this->getViewer();
@ -43,17 +51,23 @@ abstract class PhabricatorEditEngineController
$key = $request->getURIData('key'); $key = $request->getURIData('key');
if ($need_edit) {
$capabilities = array(
PhabricatorPolicyCapability::CAN_VIEW,
PhabricatorPolicyCapability::CAN_EDIT,
);
} else {
$capabilities = array(
PhabricatorPolicyCapability::CAN_VIEW,
);
}
$config = id(new PhabricatorEditEngineConfigurationQuery()) $config = id(new PhabricatorEditEngineConfigurationQuery())
->setViewer($viewer) ->setViewer($viewer)
->withEngineKeys(array($engine_key)) ->withEngineKeys(array($engine_key))
->withIdentifiers(array($key)) ->withIdentifiers(array($key))
->requireCapabilities( ->requireCapabilities($capabilities)
array(
PhabricatorPolicyCapability::CAN_VIEW,
PhabricatorPolicyCapability::CAN_EDIT,
))
->executeOne(); ->executeOne();
if ($config) { if ($config) {
$engine = $config->getEngine(); $engine = $config->getEngine();

View file

@ -628,6 +628,7 @@ abstract class PhabricatorEditEngine
$capabilities = array(); $capabilities = array();
$use_default = false; $use_default = false;
$require_create = true;
switch ($action) { switch ($action) {
case 'comment': case 'comment':
$capabilities = array( $capabilities = array(
@ -638,6 +639,11 @@ abstract class PhabricatorEditEngine
case 'parameters': case 'parameters':
$use_default = true; $use_default = true;
break; break;
case 'nodefault':
case 'nocreate':
case 'nomanage':
$require_create = false;
break;
default: default:
break; break;
} }
@ -661,6 +667,12 @@ abstract class PhabricatorEditEngine
return new Aphront404Response(); return new Aphront404Response();
} }
} else { } else {
// Make sure the viewer has permission to create new objects of
// this type if we're going to create a new object.
if ($require_create) {
$this->requireCreateCapability();
}
$this->setIsCreate(true); $this->setIsCreate(true);
$object = $this->newEditableObject(); $object = $this->newEditableObject();
} }
@ -672,6 +684,10 @@ abstract class PhabricatorEditEngine
return $this->buildParametersResponse($object); return $this->buildParametersResponse($object);
case 'nodefault': case 'nodefault':
return $this->buildNoDefaultResponse($object); return $this->buildNoDefaultResponse($object);
case 'nocreate':
return $this->buildNoCreateResponse($object);
case 'nomanage':
return $this->buildNoManageResponse($object);
case 'comment': case 'comment':
return $this->buildCommentResponse($object); return $this->buildCommentResponse($object);
default: default:
@ -757,8 +773,7 @@ abstract class PhabricatorEditEngine
$editor->applyTransactions($object, $xactions); $editor->applyTransactions($object, $xactions);
return id(new AphrontRedirectResponse()) return $this->newEditResponse($request, $object, $xactions);
->setURI($this->getObjectViewURI($object));
} catch (PhabricatorApplicationTransactionValidationException $ex) { } catch (PhabricatorApplicationTransactionValidationException $ex) {
$validation_exception = $ex; $validation_exception = $ex;
@ -847,6 +862,14 @@ abstract class PhabricatorEditEngine
->appendChild($box); ->appendChild($box);
} }
protected function newEditResponse(
AphrontRequest $request,
$object,
array $xactions) {
return id(new AphrontRedirectResponse())
->setURI($this->getObjectViewURI($object));
}
private function buildEditForm($object, array $fields) { private function buildEditForm($object, array $fields) {
$viewer = $this->getViewer(); $viewer = $this->getViewer();
@ -896,32 +919,54 @@ abstract class PhabricatorEditEngine
private function buildEditFormActions($object) { private function buildEditFormActions($object) {
$actions = array(); $actions = array();
if ($this->supportsEditEngineConfiguration()) {
$engine_key = $this->getEngineKey();
$config = $this->getEditEngineConfiguration();
$actions[] = id(new PhabricatorActionView())
->setName(pht('Manage Form Configurations'))
->setIcon('fa-list-ul')
->setHref("/transactions/editengine/{$engine_key}/");
$actions[] = id(new PhabricatorActionView())
->setName(pht('Edit Form Configuration'))
->setIcon('fa-pencil')
->setHref($config->getURI());
}
$actions[] = id(new PhabricatorActionView()) $actions[] = id(new PhabricatorActionView())
->setName(pht('Show HTTP Parameters')) ->setName(pht('Show HTTP Parameters'))
->setIcon('fa-crosshairs') ->setIcon('fa-crosshairs')
->setHref($this->getEditURI($object, 'parameters/')); ->setHref($this->getEditURI($object, 'parameters/'));
if ($this->supportsEditEngineConfiguration()) {
$engine_key = $this->getEngineKey();
$config = $this->getEditEngineConfiguration();
$can_manage = PhabricatorPolicyFilter::hasCapability(
$this->getViewer(),
$config,
PhabricatorPolicyCapability::CAN_EDIT);
if ($can_manage) {
$manage_uri = $config->getURI();
} else {
$manage_uri = $this->getEditURI(null, 'nomanage/');
}
$view_uri = "/transactions/editengine/{$engine_key}/";
$actions[] = id(new PhabricatorActionView())
->setName(pht('View Form Configurations'))
->setIcon('fa-list-ul')
->setHref($view_uri);
$actions[] = id(new PhabricatorActionView())
->setName(pht('Edit Form Configuration'))
->setIcon('fa-pencil')
->setHref($manage_uri)
->setDisabled(!$can_manage)
->setWorkflow(!$can_manage);
}
return $actions; return $actions;
} }
final public function addActionToCrumbs(PHUICrumbsView $crumbs) { final public function addActionToCrumbs(PHUICrumbsView $crumbs) {
$viewer = $this->getViewer(); $viewer = $this->getViewer();
$configs = $this->loadUsableConfigurationsForCreate(); $can_create = $this->hasCreateCapability();
if ($can_create) {
$configs = $this->loadUsableConfigurationsForCreate();
} else {
$configs = array();
}
$dropdown = null; $dropdown = null;
$disabled = false; $disabled = false;
@ -938,7 +983,12 @@ abstract class PhabricatorEditEngine
$disabled = false; $disabled = false;
} }
$workflow = true; $workflow = true;
$create_uri = $this->getEditURI(null, 'nodefault/');
if ($can_create) {
$create_uri = $this->getEditURI(null, 'nodefault/');
} else {
$create_uri = $this->getEditURI(null, 'nocreate/');
}
} else { } else {
$config = head($configs); $config = head($configs);
$form_key = $config->getIdentifier(); $form_key = $config->getIdentifier();
@ -1109,6 +1159,31 @@ abstract class PhabricatorEditEngine
->addCancelButton($cancel_uri); ->addCancelButton($cancel_uri);
} }
private function buildNoCreateResponse($object) {
$cancel_uri = $this->getObjectCreateCancelURI($object);
return $this->getController()
->newDialog()
->setTitle(pht('No Create Permission'))
->appendParagraph(
pht(
'You do not have permission to create these objects.'))
->addCancelButton($cancel_uri);
}
private function buildNoManageResponse($object) {
$cancel_uri = $this->getObjectCreateCancelURI($object);
return $this->getController()
->newDialog()
->setTitle(pht('No Manage Permission'))
->appendParagraph(
pht(
'You do not have permission to configure forms for this '.
'application.'))
->addCancelButton($cancel_uri);
}
private function buildCommentResponse($object) { private function buildCommentResponse($object) {
$viewer = $this->getViewer(); $viewer = $this->getViewer();
@ -1266,6 +1341,8 @@ abstract class PhabricatorEditEngine
$this->setIsCreate(false); $this->setIsCreate(false);
$object = $this->newObjectFromIdentifier($identifier); $object = $this->newObjectFromIdentifier($identifier);
} else { } else {
$this->requireCreateCapability();
$this->setIsCreate(true); $this->setIsCreate(true);
$object = $this->newEditableObject(); $object = $this->newEditableObject();
} }
@ -1429,10 +1506,14 @@ abstract class PhabricatorEditEngine
} }
public function loadQuickCreateItems() { public function loadQuickCreateItems() {
$configs = $this->loadUsableConfigurationsForCreate();
$items = array(); $items = array();
if (!$this->hasCreateCapability()) {
return $items;
}
$configs = $this->loadUsableConfigurationsForCreate();
if (!$configs) { if (!$configs) {
// No items to add. // No items to add.
} else if (count($configs) == 1) { } else if (count($configs) == 1) {
@ -1456,12 +1537,16 @@ abstract class PhabricatorEditEngine
private function loadUsableConfigurationsForCreate() { private function loadUsableConfigurationsForCreate() {
$viewer = $this->getViewer(); $viewer = $this->getViewer();
return id(new PhabricatorEditEngineConfigurationQuery()) $configs = id(new PhabricatorEditEngineConfigurationQuery())
->setViewer($viewer) ->setViewer($viewer)
->withEngineKeys(array($this->getEngineKey())) ->withEngineKeys(array($this->getEngineKey()))
->withIsDefault(true) ->withIsDefault(true)
->withIsDisabled(false) ->withIsDisabled(false)
->execute(); ->execute();
$configs = msort($configs, 'getCreateSortKey');
return $configs;
} }
private function newQuickCreateItem( private function newQuickCreateItem(
@ -1478,6 +1563,24 @@ abstract class PhabricatorEditEngine
->setHref($item_uri); ->setHref($item_uri);
} }
protected function getCreateNewObjectPolicy() {
return PhabricatorPolicies::POLICY_USER;
}
private function requireCreateCapability() {
PhabricatorPolicyFilter::requireCapability(
$this->getViewer(),
$this,
PhabricatorPolicyCapability::CAN_EDIT);
}
private function hasCreateCapability() {
return PhabricatorPolicyFilter::hasCapability(
$this->getViewer(),
$this,
PhabricatorPolicyCapability::CAN_EDIT);
}
/* -( PhabricatorPolicyInterface )----------------------------------------- */ /* -( PhabricatorPolicyInterface )----------------------------------------- */
@ -1489,6 +1592,7 @@ abstract class PhabricatorEditEngine
public function getCapabilities() { public function getCapabilities() {
return array( return array(
PhabricatorPolicyCapability::CAN_VIEW, PhabricatorPolicyCapability::CAN_VIEW,
PhabricatorPolicyCapability::CAN_EDIT,
); );
} }
@ -1496,6 +1600,8 @@ abstract class PhabricatorEditEngine
switch ($capability) { switch ($capability) {
case PhabricatorPolicyCapability::CAN_VIEW: case PhabricatorPolicyCapability::CAN_VIEW:
return PhabricatorPolicies::getMostOpenPolicy(); return PhabricatorPolicies::getMostOpenPolicy();
case PhabricatorPolicyCapability::CAN_EDIT:
return $this->getCreateNewObjectPolicy();
} }
} }

View file

@ -19,6 +19,12 @@ final class PhabricatorEditEngineConfigurationEditEngine
return $this->targetEngine; return $this->targetEngine;
} }
protected function getCreateNewObjectPolicy() {
return $this->getTargetEngine()
->getApplication()
->getPolicy(PhabricatorPolicyCapability::CAN_EDIT);
}
public function getEngineName() { public function getEngineName() {
return pht('Edit Configurations'); return pht('Edit Configurations');
} }

View file

@ -15,7 +15,6 @@ final class PhabricatorEditEngineConfigurationEditor
$types = parent::getTransactionTypes(); $types = parent::getTransactionTypes();
$types[] = PhabricatorTransactions::TYPE_VIEW_POLICY; $types[] = PhabricatorTransactions::TYPE_VIEW_POLICY;
$types[] = PhabricatorTransactions::TYPE_EDIT_POLICY;
$types[] = PhabricatorEditEngineConfigurationTransaction::TYPE_NAME; $types[] = PhabricatorEditEngineConfigurationTransaction::TYPE_NAME;
$types[] = PhabricatorEditEngineConfigurationTransaction::TYPE_PREAMBLE; $types[] = PhabricatorEditEngineConfigurationTransaction::TYPE_PREAMBLE;

View file

@ -10,10 +10,12 @@ final class PhabricatorEditEngineConfiguration
protected $builtinKey; protected $builtinKey;
protected $name; protected $name;
protected $viewPolicy; protected $viewPolicy;
protected $editPolicy;
protected $properties = array(); protected $properties = array();
protected $isDisabled = 0; protected $isDisabled = 0;
protected $isDefault = 0; protected $isDefault = 0;
protected $isEdit = 0;
protected $createOrder = 0;
protected $editOrder = 0;
private $engine = self::ATTACHABLE; private $engine = self::ATTACHABLE;
@ -29,14 +31,10 @@ final class PhabricatorEditEngineConfiguration
PhabricatorUser $actor, PhabricatorUser $actor,
PhabricatorEditEngine $engine) { PhabricatorEditEngine $engine) {
// TODO: This should probably be controlled by a new default capability.
$edit_policy = PhabricatorPolicies::POLICY_ADMIN;
return id(new PhabricatorEditEngineConfiguration()) return id(new PhabricatorEditEngineConfiguration())
->setEngineKey($engine->getEngineKey()) ->setEngineKey($engine->getEngineKey())
->attachEngine($engine) ->attachEngine($engine)
->setViewPolicy(PhabricatorPolicies::getMostOpenPolicy()) ->setViewPolicy(PhabricatorPolicies::getMostOpenPolicy());
->setEditPolicy($edit_policy);
} }
public function generatePHID() { public function generatePHID() {
@ -44,6 +42,33 @@ final class PhabricatorEditEngineConfiguration
PhabricatorEditEngineConfigurationPHIDType::TYPECONST); PhabricatorEditEngineConfigurationPHIDType::TYPECONST);
} }
public function getCreateSortKey() {
return $this->getSortKey($this->createOrder);
}
public function getEditSortKey() {
return $this->getSortKey($this->editOrder);
}
private function getSortKey($order) {
// Put objects at the bottom by default if they haven't previously been
// reordered. When they're explicitly reordered, the smallest sort key we
// assign is 1, so if the object has a value of 0 it means it hasn't been
// ordered yet.
if ($order != 0) {
$group = 'A';
} else {
$group = 'B';
}
return sprintf(
"%s%012d%s\0%012d",
$group,
$order,
$this->getName(),
$this->getID());
}
protected function getConfiguration() { protected function getConfiguration() {
return array( return array(
self::CONFIG_AUX_PHID => true, self::CONFIG_AUX_PHID => true,
@ -56,6 +81,9 @@ final class PhabricatorEditEngineConfiguration
'name' => 'text255', 'name' => 'text255',
'isDisabled' => 'bool', 'isDisabled' => 'bool',
'isDefault' => 'bool', 'isDefault' => 'bool',
'isEdit' => 'bool',
'createOrder' => 'uint32',
'editOrder' => 'uint32',
), ),
self::CONFIG_KEY_SCHEMA => array( self::CONFIG_KEY_SCHEMA => array(
'key_engine' => array( 'key_engine' => array(
@ -65,6 +93,9 @@ final class PhabricatorEditEngineConfiguration
'key_default' => array( 'key_default' => array(
'columns' => array('engineKey', 'isDefault', 'isDisabled'), 'columns' => array('engineKey', 'isDefault', 'isDisabled'),
), ),
'key_edit' => array(
'columns' => array('engineKey', 'isEdit', 'isDisabled'),
),
), ),
) + parent::getConfiguration(); ) + parent::getConfiguration();
} }
@ -233,11 +264,21 @@ final class PhabricatorEditEngineConfiguration
case PhabricatorPolicyCapability::CAN_VIEW: case PhabricatorPolicyCapability::CAN_VIEW:
return $this->getViewPolicy(); return $this->getViewPolicy();
case PhabricatorPolicyCapability::CAN_EDIT: case PhabricatorPolicyCapability::CAN_EDIT:
return $this->getEditPolicy(); return $this->getEngine()
->getApplication()
->getPolicy($capability);
} }
} }
public function hasAutomaticCapability($capability, PhabricatorUser $viewer) { public function hasAutomaticCapability($capability, PhabricatorUser $viewer) {
switch ($capability) {
case PhabricatorPolicyCapability::CAN_VIEW:
return PhabricatorPolicyFilter::hasCapability(
$viewer,
$this->getEngine()->getApplication(),
PhabricatorPolicyCapability::CAN_EDIT);
}
return false; return false;
} }