mirror of
https://we.phorge.it/source/phorge.git
synced 2025-01-05 12:21:02 +01:00
Don't re-mention users for comment edits
Summary: Ref T11035. This only fixes half of the issue: comment editing has been fixed, but normal transactions which edit things like descriptions haven't yet. The normal edits aren't fixed because the "oldValues" are populated too late. The code should start working once they get populated sooner, but I don't want to jump the gun on that since it'll probably have some spooky effects. I have some other transaction changes coming down the pipe which should provide a better context for testing "oldValue" population order. Test Plan: - Mentioned `@dog` in a comment. - Removed `@dog` as a subscriber. - Edited the comment, adding some unrelated text at the end (e.g., fixing a typo). - Before change: `@dog` re-added as subscriber. - After change: no re-add. Reviewers: chad Reviewed By: chad Maniphest Tasks: T11035 Differential Revision: https://secure.phabricator.com/D16108
This commit is contained in:
parent
74682d46ae
commit
65634781b4
10 changed files with 167 additions and 42 deletions
|
@ -3591,6 +3591,8 @@ phutil_register_library_map(array(
|
||||||
'PhabricatorTokensCurtainExtension' => 'applications/tokens/engineextension/PhabricatorTokensCurtainExtension.php',
|
'PhabricatorTokensCurtainExtension' => 'applications/tokens/engineextension/PhabricatorTokensCurtainExtension.php',
|
||||||
'PhabricatorTokensSettingsPanel' => 'applications/settings/panel/PhabricatorTokensSettingsPanel.php',
|
'PhabricatorTokensSettingsPanel' => 'applications/settings/panel/PhabricatorTokensSettingsPanel.php',
|
||||||
'PhabricatorTooltipUIExample' => 'applications/uiexample/examples/PhabricatorTooltipUIExample.php',
|
'PhabricatorTooltipUIExample' => 'applications/uiexample/examples/PhabricatorTooltipUIExample.php',
|
||||||
|
'PhabricatorTransactionChange' => 'applications/transactions/data/PhabricatorTransactionChange.php',
|
||||||
|
'PhabricatorTransactionRemarkupChange' => 'applications/transactions/data/PhabricatorTransactionRemarkupChange.php',
|
||||||
'PhabricatorTransactions' => 'applications/transactions/constants/PhabricatorTransactions.php',
|
'PhabricatorTransactions' => 'applications/transactions/constants/PhabricatorTransactions.php',
|
||||||
'PhabricatorTransactionsApplication' => 'applications/transactions/application/PhabricatorTransactionsApplication.php',
|
'PhabricatorTransactionsApplication' => 'applications/transactions/application/PhabricatorTransactionsApplication.php',
|
||||||
'PhabricatorTransactionsDestructionEngineExtension' => 'applications/transactions/engineextension/PhabricatorTransactionsDestructionEngineExtension.php',
|
'PhabricatorTransactionsDestructionEngineExtension' => 'applications/transactions/engineextension/PhabricatorTransactionsDestructionEngineExtension.php',
|
||||||
|
@ -8397,6 +8399,8 @@ phutil_register_library_map(array(
|
||||||
'PhabricatorTokensCurtainExtension' => 'PHUICurtainExtension',
|
'PhabricatorTokensCurtainExtension' => 'PHUICurtainExtension',
|
||||||
'PhabricatorTokensSettingsPanel' => 'PhabricatorSettingsPanel',
|
'PhabricatorTokensSettingsPanel' => 'PhabricatorSettingsPanel',
|
||||||
'PhabricatorTooltipUIExample' => 'PhabricatorUIExample',
|
'PhabricatorTooltipUIExample' => 'PhabricatorUIExample',
|
||||||
|
'PhabricatorTransactionChange' => 'Phobject',
|
||||||
|
'PhabricatorTransactionRemarkupChange' => 'PhabricatorTransactionChange',
|
||||||
'PhabricatorTransactions' => 'Phobject',
|
'PhabricatorTransactions' => 'Phobject',
|
||||||
'PhabricatorTransactionsApplication' => 'PhabricatorApplication',
|
'PhabricatorTransactionsApplication' => 'PhabricatorApplication',
|
||||||
'PhabricatorTransactionsDestructionEngineExtension' => 'PhabricatorDestructionEngineExtension',
|
'PhabricatorTransactionsDestructionEngineExtension' => 'PhabricatorDestructionEngineExtension',
|
||||||
|
|
|
@ -542,7 +542,7 @@ final class PhabricatorAuditEditor
|
||||||
protected function expandCustomRemarkupBlockTransactions(
|
protected function expandCustomRemarkupBlockTransactions(
|
||||||
PhabricatorLiskDAO $object,
|
PhabricatorLiskDAO $object,
|
||||||
array $xactions,
|
array $xactions,
|
||||||
$blocks,
|
array $changes,
|
||||||
PhutilMarkupEngine $engine) {
|
PhutilMarkupEngine $engine) {
|
||||||
|
|
||||||
// we are only really trying to find unmentionable phids here...
|
// we are only really trying to find unmentionable phids here...
|
||||||
|
@ -563,7 +563,7 @@ final class PhabricatorAuditEditor
|
||||||
return $result;
|
return $result;
|
||||||
}
|
}
|
||||||
|
|
||||||
$flat_blocks = array_mergev($blocks);
|
$flat_blocks = mpull($changes, 'getNewValue');
|
||||||
$huge_block = implode("\n\n", $flat_blocks);
|
$huge_block = implode("\n\n", $flat_blocks);
|
||||||
$phid_map = array();
|
$phid_map = array();
|
||||||
$phid_map[] = $this->getUnmentionablePHIDMap();
|
$phid_map[] = $this->getUnmentionablePHIDMap();
|
||||||
|
|
|
@ -1303,10 +1303,10 @@ final class DifferentialTransactionEditor
|
||||||
protected function expandCustomRemarkupBlockTransactions(
|
protected function expandCustomRemarkupBlockTransactions(
|
||||||
PhabricatorLiskDAO $object,
|
PhabricatorLiskDAO $object,
|
||||||
array $xactions,
|
array $xactions,
|
||||||
$blocks,
|
array $changes,
|
||||||
PhutilMarkupEngine $engine) {
|
PhutilMarkupEngine $engine) {
|
||||||
|
|
||||||
$flat_blocks = array_mergev($blocks);
|
$flat_blocks = mpull($changes, 'getNewValue');
|
||||||
$huge_block = implode("\n\n", $flat_blocks);
|
$huge_block = implode("\n\n", $flat_blocks);
|
||||||
|
|
||||||
$task_map = array();
|
$task_map = array();
|
||||||
|
|
|
@ -54,16 +54,18 @@ final class ManiphestTransaction
|
||||||
return parent::shouldGenerateOldValue();
|
return parent::shouldGenerateOldValue();
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getRemarkupBlocks() {
|
protected function newRemarkupChanges() {
|
||||||
$blocks = parent::getRemarkupBlocks();
|
$changes = array();
|
||||||
|
|
||||||
switch ($this->getTransactionType()) {
|
switch ($this->getTransactionType()) {
|
||||||
case self::TYPE_DESCRIPTION:
|
case self::TYPE_DESCRIPTION:
|
||||||
$blocks[] = $this->getNewValue();
|
$changes[] = $this->newRemarkupChange()
|
||||||
|
->setOldValue($this->getOldValue())
|
||||||
|
->setNewValue($this->getNewValue());
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
return $blocks;
|
return $changes;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getRequiredHandlePHIDs() {
|
public function getRequiredHandlePHIDs() {
|
||||||
|
|
|
@ -0,0 +1,37 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
abstract class PhabricatorTransactionChange extends Phobject {
|
||||||
|
|
||||||
|
private $transaction;
|
||||||
|
private $oldValue;
|
||||||
|
private $newValue;
|
||||||
|
|
||||||
|
final public function setTransaction(
|
||||||
|
PhabricatorApplicationTransaction $xaction) {
|
||||||
|
$this->transaction = $xaction;
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
final public function getTransaction() {
|
||||||
|
return $this->transaction;
|
||||||
|
}
|
||||||
|
|
||||||
|
final public function setOldValue($old_value) {
|
||||||
|
$this->oldValue = $old_value;
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
final public function getOldValue() {
|
||||||
|
return $this->oldValue;
|
||||||
|
}
|
||||||
|
|
||||||
|
final public function setNewValue($new_value) {
|
||||||
|
$this->newValue = $new_value;
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
final public function getNewValue() {
|
||||||
|
return $this->newValue;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,4 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
final class PhabricatorTransactionRemarkupChange
|
||||||
|
extends PhabricatorTransactionChange {}
|
|
@ -64,6 +64,9 @@ final class PhabricatorApplicationTransactionCommentEditor
|
||||||
$comment->setTransactionPHID($xaction->getPHID());
|
$comment->setTransactionPHID($xaction->getPHID());
|
||||||
$comment->save();
|
$comment->save();
|
||||||
|
|
||||||
|
$old_comment = $xaction->getComment();
|
||||||
|
$comment->attachOldComment($old_comment);
|
||||||
|
|
||||||
$xaction->setCommentVersion($new_version);
|
$xaction->setCommentVersion($new_version);
|
||||||
$xaction->setCommentPHID($comment->getPHID());
|
$xaction->setCommentPHID($comment->getPHID());
|
||||||
$xaction->setViewPolicy($comment->getViewPolicy());
|
$xaction->setViewPolicy($comment->getViewPolicy());
|
||||||
|
|
|
@ -1320,17 +1320,28 @@ abstract class PhabricatorApplicationTransactionEditor
|
||||||
private function buildSubscribeTransaction(
|
private function buildSubscribeTransaction(
|
||||||
PhabricatorLiskDAO $object,
|
PhabricatorLiskDAO $object,
|
||||||
array $xactions,
|
array $xactions,
|
||||||
array $blocks) {
|
array $changes) {
|
||||||
|
|
||||||
if (!($object instanceof PhabricatorSubscribableInterface)) {
|
if (!($object instanceof PhabricatorSubscribableInterface)) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($this->shouldEnableMentions($object, $xactions)) {
|
if ($this->shouldEnableMentions($object, $xactions)) {
|
||||||
$texts = array_mergev($blocks);
|
// Identify newly mentioned users. We ignore users who were previously
|
||||||
$phids = PhabricatorMarkupEngine::extractPHIDsFromMentions(
|
// mentioned so that we don't re-subscribe users after an edit of text
|
||||||
|
// which mentions them.
|
||||||
|
$old_texts = mpull($changes, 'getOldValue');
|
||||||
|
$new_texts = mpull($changes, 'getNewValue');
|
||||||
|
|
||||||
|
$old_phids = PhabricatorMarkupEngine::extractPHIDsFromMentions(
|
||||||
$this->getActor(),
|
$this->getActor(),
|
||||||
$texts);
|
$old_texts);
|
||||||
|
|
||||||
|
$new_phids = PhabricatorMarkupEngine::extractPHIDsFromMentions(
|
||||||
|
$this->getActor(),
|
||||||
|
$new_texts);
|
||||||
|
|
||||||
|
$phids = array_diff($new_phids, $old_phids);
|
||||||
} else {
|
} else {
|
||||||
$phids = array();
|
$phids = array();
|
||||||
}
|
}
|
||||||
|
@ -1381,11 +1392,6 @@ abstract class PhabricatorApplicationTransactionEditor
|
||||||
return $xaction;
|
return $xaction;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected function getRemarkupBlocksFromTransaction(
|
|
||||||
PhabricatorApplicationTransaction $transaction) {
|
|
||||||
return $transaction->getRemarkupBlocks();
|
|
||||||
}
|
|
||||||
|
|
||||||
protected function mergeTransactions(
|
protected function mergeTransactions(
|
||||||
PhabricatorApplicationTransaction $u,
|
PhabricatorApplicationTransaction $u,
|
||||||
PhabricatorApplicationTransaction $v) {
|
PhabricatorApplicationTransaction $v) {
|
||||||
|
@ -1464,15 +1470,12 @@ abstract class PhabricatorApplicationTransactionEditor
|
||||||
|
|
||||||
$xactions = $this->applyImplicitCC($object, $xactions);
|
$xactions = $this->applyImplicitCC($object, $xactions);
|
||||||
|
|
||||||
$blocks = array();
|
$changes = $this->getRemarkupChanges($xactions);
|
||||||
foreach ($xactions as $key => $xaction) {
|
|
||||||
$blocks[$key] = $this->getRemarkupBlocksFromTransaction($xaction);
|
|
||||||
}
|
|
||||||
|
|
||||||
$subscribe_xaction = $this->buildSubscribeTransaction(
|
$subscribe_xaction = $this->buildSubscribeTransaction(
|
||||||
$object,
|
$object,
|
||||||
$xactions,
|
$xactions,
|
||||||
$blocks);
|
$changes);
|
||||||
if ($subscribe_xaction) {
|
if ($subscribe_xaction) {
|
||||||
$xactions[] = $subscribe_xaction;
|
$xactions[] = $subscribe_xaction;
|
||||||
}
|
}
|
||||||
|
@ -1484,7 +1487,7 @@ abstract class PhabricatorApplicationTransactionEditor
|
||||||
$block_xactions = $this->expandRemarkupBlockTransactions(
|
$block_xactions = $this->expandRemarkupBlockTransactions(
|
||||||
$object,
|
$object,
|
||||||
$xactions,
|
$xactions,
|
||||||
$blocks,
|
$changes,
|
||||||
$engine);
|
$engine);
|
||||||
|
|
||||||
foreach ($block_xactions as $xaction) {
|
foreach ($block_xactions as $xaction) {
|
||||||
|
@ -1494,29 +1497,48 @@ abstract class PhabricatorApplicationTransactionEditor
|
||||||
return $xactions;
|
return $xactions;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function getRemarkupChanges(array $xactions) {
|
||||||
|
$changes = array();
|
||||||
|
|
||||||
|
foreach ($xactions as $key => $xaction) {
|
||||||
|
foreach ($this->getRemarkupChangesFromTransaction($xaction) as $change) {
|
||||||
|
$changes[] = $change;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $changes;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function getRemarkupChangesFromTransaction(
|
||||||
|
PhabricatorApplicationTransaction $transaction) {
|
||||||
|
return $transaction->getRemarkupChanges();
|
||||||
|
}
|
||||||
|
|
||||||
private function expandRemarkupBlockTransactions(
|
private function expandRemarkupBlockTransactions(
|
||||||
PhabricatorLiskDAO $object,
|
PhabricatorLiskDAO $object,
|
||||||
array $xactions,
|
array $xactions,
|
||||||
$blocks,
|
array $changes,
|
||||||
PhutilMarkupEngine $engine) {
|
PhutilMarkupEngine $engine) {
|
||||||
|
|
||||||
$block_xactions = $this->expandCustomRemarkupBlockTransactions(
|
$block_xactions = $this->expandCustomRemarkupBlockTransactions(
|
||||||
$object,
|
$object,
|
||||||
$xactions,
|
$xactions,
|
||||||
$blocks,
|
$changes,
|
||||||
$engine);
|
$engine);
|
||||||
|
|
||||||
$mentioned_phids = array();
|
$mentioned_phids = array();
|
||||||
if ($this->shouldEnableMentions($object, $xactions)) {
|
if ($this->shouldEnableMentions($object, $xactions)) {
|
||||||
foreach ($blocks as $key => $xaction_blocks) {
|
foreach ($changes as $change) {
|
||||||
foreach ($xaction_blocks as $block) {
|
// Here, we don't care about processing only new mentions after an edit
|
||||||
$engine->markupText($block);
|
// because there is no way for an object to ever "unmention" itself on
|
||||||
|
// another object, so we can ignore the old value.
|
||||||
|
$engine->markupText($change->getNewValue());
|
||||||
|
|
||||||
$mentioned_phids += $engine->getTextMetadata(
|
$mentioned_phids += $engine->getTextMetadata(
|
||||||
PhabricatorObjectRemarkupRule::KEY_MENTIONED_OBJECTS,
|
PhabricatorObjectRemarkupRule::KEY_MENTIONED_OBJECTS,
|
||||||
array());
|
array());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
if (!$mentioned_phids) {
|
if (!$mentioned_phids) {
|
||||||
return $block_xactions;
|
return $block_xactions;
|
||||||
|
@ -1559,7 +1581,7 @@ abstract class PhabricatorApplicationTransactionEditor
|
||||||
protected function expandCustomRemarkupBlockTransactions(
|
protected function expandCustomRemarkupBlockTransactions(
|
||||||
PhabricatorLiskDAO $object,
|
PhabricatorLiskDAO $object,
|
||||||
array $xactions,
|
array $xactions,
|
||||||
$blocks,
|
array $changes,
|
||||||
PhutilMarkupEngine $engine) {
|
PhutilMarkupEngine $engine) {
|
||||||
return array();
|
return array();
|
||||||
}
|
}
|
||||||
|
@ -3096,11 +3118,8 @@ abstract class PhabricatorApplicationTransactionEditor
|
||||||
PhabricatorLiskDAO $object,
|
PhabricatorLiskDAO $object,
|
||||||
array $xactions) {
|
array $xactions) {
|
||||||
|
|
||||||
$blocks = array();
|
$changes = $this->getRemarkupChanges($xactions);
|
||||||
foreach ($xactions as $xaction) {
|
$blocks = mpull($changes, 'getNewValue');
|
||||||
$blocks[] = $this->getRemarkupBlocksFromTransaction($xaction);
|
|
||||||
}
|
|
||||||
$blocks = array_mergev($blocks);
|
|
||||||
|
|
||||||
$phids = array();
|
$phids = array();
|
||||||
if ($blocks) {
|
if ($blocks) {
|
||||||
|
|
|
@ -179,6 +179,50 @@ abstract class PhabricatorApplicationTransaction
|
||||||
return $this->assertAttached($this->object);
|
return $this->assertAttached($this->object);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function getRemarkupChanges() {
|
||||||
|
$changes = $this->newRemarkupChanges();
|
||||||
|
assert_instances_of($changes, 'PhabricatorTransactionRemarkupChange');
|
||||||
|
|
||||||
|
// Convert older-style remarkup blocks into newer-style remarkup changes.
|
||||||
|
// This builds changes that do not have the correct "old value", so rules
|
||||||
|
// that operate differently against edits (like @user mentions) won't work
|
||||||
|
// properly.
|
||||||
|
foreach ($this->getRemarkupBlocks() as $block) {
|
||||||
|
$changes[] = $this->newRemarkupChange()
|
||||||
|
->setOldValue(null)
|
||||||
|
->setNewValue($block);
|
||||||
|
}
|
||||||
|
|
||||||
|
$comment = $this->getComment();
|
||||||
|
if ($comment) {
|
||||||
|
if ($comment->hasOldComment()) {
|
||||||
|
$old_value = $comment->getOldComment()->getContent();
|
||||||
|
} else {
|
||||||
|
$old_value = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$new_value = $comment->getContent();
|
||||||
|
|
||||||
|
$changes[] = $this->newRemarkupChange()
|
||||||
|
->setOldValue($old_value)
|
||||||
|
->setNewValue($new_value);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $changes;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function newRemarkupChanges() {
|
||||||
|
return array();
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function newRemarkupChange() {
|
||||||
|
return id(new PhabricatorTransactionRemarkupChange())
|
||||||
|
->setTransaction($this);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @deprecated
|
||||||
|
*/
|
||||||
public function getRemarkupBlocks() {
|
public function getRemarkupBlocks() {
|
||||||
$blocks = array();
|
$blocks = array();
|
||||||
|
|
||||||
|
@ -195,10 +239,6 @@ abstract class PhabricatorApplicationTransaction
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($this->getComment()) {
|
|
||||||
$blocks[] = $this->getComment()->getContent();
|
|
||||||
}
|
|
||||||
|
|
||||||
return $blocks;
|
return $blocks;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -18,6 +18,8 @@ abstract class PhabricatorApplicationTransactionComment
|
||||||
protected $contentSource;
|
protected $contentSource;
|
||||||
protected $isDeleted = 0;
|
protected $isDeleted = 0;
|
||||||
|
|
||||||
|
private $oldComment = self::ATTACHABLE;
|
||||||
|
|
||||||
abstract public function getApplicationTransactionObject();
|
abstract public function getApplicationTransactionObject();
|
||||||
|
|
||||||
public function generatePHID() {
|
public function generatePHID() {
|
||||||
|
@ -85,6 +87,20 @@ abstract class PhabricatorApplicationTransactionComment
|
||||||
return $this;
|
return $this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function attachOldComment(
|
||||||
|
PhabricatorApplicationTransactionComment $old_comment) {
|
||||||
|
$this->oldComment = $old_comment;
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getOldComment() {
|
||||||
|
return $this->assertAttached($this->oldComment);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function hasOldComment() {
|
||||||
|
return ($this->oldComment !== self::ATTACHABLE);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/* -( PhabricatorMarkupInterface )----------------------------------------- */
|
/* -( PhabricatorMarkupInterface )----------------------------------------- */
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue