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:
epriestley
2018-12-18 15:09:05 -08:00
parent b1e7e3a10e
commit 741fb747db
3 changed files with 128 additions and 41 deletions

View File

@@ -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());
} }

View File

@@ -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];
}
} }

View File

@@ -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;
} }
} }