mirror of
https://we.phorge.it/source/phorge.git
synced 2025-04-04 00:18:21 +02:00
Synchronize review request state to Asana
Summary: Ref T2852. Depends on D6302. This now creates, destroys, and synchronizes subtasks. - After finishing the parent task stuff, we pull a list of all known subtasks. - We load all those subtasks. - If we fail to load any, we delete their objects and edges on the Phabricator side. - Of the remaining subtasks, we find subtasks for users who aren't related to the object any more and delete them in Asana and locally (for example, if alincoln is removed as a reviewer, we delete his subtask). - For all the related users, we either synchronize their existing task or create a new one for them. - Then we write edges for any new tasks we added. This doesn't handle a few weird edge cases in any specific way: - If a subtask is moved under a different parent, we ignore it. - If a new subtask is created that we don't know about, we ignore it. - If a subtask we know about is deleted, we just respawn it. This is consistent with "DON'T EDIT THESE". You can force sync to stop by deleting the parent. Addititionally: - Make the "don't edit" warning more compelling and visceral. Test Plan: - Kind of ran it a bit. - There are like 3,000 edge cases here so this is hard to test exhaustively. - Forced a few of the edge cases to happen. - Nothing seems immediately broken in an obvious way? {F47551} Reviewers: btrahan Reviewed By: btrahan CC: aran Maniphest Tasks: T2852 Differential Revision: https://secure.phabricator.com/D6303
This commit is contained in:
parent
5a6044dbaa
commit
302da70e72
1 changed files with 176 additions and 3 deletions
|
@ -85,9 +85,8 @@ final class DoorkeeperFeedWorkerAsana extends FeedPushWorker {
|
||||||
$notes = array(
|
$notes = array(
|
||||||
$revision->getSummary(),
|
$revision->getSummary(),
|
||||||
$uri,
|
$uri,
|
||||||
"\xE2\x9A\xA0 DO NOT EDIT THIS TASK \xE2\x9A\xA0\n".
|
$this->getSynchronizationWarning(),
|
||||||
"Your changes will not be reflected in Phabricator.",
|
);
|
||||||
);
|
|
||||||
|
|
||||||
$notes = implode("\n\n", $notes);
|
$notes = implode("\n\n", $notes);
|
||||||
|
|
||||||
|
@ -97,6 +96,34 @@ final class DoorkeeperFeedWorkerAsana extends FeedPushWorker {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function getAsanaSubtaskData($object) {
|
||||||
|
$revision = $object;
|
||||||
|
|
||||||
|
$name = '[Differential] Review Request';
|
||||||
|
$uri = PhabricatorEnv::getProductionURI('/D'.$revision->getID());
|
||||||
|
|
||||||
|
$notes = array(
|
||||||
|
$revision->getSummary(),
|
||||||
|
$uri,
|
||||||
|
$this->getSynchronizationWarning(),
|
||||||
|
);
|
||||||
|
|
||||||
|
$notes = implode("\n\n", $notes);
|
||||||
|
|
||||||
|
return array(
|
||||||
|
'name' => '[Differential] Review Request',
|
||||||
|
'notes' => $notes,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
private function getSynchronizationWarning() {
|
||||||
|
return
|
||||||
|
"\xE2\x9A\xA0 DO NOT EDIT THIS TASK \xE2\x9A\xA0\n".
|
||||||
|
"\xE2\x98\xA0 Your changes will not be reflected in Phabricator.\n".
|
||||||
|
"\xE2\x98\xA0 Your changes will be destroyed the next time state ".
|
||||||
|
"is synchronized.";
|
||||||
|
}
|
||||||
|
|
||||||
protected function doWork() {
|
protected function doWork() {
|
||||||
$story = $this->getFeedStory();
|
$story = $this->getFeedStory();
|
||||||
$data = $story->getStoryData();
|
$data = $story->getStoryData();
|
||||||
|
@ -262,6 +289,152 @@ final class DoorkeeperFeedWorkerAsana extends FeedPushWorker {
|
||||||
'text' => $story->renderText(),
|
'text' => $story->renderText(),
|
||||||
));
|
));
|
||||||
|
|
||||||
|
|
||||||
|
// Now, handle the subtasks.
|
||||||
|
|
||||||
|
$sub_editor = id(new PhabricatorEdgeEditor())
|
||||||
|
->setActor($viewer);
|
||||||
|
|
||||||
|
// First, find all the object references in Phabricator for tasks that we
|
||||||
|
// know about and import their objects from Asana.
|
||||||
|
$sub_edges = $edges[$src_phid][$etype_sub];
|
||||||
|
$sub_refs = array();
|
||||||
|
$subtask_data = $this->getAsanaSubtaskData($object);
|
||||||
|
$have_phids = array();
|
||||||
|
|
||||||
|
if ($sub_edges) {
|
||||||
|
$refs = id(new DoorkeeperImportEngine())
|
||||||
|
->setViewer($possessed_user)
|
||||||
|
->withPHIDs(array_keys($sub_edges))
|
||||||
|
->execute();
|
||||||
|
|
||||||
|
foreach ($refs as $ref) {
|
||||||
|
if (!$ref->getIsVisible()) {
|
||||||
|
$ref->getExternalObject()->delete();
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
$have_phids[$ref->getExternalObject()->getPHID()] = $ref;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Remove any edges in Phabricator which don't have valid tasks in Asana.
|
||||||
|
// These are likely tasks which have been deleted. We're going to respawn
|
||||||
|
// them.
|
||||||
|
foreach ($sub_edges as $sub_phid => $sub_edge) {
|
||||||
|
if (isset($have_phids[$sub_phid])) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->log(
|
||||||
|
"Removing subtask edge to %s, foreign object is not visible.\n",
|
||||||
|
$sub_phid);
|
||||||
|
$sub_editor->removeEdge($src_phid, $etype_sub, $sub_phid);
|
||||||
|
unset($sub_edges[$sub_phid]);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// For each active or passive user, we're looking for an existing, valid
|
||||||
|
// task. If we find one we're going to update it; if we don't, we'll
|
||||||
|
// create one. We ignore extra subtasks that we didn't create (we gain
|
||||||
|
// nothing by deleting them and might be nuking something important) and
|
||||||
|
// ignore subtasks which have been moved across workspaces or replanted
|
||||||
|
// under new parents (this stuff is too edge-casey to bother checking for
|
||||||
|
// and complicated to fix, as it needs extra API calls). However, we do
|
||||||
|
// clean up subtasks we created whose owners are no longer associated
|
||||||
|
// with the object.
|
||||||
|
|
||||||
|
$subtask_states = array_fill_keys($active_phids, false) +
|
||||||
|
array_fill_keys($passive_phids, true);
|
||||||
|
|
||||||
|
// Continue with only those users who have Asana credentials.
|
||||||
|
|
||||||
|
$subtask_states = array_select_keys(
|
||||||
|
$subtask_states,
|
||||||
|
array_keys($phid_aid_map));
|
||||||
|
|
||||||
|
$need_subtasks = $subtask_states;
|
||||||
|
|
||||||
|
$user_to_ref_map = array();
|
||||||
|
$nuke_refs = array();
|
||||||
|
foreach ($sub_edges as $sub_phid => $sub_edge) {
|
||||||
|
$user_phid = idx($sub_edge['data'], 'userPHID');
|
||||||
|
|
||||||
|
if (isset($need_subtasks[$user_phid])) {
|
||||||
|
unset($need_subtasks[$user_phid]);
|
||||||
|
$user_to_ref_map[$user_phid] = $have_phids[$sub_phid];
|
||||||
|
} else {
|
||||||
|
// This user isn't associated with the object anymore, so get rid
|
||||||
|
// of their task and edge.
|
||||||
|
$nuke_refs[$sub_phid] = $have_phids[$sub_phid];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// These are tasks we know about but which are no longer relevant -- for
|
||||||
|
// example, because a user has been removed as a reviewer. Remove them and
|
||||||
|
// their edges.
|
||||||
|
|
||||||
|
foreach ($nuke_refs as $sub_phid => $ref) {
|
||||||
|
$sub_editor->removeEdge($src_phid, $etype_sub, $sub_phid);
|
||||||
|
$this->makeAsanaAPICall(
|
||||||
|
$oauth_token,
|
||||||
|
'tasks/'.$ref->getObjectID(),
|
||||||
|
'DELETE',
|
||||||
|
array());
|
||||||
|
$ref->getExternalObject()->delete();
|
||||||
|
}
|
||||||
|
|
||||||
|
// For each user that we don't have a subtask for, create a new subtask.
|
||||||
|
|
||||||
|
foreach ($need_subtasks as $user_phid => $is_completed) {
|
||||||
|
$subtask = $this->makeAsanaAPICall(
|
||||||
|
$oauth_token,
|
||||||
|
'tasks',
|
||||||
|
'POST',
|
||||||
|
$subtask_data + array(
|
||||||
|
'assignee' => $phid_aid_map[$user_phid],
|
||||||
|
'completed' => $is_completed,
|
||||||
|
'parent' => $parent_ref->getObjectID(),
|
||||||
|
));
|
||||||
|
|
||||||
|
$subtask_ref = $this->newRefFromResult(
|
||||||
|
DoorkeeperBridgeAsana::OBJTYPE_TASK,
|
||||||
|
$subtask);
|
||||||
|
|
||||||
|
$user_to_ref_map[$user_phid] = $subtask_ref;
|
||||||
|
|
||||||
|
// We don't need to synchronize this subtask's state because we just
|
||||||
|
// set it when we created it.
|
||||||
|
unset($subtask_states[$user_phid]);
|
||||||
|
|
||||||
|
// Add an edge to track this subtask.
|
||||||
|
$sub_editor->addEdge(
|
||||||
|
$src_phid,
|
||||||
|
$etype_sub,
|
||||||
|
$subtask_ref->getExternalObject()->getPHID(),
|
||||||
|
array(
|
||||||
|
'data' => array(
|
||||||
|
'userPHID' => $user_phid,
|
||||||
|
),
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Synchronize all the previously-existing subtasks.
|
||||||
|
|
||||||
|
foreach ($subtask_states as $user_phid => $is_completed) {
|
||||||
|
$this->makeAsanaAPICall(
|
||||||
|
$oauth_token,
|
||||||
|
'tasks/'.$user_to_ref_map[$user_phid]->getObjectID(),
|
||||||
|
'PUT',
|
||||||
|
$subtask_data + array(
|
||||||
|
'assignee' => $phid_aid_map[$user_phid],
|
||||||
|
'completed' => $is_completed,
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Update edges on our side.
|
||||||
|
|
||||||
|
$sub_editor->save();
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private function lookupAsanaUserIDs($all_phids) {
|
private function lookupAsanaUserIDs($all_phids) {
|
||||||
|
|
Loading…
Add table
Reference in a new issue