mirror of
https://we.phorge.it/source/phorge.git
synced 2024-11-19 05:12:41 +01:00
Fix errant rules for associating projects when dragging tasks within a milestone column
Summary: Fixes T10912. When you drag tasks within a milestone, we currently apply an overbroad, API-focused rule and add the parent board's project. This logic was added fairly recently, as part of T6027, to improve the behavior of API-originated moves. Later on, this causes the task to toggle in and out of the parent project on every alternate drag. This logic is also partially duplicated in the `MoveController`. - Add test coverage for this interaction. - Fix the logic so it accounts for subproject / milestone columns correctly. - Put all of the logic into the TransactionEditor, so the API gets the exact same rules. Test Plan: - Added a failing test and made it pass. - Dragged tasks around within a milestone column: - Before patch: they got bogus project swaps on every other move. - After patch: projects didn't change (correct). - Dragged tasks around between normal and milestone columns. - Before patch: worked properly. - After patch: still works properly. Here's what the bad changes look like, the task is swapping projects with every other move: {F1255957} The "every other" is because the logic was trying to do this: - Add both the parent and milestone project. - Whichever one exists already gets dropped from the change list because it would have no effect. - The other one then applies. - In applying, it forces removal of the first one. - Then this process repeats in the other direction the next time through. Reviewers: chad Reviewed By: chad Maniphest Tasks: T10912 Differential Revision: https://secure.phabricator.com/D15834
This commit is contained in:
parent
c0d42a8943
commit
cac26c8824
3 changed files with 119 additions and 43 deletions
|
@ -868,23 +868,9 @@ final class ManiphestTransactionEditor
|
||||||
switch ($type) {
|
switch ($type) {
|
||||||
case PhabricatorTransactions::TYPE_COLUMNS:
|
case PhabricatorTransactions::TYPE_COLUMNS:
|
||||||
try {
|
try {
|
||||||
$this->buildMoveTransaction($object, $xaction);
|
$more_xactions = $this->buildMoveTransaction($object, $xaction);
|
||||||
|
foreach ($more_xactions as $more_xaction) {
|
||||||
// Implicilty add the task to any boards that we're moving it
|
$results[] = $more_xaction;
|
||||||
// on, since moves on a board the task isn't part of are not
|
|
||||||
// meaningful.
|
|
||||||
$board_phids = ipull($xaction->getNewValue(), 'boardPHID');
|
|
||||||
if ($board_phids) {
|
|
||||||
$results[] = id(new ManiphestTransaction())
|
|
||||||
->setTransactionType(PhabricatorTransactions::TYPE_EDGE)
|
|
||||||
->setMetadataValue(
|
|
||||||
'edge:type',
|
|
||||||
PhabricatorProjectObjectHasProjectEdgeType::EDGECONST)
|
|
||||||
->setIgnoreOnNoEffect(true)
|
|
||||||
->setNewValue(
|
|
||||||
array(
|
|
||||||
'+' => array_fuse($board_phids),
|
|
||||||
));
|
|
||||||
}
|
}
|
||||||
} catch (Exception $ex) {
|
} catch (Exception $ex) {
|
||||||
$error = new PhabricatorApplicationTransactionValidationError(
|
$error = new PhabricatorApplicationTransactionValidationError(
|
||||||
|
@ -1098,6 +1084,89 @@ final class ManiphestTransactionEditor
|
||||||
|
|
||||||
$new = array_values($new);
|
$new = array_values($new);
|
||||||
$xaction->setNewValue($new);
|
$xaction->setNewValue($new);
|
||||||
|
|
||||||
|
|
||||||
|
$more = array();
|
||||||
|
|
||||||
|
// If we're moving the object into a column and it does not already belong
|
||||||
|
// in the column, add the appropriate board. For normal columns, this
|
||||||
|
// is the board PHID. For proxy columns, it is the proxy PHID, unless the
|
||||||
|
// object is already a member of some descendant of the proxy PHID.
|
||||||
|
|
||||||
|
// The major case where this can happen is moves via the API, but it also
|
||||||
|
// happens when a user drags a task from the "Backlog" to a milestone
|
||||||
|
// column.
|
||||||
|
|
||||||
|
if ($object_phid) {
|
||||||
|
$current_phids = PhabricatorEdgeQuery::loadDestinationPHIDs(
|
||||||
|
$object_phid,
|
||||||
|
PhabricatorProjectObjectHasProjectEdgeType::EDGECONST);
|
||||||
|
$current_phids = array_fuse($current_phids);
|
||||||
|
} else {
|
||||||
|
$current_phids = array();
|
||||||
|
}
|
||||||
|
|
||||||
|
$add_boards = array();
|
||||||
|
foreach ($new as $move) {
|
||||||
|
$column_phid = $move['columnPHID'];
|
||||||
|
$board_phid = $move['boardPHID'];
|
||||||
|
$column = $columns[$column_phid];
|
||||||
|
$proxy_phid = $column->getProxyPHID();
|
||||||
|
|
||||||
|
// If this is a normal column, add the board if the object isn't already
|
||||||
|
// associated.
|
||||||
|
if (!$proxy_phid) {
|
||||||
|
if (!isset($current_phids[$board_phid])) {
|
||||||
|
$add_boards[] = $board_phid;
|
||||||
|
}
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// If this is a proxy column but the object is already associated with
|
||||||
|
// the proxy board, we don't need to do anything.
|
||||||
|
if (isset($current_phids[$proxy_phid])) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// If this a proxy column and the object is already associated with some
|
||||||
|
// descendant of the proxy board, we also don't need to do anything.
|
||||||
|
$descendants = id(new PhabricatorProjectQuery())
|
||||||
|
->setViewer(PhabricatorUser::getOmnipotentUser())
|
||||||
|
->withAncestorProjectPHIDs(array($proxy_phid))
|
||||||
|
->execute();
|
||||||
|
|
||||||
|
$found_descendant = false;
|
||||||
|
foreach ($descendants as $descendant) {
|
||||||
|
if (isset($current_phids[$descendant->getPHID()])) {
|
||||||
|
$found_descendant = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($found_descendant) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Otherwise, we're moving the object to a proxy column which it is not
|
||||||
|
// a member of yet, so add an association to the column's proxy board.
|
||||||
|
|
||||||
|
$add_boards[] = $proxy_phid;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($add_boards) {
|
||||||
|
$more[] = id(new ManiphestTransaction())
|
||||||
|
->setTransactionType(PhabricatorTransactions::TYPE_EDGE)
|
||||||
|
->setMetadataValue(
|
||||||
|
'edge:type',
|
||||||
|
PhabricatorProjectObjectHasProjectEdgeType::EDGECONST)
|
||||||
|
->setIgnoreOnNoEffect(true)
|
||||||
|
->setNewValue(
|
||||||
|
array(
|
||||||
|
'+' => array_fuse($add_boards),
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
return $more;
|
||||||
}
|
}
|
||||||
|
|
||||||
private function applyBoardMove($object, array $move) {
|
private function applyBoardMove($object, array $move) {
|
||||||
|
|
|
@ -1011,6 +1011,39 @@ final class PhabricatorProjectCoreTestCase extends PhabricatorTestCase {
|
||||||
$column->getPHID(),
|
$column->getPHID(),
|
||||||
);
|
);
|
||||||
$this->assertColumns($expect, $user, $board, $task);
|
$this->assertColumns($expect, $user, $board, $task);
|
||||||
|
|
||||||
|
|
||||||
|
// Move the task within the "Milestone" column. This should not affect
|
||||||
|
// the projects the task is tagged with. See T10912.
|
||||||
|
$task_a = $task;
|
||||||
|
|
||||||
|
$task_b = $this->newTask($user, array($backlog));
|
||||||
|
$this->moveToColumn($user, $board, $task_b, $backlog, $column);
|
||||||
|
|
||||||
|
$a_options = array(
|
||||||
|
'beforePHID' => $task_b->getPHID(),
|
||||||
|
);
|
||||||
|
|
||||||
|
$b_options = array(
|
||||||
|
'beforePHID' => $task_a->getPHID(),
|
||||||
|
);
|
||||||
|
|
||||||
|
$old_projects = $this->getTaskProjects($task);
|
||||||
|
|
||||||
|
// Move the target task to the top.
|
||||||
|
$this->moveToColumn($user, $board, $task_a, $column, $column, $a_options);
|
||||||
|
$new_projects = $this->getTaskProjects($task_a);
|
||||||
|
$this->assertEqual($old_projects, $new_projects);
|
||||||
|
|
||||||
|
// Move the other task.
|
||||||
|
$this->moveToColumn($user, $board, $task_b, $column, $column, $b_options);
|
||||||
|
$new_projects = $this->getTaskProjects($task_a);
|
||||||
|
$this->assertEqual($old_projects, $new_projects);
|
||||||
|
|
||||||
|
// Move the target task again.
|
||||||
|
$this->moveToColumn($user, $board, $task_a, $column, $column, $a_options);
|
||||||
|
$new_projects = $this->getTaskProjects($task_a);
|
||||||
|
$this->assertEqual($old_projects, $new_projects);
|
||||||
}
|
}
|
||||||
|
|
||||||
public function testColumnExtendedPolicies() {
|
public function testColumnExtendedPolicies() {
|
||||||
|
|
|
@ -96,32 +96,6 @@ final class PhabricatorProjectMoveController
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
$proxy = $column->getProxy();
|
|
||||||
if ($proxy) {
|
|
||||||
// We're moving the task into a subproject or milestone column, so add
|
|
||||||
// the subproject or milestone.
|
|
||||||
$add_projects = array($proxy->getPHID());
|
|
||||||
} else if ($project->getHasSubprojects() || $project->getHasMilestones()) {
|
|
||||||
// We're moving the task into the "Backlog" column on the parent project,
|
|
||||||
// so add the parent explicitly. This gets rid of any subproject or
|
|
||||||
// milestone tags.
|
|
||||||
$add_projects = array($project->getPHID());
|
|
||||||
} else {
|
|
||||||
$add_projects = array();
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($add_projects) {
|
|
||||||
$project_type = PhabricatorProjectObjectHasProjectEdgeType::EDGECONST;
|
|
||||||
|
|
||||||
$xactions[] = id(new ManiphestTransaction())
|
|
||||||
->setTransactionType(PhabricatorTransactions::TYPE_EDGE)
|
|
||||||
->setMetadataValue('edge:type', $project_type)
|
|
||||||
->setNewValue(
|
|
||||||
array(
|
|
||||||
'+' => array_fuse($add_projects),
|
|
||||||
));
|
|
||||||
}
|
|
||||||
|
|
||||||
$editor = id(new ManiphestTransactionEditor())
|
$editor = id(new ManiphestTransactionEditor())
|
||||||
->setActor($viewer)
|
->setActor($viewer)
|
||||||
->setContinueOnMissingFields(true)
|
->setContinueOnMissingFields(true)
|
||||||
|
|
Loading…
Reference in a new issue