mirror of
https://we.phorge.it/source/phorge.git
synced 2025-01-05 04:11:01 +01:00
Wrap basic diff/revision association in a transaction
Summary: This doesn't cover every case exhaustively (see comments) but should cover like 98% of the practical cases. This makes one workflow modification: willWriteRevision() was previously guaranteed to have a revisionID / revisionPHID and no longer is. I verified that no field implementations depend on this behavior. Fields which depend on IDs should be using didWriteRevision() instead. Test Plan: Inserted a "throw" into the middle of the transactions and created revisions; they didn't orphan. Created revisions normally, they worked correctly. Reviewers: btrahan, nh Reviewed By: btrahan CC: aran, epriestley Maniphest Tasks: T605 Differential Revision: https://secure.phabricator.com/D1541
This commit is contained in:
parent
dc36317ea4
commit
c86dfd84d9
1 changed files with 54 additions and 32 deletions
|
@ -166,31 +166,9 @@ class DifferentialRevisionEditor {
|
||||||
public function save() {
|
public function save() {
|
||||||
$revision = $this->getRevision();
|
$revision = $this->getRevision();
|
||||||
|
|
||||||
// TODO
|
|
||||||
// $revision->openTransaction();
|
|
||||||
|
|
||||||
$is_new = $this->isNewRevision();
|
$is_new = $this->isNewRevision();
|
||||||
if ($is_new) {
|
if ($is_new) {
|
||||||
// These fields aren't nullable; set them to sensible defaults if they
|
$this->initializeNewRevision($revision);
|
||||||
// haven't been configured. We're just doing this so we can generate an
|
|
||||||
// ID for the revision if we don't have one already.
|
|
||||||
$revision->setLineCount(0);
|
|
||||||
if ($revision->getStatus() === null) {
|
|
||||||
$revision->setStatus(ArcanistDifferentialRevisionStatus::NEEDS_REVIEW);
|
|
||||||
}
|
|
||||||
if ($revision->getTitle() === null) {
|
|
||||||
$revision->setTitle('Untitled Revision');
|
|
||||||
}
|
|
||||||
if ($revision->getAuthorPHID() === null) {
|
|
||||||
$revision->setAuthorPHID($this->getActorPHID());
|
|
||||||
}
|
|
||||||
if ($revision->getSummary() === null) {
|
|
||||||
$revision->setSummary('');
|
|
||||||
}
|
|
||||||
if ($revision->getTestPlan() === null) {
|
|
||||||
$revision->setTestPlan('');
|
|
||||||
}
|
|
||||||
$revision->save();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
$revision->loadRelationships();
|
$revision->loadRelationships();
|
||||||
|
@ -205,6 +183,37 @@ class DifferentialRevisionEditor {
|
||||||
$this->cc = $revision->getCCPHIDs();
|
$this->cc = $revision->getCCPHIDs();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$diff = $this->getDiff();
|
||||||
|
if ($diff) {
|
||||||
|
$revision->setLineCount($diff->getLineCount());
|
||||||
|
}
|
||||||
|
|
||||||
|
// Save the revision, to generate its ID and PHID if it is new. We need
|
||||||
|
// the ID/PHID in order to record them in Herald transcripts, but don't
|
||||||
|
// want to hold a transaction open while running Herald because it is
|
||||||
|
// potentially somewhat slow. The downside is that we may end up with a
|
||||||
|
// saved revision/diff pair without appropriate CCs. We could be better
|
||||||
|
// about this -- for example:
|
||||||
|
//
|
||||||
|
// - Herald can't affect reviewers, so we could compute them before
|
||||||
|
// opening the transaction and then save them in the transaction.
|
||||||
|
// - Herald doesn't *really* need PHIDs to compute its effects, we could
|
||||||
|
// run it before saving these objects and then hand over the PHIDs later.
|
||||||
|
//
|
||||||
|
// But this should address the problem of orphaned revisions, which is
|
||||||
|
// currently the only problem we experience in practice.
|
||||||
|
|
||||||
|
$revision->openTransaction();
|
||||||
|
|
||||||
|
$revision->save();
|
||||||
|
if ($diff) {
|
||||||
|
$diff->setRevisionID($revision->getID());
|
||||||
|
$diff->save();
|
||||||
|
}
|
||||||
|
|
||||||
|
$revision->saveTransaction();
|
||||||
|
|
||||||
|
|
||||||
// We're going to build up three dictionaries: $add, $rem, and $stable. The
|
// We're going to build up three dictionaries: $add, $rem, and $stable. The
|
||||||
// $add dictionary has added reviewers/CCs. The $rem dictionary has
|
// $add dictionary has added reviewers/CCs. The $rem dictionary has
|
||||||
// reviewers/CCs who have been removed, and the $stable array is
|
// reviewers/CCs who have been removed, and the $stable array is
|
||||||
|
@ -216,8 +225,6 @@ class DifferentialRevisionEditor {
|
||||||
'ccs' => array_fill_keys($revision->getCCPHIDs(), true),
|
'ccs' => array_fill_keys($revision->getCCPHIDs(), true),
|
||||||
);
|
);
|
||||||
|
|
||||||
$diff = $this->getDiff();
|
|
||||||
|
|
||||||
$xscript_header = null;
|
$xscript_header = null;
|
||||||
$xscript_uri = null;
|
$xscript_uri = null;
|
||||||
|
|
||||||
|
@ -226,12 +233,8 @@ class DifferentialRevisionEditor {
|
||||||
'ccs' => array_fill_keys($this->cc, true),
|
'ccs' => array_fill_keys($this->cc, true),
|
||||||
);
|
);
|
||||||
|
|
||||||
|
|
||||||
$rem_ccs = array();
|
$rem_ccs = array();
|
||||||
if ($diff) {
|
if ($diff) {
|
||||||
$diff->setRevisionID($revision->getID());
|
|
||||||
$revision->setLineCount($diff->getLineCount());
|
|
||||||
|
|
||||||
$adapter = new HeraldDifferentialRevisionAdapter(
|
$adapter = new HeraldDifferentialRevisionAdapter(
|
||||||
$revision,
|
$revision,
|
||||||
$diff);
|
$diff);
|
||||||
|
@ -427,9 +430,6 @@ class DifferentialRevisionEditor {
|
||||||
))
|
))
|
||||||
->publish();
|
->publish();
|
||||||
|
|
||||||
// TODO
|
|
||||||
// $revision->saveTransaction();
|
|
||||||
|
|
||||||
// TODO: Move this into a worker task thing.
|
// TODO: Move this into a worker task thing.
|
||||||
PhabricatorSearchDifferentialIndexer::indexRevision($revision);
|
PhabricatorSearchDifferentialIndexer::indexRevision($revision);
|
||||||
|
|
||||||
|
@ -908,5 +908,27 @@ class DifferentialRevisionEditor {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function initializeNewRevision(DifferentialRevision $revision) {
|
||||||
|
// These fields aren't nullable; set them to sensible defaults if they
|
||||||
|
// haven't been configured. We're just doing this so we can generate an
|
||||||
|
// ID for the revision if we don't have one already.
|
||||||
|
$revision->setLineCount(0);
|
||||||
|
if ($revision->getStatus() === null) {
|
||||||
|
$revision->setStatus(ArcanistDifferentialRevisionStatus::NEEDS_REVIEW);
|
||||||
|
}
|
||||||
|
if ($revision->getTitle() === null) {
|
||||||
|
$revision->setTitle('Untitled Revision');
|
||||||
|
}
|
||||||
|
if ($revision->getAuthorPHID() === null) {
|
||||||
|
$revision->setAuthorPHID($this->getActorPHID());
|
||||||
|
}
|
||||||
|
if ($revision->getSummary() === null) {
|
||||||
|
$revision->setSummary('');
|
||||||
|
}
|
||||||
|
if ($revision->getTestPlan() === null) {
|
||||||
|
$revision->setTestPlan('');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue