mirror of
https://we.phorge.it/source/phorge.git
synced 2024-12-31 18:01:00 +01:00
Differential - make DifferentialDiffEditor into a real transaction editor.
Summary: Ref T6237. This sets us up for some future work like T6152, T6200 and generally cleaning up this workflow a bit. Tried to do as little as possible so not exposing transaction view yet. (Though that timeline is going to be a little funky in the common case of just the lone create transaction.) Test Plan: made a diff from web ui and it worked. made a herald rule to block certain diffs then tried to make such a diff and saw UI letting me know i was blocked Reviewers: epriestley Reviewed By: epriestley Subscribers: Korvin, epriestley Maniphest Tasks: T6237 Differential Revision: https://secure.phabricator.com/D10869
This commit is contained in:
parent
edf88225f5
commit
ffe0765b50
8 changed files with 376 additions and 143 deletions
19
resources/sql/autopatches/20141118.diffxaction.sql
Normal file
19
resources/sql/autopatches/20141118.diffxaction.sql
Normal file
|
@ -0,0 +1,19 @@
|
||||||
|
CREATE TABLE {$NAMESPACE}_differential.differential_difftransaction (
|
||||||
|
id INT UNSIGNED NOT NULL AUTO_INCREMENT PRIMARY KEY,
|
||||||
|
phid VARBINARY(64) NOT NULL,
|
||||||
|
authorPHID VARBINARY(64) NOT NULL,
|
||||||
|
objectPHID VARBINARY(64) NOT NULL,
|
||||||
|
viewPolicy VARBINARY(64) NOT NULL,
|
||||||
|
editPolicy VARBINARY(64) NOT NULL,
|
||||||
|
commentPHID VARBINARY(64),
|
||||||
|
commentVersion INT UNSIGNED NOT NULL,
|
||||||
|
transactionType VARCHAR(32) COLLATE {$COLLATE_TEXT} NOT NULL,
|
||||||
|
oldValue LONGTEXT COLLATE {$COLLATE_TEXT} NOT NULL,
|
||||||
|
newValue LONGTEXT COLLATE {$COLLATE_TEXT} NOT NULL,
|
||||||
|
contentSource LONGTEXT COLLATE {$COLLATE_TEXT} NOT NULL,
|
||||||
|
metadata LONGTEXT COLLATE {$COLLATE_TEXT} NOT NULL,
|
||||||
|
dateCreated INT UNSIGNED NOT NULL,
|
||||||
|
dateModified INT UNSIGNED NOT NULL,
|
||||||
|
UNIQUE KEY `key_phid` (`phid`),
|
||||||
|
KEY `key_object` (`objectPHID`)
|
||||||
|
) ENGINE=InnoDB, COLLATE {$COLLATE_TEXT};
|
|
@ -325,6 +325,7 @@ phutil_register_library_map(array(
|
||||||
'DifferentialDiffQuery' => 'applications/differential/query/DifferentialDiffQuery.php',
|
'DifferentialDiffQuery' => 'applications/differential/query/DifferentialDiffQuery.php',
|
||||||
'DifferentialDiffTableOfContentsView' => 'applications/differential/view/DifferentialDiffTableOfContentsView.php',
|
'DifferentialDiffTableOfContentsView' => 'applications/differential/view/DifferentialDiffTableOfContentsView.php',
|
||||||
'DifferentialDiffTestCase' => 'applications/differential/storage/__tests__/DifferentialDiffTestCase.php',
|
'DifferentialDiffTestCase' => 'applications/differential/storage/__tests__/DifferentialDiffTestCase.php',
|
||||||
|
'DifferentialDiffTransaction' => 'applications/differential/storage/DifferentialDiffTransaction.php',
|
||||||
'DifferentialDiffViewController' => 'applications/differential/controller/DifferentialDiffViewController.php',
|
'DifferentialDiffViewController' => 'applications/differential/controller/DifferentialDiffViewController.php',
|
||||||
'DifferentialDoorkeeperRevisionFeedStoryPublisher' => 'applications/differential/doorkeeper/DifferentialDoorkeeperRevisionFeedStoryPublisher.php',
|
'DifferentialDoorkeeperRevisionFeedStoryPublisher' => 'applications/differential/doorkeeper/DifferentialDoorkeeperRevisionFeedStoryPublisher.php',
|
||||||
'DifferentialDraft' => 'applications/differential/storage/DifferentialDraft.php',
|
'DifferentialDraft' => 'applications/differential/storage/DifferentialDraft.php',
|
||||||
|
@ -3313,12 +3314,13 @@ phutil_register_library_map(array(
|
||||||
),
|
),
|
||||||
'DifferentialDiffCreateController' => 'DifferentialController',
|
'DifferentialDiffCreateController' => 'DifferentialController',
|
||||||
'DifferentialDiffCreationRejectException' => 'Exception',
|
'DifferentialDiffCreationRejectException' => 'Exception',
|
||||||
'DifferentialDiffEditor' => 'PhabricatorEditor',
|
'DifferentialDiffEditor' => 'PhabricatorApplicationTransactionEditor',
|
||||||
'DifferentialDiffPHIDType' => 'PhabricatorPHIDType',
|
'DifferentialDiffPHIDType' => 'PhabricatorPHIDType',
|
||||||
'DifferentialDiffProperty' => 'DifferentialDAO',
|
'DifferentialDiffProperty' => 'DifferentialDAO',
|
||||||
'DifferentialDiffQuery' => 'PhabricatorCursorPagedPolicyAwareQuery',
|
'DifferentialDiffQuery' => 'PhabricatorCursorPagedPolicyAwareQuery',
|
||||||
'DifferentialDiffTableOfContentsView' => 'AphrontView',
|
'DifferentialDiffTableOfContentsView' => 'AphrontView',
|
||||||
'DifferentialDiffTestCase' => 'ArcanistPhutilTestCase',
|
'DifferentialDiffTestCase' => 'ArcanistPhutilTestCase',
|
||||||
|
'DifferentialDiffTransaction' => 'PhabricatorApplicationTransaction',
|
||||||
'DifferentialDiffViewController' => 'DifferentialController',
|
'DifferentialDiffViewController' => 'DifferentialController',
|
||||||
'DifferentialDoorkeeperRevisionFeedStoryPublisher' => 'DoorkeeperFeedStoryPublisher',
|
'DifferentialDoorkeeperRevisionFeedStoryPublisher' => 'DoorkeeperFeedStoryPublisher',
|
||||||
'DifferentialDraft' => 'DifferentialDAO',
|
'DifferentialDraft' => 'DifferentialDAO',
|
||||||
|
|
|
@ -70,19 +70,13 @@ final class DifferentialCreateDiffConduitAPIMethod
|
||||||
}
|
}
|
||||||
|
|
||||||
$diff = DifferentialDiff::newFromRawChanges($changes);
|
$diff = DifferentialDiff::newFromRawChanges($changes);
|
||||||
$diff->setSourcePath($request->getValue('sourcePath'));
|
|
||||||
$diff->setSourceMachine($request->getValue('sourceMachine'));
|
|
||||||
|
|
||||||
$diff->setBranch($request->getValue('branch'));
|
// TODO: Remove repository UUID eventually; for now continue writing
|
||||||
$diff->setCreationMethod($request->getValue('creationMethod'));
|
// the UUID. Note that we'll overwrite it below if we identify a
|
||||||
$diff->setAuthorPHID($viewer->getPHID());
|
// repository, and `arc` no longer sends it. This stuff is retained for
|
||||||
$diff->setBookmark($request->getValue('bookmark'));
|
// backward compatibility.
|
||||||
|
|
||||||
// TODO: Remove this eventually; for now continue writing the UUID. Note
|
|
||||||
// that we'll overwrite it below if we identify a repository, and `arc`
|
|
||||||
// no longer sends it. This stuff is retained for backward compatibility.
|
|
||||||
$diff->setRepositoryUUID($request->getValue('repositoryUUID'));
|
|
||||||
|
|
||||||
|
$repository_uuid = $request->getValue('repositoryUUID');
|
||||||
$repository_phid = $request->getValue('repositoryPHID');
|
$repository_phid = $request->getValue('repositoryPHID');
|
||||||
if ($repository_phid) {
|
if ($repository_phid) {
|
||||||
$repository = id(new PhabricatorRepositoryQuery())
|
$repository = id(new PhabricatorRepositoryQuery())
|
||||||
|
@ -90,17 +84,11 @@ final class DifferentialCreateDiffConduitAPIMethod
|
||||||
->withPHIDs(array($repository_phid))
|
->withPHIDs(array($repository_phid))
|
||||||
->executeOne();
|
->executeOne();
|
||||||
if ($repository) {
|
if ($repository) {
|
||||||
$diff->setRepositoryPHID($repository->getPHID());
|
$repository_phid = $repository->getPHID();
|
||||||
$diff->setRepositoryUUID($repository->getUUID());
|
$repository_uuid = $repository->getUUID();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
$system = $request->getValue('sourceControlSystem');
|
|
||||||
$diff->setSourceControlSystem($system);
|
|
||||||
$diff->setSourceControlPath($request->getValue('sourceControlPath'));
|
|
||||||
$diff->setSourceControlBaseRevision(
|
|
||||||
$request->getValue('sourceControlBaseRevision'));
|
|
||||||
|
|
||||||
$project_name = $request->getValue('arcanistProject');
|
$project_name = $request->getValue('arcanistProject');
|
||||||
$project_phid = null;
|
$project_phid = null;
|
||||||
if ($project_name) {
|
if ($project_name) {
|
||||||
|
@ -116,73 +104,75 @@ final class DifferentialCreateDiffConduitAPIMethod
|
||||||
$project_phid = $arcanist_project->getPHID();
|
$project_phid = $arcanist_project->getPHID();
|
||||||
}
|
}
|
||||||
|
|
||||||
$diff->setArcanistProjectPHID($project_phid);
|
|
||||||
|
|
||||||
switch ($request->getValue('lintStatus')) {
|
switch ($request->getValue('lintStatus')) {
|
||||||
case 'skip':
|
case 'skip':
|
||||||
$diff->setLintStatus(DifferentialLintStatus::LINT_SKIP);
|
$lint_status = DifferentialLintStatus::LINT_SKIP;
|
||||||
break;
|
break;
|
||||||
case 'okay':
|
case 'okay':
|
||||||
$diff->setLintStatus(DifferentialLintStatus::LINT_OKAY);
|
$lint_status = DifferentialLintStatus::LINT_OKAY;
|
||||||
break;
|
break;
|
||||||
case 'warn':
|
case 'warn':
|
||||||
$diff->setLintStatus(DifferentialLintStatus::LINT_WARN);
|
$lint_status = DifferentialLintStatus::LINT_WARN;
|
||||||
break;
|
break;
|
||||||
case 'fail':
|
case 'fail':
|
||||||
$diff->setLintStatus(DifferentialLintStatus::LINT_FAIL);
|
$lint_status = DifferentialLintStatus::LINT_FAIL;
|
||||||
break;
|
break;
|
||||||
case 'postponed':
|
case 'postponed':
|
||||||
$diff->setLintStatus(DifferentialLintStatus::LINT_POSTPONED);
|
$lint_status = DifferentialLintStatus::LINT_POSTPONED;
|
||||||
break;
|
break;
|
||||||
case 'none':
|
case 'none':
|
||||||
default:
|
default:
|
||||||
$diff->setLintStatus(DifferentialLintStatus::LINT_NONE);
|
$lint_status = DifferentialLintStatus::LINT_NONE;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
switch ($request->getValue('unitStatus')) {
|
switch ($request->getValue('unitStatus')) {
|
||||||
case 'skip':
|
case 'skip':
|
||||||
$diff->setUnitStatus(DifferentialUnitStatus::UNIT_SKIP);
|
$unit_status = DifferentialUnitStatus::UNIT_SKIP;
|
||||||
break;
|
break;
|
||||||
case 'okay':
|
case 'okay':
|
||||||
$diff->setUnitStatus(DifferentialUnitStatus::UNIT_OKAY);
|
$unit_status = DifferentialUnitStatus::UNIT_OKAY;
|
||||||
break;
|
break;
|
||||||
case 'warn':
|
case 'warn':
|
||||||
$diff->setUnitStatus(DifferentialUnitStatus::UNIT_WARN);
|
$unit_status = DifferentialUnitStatus::UNIT_WARN;
|
||||||
break;
|
break;
|
||||||
case 'fail':
|
case 'fail':
|
||||||
$diff->setUnitStatus(DifferentialUnitStatus::UNIT_FAIL);
|
$unit_status = DifferentialUnitStatus::UNIT_FAIL;
|
||||||
break;
|
break;
|
||||||
case 'postponed':
|
case 'postponed':
|
||||||
$diff->setUnitStatus(DifferentialUnitStatus::UNIT_POSTPONED);
|
$unit_status = DifferentialUnitStatus::UNIT_POSTPONED;
|
||||||
break;
|
break;
|
||||||
case 'none':
|
case 'none':
|
||||||
default:
|
default:
|
||||||
$diff->setUnitStatus(DifferentialUnitStatus::UNIT_NONE);
|
$unit_status = DifferentialUnitStatus::UNIT_NONE;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$diff_data_dict = array(
|
||||||
|
'sourcePath' => $request->getValue('sourcePath'),
|
||||||
|
'sourceMachine' => $request->getValue('sourceMachine'),
|
||||||
|
'branch' => $request->getValue('branch'),
|
||||||
|
'creationMethod' => $request->getValue('creationMethod'),
|
||||||
|
'authorPHID' => $viewer->getPHID(),
|
||||||
|
'bookmark' => $request->getValue('bookmark'),
|
||||||
|
'repositoryUUID' => $repository_uuid,
|
||||||
|
'repositoryPHID' => $repository_phid,
|
||||||
|
'sourceControlSystem' => $request->getValue('sourceControlSystem'),
|
||||||
|
'sourceControlPath' => $request->getValue('sourceControlPath'),
|
||||||
|
'sourceControlBaseRevision' =>
|
||||||
|
$request->getValue('sourceControlBaseRevision'),
|
||||||
|
'arcanistProjectPHID' => $project_phid,
|
||||||
|
'lintStatus' => $lint_status,
|
||||||
|
'unitStatus' => $unit_status,);
|
||||||
|
|
||||||
|
$xactions = array(id(new DifferentialTransaction())
|
||||||
|
->setTransactionType(DifferentialDiffTransaction::TYPE_DIFF_CREATE)
|
||||||
|
->setNewValue($diff_data_dict),);
|
||||||
|
|
||||||
id(new DifferentialDiffEditor())
|
id(new DifferentialDiffEditor())
|
||||||
->setActor($viewer)
|
->setActor($viewer)
|
||||||
->setContentSource(
|
->setContentSourceFromConduitRequest($request)
|
||||||
PhabricatorContentSource::newFromConduitRequest($request))
|
->applyTransactions($diff, $xactions);
|
||||||
->saveDiff($diff);
|
|
||||||
|
|
||||||
// If we didn't get an explicit `repositoryPHID` (which means the client is
|
|
||||||
// old, or couldn't figure out which repository the working copy belongs
|
|
||||||
// to), apply heuristics to try to figure it out.
|
|
||||||
|
|
||||||
if (!$repository_phid) {
|
|
||||||
$repository = id(new DifferentialRepositoryLookup())
|
|
||||||
->setDiff($diff)
|
|
||||||
->setViewer($viewer)
|
|
||||||
->lookupRepository();
|
|
||||||
if ($repository) {
|
|
||||||
$diff->setRepositoryPHID($repository->getPHID());
|
|
||||||
$diff->setRepositoryUUID($repository->getUUID());
|
|
||||||
$diff->save();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
$path = '/differential/diff/'.$diff->getID().'/';
|
$path = '/differential/diff/'.$diff->getID().'/';
|
||||||
$uri = PhabricatorEnv::getURI($path);
|
$uri = PhabricatorEnv::getURI($path);
|
||||||
|
|
|
@ -41,29 +41,27 @@ final class DifferentialCreateRawDiffConduitAPIMethod
|
||||||
throw new Exception(
|
throw new Exception(
|
||||||
pht('No such repository "%s"!', $repository_phid));
|
pht('No such repository "%s"!', $repository_phid));
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
$repository = null;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
$parser = new ArcanistDiffParser();
|
$parser = new ArcanistDiffParser();
|
||||||
$changes = $parser->parseDiff($raw_diff);
|
$changes = $parser->parseDiff($raw_diff);
|
||||||
$diff = DifferentialDiff::newFromRawChanges($changes);
|
$diff = DifferentialDiff::newFromRawChanges($changes);
|
||||||
|
|
||||||
$diff->setLintStatus(DifferentialLintStatus::LINT_SKIP);
|
$diff_data_dict = array(
|
||||||
$diff->setUnitStatus(DifferentialUnitStatus::UNIT_SKIP);
|
'creationMethod' => 'web',
|
||||||
|
'authorPHID' => $viewer->getPHID(),
|
||||||
|
'repositoryPHID' => $repository_phid,
|
||||||
|
'lintStatus' => DifferentialLintStatus::LINT_SKIP,
|
||||||
|
'unitStatus' => DifferentialUnitStatus::UNIT_SKIP,);
|
||||||
|
|
||||||
$diff->setAuthorPHID($viewer->getPHID());
|
$xactions = array(id(new DifferentialTransaction())
|
||||||
$diff->setCreationMethod('web');
|
->setTransactionType(DifferentialDiffTransaction::TYPE_DIFF_CREATE)
|
||||||
|
->setNewValue($diff_data_dict),);
|
||||||
if ($repository) {
|
|
||||||
$diff->setRepositoryPHID($repository->getPHID());
|
|
||||||
}
|
|
||||||
|
|
||||||
id(new DifferentialDiffEditor())
|
id(new DifferentialDiffEditor())
|
||||||
->setActor($viewer)
|
->setActor($viewer)
|
||||||
->setContentSource(
|
->setContentSourceFromConduitRequest($request)
|
||||||
PhabricatorContentSource::newFromConduitRequest($request))
|
->applyTransactions($diff, $xactions);
|
||||||
->saveDiff($diff);
|
|
||||||
|
|
||||||
return $this->buildDiffInfoDictionary($diff);
|
return $this->buildDiffInfoDictionary($diff);
|
||||||
}
|
}
|
||||||
|
|
|
@ -6,11 +6,12 @@ final class DifferentialDiffCreateController extends DifferentialController {
|
||||||
|
|
||||||
$request = $this->getRequest();
|
$request = $this->getRequest();
|
||||||
|
|
||||||
|
$diff = null;
|
||||||
$errors = array();
|
$errors = array();
|
||||||
$e_diff = null;
|
$e_diff = null;
|
||||||
$e_file = null;
|
$e_file = null;
|
||||||
|
$validation_exception = null;
|
||||||
if ($request->isFormPost()) {
|
if ($request->isFormPost()) {
|
||||||
$diff = null;
|
|
||||||
|
|
||||||
if ($request->getFileExists('diff-file')) {
|
if ($request->getFileExists('diff-file')) {
|
||||||
$diff = PhabricatorFile::readUploadedFileData($_FILES['diff-file']);
|
$diff = PhabricatorFile::readUploadedFileData($_FILES['diff-file']);
|
||||||
|
@ -27,16 +28,19 @@ final class DifferentialDiffCreateController extends DifferentialController {
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!$errors) {
|
if (!$errors) {
|
||||||
$call = new ConduitCall(
|
try {
|
||||||
'differential.createrawdiff',
|
$call = new ConduitCall(
|
||||||
array(
|
'differential.createrawdiff',
|
||||||
'diff' => $diff,
|
array(
|
||||||
|
'diff' => $diff,
|
||||||
));
|
));
|
||||||
$call->setUser($request->getUser());
|
$call->setUser($request->getUser());
|
||||||
$result = $call->execute();
|
$result = $call->execute();
|
||||||
|
$path = id(new PhutilURI($result['uri']))->getPath();
|
||||||
$path = id(new PhutilURI($result['uri']))->getPath();
|
return id(new AphrontRedirectResponse())->setURI($path);
|
||||||
return id(new AphrontRedirectResponse())->setURI($path);
|
} catch (PhabricatorApplicationTransactionValidationException $ex) {
|
||||||
|
$validation_exception = $ex;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -69,6 +73,7 @@ final class DifferentialDiffCreateController extends DifferentialController {
|
||||||
id(new AphrontFormTextAreaControl())
|
id(new AphrontFormTextAreaControl())
|
||||||
->setLabel(pht('Raw Diff'))
|
->setLabel(pht('Raw Diff'))
|
||||||
->setName('diff')
|
->setName('diff')
|
||||||
|
->setValue($diff)
|
||||||
->setHeight(AphrontFormTextAreaControl::HEIGHT_VERY_TALL)
|
->setHeight(AphrontFormTextAreaControl::HEIGHT_VERY_TALL)
|
||||||
->setError($e_diff))
|
->setError($e_diff))
|
||||||
->appendChild(
|
->appendChild(
|
||||||
|
@ -83,6 +88,7 @@ final class DifferentialDiffCreateController extends DifferentialController {
|
||||||
|
|
||||||
$form_box = id(new PHUIObjectBoxView())
|
$form_box = id(new PHUIObjectBoxView())
|
||||||
->setHeaderText(pht('Create New Diff'))
|
->setHeaderText(pht('Create New Diff'))
|
||||||
|
->setValidationException($validation_exception)
|
||||||
->setForm($form)
|
->setForm($form)
|
||||||
->setFormErrors($errors);
|
->setFormErrors($errors);
|
||||||
|
|
||||||
|
|
|
@ -1,80 +1,236 @@
|
||||||
<?php
|
<?php
|
||||||
|
|
||||||
final class DifferentialDiffEditor extends PhabricatorEditor {
|
final class DifferentialDiffEditor
|
||||||
|
extends PhabricatorApplicationTransactionEditor {
|
||||||
|
|
||||||
private $contentSource;
|
private $diffDataDict;
|
||||||
|
|
||||||
public function setContentSource($content_source) {
|
public function getEditorApplicationClass() {
|
||||||
$this->contentSource = $content_source;
|
return 'PhabricatorDifferentialApplication';
|
||||||
return $this;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getContentSource() {
|
public function getEditorObjectsDescription() {
|
||||||
return $this->contentSource;
|
return pht('Differential Diffs');
|
||||||
}
|
}
|
||||||
|
|
||||||
public function saveDiff(DifferentialDiff $diff) {
|
public function getTransactionTypes() {
|
||||||
$actor = $this->requireActor();
|
$types = parent::getTransactionTypes();
|
||||||
|
|
||||||
// Generate a PHID first, so the transcript will point at the object if
|
$types[] = DifferentialDiffTransaction::TYPE_DIFF_CREATE;
|
||||||
// we deicde to preserve it.
|
|
||||||
$phid = $diff->generatePHID();
|
return $types;
|
||||||
$diff->setPHID($phid);
|
}
|
||||||
|
|
||||||
|
protected function getCustomTransactionOldValue(
|
||||||
|
PhabricatorLiskDAO $object,
|
||||||
|
PhabricatorApplicationTransaction $xaction) {
|
||||||
|
|
||||||
|
switch ($xaction->getTransactionType()) {
|
||||||
|
case DifferentialDiffTransaction::TYPE_DIFF_CREATE:
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return parent::getCustomTransactionOldValue($object, $xaction);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function getCustomTransactionNewValue(
|
||||||
|
PhabricatorLiskDAO $object,
|
||||||
|
PhabricatorApplicationTransaction $xaction) {
|
||||||
|
|
||||||
|
switch ($xaction->getTransactionType()) {
|
||||||
|
case DifferentialDiffTransaction::TYPE_DIFF_CREATE:
|
||||||
|
$this->diffDataDict = $xaction->getNewValue();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return parent::getCustomTransactionNewValue($object, $xaction);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function applyCustomInternalTransaction(
|
||||||
|
PhabricatorLiskDAO $object,
|
||||||
|
PhabricatorApplicationTransaction $xaction) {
|
||||||
|
|
||||||
|
switch ($xaction->getTransactionType()) {
|
||||||
|
case DifferentialDiffTransaction::TYPE_DIFF_CREATE:
|
||||||
|
$dict = $this->diffDataDict;
|
||||||
|
$this->updateDiffFromDict($object, $dict);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
return parent::applyCustomInternalTransaction($object, $xaction);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function applyCustomExternalTransaction(
|
||||||
|
PhabricatorLiskDAO $object,
|
||||||
|
PhabricatorApplicationTransaction $xaction) {
|
||||||
|
|
||||||
|
switch ($xaction->getTransactionType()) {
|
||||||
|
case DifferentialDiffTransaction::TYPE_DIFF_CREATE:
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
return parent::applyCustomExternalTransaction($object, $xaction);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function applyFinalEffects(
|
||||||
|
PhabricatorLiskDAO $object,
|
||||||
|
array $xactions) {
|
||||||
|
|
||||||
|
// If we didn't get an explicit `repositoryPHID` (which means the client
|
||||||
|
// is old, or couldn't figure out which repository the working copy
|
||||||
|
// belongs to), apply heuristics to try to figure it out.
|
||||||
|
|
||||||
|
if (!$object->getRepositoryPHID()) {
|
||||||
|
$repository = id(new DifferentialRepositoryLookup())
|
||||||
|
->setDiff($object)
|
||||||
|
->setViewer($this->getActor())
|
||||||
|
->lookupRepository();
|
||||||
|
if ($repository) {
|
||||||
|
$object->setRepositoryPHID($repository->getPHID());
|
||||||
|
$object->setRepositoryUUID($repository->getUUID());
|
||||||
|
$object->save();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $xactions;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* We run Herald as part of transaction validation because Herald can
|
||||||
|
* block diff creation for Differential diffs. Its important to do this
|
||||||
|
* separately so no Herald logs are saved; these logs could expose
|
||||||
|
* information the Herald rules are inteneded to block.
|
||||||
|
*/
|
||||||
|
protected function validateTransaction(
|
||||||
|
PhabricatorLiskDAO $object,
|
||||||
|
$type,
|
||||||
|
array $xactions) {
|
||||||
|
|
||||||
|
$errors = parent::validateTransaction($object, $type, $xactions);
|
||||||
|
|
||||||
|
foreach ($xactions as $xaction) {
|
||||||
|
switch ($type) {
|
||||||
|
case DifferentialDiffTransaction::TYPE_DIFF_CREATE:
|
||||||
|
$diff = clone $object;
|
||||||
|
$diff = $this->updateDiffFromDict($diff, $xaction->getNewValue());
|
||||||
|
|
||||||
|
$adapter = $this->buildHeraldAdapter($diff, $xactions);
|
||||||
|
$adapter->setContentSource($this->getContentSource());
|
||||||
|
$adapter->setIsNewObject($this->getIsNewObject());
|
||||||
|
|
||||||
|
$engine = new HeraldEngine();
|
||||||
|
|
||||||
|
$rules = $engine->loadRulesForAdapter($adapter);
|
||||||
|
$rules = mpull($rules, null, 'getID');
|
||||||
|
|
||||||
|
$effects = $engine->applyRules($rules, $adapter);
|
||||||
|
|
||||||
|
$blocking_effect = null;
|
||||||
|
foreach ($effects as $effect) {
|
||||||
|
if ($effect->getAction() == HeraldAdapter::ACTION_BLOCK) {
|
||||||
|
$blocking_effect = $effect;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($blocking_effect) {
|
||||||
|
$rule = idx($rules, $effect->getRuleID());
|
||||||
|
if ($rule && strlen($rule->getName())) {
|
||||||
|
$rule_name = $rule->getName();
|
||||||
|
} else {
|
||||||
|
$rule_name = pht('Unnamed Herald Rule');
|
||||||
|
}
|
||||||
|
|
||||||
|
$message = $effect->getTarget();
|
||||||
|
if (!strlen($message)) {
|
||||||
|
$message = pht('(None.)');
|
||||||
|
}
|
||||||
|
|
||||||
|
$errors[] = new PhabricatorApplicationTransactionValidationError(
|
||||||
|
$type,
|
||||||
|
pht('Rejected by Herald'),
|
||||||
|
pht(
|
||||||
|
"Creation of this diff was rejected by Herald rule %s.\n".
|
||||||
|
" Rule: %s\n".
|
||||||
|
"Reason: %s",
|
||||||
|
'H'.$effect->getRuleID(),
|
||||||
|
$rule_name,
|
||||||
|
$message));
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $errors;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
protected function shouldPublishFeedStory(
|
||||||
|
PhabricatorLiskDAO $object,
|
||||||
|
array $xactions) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function shouldSendMail(
|
||||||
|
PhabricatorLiskDAO $object,
|
||||||
|
array $xactions) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function supportsSearch() {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* -( Herald Integration )------------------------------------------------- */
|
||||||
|
|
||||||
|
/**
|
||||||
|
* See @{method:validateTransaction}. The only Herald action is to block
|
||||||
|
* the creation of Diffs. We thus have to be careful not to save any
|
||||||
|
* data and do this validation very early.
|
||||||
|
*/
|
||||||
|
protected function shouldApplyHeraldRules(
|
||||||
|
PhabricatorLiskDAO $object,
|
||||||
|
array $xactions) {
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function buildHeraldAdapter(
|
||||||
|
PhabricatorLiskDAO $object,
|
||||||
|
array $xactions) {
|
||||||
|
|
||||||
$adapter = id(new HeraldDifferentialDiffAdapter())
|
$adapter = id(new HeraldDifferentialDiffAdapter())
|
||||||
->setDiff($diff);
|
->setDiff($object);
|
||||||
|
|
||||||
$adapter->setContentSource($this->getContentSource());
|
|
||||||
$adapter->setIsNewObject(true);
|
|
||||||
|
|
||||||
$engine = new HeraldEngine();
|
|
||||||
|
|
||||||
$rules = $engine->loadRulesForAdapter($adapter);
|
|
||||||
$rules = mpull($rules, null, 'getID');
|
|
||||||
|
|
||||||
$effects = $engine->applyRules($rules, $adapter);
|
|
||||||
|
|
||||||
$blocking_effect = null;
|
|
||||||
foreach ($effects as $effect) {
|
|
||||||
if ($effect->getAction() == HeraldAdapter::ACTION_BLOCK) {
|
|
||||||
$blocking_effect = $effect;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($blocking_effect) {
|
|
||||||
$rule = idx($rules, $effect->getRuleID());
|
|
||||||
if ($rule && strlen($rule->getName())) {
|
|
||||||
$rule_name = $rule->getName();
|
|
||||||
} else {
|
|
||||||
$rule_name = pht('Unnamed Herald Rule');
|
|
||||||
}
|
|
||||||
|
|
||||||
$message = $effect->getTarget();
|
|
||||||
if (!strlen($message)) {
|
|
||||||
$message = pht('(None.)');
|
|
||||||
}
|
|
||||||
|
|
||||||
throw new DifferentialDiffCreationRejectException(
|
|
||||||
pht(
|
|
||||||
"Creation of this diff was rejected by Herald rule %s.\n".
|
|
||||||
" Rule: %s\n".
|
|
||||||
"Reason: %s",
|
|
||||||
'H'.$effect->getRuleID(),
|
|
||||||
$rule_name,
|
|
||||||
$message));
|
|
||||||
}
|
|
||||||
|
|
||||||
$diff->save();
|
|
||||||
|
|
||||||
// NOTE: We only save the transcript if we didn't block the diff.
|
|
||||||
// Otherwise, we might save some of the diff's content in the transcript
|
|
||||||
// table, which would defeat the purpose of allowing rules to block
|
|
||||||
// storage of key material.
|
|
||||||
|
|
||||||
$engine->applyEffects($effects, $adapter, $rules);
|
|
||||||
$xscript = $engine->getTranscript();
|
|
||||||
|
|
||||||
|
return $adapter;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected function didApplyHeraldRules(
|
||||||
|
PhabricatorLiskDAO $object,
|
||||||
|
HeraldAdapter $adapter,
|
||||||
|
HeraldTranscript $transcript) {
|
||||||
|
|
||||||
|
$xactions = array();
|
||||||
|
return $xactions;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function updateDiffFromDict(DifferentialDiff $diff, $dict) {
|
||||||
|
$diff
|
||||||
|
->setSourcePath(idx($dict, 'sourcePath'))
|
||||||
|
->setSourceMachine(idx($dict, 'sourceMachine'))
|
||||||
|
->setBranch(idx($dict, 'branch'))
|
||||||
|
->setCreationMethod(idx($dict, 'creationMethod'))
|
||||||
|
->setAuthorPHID(idx($dict, 'authorPHID', $this->getActor()))
|
||||||
|
->setBookmark(idx($dict, 'bookmark'))
|
||||||
|
->setRepositoryPHID(idx($dict, 'repositoryPHID'))
|
||||||
|
->setRepositoryUUID(idx($dict, 'repositoryUUID'))
|
||||||
|
->setSourceControlSystem(idx($dict, 'sourceControlSystem'))
|
||||||
|
->setSourceControlPath(idx($dict, 'sourceControlPath'))
|
||||||
|
->setSourceControlBaseRevision(idx($dict, 'sourceControlBaseRevision'))
|
||||||
|
->setLintStatus(idx($dict, 'lintStatus'))
|
||||||
|
->setUnitStatus(idx($dict, 'unitStatus'))
|
||||||
|
->setArcanistProjectPHID(idx($dict, 'arcanistProjectPHID'));
|
||||||
|
|
||||||
|
return $diff;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1523,8 +1523,6 @@ final class DifferentialTransactionEditor
|
||||||
$adapter->setExplicitReviewers($reviewer_phids);
|
$adapter->setExplicitReviewers($reviewer_phids);
|
||||||
$adapter->setForbiddenCCs($unsubscribed_phids);
|
$adapter->setForbiddenCCs($unsubscribed_phids);
|
||||||
|
|
||||||
$adapter->setIsNewObject($this->getIsNewObject());
|
|
||||||
|
|
||||||
return $adapter;
|
return $adapter;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,64 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
final class DifferentialDiffTransaction
|
||||||
|
extends PhabricatorApplicationTransaction {
|
||||||
|
|
||||||
|
const TYPE_DIFF_CREATE = 'differential:diff:create';
|
||||||
|
|
||||||
|
public function getApplicationName() {
|
||||||
|
return 'differential';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getApplicationTransactionType() {
|
||||||
|
return DifferentialDiffPHIDType::TYPECONST;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function shouldHideForMail(array $xactions) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getActionName() {
|
||||||
|
switch ($this->getTransactionType()) {
|
||||||
|
case self::TYPE_DIFF_CREATE;
|
||||||
|
return pht('Created');
|
||||||
|
}
|
||||||
|
|
||||||
|
return parent::getActionName();
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getTitle() {
|
||||||
|
$author_phid = $this->getAuthorPHID();
|
||||||
|
$author_handle = $this->renderHandleLink($author_phid);
|
||||||
|
|
||||||
|
$old = $this->getOldValue();
|
||||||
|
$new = $this->getNewValue();
|
||||||
|
|
||||||
|
switch ($this->getTransactionType()) {
|
||||||
|
case self::TYPE_DIFF_CREATE;
|
||||||
|
return pht(
|
||||||
|
'%s created this diff.',
|
||||||
|
$author_handle);
|
||||||
|
}
|
||||||
|
|
||||||
|
return parent::getTitle();
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getIcon() {
|
||||||
|
switch ($this->getTransactionType()) {
|
||||||
|
case self::TYPE_DIFF_CREATE:
|
||||||
|
return 'fa-refresh';
|
||||||
|
}
|
||||||
|
|
||||||
|
return parent::getIcon();
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getColor() {
|
||||||
|
switch ($this->getTransactionType()) {
|
||||||
|
case self::TYPE_DIFF_CREATE:
|
||||||
|
return PhabricatorTransactions::COLOR_SKY;
|
||||||
|
}
|
||||||
|
|
||||||
|
return parent::getColor();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
Loading…
Reference in a new issue