From aef2a39a81b0c32e776b1e58cc63c4043e58ffe4 Mon Sep 17 00:00:00 2001 From: Chad Little Date: Fri, 17 Mar 2017 10:32:07 -0700 Subject: [PATCH] Add Badges to UserCache Summary: Ref T12270. Builds out a BadgeCache for PhabricatorUser, primarily for Timeline, potentially feed? This should still work if we later let people pick which two, just switch query in BadgeCache. Test Plan: Give out badges, test timeline for displaying badges from handles and without queries. Revoke a badge, see cache change. Reviewers: epriestley Reviewed By: epriestley Subscribers: Korvin Maniphest Tasks: T12270 Differential Revision: https://secure.phabricator.com/D17503 --- src/__phutil_library_map__.php | 2 + .../badges/editor/PhabricatorBadgesEditor.php | 41 +++++++++++++ .../cache/PhabricatorUserBadgesCacheType.php | 61 +++++++++++++++++++ .../people/query/PhabricatorPeopleQuery.php | 13 ++++ .../people/storage/PhabricatorUser.php | 5 ++ ...catorApplicationTransactionCommentView.php | 19 ++---- src/view/phui/PHUITimelineView.php | 37 ++++------- 7 files changed, 141 insertions(+), 37 deletions(-) create mode 100644 src/applications/people/cache/PhabricatorUserBadgesCacheType.php diff --git a/src/__phutil_library_map__.php b/src/__phutil_library_map__.php index cfd11d630f..29bf2e418f 100644 --- a/src/__phutil_library_map__.php +++ b/src/__phutil_library_map__.php @@ -4067,6 +4067,7 @@ phutil_register_library_map(array( 'PhabricatorUnknownContentSource' => 'infrastructure/contentsource/PhabricatorUnknownContentSource.php', 'PhabricatorUnsubscribedFromObjectEdgeType' => 'applications/transactions/edges/PhabricatorUnsubscribedFromObjectEdgeType.php', 'PhabricatorUser' => 'applications/people/storage/PhabricatorUser.php', + 'PhabricatorUserBadgesCacheType' => 'applications/people/cache/PhabricatorUserBadgesCacheType.php', 'PhabricatorUserBlurbField' => 'applications/people/customfield/PhabricatorUserBlurbField.php', 'PhabricatorUserCache' => 'applications/people/storage/PhabricatorUserCache.php', 'PhabricatorUserCacheType' => 'applications/people/cache/PhabricatorUserCacheType.php', @@ -9415,6 +9416,7 @@ phutil_register_library_map(array( 'PhabricatorFulltextInterface', 'PhabricatorConduitResultInterface', ), + 'PhabricatorUserBadgesCacheType' => 'PhabricatorUserCacheType', 'PhabricatorUserBlurbField' => 'PhabricatorUserCustomField', 'PhabricatorUserCache' => 'PhabricatorUserDAO', 'PhabricatorUserCacheType' => 'Phobject', diff --git a/src/applications/badges/editor/PhabricatorBadgesEditor.php b/src/applications/badges/editor/PhabricatorBadgesEditor.php index a077f4c3ba..fddc55747c 100644 --- a/src/applications/badges/editor/PhabricatorBadgesEditor.php +++ b/src/applications/badges/editor/PhabricatorBadgesEditor.php @@ -118,4 +118,45 @@ final class PhabricatorBadgesEditor return pht('[Badge]'); } + protected function applyFinalEffects( + PhabricatorLiskDAO $object, + array $xactions) { + + $badge_phid = $object->getPHID(); + $user_phids = array(); + $clear_everything = false; + + foreach ($xactions as $xaction) { + switch ($xaction->getTransactionType()) { + case PhabricatorBadgesBadgeAwardTransaction::TRANSACTIONTYPE: + case PhabricatorBadgesBadgeRevokeTransaction::TRANSACTIONTYPE: + foreach ($xaction->getNewValue() as $user_phid) { + $user_phids[] = $user_phid; + } + break; + default: + $clear_everything = true; + break; + } + } + + if ($clear_everything) { + $awards = id(new PhabricatorBadgesAwardQuery()) + ->setViewer($this->getActor()) + ->withBadgePHIDs(array($badge_phid)) + ->execute(); + foreach ($awards as $award) { + $user_phids[] = $award->getRecipientPHID(); + } + } + + if ($user_phids) { + PhabricatorUserCache::clearCaches( + PhabricatorUserBadgesCacheType::KEY_BADGES, + $user_phids); + } + + return $xactions; + } + } diff --git a/src/applications/people/cache/PhabricatorUserBadgesCacheType.php b/src/applications/people/cache/PhabricatorUserBadgesCacheType.php new file mode 100644 index 0000000000..28eef59777 --- /dev/null +++ b/src/applications/people/cache/PhabricatorUserBadgesCacheType.php @@ -0,0 +1,61 @@ +setViewer($this->getViewer()) + ->withRecipientPHIDs(array($user_phid)) + ->withBadgeStatuses(array(PhabricatorBadgesBadge::STATUS_ACTIVE)) + ->setLimit(self::BADGE_COUNT) + ->execute(); + + $award_data = array(); + if ($awards) { + foreach ($awards as $award) { + $badge = $award->getBadge(); + $award_data[] = array( + 'icon' => $badge->getIcon(), + 'name' => $badge->getName(), + 'quality' => $badge->getQuality(), + 'id' => $badge->getID(), + ); + } + } + $results[$user_phid] = phutil_json_encode($award_data); + + } + + return $results; + } + +} diff --git a/src/applications/people/query/PhabricatorPeopleQuery.php b/src/applications/people/query/PhabricatorPeopleQuery.php index 6cb2922083..7367382e50 100644 --- a/src/applications/people/query/PhabricatorPeopleQuery.php +++ b/src/applications/people/query/PhabricatorPeopleQuery.php @@ -24,6 +24,7 @@ final class PhabricatorPeopleQuery private $needProfile; private $needProfileImage; private $needAvailability; + private $needBadgeAwards; private $cacheKeys = array(); public function withIDs(array $ids) { @@ -145,6 +146,18 @@ final class PhabricatorPeopleQuery return $this; } + public function needBadgeAwards($need) { + $cache_key = PhabricatorUserBadgesCacheType::KEY_BADGES; + + if ($need) { + $this->cacheKeys[$cache_key] = true; + } else { + unset($this->cacheKeys[$cache_key]); + } + + return $this; + } + public function newResultObject() { return new PhabricatorUser(); } diff --git a/src/applications/people/storage/PhabricatorUser.php b/src/applications/people/storage/PhabricatorUser.php index 215f8b9eed..7edbcd4e52 100644 --- a/src/applications/people/storage/PhabricatorUser.php +++ b/src/applications/people/storage/PhabricatorUser.php @@ -848,6 +848,11 @@ final class PhabricatorUser return $this->requireCacheData($message_key); } + public function getRecentBadgeAwards() { + $badges_key = PhabricatorUserBadgesCacheType::KEY_BADGES; + return $this->requireCacheData($badges_key); + } + public function getFullName() { if (strlen($this->getRealName())) { return $this->getUsername().' ('.$this->getRealName().')'; diff --git a/src/applications/transactions/view/PhabricatorApplicationTransactionCommentView.php b/src/applications/transactions/view/PhabricatorApplicationTransactionCommentView.php index b91155183c..232420d4c2 100644 --- a/src/applications/transactions/view/PhabricatorApplicationTransactionCommentView.php +++ b/src/applications/transactions/view/PhabricatorApplicationTransactionCommentView.php @@ -525,25 +525,18 @@ class PhabricatorApplicationTransactionCommentView extends AphrontView { return null; } - $awards = id(new PhabricatorBadgesAwardQuery()) - ->setViewer($this->getUser()) - ->withRecipientPHIDs(array($user->getPHID())) - ->withBadgeStatuses(array(PhabricatorBadgesBadge::STATUS_ACTIVE)) - ->setLimit(2) - ->execute(); - - $badges = mpull($awards, 'getBadge'); - + // Pull Badges from UserCache + $badges = $user->getRecentBadgeAwards(); $badge_view = null; if ($badges) { $badge_list = array(); foreach ($badges as $badge) { $badge_view = id(new PHUIBadgeMiniView()) - ->setIcon($badge->getIcon()) - ->setQuality($badge->getQuality()) - ->setHeader($badge->getName()) + ->setIcon($badge['icon']) + ->setQuality($badge['quality']) + ->setHeader($badge['name']) ->setTipDirection('E') - ->setHref('/badges/view/'.$badge->getID()); + ->setHref('/badges/view/'.$badge['id'].'/'); $badge_list[] = $badge_view; } diff --git a/src/view/phui/PHUITimelineView.php b/src/view/phui/PHUITimelineView.php index 57b07a94bf..804b060d55 100644 --- a/src/view/phui/PHUITimelineView.php +++ b/src/view/phui/PHUITimelineView.php @@ -243,37 +243,26 @@ final class PHUITimelineView extends AphrontView { return; } - - $awards = id(new PhabricatorBadgesAwardQuery()) - ->setViewer($this->getViewer()) - ->withRecipientPHIDs($user_phids) - ->withBadgeStatuses(array(PhabricatorBadgesBadge::STATUS_ACTIVE)) + $users = id(new PhabricatorPeopleQuery()) + ->setViewer($viewer) + ->withPHIDs($user_phids) + ->needBadgeAwards(true) ->execute(); - - $awards = mgroup($awards, 'getRecipientPHID'); + $users = mpull($users, null, 'getPHID'); foreach ($events as $event) { - - $author_awards = idx($awards, $event->getAuthorPHID(), array()); - - $badges = array(); - foreach ($author_awards as $award) { - $badge = $award->getBadge(); - $badges[$award->getBadgePHID()] = $badge; + $user_phid = $event->getAuthorPHID(); + if (!array_key_exists($user_phid, $users)) { + continue; } - - // TODO: Pick the "best" badges in some smart way. For now, just pick - // the first two. - $badges = array_slice($badges, 0, 2); - + $badges = $users[$user_phid]->getRecentBadgeAwards(); foreach ($badges as $badge) { $badge_view = id(new PHUIBadgeMiniView()) - ->setIcon($badge->getIcon()) - ->setQuality($badge->getQuality()) - ->setHeader($badge->getName()) + ->setIcon($badge['icon']) + ->setQuality($badge['quality']) + ->setHeader($badge['name']) ->setTipDirection('E') - ->setHref('/badges/view/'.$badge->getID()); - + ->setHref('/badges/view/'.$badge['id'].'/'); $event->addBadge($badge_view); } }