123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641 |
- <?php
- /**
- * SPDX-FileCopyrightText: 2016 Nextcloud GmbH and Nextcloud contributors
- * SPDX-License-Identifier: AGPL-3.0-or-later
- */
- namespace OCA\DAV\CalDAV\Activity;
- use OCA\DAV\CalDAV\Activity\Provider\Calendar;
- use OCA\DAV\CalDAV\Activity\Provider\Event;
- use OCA\DAV\CalDAV\CalDavBackend;
- use OCP\Activity\IEvent;
- use OCP\Activity\IManager as IActivityManager;
- use OCP\App\IAppManager;
- use OCP\IGroup;
- use OCP\IGroupManager;
- use OCP\IUser;
- use OCP\IUserManager;
- use OCP\IUserSession;
- use Sabre\VObject\Reader;
- /**
- * Class Backend
- *
- * @package OCA\DAV\CalDAV\Activity
- */
- class Backend {
- /** @var IActivityManager */
- protected $activityManager;
- /** @var IGroupManager */
- protected $groupManager;
- /** @var IUserSession */
- protected $userSession;
- /** @var IAppManager */
- protected $appManager;
- /** @var IUserManager */
- protected $userManager;
- public function __construct(IActivityManager $activityManager, IGroupManager $groupManager, IUserSession $userSession, IAppManager $appManager, IUserManager $userManager) {
- $this->activityManager = $activityManager;
- $this->groupManager = $groupManager;
- $this->userSession = $userSession;
- $this->appManager = $appManager;
- $this->userManager = $userManager;
- }
- /**
- * Creates activities when a calendar was creates
- *
- * @param array $calendarData
- */
- public function onCalendarAdd(array $calendarData) {
- $this->triggerCalendarActivity(Calendar::SUBJECT_ADD, $calendarData);
- }
- /**
- * Creates activities when a calendar was updated
- *
- * @param array $calendarData
- * @param array $shares
- * @param array $properties
- */
- public function onCalendarUpdate(array $calendarData, array $shares, array $properties) {
- $this->triggerCalendarActivity(Calendar::SUBJECT_UPDATE, $calendarData, $shares, $properties);
- }
- /**
- * Creates activities when a calendar was moved to trash
- *
- * @param array $calendarData
- * @param array $shares
- */
- public function onCalendarMovedToTrash(array $calendarData, array $shares): void {
- $this->triggerCalendarActivity(Calendar::SUBJECT_MOVE_TO_TRASH, $calendarData, $shares);
- }
- /**
- * Creates activities when a calendar was restored
- *
- * @param array $calendarData
- * @param array $shares
- */
- public function onCalendarRestored(array $calendarData, array $shares): void {
- $this->triggerCalendarActivity(Calendar::SUBJECT_RESTORE, $calendarData, $shares);
- }
- /**
- * Creates activities when a calendar was deleted
- *
- * @param array $calendarData
- * @param array $shares
- */
- public function onCalendarDelete(array $calendarData, array $shares): void {
- $this->triggerCalendarActivity(Calendar::SUBJECT_DELETE, $calendarData, $shares);
- }
- /**
- * Creates activities when a calendar was (un)published
- *
- * @param array $calendarData
- * @param bool $publishStatus
- */
- public function onCalendarPublication(array $calendarData, bool $publishStatus): void {
- $this->triggerCalendarActivity($publishStatus ? Calendar::SUBJECT_PUBLISH : Calendar::SUBJECT_UNPUBLISH, $calendarData);
- }
- /**
- * Creates activities for all related users when a calendar was touched
- *
- * @param string $action
- * @param array $calendarData
- * @param array $shares
- * @param array $changedProperties
- */
- protected function triggerCalendarActivity($action, array $calendarData, array $shares = [], array $changedProperties = []) {
- if (!isset($calendarData['principaluri'])) {
- return;
- }
- $principal = explode('/', $calendarData['principaluri']);
- $owner = array_pop($principal);
- $currentUser = $this->userSession->getUser();
- if ($currentUser instanceof IUser) {
- $currentUser = $currentUser->getUID();
- } else {
- $currentUser = $owner;
- }
- $event = $this->activityManager->generateEvent();
- $event->setApp('dav')
- ->setObject('calendar', (int) $calendarData['id'])
- ->setType('calendar')
- ->setAuthor($currentUser);
- $changedVisibleInformation = array_intersect([
- '{DAV:}displayname',
- '{http://apple.com/ns/ical/}calendar-color'
- ], array_keys($changedProperties));
- if (empty($shares) || ($action === Calendar::SUBJECT_UPDATE && empty($changedVisibleInformation))) {
- $users = [$owner];
- } else {
- $users = $this->getUsersForShares($shares);
- $users[] = $owner;
- }
- foreach ($users as $user) {
- if ($action === Calendar::SUBJECT_DELETE && !$this->userManager->userExists($user)) {
- // Avoid creating calendar_delete activities for deleted users
- continue;
- }
- $event->setAffectedUser($user)
- ->setSubject(
- $user === $currentUser ? $action . '_self' : $action,
- [
- 'actor' => $currentUser,
- 'calendar' => [
- 'id' => (int) $calendarData['id'],
- 'uri' => $calendarData['uri'],
- 'name' => $calendarData['{DAV:}displayname'],
- ],
- ]
- );
- $this->activityManager->publish($event);
- }
- }
- /**
- * Creates activities for all related users when a calendar was (un-)shared
- *
- * @param array $calendarData
- * @param array $shares
- * @param array $add
- * @param array $remove
- */
- public function onCalendarUpdateShares(array $calendarData, array $shares, array $add, array $remove) {
- $principal = explode('/', $calendarData['principaluri']);
- $owner = $principal[2];
- $currentUser = $this->userSession->getUser();
- if ($currentUser instanceof IUser) {
- $currentUser = $currentUser->getUID();
- } else {
- $currentUser = $owner;
- }
- $event = $this->activityManager->generateEvent();
- $event->setApp('dav')
- ->setObject('calendar', (int) $calendarData['id'])
- ->setType('calendar')
- ->setAuthor($currentUser);
- foreach ($remove as $principal) {
- // principal:principals/users/test
- $parts = explode(':', $principal, 2);
- if ($parts[0] !== 'principal') {
- continue;
- }
- $principal = explode('/', $parts[1]);
- if ($principal[1] === 'users') {
- $this->triggerActivityUser(
- $principal[2],
- $event,
- $calendarData,
- Calendar::SUBJECT_UNSHARE_USER,
- Calendar::SUBJECT_DELETE . '_self'
- );
- if ($owner !== $principal[2]) {
- $parameters = [
- 'actor' => $event->getAuthor(),
- 'calendar' => [
- 'id' => (int) $calendarData['id'],
- 'uri' => $calendarData['uri'],
- 'name' => $calendarData['{DAV:}displayname'],
- ],
- 'user' => $principal[2],
- ];
- if ($owner === $event->getAuthor()) {
- $subject = Calendar::SUBJECT_UNSHARE_USER . '_you';
- } elseif ($principal[2] === $event->getAuthor()) {
- $subject = Calendar::SUBJECT_UNSHARE_USER . '_self';
- } else {
- $event->setAffectedUser($event->getAuthor())
- ->setSubject(Calendar::SUBJECT_UNSHARE_USER . '_you', $parameters);
- $this->activityManager->publish($event);
- $subject = Calendar::SUBJECT_UNSHARE_USER . '_by';
- }
- $event->setAffectedUser($owner)
- ->setSubject($subject, $parameters);
- $this->activityManager->publish($event);
- }
- } elseif ($principal[1] === 'groups') {
- $this->triggerActivityGroup($principal[2], $event, $calendarData, Calendar::SUBJECT_UNSHARE_USER);
- $parameters = [
- 'actor' => $event->getAuthor(),
- 'calendar' => [
- 'id' => (int) $calendarData['id'],
- 'uri' => $calendarData['uri'],
- 'name' => $calendarData['{DAV:}displayname'],
- ],
- 'group' => $principal[2],
- ];
- if ($owner === $event->getAuthor()) {
- $subject = Calendar::SUBJECT_UNSHARE_GROUP . '_you';
- } else {
- $event->setAffectedUser($event->getAuthor())
- ->setSubject(Calendar::SUBJECT_UNSHARE_GROUP . '_you', $parameters);
- $this->activityManager->publish($event);
- $subject = Calendar::SUBJECT_UNSHARE_GROUP . '_by';
- }
- $event->setAffectedUser($owner)
- ->setSubject($subject, $parameters);
- $this->activityManager->publish($event);
- }
- }
- foreach ($add as $share) {
- if ($this->isAlreadyShared($share['href'], $shares)) {
- continue;
- }
- // principal:principals/users/test
- $parts = explode(':', $share['href'], 2);
- if ($parts[0] !== 'principal') {
- continue;
- }
- $principal = explode('/', $parts[1]);
- if ($principal[1] === 'users') {
- $this->triggerActivityUser($principal[2], $event, $calendarData, Calendar::SUBJECT_SHARE_USER);
- if ($owner !== $principal[2]) {
- $parameters = [
- 'actor' => $event->getAuthor(),
- 'calendar' => [
- 'id' => (int) $calendarData['id'],
- 'uri' => $calendarData['uri'],
- 'name' => $calendarData['{DAV:}displayname'],
- ],
- 'user' => $principal[2],
- ];
- if ($owner === $event->getAuthor()) {
- $subject = Calendar::SUBJECT_SHARE_USER . '_you';
- } else {
- $event->setAffectedUser($event->getAuthor())
- ->setSubject(Calendar::SUBJECT_SHARE_USER . '_you', $parameters);
- $this->activityManager->publish($event);
- $subject = Calendar::SUBJECT_SHARE_USER . '_by';
- }
- $event->setAffectedUser($owner)
- ->setSubject($subject, $parameters);
- $this->activityManager->publish($event);
- }
- } elseif ($principal[1] === 'groups') {
- $this->triggerActivityGroup($principal[2], $event, $calendarData, Calendar::SUBJECT_SHARE_USER);
- $parameters = [
- 'actor' => $event->getAuthor(),
- 'calendar' => [
- 'id' => (int) $calendarData['id'],
- 'uri' => $calendarData['uri'],
- 'name' => $calendarData['{DAV:}displayname'],
- ],
- 'group' => $principal[2],
- ];
- if ($owner === $event->getAuthor()) {
- $subject = Calendar::SUBJECT_SHARE_GROUP . '_you';
- } else {
- $event->setAffectedUser($event->getAuthor())
- ->setSubject(Calendar::SUBJECT_SHARE_GROUP . '_you', $parameters);
- $this->activityManager->publish($event);
- $subject = Calendar::SUBJECT_SHARE_GROUP . '_by';
- }
- $event->setAffectedUser($owner)
- ->setSubject($subject, $parameters);
- $this->activityManager->publish($event);
- }
- }
- }
- /**
- * Checks if a calendar is already shared with a principal
- *
- * @param string $principal
- * @param array[] $shares
- * @return bool
- */
- protected function isAlreadyShared($principal, $shares) {
- foreach ($shares as $share) {
- if ($principal === $share['href']) {
- return true;
- }
- }
- return false;
- }
- /**
- * Creates the given activity for all members of the given group
- *
- * @param string $gid
- * @param IEvent $event
- * @param array $properties
- * @param string $subject
- */
- protected function triggerActivityGroup($gid, IEvent $event, array $properties, $subject) {
- $group = $this->groupManager->get($gid);
- if ($group instanceof IGroup) {
- foreach ($group->getUsers() as $user) {
- // Exclude current user
- if ($user->getUID() !== $event->getAuthor()) {
- $this->triggerActivityUser($user->getUID(), $event, $properties, $subject);
- }
- }
- }
- }
- /**
- * Creates the given activity for the given user
- *
- * @param string $user
- * @param IEvent $event
- * @param array $properties
- * @param string $subject
- * @param string $subjectSelf
- */
- protected function triggerActivityUser($user, IEvent $event, array $properties, $subject, $subjectSelf = '') {
- $event->setAffectedUser($user)
- ->setSubject(
- $user === $event->getAuthor() && $subjectSelf ? $subjectSelf : $subject,
- [
- 'actor' => $event->getAuthor(),
- 'calendar' => [
- 'id' => (int) $properties['id'],
- 'uri' => $properties['uri'],
- 'name' => $properties['{DAV:}displayname'],
- ],
- ]
- );
- $this->activityManager->publish($event);
- }
- /**
- * Creates activities when a calendar object was created/updated/deleted
- *
- * @param string $action
- * @param array $calendarData
- * @param array $shares
- * @param array $objectData
- */
- public function onTouchCalendarObject($action, array $calendarData, array $shares, array $objectData) {
- if (!isset($calendarData['principaluri'])) {
- return;
- }
- $principal = explode('/', $calendarData['principaluri']);
- $owner = array_pop($principal);
- $currentUser = $this->userSession->getUser();
- if ($currentUser instanceof IUser) {
- $currentUser = $currentUser->getUID();
- } else {
- $currentUser = $owner;
- }
- $classification = $objectData['classification'] ?? CalDavBackend::CLASSIFICATION_PUBLIC;
- $object = $this->getObjectNameAndType($objectData);
- if (!$object) {
- return;
- }
- $action = $action . '_' . $object['type'];
- if ($object['type'] === 'todo' && str_starts_with($action, Event::SUBJECT_OBJECT_UPDATE) && $object['status'] === 'COMPLETED') {
- $action .= '_completed';
- } elseif ($object['type'] === 'todo' && str_starts_with($action, Event::SUBJECT_OBJECT_UPDATE) && $object['status'] === 'NEEDS-ACTION') {
- $action .= '_needs_action';
- }
- $event = $this->activityManager->generateEvent();
- $event->setApp('dav')
- ->setObject('calendar', (int) $calendarData['id'])
- ->setType($object['type'] === 'event' ? 'calendar_event' : 'calendar_todo')
- ->setAuthor($currentUser);
- $users = $this->getUsersForShares($shares);
- $users[] = $owner;
- // Users for share can return the owner itself if the calendar is published
- foreach (array_unique($users) as $user) {
- if ($classification === CalDavBackend::CLASSIFICATION_PRIVATE && $user !== $owner) {
- // Private events are only shown to the owner
- continue;
- }
- $params = [
- 'actor' => $event->getAuthor(),
- 'calendar' => [
- 'id' => (int) $calendarData['id'],
- 'uri' => $calendarData['uri'],
- 'name' => $calendarData['{DAV:}displayname'],
- ],
- 'object' => [
- 'id' => $object['id'],
- 'name' => $classification === CalDavBackend::CLASSIFICATION_CONFIDENTIAL && $user !== $owner ? 'Busy' : $object['name'],
- 'classified' => $classification === CalDavBackend::CLASSIFICATION_CONFIDENTIAL && $user !== $owner,
- ],
- ];
- if ($object['type'] === 'event' && !str_contains($action, Event::SUBJECT_OBJECT_DELETE) && $this->appManager->isEnabledForUser('calendar')) {
- $params['object']['link']['object_uri'] = $objectData['uri'];
- $params['object']['link']['calendar_uri'] = $calendarData['uri'];
- $params['object']['link']['owner'] = $owner;
- }
- $event->setAffectedUser($user)
- ->setSubject(
- $user === $currentUser ? $action . '_self' : $action,
- $params
- );
- $this->activityManager->publish($event);
- }
- }
- /**
- * Creates activities when a calendar object was moved
- */
- public function onMovedCalendarObject(array $sourceCalendarData, array $targetCalendarData, array $sourceShares, array $targetShares, array $objectData): void {
- if (!isset($targetCalendarData['principaluri'])) {
- return;
- }
- $sourcePrincipal = explode('/', $sourceCalendarData['principaluri']);
- $sourceOwner = array_pop($sourcePrincipal);
- $targetPrincipal = explode('/', $targetCalendarData['principaluri']);
- $targetOwner = array_pop($targetPrincipal);
- if ($sourceOwner !== $targetOwner) {
- $this->onTouchCalendarObject(
- Event::SUBJECT_OBJECT_DELETE,
- $sourceCalendarData,
- $sourceShares,
- $objectData
- );
- $this->onTouchCalendarObject(
- Event::SUBJECT_OBJECT_ADD,
- $targetCalendarData,
- $targetShares,
- $objectData
- );
- return;
- }
- $currentUser = $this->userSession->getUser();
- if ($currentUser instanceof IUser) {
- $currentUser = $currentUser->getUID();
- } else {
- $currentUser = $targetOwner;
- }
- $classification = $objectData['classification'] ?? CalDavBackend::CLASSIFICATION_PUBLIC;
- $object = $this->getObjectNameAndType($objectData);
- if (!$object) {
- return;
- }
- $event = $this->activityManager->generateEvent();
- $event->setApp('dav')
- ->setObject('calendar', (int) $targetCalendarData['id'])
- ->setType($object['type'] === 'event' ? 'calendar_event' : 'calendar_todo')
- ->setAuthor($currentUser);
- $users = $this->getUsersForShares(array_intersect($sourceShares, $targetShares));
- $users[] = $targetOwner;
- // Users for share can return the owner itself if the calendar is published
- foreach (array_unique($users) as $user) {
- if ($classification === CalDavBackend::CLASSIFICATION_PRIVATE && $user !== $targetOwner) {
- // Private events are only shown to the owner
- continue;
- }
- $params = [
- 'actor' => $event->getAuthor(),
- 'sourceCalendar' => [
- 'id' => (int) $sourceCalendarData['id'],
- 'uri' => $sourceCalendarData['uri'],
- 'name' => $sourceCalendarData['{DAV:}displayname'],
- ],
- 'targetCalendar' => [
- 'id' => (int) $targetCalendarData['id'],
- 'uri' => $targetCalendarData['uri'],
- 'name' => $targetCalendarData['{DAV:}displayname'],
- ],
- 'object' => [
- 'id' => $object['id'],
- 'name' => $classification === CalDavBackend::CLASSIFICATION_CONFIDENTIAL && $user !== $targetOwner ? 'Busy' : $object['name'],
- 'classified' => $classification === CalDavBackend::CLASSIFICATION_CONFIDENTIAL && $user !== $targetOwner,
- ],
- ];
- if ($object['type'] === 'event' && $this->appManager->isEnabledForUser('calendar')) {
- $params['object']['link']['object_uri'] = $objectData['uri'];
- $params['object']['link']['calendar_uri'] = $targetCalendarData['uri'];
- $params['object']['link']['owner'] = $targetOwner;
- }
- $event->setAffectedUser($user)
- ->setSubject(
- $user === $currentUser ? Event::SUBJECT_OBJECT_MOVE . '_' . $object['type'] . '_self' : Event::SUBJECT_OBJECT_MOVE . '_' . $object['type'],
- $params
- );
- $this->activityManager->publish($event);
- }
- }
- /**
- * @param array $objectData
- * @return string[]|false
- */
- protected function getObjectNameAndType(array $objectData) {
- $vObject = Reader::read($objectData['calendardata']);
- $component = $componentType = null;
- foreach ($vObject->getComponents() as $component) {
- if (in_array($component->name, ['VEVENT', 'VTODO'])) {
- $componentType = $component->name;
- break;
- }
- }
- if (!$componentType) {
- // Calendar objects must have a VEVENT or VTODO component
- return false;
- }
- if ($componentType === 'VEVENT') {
- return ['id' => (string) $component->UID, 'name' => (string) $component->SUMMARY, 'type' => 'event'];
- }
- return ['id' => (string) $component->UID, 'name' => (string) $component->SUMMARY, 'type' => 'todo', 'status' => (string) $component->STATUS];
- }
- /**
- * Get all users that have access to a given calendar
- *
- * @param array $shares
- * @return string[]
- */
- protected function getUsersForShares(array $shares) {
- $users = $groups = [];
- foreach ($shares as $share) {
- $principal = explode('/', $share['{http://owncloud.org/ns}principal']);
- if ($principal[1] === 'users') {
- $users[] = $principal[2];
- } elseif ($principal[1] === 'groups') {
- $groups[] = $principal[2];
- }
- }
- if (!empty($groups)) {
- foreach ($groups as $gid) {
- $group = $this->groupManager->get($gid);
- if ($group instanceof IGroup) {
- foreach ($group->getUsers() as $user) {
- $users[] = $user->getUID();
- }
- }
- }
- }
- return array_unique($users);
- }
- }
|