Implement "replace" transactions in Pholio without "applyInitialEffects"
Summary: Depends on D19923. Ref T11351. Currently, this transaction takes an `Image` as the `newValue` and uses some magic to reduce it into PHIDs by the time we're done. This creates some problems today where I'd like to get rid of `applyInitialEffects` for MFA code. In the future, it creates a problem becuase there's no way to pass an entire `Image` object over the API. Instead, create the `Image` first, then just provide the PHID. This is generally simpler, will work well with the API in the future, and stops us from needing any `applyInitialEffects` stuff. Test Plan: Replaced images in a Pholio mock. Reviewers: amckinley Reviewed By: amckinley Maniphest Tasks: T11351 Differential Revision: https://secure.phabricator.com/D19924
This commit is contained in:
		| @@ -143,20 +143,22 @@ final class PholioMockEditController extends PholioController { | |||||||
|           $replace_image = PholioImage::initializeNewImage() |           $replace_image = PholioImage::initializeNewImage() | ||||||
|             ->setAuthorPHID($viewer->getPHID()) |             ->setAuthorPHID($viewer->getPHID()) | ||||||
|             ->setReplacesImagePHID($replaces_image_phid) |             ->setReplacesImagePHID($replaces_image_phid) | ||||||
|             ->setFilePhid($file_phid) |             ->setFilePHID($file_phid) | ||||||
|             ->attachFile($file) |             ->attachFile($file) | ||||||
|             ->setName(strlen($title) ? $title : $file->getName()) |             ->setName(strlen($title) ? $title : $file->getName()) | ||||||
|             ->setDescription($description) |             ->setDescription($description) | ||||||
|             ->setSequence($sequence); |             ->setSequence($sequence) | ||||||
|  |             ->save(); | ||||||
|  |  | ||||||
|           $xactions[] = id(new PholioTransaction()) |           $xactions[] = id(new PholioTransaction()) | ||||||
|             ->setTransactionType( |             ->setTransactionType(PholioImageReplaceTransaction::TRANSACTIONTYPE) | ||||||
|               PholioImageReplaceTransaction::TRANSACTIONTYPE) |             ->setNewValue($replace_image->getPHID()); | ||||||
|             ->setNewValue($replace_image); |  | ||||||
|           $posted_mock_images[] = $replace_image; |           $posted_mock_images[] = $replace_image; | ||||||
|         } else if (!$existing_image) { // this is an add |         } else if (!$existing_image) { // this is an add | ||||||
|           $add_image = PholioImage::initializeNewImage() |           $add_image = PholioImage::initializeNewImage() | ||||||
|             ->setAuthorPHID($viewer->getPHID()) |             ->setAuthorPHID($viewer->getPHID()) | ||||||
|             ->setFilePhid($file_phid) |             ->setFilePHID($file_phid) | ||||||
|             ->attachFile($file) |             ->attachFile($file) | ||||||
|             ->setName(strlen($title) ? $title : $file->getName()) |             ->setName(strlen($title) ? $title : $file->getName()) | ||||||
|             ->setDescription($description) |             ->setDescription($description) | ||||||
| @@ -202,7 +204,6 @@ final class PholioMockEditController extends PholioController { | |||||||
|           ->setMetadataValue('edge:type', $proj_edge_type) |           ->setMetadataValue('edge:type', $proj_edge_type) | ||||||
|           ->setNewValue(array('=' => array_fuse($v_projects))); |           ->setNewValue(array('=' => array_fuse($v_projects))); | ||||||
|  |  | ||||||
|         $mock->openTransaction(); |  | ||||||
|         $editor = id(new PholioMockEditor()) |         $editor = id(new PholioMockEditor()) | ||||||
|           ->setContentSourceFromRequest($request) |           ->setContentSourceFromRequest($request) | ||||||
|           ->setContinueOnNoEffect(true) |           ->setContinueOnNoEffect(true) | ||||||
| @@ -210,8 +211,6 @@ final class PholioMockEditController extends PholioController { | |||||||
|  |  | ||||||
|         $xactions = $editor->applyTransactions($mock, $xactions); |         $xactions = $editor->applyTransactions($mock, $xactions); | ||||||
|  |  | ||||||
|         $mock->saveTransaction(); |  | ||||||
|  |  | ||||||
|         return id(new AphrontRedirectResponse()) |         return id(new AphrontRedirectResponse()) | ||||||
|           ->setURI('/M'.$mock->getID()); |           ->setURI('/M'.$mock->getID()); | ||||||
|       } |       } | ||||||
|   | |||||||
| @@ -4,6 +4,8 @@ final class PholioMockEditor extends PhabricatorApplicationTransactionEditor { | |||||||
|  |  | ||||||
|   private $newImages = array(); |   private $newImages = array(); | ||||||
|  |  | ||||||
|  |   private $images = array(); | ||||||
|  |  | ||||||
|   public function getEditorApplicationClass() { |   public function getEditorApplicationClass() { | ||||||
|     return 'PhabricatorPholioApplication'; |     return 'PhabricatorPholioApplication'; | ||||||
|   } |   } | ||||||
| @@ -48,9 +50,7 @@ final class PholioMockEditor extends PhabricatorApplicationTransactionEditor { | |||||||
|     foreach ($xactions as $xaction) { |     foreach ($xactions as $xaction) { | ||||||
|       switch ($xaction->getTransactionType()) { |       switch ($xaction->getTransactionType()) { | ||||||
|         case PholioImageFileTransaction::TRANSACTIONTYPE: |         case PholioImageFileTransaction::TRANSACTIONTYPE: | ||||||
|         case PholioImageReplaceTransaction::TRANSACTIONTYPE: |  | ||||||
|           return true; |           return true; | ||||||
|           break; |  | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|     return false; |     return false; | ||||||
| @@ -75,11 +75,6 @@ final class PholioMockEditor extends PhabricatorApplicationTransactionEditor { | |||||||
|             } |             } | ||||||
|           } |           } | ||||||
|           break; |           break; | ||||||
|         case PholioImageReplaceTransaction::TRANSACTIONTYPE: |  | ||||||
|           $image = $xaction->getNewValue(); |  | ||||||
|           $image->save(); |  | ||||||
|           $new_images[] = $image; |  | ||||||
|           break; |  | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|     $this->setNewImages($new_images); |     $this->setNewImages($new_images); | ||||||
| @@ -275,4 +270,37 @@ final class PholioMockEditor extends PhabricatorApplicationTransactionEditor { | |||||||
|     return parent::shouldImplyCC($object, $xaction); |     return parent::shouldImplyCC($object, $xaction); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   public function loadPholioImage($object, $phid) { | ||||||
|  |     if (!isset($this->images[$phid])) { | ||||||
|  |  | ||||||
|  |       $image = id(new PholioImageQuery()) | ||||||
|  |         ->setViewer($this->getActor()) | ||||||
|  |         ->withPHIDs(array($phid)) | ||||||
|  |         ->executeOne(); | ||||||
|  |  | ||||||
|  |       if (!$image) { | ||||||
|  |         throw new Exception( | ||||||
|  |           pht( | ||||||
|  |             'No image exists with PHID "%s".', | ||||||
|  |             $phid)); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       $mock_phid = $image->getMockPHID(); | ||||||
|  |       if ($mock_phid) { | ||||||
|  |         if ($mock_phid !== $object->getPHID()) { | ||||||
|  |           throw new Exception( | ||||||
|  |             pht( | ||||||
|  |               'Image ("%s") belongs to the wrong object ("%s", expected "%s").', | ||||||
|  |               $phid, | ||||||
|  |               $mock_phid, | ||||||
|  |               $object->getPHID())); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       $this->images[$phid] = $image; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return $this->images[$phid]; | ||||||
|  |   } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -6,25 +6,25 @@ final class PholioImageReplaceTransaction | |||||||
|   const TRANSACTIONTYPE = 'image-replace'; |   const TRANSACTIONTYPE = 'image-replace'; | ||||||
|  |  | ||||||
|   public function generateOldValue($object) { |   public function generateOldValue($object) { | ||||||
|     $new_image = $this->getNewValue(); |     $editor = $this->getEditor(); | ||||||
|     return $new_image->getReplacesImagePHID(); |     $new_phid = $this->getNewValue(); | ||||||
|  |  | ||||||
|  |     return $editor->loadPholioImage($object, $new_phid) | ||||||
|  |       ->getReplacesImagePHID(); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   public function generateNewValue($object, $value) { |   public function applyExternalEffects($object, $value) { | ||||||
|     return $value->getPHID(); |     $editor = $this->getEditor(); | ||||||
|   } |     $old_phid = $this->getOldValue(); | ||||||
|  |  | ||||||
|   public function applyInternalEffects($object, $value) { |     $old_image = $editor->loadPholioImage($object, $old_phid) | ||||||
|     $old = $this->getOldValue(); |       ->setIsObsolete(1) | ||||||
|     $images = $object->getImages(); |       ->save(); | ||||||
|     foreach ($images as $seq => $image) { |  | ||||||
|       if ($image->getPHID() == $old) { |     $editor->loadPholioImage($object, $value) | ||||||
|         $image->setIsObsolete(1); |       ->setMockPHID($object->getPHID()) | ||||||
|         $image->save(); |       ->setSequence($old_image->getSequence()) | ||||||
|         unset($images[$seq]); |       ->save(); | ||||||
|       } |  | ||||||
|     } |  | ||||||
|     $object->attachImages($images); |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   public function getTitle() { |   public function getTitle() { | ||||||
| @@ -54,27 +54,87 @@ final class PholioImageReplaceTransaction | |||||||
|     $object, |     $object, | ||||||
|     PhabricatorApplicationTransaction $u, |     PhabricatorApplicationTransaction $u, | ||||||
|     PhabricatorApplicationTransaction $v) { |     PhabricatorApplicationTransaction $v) { | ||||||
|     $u_img = $u->getNewValue(); |  | ||||||
|     $v_img = $v->getNewValue(); |     $u_phid = $u->getOldValue(); | ||||||
|     if ($u_img->getReplacesImagePHID() == $v_img->getReplacesImagePHID()) { |     $v_phid = $v->getOldValue(); | ||||||
|  |  | ||||||
|  |     if ($u_phid === $v_phid) { | ||||||
|       return $v; |       return $v; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     return null; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   public function extractFilePHIDs($object, $value) { |   public function extractFilePHIDs($object, $value) { | ||||||
|     $file_phids = array(); |     $editor = $this->getEditor(); | ||||||
|  |  | ||||||
|  |     $file_phid = $editor->loadPholioImage($object, $value) | ||||||
|  |       ->getFilePHID(); | ||||||
|  |  | ||||||
|  |     return array($file_phid); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public function validateTransactions($object, array $xactions) { | ||||||
|  |     $errors = array(); | ||||||
|  |  | ||||||
|  |     $mock_phid = $object->getPHID(); | ||||||
|  |  | ||||||
|     $editor = $this->getEditor(); |     $editor = $this->getEditor(); | ||||||
|     $images = $editor->getNewImages(); |     foreach ($xactions as $xaction) { | ||||||
|     foreach ($images as $image) { |       $new_phid = $xaction->getNewValue(); | ||||||
|       if ($image->getPHID() !== $value) { |  | ||||||
|  |       try { | ||||||
|  |         $new_image = $editor->loadPholioImage($object, $new_phid); | ||||||
|  |       } catch (Exception $ex) { | ||||||
|  |         $errors[] = $this->newInvalidError( | ||||||
|  |           pht( | ||||||
|  |             'Unable to load replacement image ("%s"): %s', | ||||||
|  |             $new_phid, | ||||||
|  |             $ex->getMessage()), | ||||||
|  |           $xaction); | ||||||
|         continue; |         continue; | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       $file_phids[] = $image->getFilePHID(); |       $old_phid = $new_image->getReplacesImagePHID(); | ||||||
|  |       if (!$old_phid) { | ||||||
|  |         $errors[] = $this->newInvalidError( | ||||||
|  |           pht( | ||||||
|  |             'Image ("%s") does not specify which image it replaces.', | ||||||
|  |             $new_phid), | ||||||
|  |           $xaction); | ||||||
|  |         continue; | ||||||
|       } |       } | ||||||
|  |  | ||||||
|     return $file_phids; |       try { | ||||||
|  |         $old_image = $editor->loadPholioImage($object, $old_phid); | ||||||
|  |       } catch (Exception $ex) { | ||||||
|  |         $errors[] = $this->newInvalidError( | ||||||
|  |           pht( | ||||||
|  |             'Unable to load replaced image ("%s"): %s', | ||||||
|  |             $old_phid, | ||||||
|  |             $ex->getMessage()), | ||||||
|  |           $xaction); | ||||||
|  |         continue; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       if ($old_image->getMockPHID() !== $mock_phid) { | ||||||
|  |         $errors[] = $this->newInvalidError( | ||||||
|  |           pht( | ||||||
|  |             'Replaced image ("%s") belongs to the wrong mock ("%s", expected '. | ||||||
|  |             '"%s").', | ||||||
|  |             $old_phid, | ||||||
|  |             $old_image->getMockPHID(), | ||||||
|  |             $mock_phid), | ||||||
|  |           $xaction); | ||||||
|  |         continue; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       // TODO: You shouldn't be able to replace an image if it has already | ||||||
|  |       // been replaced. | ||||||
|  |  | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return $errors; | ||||||
|   } |   } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 epriestley
					epriestley