mirror of
https://we.phorge.it/source/phorge.git
synced 2025-02-17 01:08:41 +01:00
Modularize Almanac Namespace transactions
Summary: Depends on D19318. Ref T13120. Ref T12414. Move transactions for Almanac Namespaces ("name" is the only meaningful one) to ModularTransactions. Test Plan: - Created a new namespace. - Edited a namespace. - Tried to choose no name, an invalid name, a duplicate name, and a name in a namespace I can't edit; got appropriate errors. - Grepped for `AlmanacNamespaceTransaction::TYPE_NAME`. Reviewers: amckinley Reviewed By: amckinley Maniphest Tasks: T13120, T12414 Differential Revision: https://secure.phabricator.com/D19320
This commit is contained in:
parent
6983479e4f
commit
5ada1211cd
6 changed files with 112 additions and 163 deletions
|
@ -75,11 +75,13 @@ phutil_register_library_map(array(
|
||||||
'AlmanacNamespaceEditor' => 'applications/almanac/editor/AlmanacNamespaceEditor.php',
|
'AlmanacNamespaceEditor' => 'applications/almanac/editor/AlmanacNamespaceEditor.php',
|
||||||
'AlmanacNamespaceListController' => 'applications/almanac/controller/AlmanacNamespaceListController.php',
|
'AlmanacNamespaceListController' => 'applications/almanac/controller/AlmanacNamespaceListController.php',
|
||||||
'AlmanacNamespaceNameNgrams' => 'applications/almanac/storage/AlmanacNamespaceNameNgrams.php',
|
'AlmanacNamespaceNameNgrams' => 'applications/almanac/storage/AlmanacNamespaceNameNgrams.php',
|
||||||
|
'AlmanacNamespaceNameTransaction' => 'applications/almanac/xaction/AlmanacNamespaceNameTransaction.php',
|
||||||
'AlmanacNamespacePHIDType' => 'applications/almanac/phid/AlmanacNamespacePHIDType.php',
|
'AlmanacNamespacePHIDType' => 'applications/almanac/phid/AlmanacNamespacePHIDType.php',
|
||||||
'AlmanacNamespaceQuery' => 'applications/almanac/query/AlmanacNamespaceQuery.php',
|
'AlmanacNamespaceQuery' => 'applications/almanac/query/AlmanacNamespaceQuery.php',
|
||||||
'AlmanacNamespaceSearchEngine' => 'applications/almanac/query/AlmanacNamespaceSearchEngine.php',
|
'AlmanacNamespaceSearchEngine' => 'applications/almanac/query/AlmanacNamespaceSearchEngine.php',
|
||||||
'AlmanacNamespaceTransaction' => 'applications/almanac/storage/AlmanacNamespaceTransaction.php',
|
'AlmanacNamespaceTransaction' => 'applications/almanac/storage/AlmanacNamespaceTransaction.php',
|
||||||
'AlmanacNamespaceTransactionQuery' => 'applications/almanac/query/AlmanacNamespaceTransactionQuery.php',
|
'AlmanacNamespaceTransactionQuery' => 'applications/almanac/query/AlmanacNamespaceTransactionQuery.php',
|
||||||
|
'AlmanacNamespaceTransactionType' => 'applications/almanac/xaction/AlmanacNamespaceTransactionType.php',
|
||||||
'AlmanacNamespaceViewController' => 'applications/almanac/controller/AlmanacNamespaceViewController.php',
|
'AlmanacNamespaceViewController' => 'applications/almanac/controller/AlmanacNamespaceViewController.php',
|
||||||
'AlmanacNetwork' => 'applications/almanac/storage/AlmanacNetwork.php',
|
'AlmanacNetwork' => 'applications/almanac/storage/AlmanacNetwork.php',
|
||||||
'AlmanacNetworkController' => 'applications/almanac/controller/AlmanacNetworkController.php',
|
'AlmanacNetworkController' => 'applications/almanac/controller/AlmanacNetworkController.php',
|
||||||
|
@ -5263,11 +5265,13 @@ phutil_register_library_map(array(
|
||||||
'AlmanacNamespaceEditor' => 'PhabricatorApplicationTransactionEditor',
|
'AlmanacNamespaceEditor' => 'PhabricatorApplicationTransactionEditor',
|
||||||
'AlmanacNamespaceListController' => 'AlmanacNamespaceController',
|
'AlmanacNamespaceListController' => 'AlmanacNamespaceController',
|
||||||
'AlmanacNamespaceNameNgrams' => 'PhabricatorSearchNgrams',
|
'AlmanacNamespaceNameNgrams' => 'PhabricatorSearchNgrams',
|
||||||
|
'AlmanacNamespaceNameTransaction' => 'AlmanacNamespaceTransactionType',
|
||||||
'AlmanacNamespacePHIDType' => 'PhabricatorPHIDType',
|
'AlmanacNamespacePHIDType' => 'PhabricatorPHIDType',
|
||||||
'AlmanacNamespaceQuery' => 'AlmanacQuery',
|
'AlmanacNamespaceQuery' => 'AlmanacQuery',
|
||||||
'AlmanacNamespaceSearchEngine' => 'PhabricatorApplicationSearchEngine',
|
'AlmanacNamespaceSearchEngine' => 'PhabricatorApplicationSearchEngine',
|
||||||
'AlmanacNamespaceTransaction' => 'PhabricatorApplicationTransaction',
|
'AlmanacNamespaceTransaction' => 'PhabricatorModularTransaction',
|
||||||
'AlmanacNamespaceTransactionQuery' => 'PhabricatorApplicationTransactionQuery',
|
'AlmanacNamespaceTransactionQuery' => 'PhabricatorApplicationTransactionQuery',
|
||||||
|
'AlmanacNamespaceTransactionType' => 'AlmanacTransactionType',
|
||||||
'AlmanacNamespaceViewController' => 'AlmanacNamespaceController',
|
'AlmanacNamespaceViewController' => 'AlmanacNamespaceController',
|
||||||
'AlmanacNetwork' => array(
|
'AlmanacNetwork' => array(
|
||||||
'AlmanacDAO',
|
'AlmanacDAO',
|
||||||
|
|
|
@ -81,7 +81,7 @@ final class AlmanacNamespaceEditEngine
|
||||||
->setKey('name')
|
->setKey('name')
|
||||||
->setLabel(pht('Name'))
|
->setLabel(pht('Name'))
|
||||||
->setDescription(pht('Name of the namespace.'))
|
->setDescription(pht('Name of the namespace.'))
|
||||||
->setTransactionType(AlmanacNamespaceTransaction::TYPE_NAME)
|
->setTransactionType(AlmanacNamespaceNameTransaction::TRANSACTIONTYPE)
|
||||||
->setIsRequired(true)
|
->setIsRequired(true)
|
||||||
->setValue($object->getName()),
|
->setValue($object->getName()),
|
||||||
);
|
);
|
||||||
|
|
|
@ -11,6 +11,14 @@ final class AlmanacNamespaceEditor
|
||||||
return pht('Almanac Namespace');
|
return pht('Almanac Namespace');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function getCreateObjectTitle($author, $object) {
|
||||||
|
return pht('%s created this namespace.', $author);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getCreateObjectTitleForFeed($author, $object) {
|
||||||
|
return pht('%s created %s.', $author, $object);
|
||||||
|
}
|
||||||
|
|
||||||
protected function supportsSearch() {
|
protected function supportsSearch() {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
@ -18,149 +26,12 @@ final class AlmanacNamespaceEditor
|
||||||
public function getTransactionTypes() {
|
public function getTransactionTypes() {
|
||||||
$types = parent::getTransactionTypes();
|
$types = parent::getTransactionTypes();
|
||||||
|
|
||||||
$types[] = AlmanacNamespaceTransaction::TYPE_NAME;
|
|
||||||
$types[] = PhabricatorTransactions::TYPE_VIEW_POLICY;
|
$types[] = PhabricatorTransactions::TYPE_VIEW_POLICY;
|
||||||
$types[] = PhabricatorTransactions::TYPE_EDIT_POLICY;
|
$types[] = PhabricatorTransactions::TYPE_EDIT_POLICY;
|
||||||
|
|
||||||
return $types;
|
return $types;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected function getCustomTransactionOldValue(
|
|
||||||
PhabricatorLiskDAO $object,
|
|
||||||
PhabricatorApplicationTransaction $xaction) {
|
|
||||||
switch ($xaction->getTransactionType()) {
|
|
||||||
case AlmanacNamespaceTransaction::TYPE_NAME:
|
|
||||||
return $object->getName();
|
|
||||||
}
|
|
||||||
|
|
||||||
return parent::getCustomTransactionOldValue($object, $xaction);
|
|
||||||
}
|
|
||||||
|
|
||||||
protected function getCustomTransactionNewValue(
|
|
||||||
PhabricatorLiskDAO $object,
|
|
||||||
PhabricatorApplicationTransaction $xaction) {
|
|
||||||
|
|
||||||
switch ($xaction->getTransactionType()) {
|
|
||||||
case AlmanacNamespaceTransaction::TYPE_NAME:
|
|
||||||
return $xaction->getNewValue();
|
|
||||||
}
|
|
||||||
|
|
||||||
return parent::getCustomTransactionNewValue($object, $xaction);
|
|
||||||
}
|
|
||||||
|
|
||||||
protected function applyCustomInternalTransaction(
|
|
||||||
PhabricatorLiskDAO $object,
|
|
||||||
PhabricatorApplicationTransaction $xaction) {
|
|
||||||
|
|
||||||
switch ($xaction->getTransactionType()) {
|
|
||||||
case AlmanacNamespaceTransaction::TYPE_NAME:
|
|
||||||
$object->setName($xaction->getNewValue());
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
return parent::applyCustomInternalTransaction($object, $xaction);
|
|
||||||
}
|
|
||||||
|
|
||||||
protected function applyCustomExternalTransaction(
|
|
||||||
PhabricatorLiskDAO $object,
|
|
||||||
PhabricatorApplicationTransaction $xaction) {
|
|
||||||
|
|
||||||
switch ($xaction->getTransactionType()) {
|
|
||||||
case AlmanacNamespaceTransaction::TYPE_NAME:
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
return parent::applyCustomExternalTransaction($object, $xaction);
|
|
||||||
}
|
|
||||||
|
|
||||||
protected function validateTransaction(
|
|
||||||
PhabricatorLiskDAO $object,
|
|
||||||
$type,
|
|
||||||
array $xactions) {
|
|
||||||
|
|
||||||
$errors = parent::validateTransaction($object, $type, $xactions);
|
|
||||||
|
|
||||||
switch ($type) {
|
|
||||||
case AlmanacNamespaceTransaction::TYPE_NAME:
|
|
||||||
$missing = $this->validateIsEmptyTextField(
|
|
||||||
$object->getName(),
|
|
||||||
$xactions);
|
|
||||||
|
|
||||||
if ($missing) {
|
|
||||||
$error = new PhabricatorApplicationTransactionValidationError(
|
|
||||||
$type,
|
|
||||||
pht('Required'),
|
|
||||||
pht('Namespace name is required.'),
|
|
||||||
nonempty(last($xactions), null));
|
|
||||||
|
|
||||||
$error->setIsMissingFieldError(true);
|
|
||||||
$errors[] = $error;
|
|
||||||
} else {
|
|
||||||
foreach ($xactions as $xaction) {
|
|
||||||
$name = $xaction->getNewValue();
|
|
||||||
|
|
||||||
$message = null;
|
|
||||||
try {
|
|
||||||
AlmanacNames::validateName($name);
|
|
||||||
} catch (Exception $ex) {
|
|
||||||
$message = $ex->getMessage();
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($message !== null) {
|
|
||||||
$error = new PhabricatorApplicationTransactionValidationError(
|
|
||||||
$type,
|
|
||||||
pht('Invalid'),
|
|
||||||
$message,
|
|
||||||
$xaction);
|
|
||||||
$errors[] = $error;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
$other = id(new AlmanacNamespaceQuery())
|
|
||||||
->setViewer(PhabricatorUser::getOmnipotentUser())
|
|
||||||
->withNames(array($name))
|
|
||||||
->executeOne();
|
|
||||||
if ($other && ($other->getID() != $object->getID())) {
|
|
||||||
$error = new PhabricatorApplicationTransactionValidationError(
|
|
||||||
$type,
|
|
||||||
pht('Not Unique'),
|
|
||||||
pht(
|
|
||||||
'The namespace name "%s" is already in use by another '.
|
|
||||||
'namespace. Each namespace must have a unique name.',
|
|
||||||
$name),
|
|
||||||
$xaction);
|
|
||||||
$errors[] = $error;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($name === $object->getName()) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
$namespace = AlmanacNamespace::loadRestrictedNamespace(
|
|
||||||
$this->getActor(),
|
|
||||||
$name);
|
|
||||||
if ($namespace) {
|
|
||||||
$error = new PhabricatorApplicationTransactionValidationError(
|
|
||||||
$type,
|
|
||||||
pht('Restricted'),
|
|
||||||
pht(
|
|
||||||
'You do not have permission to create Almanac namespaces '.
|
|
||||||
'within the "%s" namespace.',
|
|
||||||
$namespace->getName()),
|
|
||||||
$xaction);
|
|
||||||
$errors[] = $error;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
return $errors;
|
|
||||||
}
|
|
||||||
|
|
||||||
protected function didCatchDuplicateKeyException(
|
protected function didCatchDuplicateKeyException(
|
||||||
PhabricatorLiskDAO $object,
|
PhabricatorLiskDAO $object,
|
||||||
array $xactions,
|
array $xactions,
|
||||||
|
|
|
@ -1,9 +1,7 @@
|
||||||
<?php
|
<?php
|
||||||
|
|
||||||
final class AlmanacNamespaceTransaction
|
final class AlmanacNamespaceTransaction
|
||||||
extends PhabricatorApplicationTransaction {
|
extends PhabricatorModularTransaction {
|
||||||
|
|
||||||
const TYPE_NAME = 'almanac:namespace:name';
|
|
||||||
|
|
||||||
public function getApplicationName() {
|
public function getApplicationName() {
|
||||||
return 'almanac';
|
return 'almanac';
|
||||||
|
@ -17,27 +15,8 @@ final class AlmanacNamespaceTransaction
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getTitle() {
|
public function getBaseTransactionClass() {
|
||||||
$author_phid = $this->getAuthorPHID();
|
return 'AlmanacNamespaceTransactionType';
|
||||||
|
|
||||||
$old = $this->getOldValue();
|
|
||||||
$new = $this->getNewValue();
|
|
||||||
|
|
||||||
switch ($this->getTransactionType()) {
|
|
||||||
case PhabricatorTransactions::TYPE_CREATE:
|
|
||||||
return pht(
|
|
||||||
'%s created this namespace.',
|
|
||||||
$this->renderHandleLink($author_phid));
|
|
||||||
break;
|
|
||||||
case self::TYPE_NAME:
|
|
||||||
return pht(
|
|
||||||
'%s renamed this namespace from "%s" to "%s".',
|
|
||||||
$this->renderHandleLink($author_phid),
|
|
||||||
$old,
|
|
||||||
$new);
|
|
||||||
}
|
|
||||||
|
|
||||||
return parent::getTitle();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,91 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
final class AlmanacNamespaceNameTransaction
|
||||||
|
extends AlmanacNamespaceTransactionType {
|
||||||
|
|
||||||
|
const TRANSACTIONTYPE = 'almanac:namespace:name';
|
||||||
|
|
||||||
|
public function generateOldValue($object) {
|
||||||
|
return $object->getName();
|
||||||
|
}
|
||||||
|
|
||||||
|
public function applyInternalEffects($object, $value) {
|
||||||
|
$object->setName($value);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getTitle() {
|
||||||
|
return pht(
|
||||||
|
'%s renamed this namespace from %s to %s.',
|
||||||
|
$this->renderAuthor(),
|
||||||
|
$this->renderOldValue(),
|
||||||
|
$this->renderNewValue());
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getTitleForFeed() {
|
||||||
|
return pht(
|
||||||
|
'%s renamed %s from %s to %s.',
|
||||||
|
$this->renderAuthor(),
|
||||||
|
$this->renderObject(),
|
||||||
|
$this->renderOldValue(),
|
||||||
|
$this->renderNewValue());
|
||||||
|
}
|
||||||
|
|
||||||
|
public function validateTransactions($object, array $xactions) {
|
||||||
|
$errors = array();
|
||||||
|
|
||||||
|
if ($this->isEmptyTextTransaction($object->getName(), $xactions)) {
|
||||||
|
$errors[] = $this->newRequiredError(
|
||||||
|
pht('Namespace name is required.'));
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($xactions as $xaction) {
|
||||||
|
$name = $xaction->getNewValue();
|
||||||
|
|
||||||
|
$message = null;
|
||||||
|
try {
|
||||||
|
AlmanacNames::validateName($name);
|
||||||
|
} catch (Exception $ex) {
|
||||||
|
$message = $ex->getMessage();
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($message !== null) {
|
||||||
|
$errors[] = $this->newInvalidError($message, $xaction);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($name === $object->getName()) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$other = id(new AlmanacNamespaceQuery())
|
||||||
|
->setViewer(PhabricatorUser::getOmnipotentUser())
|
||||||
|
->withNames(array($name))
|
||||||
|
->executeOne();
|
||||||
|
if ($other && ($other->getID() != $object->getID())) {
|
||||||
|
$errors[] = $this->newInvalidError(
|
||||||
|
pht(
|
||||||
|
'The namespace name "%s" is already in use by another '.
|
||||||
|
'namespace. Each namespace must have a unique name.',
|
||||||
|
$name),
|
||||||
|
$xaction);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$namespace = AlmanacNamespace::loadRestrictedNamespace(
|
||||||
|
$this->getActor(),
|
||||||
|
$name);
|
||||||
|
if ($namespace) {
|
||||||
|
$errors[] = $this->newInvalidError(
|
||||||
|
pht(
|
||||||
|
'You do not have permission to create Almanac namespaces '.
|
||||||
|
'within the "%s" namespace.',
|
||||||
|
$namespace->getName()),
|
||||||
|
$xaction);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $errors;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,4 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
abstract class AlmanacNamespaceTransactionType
|
||||||
|
extends AlmanacTransactionType {}
|
Loading…
Add table
Reference in a new issue