Implement "words of power" against ApplicationTransactions
Summary:
Ref T2222. Differential has certain "words of power" (like `Ref T123` or `Depends on D345`) which should expand into a separate transaction when they appear anywhere in text.
Currently, they're respected in only some fields. I'm expanding them to work in any remarkup field, including comments and inline comments.
This partially generalizes transaction expansion/extraction in comments. Eventually, I'll probably implement some very soft sort of reference edge for T4036, maybe.
Test Plan: {F119368}
Reviewers: btrahan
Reviewed By: btrahan
CC: aran
Maniphest Tasks: T2222
Differential Revision: https://secure.phabricator.com/D8369
			
			
This commit is contained in:
		| @@ -126,4 +126,9 @@ final class DifferentialRevertPlanField | |||||||
|       $xaction->getNewValue()); |       $xaction->getNewValue()); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   public function getApplicationTransactionRemarkupBlocks( | ||||||
|  |     PhabricatorApplicationTransaction $xaction) { | ||||||
|  |     return array($xaction->getNewValue()); | ||||||
|  |   } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -118,4 +118,9 @@ final class DifferentialSummaryField | |||||||
|       $this->getViewer()); |       $this->getViewer()); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   public function getApplicationTransactionRemarkupBlocks( | ||||||
|  |     PhabricatorApplicationTransaction $xaction) { | ||||||
|  |     return array($xaction->getNewValue()); | ||||||
|  |   } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -133,4 +133,9 @@ final class DifferentialTestPlanField | |||||||
|       $this->getViewer()); |       $this->getViewer()); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   public function getApplicationTransactionRemarkupBlocks( | ||||||
|  |     PhabricatorApplicationTransaction $xaction) { | ||||||
|  |     return array($xaction->getNewValue()); | ||||||
|  |   } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -706,10 +706,11 @@ final class DifferentialTransactionEditor | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   protected function sortTransactions(array $xactions) { |   protected function sortTransactions(array $xactions) { | ||||||
|  |     $xactions = parent::sortTransactions(); | ||||||
|  |  | ||||||
|     $head = array(); |     $head = array(); | ||||||
|     $tail = array(); |     $tail = array(); | ||||||
|  |  | ||||||
|     // Move bare comments to the end, so the actions precede them. |  | ||||||
|     foreach ($xactions as $xaction) { |     foreach ($xactions as $xaction) { | ||||||
|       $type = $xaction->getTransactionType(); |       $type = $xaction->getTransactionType(); | ||||||
|       if ($type == DifferentialTransaction::TYPE_INLINE) { |       if ($type == DifferentialTransaction::TYPE_INLINE) { | ||||||
| @@ -823,6 +824,78 @@ final class DifferentialTransactionEditor | |||||||
|     return parent::extractFilePHIDsFromCustomTransaction($object, $xaction); |     return parent::extractFilePHIDsFromCustomTransaction($object, $xaction); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   protected function expandCustomRemarkupBlockTransactions( | ||||||
|  |     PhabricatorLiskDAO $object, | ||||||
|  |     array $xactions, | ||||||
|  |     $blocks, | ||||||
|  |     PhutilMarkupEngine $engine) { | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     $flat_blocks = array_mergev($blocks); | ||||||
|  |     $huge_block = implode("\n\n", $flat_blocks); | ||||||
|  |  | ||||||
|  |     $task_map = array(); | ||||||
|  |     $task_refs = id(new ManiphestCustomFieldStatusParser()) | ||||||
|  |       ->parseCorpus($huge_block); | ||||||
|  |     foreach ($task_refs as $match) { | ||||||
|  |       foreach ($match['monograms'] as $monogram) { | ||||||
|  |         $task_id = (int)trim($monogram, 'tT'); | ||||||
|  |         $task_map[$task_id] = true; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     $rev_map = array(); | ||||||
|  |     $rev_refs = id(new DifferentialCustomFieldDependsOnParser()) | ||||||
|  |       ->parseCorpus($huge_block); | ||||||
|  |     foreach ($rev_refs as $match) { | ||||||
|  |       foreach ($match['monograms'] as $monogram) { | ||||||
|  |         $rev_id = (int)trim($monogram, 'dD'); | ||||||
|  |         $rev_map[$rev_id] = true; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     $edges = array(); | ||||||
|  |  | ||||||
|  |     if ($task_map) { | ||||||
|  |       $tasks = id(new ManiphestTaskQuery()) | ||||||
|  |         ->setViewer($this->getActor()) | ||||||
|  |         ->withIDs(array_keys($task_map)) | ||||||
|  |         ->execute(); | ||||||
|  |  | ||||||
|  |       if ($tasks) { | ||||||
|  |         $edge_related = PhabricatorEdgeConfig::TYPE_DREV_HAS_RELATED_TASK; | ||||||
|  |         $edges[$edge_related] = mpull($tasks, 'getPHID', 'getPHID'); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if ($rev_map) { | ||||||
|  |       $revs = id(new DifferentialRevisionQuery()) | ||||||
|  |         ->setViewer($this->getActor()) | ||||||
|  |         ->withIDs(array_keys($rev_map)) | ||||||
|  |         ->execute(); | ||||||
|  |       $rev_phids = mpull($revs, 'getPHID', 'getPHID'); | ||||||
|  |  | ||||||
|  |       // NOTE: Skip any write attempts if a user cleverly implies a revision | ||||||
|  |       // depends upon itself. | ||||||
|  |       unset($rev_phids[$object->getPHID()]); | ||||||
|  |  | ||||||
|  |       if ($revs) { | ||||||
|  |         $edge_depends = PhabricatorEdgeConfig::TYPE_DREV_DEPENDS_ON_DREV; | ||||||
|  |         $edges[$edge_depends] = $rev_phids; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     $result = array(); | ||||||
|  |     foreach ($edges as $type => $specs) { | ||||||
|  |       $result[] = id(new DifferentialTransaction()) | ||||||
|  |         ->setTransactionType(PhabricatorTransactions::TYPE_EDGE) | ||||||
|  |         ->setMetadataValue('edge:type', $type) | ||||||
|  |         ->setNewValue(array('+' => $specs)); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return $result; | ||||||
|  |   } | ||||||
|  |  | ||||||
|   private function renderInlineCommentsForMail( |   private function renderInlineCommentsForMail( | ||||||
|     PhabricatorLiskDAO $object, |     PhabricatorLiskDAO $object, | ||||||
|     array $inlines) { |     array $inlines) { | ||||||
|   | |||||||
| @@ -435,16 +435,15 @@ abstract class PhabricatorApplicationTransactionEditor | |||||||
|  |  | ||||||
|     $actor = $this->requireActor(); |     $actor = $this->requireActor(); | ||||||
|  |  | ||||||
|     $this->loadSubscribers($object); |     // NOTE: Some transaction expansion requires that the edited object be | ||||||
|  |     // attached. | ||||||
|     $xactions = $this->applyImplicitCC($object, $xactions); |     foreach ($xactions as $xaction) { | ||||||
|  |       $xaction->attachObject($object); | ||||||
|     $mention_xaction = $this->buildMentionTransaction($object, $xactions); |       $xaction->attachViewer($actor); | ||||||
|     if ($mention_xaction) { |  | ||||||
|       $xactions[] = $mention_xaction; |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     $xactions = $this->expandTransactions($object, $xactions); |     $xactions = $this->expandTransactions($object, $xactions); | ||||||
|  |     $xactions = $this->expandSupportTransactions($object, $xactions); | ||||||
|     $xactions = $this->combineTransactions($xactions); |     $xactions = $this->combineTransactions($xactions); | ||||||
|  |  | ||||||
|     foreach ($xactions as $xaction) { |     foreach ($xactions as $xaction) { | ||||||
| @@ -802,18 +801,14 @@ abstract class PhabricatorApplicationTransactionEditor | |||||||
|  |  | ||||||
|   private function buildMentionTransaction( |   private function buildMentionTransaction( | ||||||
|     PhabricatorLiskDAO $object, |     PhabricatorLiskDAO $object, | ||||||
|     array $xactions) { |     array $xactions, | ||||||
|  |     array $blocks) { | ||||||
|  |  | ||||||
|     if (!($object instanceof PhabricatorSubscribableInterface)) { |     if (!($object instanceof PhabricatorSubscribableInterface)) { | ||||||
|       return null; |       return null; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     $texts = array(); |     $texts = array_mergev($blocks); | ||||||
|     foreach ($xactions as $xaction) { |  | ||||||
|       $texts[] = $this->getRemarkupBlocksFromTransaction($xaction); |  | ||||||
|     } |  | ||||||
|     $texts = array_mergev($texts); |  | ||||||
|  |  | ||||||
|     $phids = PhabricatorMarkupEngine::extractPHIDsFromMentions($texts); |     $phids = PhabricatorMarkupEngine::extractPHIDsFromMentions($texts); | ||||||
|  |  | ||||||
|     $this->mentionedPHIDs = $phids; |     $this->mentionedPHIDs = $phids; | ||||||
| @@ -845,11 +840,7 @@ abstract class PhabricatorApplicationTransactionEditor | |||||||
|  |  | ||||||
|   protected function getRemarkupBlocksFromTransaction( |   protected function getRemarkupBlocksFromTransaction( | ||||||
|     PhabricatorApplicationTransaction $transaction) { |     PhabricatorApplicationTransaction $transaction) { | ||||||
|     $texts = array(); |     return $transaction->getRemarkupBlocks(); | ||||||
|     if ($transaction->getComment()) { |  | ||||||
|       $texts[] = $transaction->getComment()->getContent(); |  | ||||||
|     } |  | ||||||
|     return $texts; |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   protected function mergeTransactions( |   protected function mergeTransactions( | ||||||
| @@ -899,6 +890,64 @@ abstract class PhabricatorApplicationTransactionEditor | |||||||
|     return array($xaction); |     return array($xaction); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |   private function expandSupportTransactions( | ||||||
|  |     PhabricatorLiskDAO $object, | ||||||
|  |     array $xactions) { | ||||||
|  |     $this->loadSubscribers($object); | ||||||
|  |  | ||||||
|  |     $xactions = $this->applyImplicitCC($object, $xactions); | ||||||
|  |  | ||||||
|  |     $blocks = array(); | ||||||
|  |     foreach ($xactions as $key => $xaction) { | ||||||
|  |       $blocks[$key] = $this->getRemarkupBlocksFromTransaction($xaction); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     $mention_xaction = $this->buildMentionTransaction( | ||||||
|  |       $object, | ||||||
|  |       $xactions, | ||||||
|  |       $blocks); | ||||||
|  |     if ($mention_xaction) { | ||||||
|  |       $xactions[] = $mention_xaction; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // TODO: For now, this is just a placeholder. | ||||||
|  |     $engine = PhabricatorMarkupEngine::getEngine('extract'); | ||||||
|  |  | ||||||
|  |     $block_xactions = $this->expandRemarkupBlockTransactions( | ||||||
|  |       $object, | ||||||
|  |       $xactions, | ||||||
|  |       $blocks, | ||||||
|  |       $engine); | ||||||
|  |  | ||||||
|  |     foreach ($block_xactions as $xaction) { | ||||||
|  |       $xactions[] = $xaction; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return $xactions; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   private function expandRemarkupBlockTransactions( | ||||||
|  |     PhabricatorLiskDAO $object, | ||||||
|  |     array $xactions, | ||||||
|  |     $blocks, | ||||||
|  |     PhutilMarkupEngine $engine) { | ||||||
|  |     return $this->expandCustomRemarkupBlockTransactions( | ||||||
|  |       $object, | ||||||
|  |       $xactions, | ||||||
|  |       $blocks, | ||||||
|  |       $engine); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   protected function expandCustomRemarkupBlockTransactions( | ||||||
|  |     PhabricatorLiskDAO $object, | ||||||
|  |     array $xactions, | ||||||
|  |     $blocks, | ||||||
|  |     PhutilMarkupEngine $engine) { | ||||||
|  |     return array(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |  | ||||||
|   /** |   /** | ||||||
|    * Attempt to combine similar transactions into a smaller number of total |    * Attempt to combine similar transactions into a smaller number of total | ||||||
|    * transactions. For example, two transactions which edit the title of an |    * transactions. For example, two transactions which edit the title of an | ||||||
| @@ -1830,7 +1879,6 @@ abstract class PhabricatorApplicationTransactionEditor | |||||||
|     } |     } | ||||||
|     $blocks = array_mergev($blocks); |     $blocks = array_mergev($blocks); | ||||||
|  |  | ||||||
|  |  | ||||||
|     $phids = array(); |     $phids = array(); | ||||||
|     if ($blocks) { |     if ($blocks) { | ||||||
|       $phids[] = PhabricatorMarkupEngine::extractFilePHIDsFromEmbeddedFiles( |       $phids[] = PhabricatorMarkupEngine::extractFilePHIDsFromEmbeddedFiles( | ||||||
|   | |||||||
| @@ -138,6 +138,30 @@ abstract class PhabricatorApplicationTransaction | |||||||
|     return $this->assertAttached($this->object); |     return $this->assertAttached($this->object); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   public function getRemarkupBlocks() { | ||||||
|  |     $blocks = array(); | ||||||
|  |  | ||||||
|  |     switch ($this->getTransactionType()) { | ||||||
|  |       case PhabricatorTransactions::TYPE_CUSTOMFIELD: | ||||||
|  |         $field = $this->getTransactionCustomField(); | ||||||
|  |         if ($field) { | ||||||
|  |           $custom_blocks = $field->getApplicationTransactionRemarkupBlocks( | ||||||
|  |             $this); | ||||||
|  |           foreach ($custom_blocks as $custom_block) { | ||||||
|  |             $blocks[] = $custom_block; | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |         break; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if ($this->getComment()) { | ||||||
|  |       $blocks[] = $this->getComment()->getContent(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return $blocks; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |  | ||||||
| /* -(  Rendering  )---------------------------------------------------------- */ | /* -(  Rendering  )---------------------------------------------------------- */ | ||||||
|  |  | ||||||
|   public function setRenderingTarget($rendering_target) { |   public function setRenderingTarget($rendering_target) { | ||||||
|   | |||||||
| @@ -848,6 +848,18 @@ abstract class PhabricatorCustomField { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * @task appxaction | ||||||
|  |    */ | ||||||
|  |   public function getApplicationTransactionRemarkupBlocks( | ||||||
|  |     PhabricatorApplicationTransaction $xaction) { | ||||||
|  |     if ($this->proxy) { | ||||||
|  |       return $this->proxy->getApplicationTransactionRemarkupBlocks($xaction); | ||||||
|  |     } | ||||||
|  |     return array(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |  | ||||||
|   /** |   /** | ||||||
|    * @task appxaction |    * @task appxaction | ||||||
|    */ |    */ | ||||||
|   | |||||||
| @@ -379,6 +379,12 @@ final class PhabricatorMarkupEngine { | |||||||
|   //    $engine->setConfig('diviner.renderer', new DivinerDefaultRenderer()); |   //    $engine->setConfig('diviner.renderer', new DivinerDefaultRenderer()); | ||||||
|         $engine->setConfig('header.generate-toc', true); |         $engine->setConfig('header.generate-toc', true); | ||||||
|         break; |         break; | ||||||
|  |       case 'extract': | ||||||
|  |         // Engine used for reference/edge extraction. Turn off anything which | ||||||
|  |         // is slow and doesn't change reference extraction. | ||||||
|  |         $engine = self::newMarkupEngine(array()); | ||||||
|  |         $engine->setConfig('pygments.enabled', false); | ||||||
|  |         break; | ||||||
|       default: |       default: | ||||||
|         throw new Exception("Unknown engine ruleset: {$ruleset}!"); |         throw new Exception("Unknown engine ruleset: {$ruleset}!"); | ||||||
|     } |     } | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 epriestley
					epriestley