From b5c753a746cf9df85ef38548219d26748e3d4d42 Mon Sep 17 00:00:00 2001 From: Ferdinand Thiessen Date: Sat, 24 Aug 2024 19:09:36 +0200 Subject: [PATCH] feat: Add sharing activity for teams This adds support for Teams (previously "Circles"), the user notification already work, but the notification for the owner and sharer need support in the `files_sharing` app. Signed-off-by: Ferdinand Thiessen --- lib/FilesHooks.php | 65 +++++++++++++++++++++++++++++++++++++++++----- 1 file changed, 58 insertions(+), 7 deletions(-) diff --git a/lib/FilesHooks.php b/lib/FilesHooks.php index 5d58deb24..1d2d7cdf3 100755 --- a/lib/FilesHooks.php +++ b/lib/FilesHooks.php @@ -11,6 +11,8 @@ use OCA\Activity\BackgroundJob\RemoteActivity; use OCA\Activity\Extension\Files; use OCA\Activity\Extension\Files_Sharing; +use OCA\Circles\CirclesManager; +use OCA\Circles\Model\Member; use OCP\Activity\IManager; use OCP\Constants; use OCP\Files\Config\IUserMountCache; @@ -60,7 +62,8 @@ public function __construct( protected IUserMountCache $userMountCache, protected IConfig $config, protected NotificationGenerator $notificationGenerator, - protected ITagManager $tagManager + protected ITagManager $tagManager, + protected ?CirclesManager $teamManager, ) { } @@ -664,6 +667,16 @@ public function share($share) { (int)$share->getId() ); break; + case IShare::TYPE_CIRCLE: + $this->shareWithTeam( + $share->getSharedWith(), + $share->getNodeId(), + $share->getNodeType(), + $share->getTarget(), + (int)$share->getId(), + $share->getSharedBy(), + ); + break; case IShare::TYPE_LINK: $this->shareByLink( $share->getNodeId(), @@ -726,7 +739,8 @@ protected function shareWithGroup($shareWith, $fileSource, $itemType, $fileTarge $offset = 0; $users = $group->searchUsers('', self::USER_BATCH_SIZE, $offset); while (!empty($users)) { - $this->addNotificationsForGroupUsers($users, 'shared_with_by', $fileSource, $itemType, $fileTarget, $shareId); + $users = array_map(fn (IUser $user) => $user->getUID(), $users); + $this->addNotificationsForUsers($users, 'shared_with_by', $fileSource, $itemType, $fileTarget, $shareId); $offset += self::USER_BATCH_SIZE; $users = $group->searchUsers('', self::USER_BATCH_SIZE, $offset); } @@ -758,6 +772,42 @@ protected function shareByLink($fileSource, $itemType, $linkOwner) { ); } + /** + * Sharing a file or folder with a team + * + * @param string $shareWith + * @param int $fileSource File ID that is being shared + * @param string $itemType File type that is being shared (file or folder) + * @param string $fileTarget File path + * @param int $shareId The Share ID of this share + */ + protected function shareWithTeam(string $shareWith, int $fileSource, string $itemType, string $fileTarget, int $shareId, string $sharer): void { + if ($this->teamManager === null) { + return; + } + + try { + $this->teamManager->startSuperSession(); + $team = $this->teamManager->getCircle($shareWith); + $members = $team->getInheritedMembers(); + $members = array_filter($members, fn ($member) => $member->getUserType() === Member::TYPE_USER); + $users = array_map(fn ($member) => $member->getUserId(), $members); + } catch (\Throwable $e) { + $this->logger->debug('Fetching team members for share activity failed', ['exception' => $e]); + // error in teams app - setting users list to empty + $users = []; + } + + // Activity for user performing the share + $this->shareNotificationForSharer('shared_team_self', $shareWith, $fileSource, $itemType); + // Activity for original owner of the file (re-sharing) + if ($this->currentUser->getUID() !== null) { + $this->shareNotificationForOriginalOwners($this->currentUser->getUID(), 're-shared_team_by', $shareWith, $fileSource, $itemType); + } + // Activity for all affected users + $this->addNotificationsForUsers($users, 'shared_with_by', $fileSource, $itemType, $fileTarget, $shareId); + } + /** * Manage unsharing events * @@ -878,9 +928,10 @@ protected function unshareFromGroup(IShare $share) { $offset = 0; $users = $group->searchUsers('', self::USER_BATCH_SIZE, $offset); + $users = array_map(fn (IUser $user) => $user->getUID(), $users); $shouldFlush = $this->startActivityTransaction(); while (!empty($users)) { - $this->addNotificationsForGroupUsers($users, $actionUser, $share->getNodeId(), $share->getNodeType(), $share->getTarget(), (int) $share->getId()); + $this->addNotificationsForUsers($users, $actionUser, $share->getNodeId(), $share->getNodeType(), $share->getTarget(), (int)$share->getId()); $offset += self::USER_BATCH_SIZE; $users = $group->searchUsers('', self::USER_BATCH_SIZE, $offset); } @@ -940,18 +991,18 @@ protected function unshareLink(IShare $share) { } /** - * @param IUser[] $usersInGroup + * @param string[] $usersIds * @param string $actionUser * @param int $fileSource File ID that is being shared * @param string $itemType File type that is being shared (file or folder) * @param string $fileTarget File path * @param int $shareId The Share ID of this share */ - protected function addNotificationsForGroupUsers(array $usersInGroup, $actionUser, $fileSource, $itemType, $fileTarget, $shareId) { + protected function addNotificationsForUsers(array $usersIds, $actionUser, $fileSource, $itemType, $fileTarget, $shareId) { $affectedUsers = []; - foreach ($usersInGroup as $user) { - $affectedUsers[$user->getUID()] = $fileTarget; + foreach ($usersIds as $user) { + $affectedUsers[$user] = $fileTarget; } // Remove the triggering user, we already managed his notifications