You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
696 lines
23 KiB
696 lines
23 KiB
<?php
|
|
|
|
declare(strict_types=1);
|
|
/**
|
|
*
|
|
* @copyright Copyright (c) 2017, Daniel Calviño Sánchez (danxuliu@gmail.com)
|
|
*
|
|
* @license GNU AGPL version 3 or any later version
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU Affero General Public License as
|
|
* published by the Free Software Foundation, either version 3 of the
|
|
* License, or (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU Affero General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Affero General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*
|
|
*/
|
|
|
|
namespace OCA\Talk\Chat;
|
|
|
|
use OC\Memcache\ArrayCache;
|
|
use OC\Memcache\NullCache;
|
|
use OCA\Talk\Events\ChatEvent;
|
|
use OCA\Talk\Events\ChatParticipantEvent;
|
|
use OCA\Talk\Model\Attendee;
|
|
use OCA\Talk\Participant;
|
|
use OCA\Talk\Room;
|
|
use OCA\Talk\Service\AttachmentService;
|
|
use OCA\Talk\Service\ParticipantService;
|
|
use OCA\Talk\Service\PollService;
|
|
use OCA\Talk\Share\RoomShareProvider;
|
|
use OCP\AppFramework\Utility\ITimeFactory;
|
|
use OCP\Comments\IComment;
|
|
use OCP\Comments\ICommentsManager;
|
|
use OCP\Comments\NotFoundException;
|
|
use OCP\EventDispatcher\IEventDispatcher;
|
|
use OCP\ICache;
|
|
use OCP\ICacheFactory;
|
|
use OCP\IDBConnection;
|
|
use OCP\IUser;
|
|
use OCP\Notification\IManager as INotificationManager;
|
|
use OCP\Share\Exceptions\ShareNotFound;
|
|
use OCP\Share\IManager;
|
|
use OCP\Share\IShare;
|
|
|
|
/**
|
|
* Basic polling chat manager.
|
|
*
|
|
* sendMessage() saves a comment using the ICommentsManager, while
|
|
* receiveMessages() tries to read comments from ICommentsManager (with a little
|
|
* wait between reads) until comments are found or until the timeout expires.
|
|
*
|
|
* When a message is saved the mentioned users are notified as needed, and
|
|
* pending notifications are removed if the messages are deleted.
|
|
*/
|
|
class ChatManager {
|
|
public const EVENT_BEFORE_SYSTEM_MESSAGE_SEND = self::class . '::preSendSystemMessage';
|
|
public const EVENT_AFTER_SYSTEM_MESSAGE_SEND = self::class . '::postSendSystemMessage';
|
|
public const EVENT_AFTER_MULTIPLE_SYSTEM_MESSAGE_SEND = self::class . '::postSendMultipleSystemMessage';
|
|
public const EVENT_BEFORE_MESSAGE_SEND = self::class . '::preSendMessage';
|
|
public const EVENT_AFTER_MESSAGE_SEND = self::class . '::postSendMessage';
|
|
|
|
public const MAX_CHAT_LENGTH = 32000;
|
|
|
|
public const GEO_LOCATION_VALIDATOR = '/^geo:-?\d{1,2}(\.\d+)?,-?\d{1,3}(\.\d+)?(,-?\d+(\.\d+)?)?(;crs=wgs84)?(;u=\d+(\.\d+)?)?$/i';
|
|
public const VERB_MESSAGE = 'comment';
|
|
public const VERB_SYSTEM = 'system';
|
|
public const VERB_OBJECT_SHARED = 'object_shared';
|
|
public const VERB_COMMAND = 'command';
|
|
public const VERB_MESSAGE_DELETED = 'comment_deleted';
|
|
public const VERB_REACTION = 'reaction';
|
|
public const VERB_REACTION_DELETED = 'reaction_deleted';
|
|
|
|
/** @var ICommentsManager|CommentsManager
|
|
*/
|
|
private $commentsManager;
|
|
private IEventDispatcher $dispatcher;
|
|
private IDBConnection $connection;
|
|
private INotificationManager $notificationManager;
|
|
private IManager $shareManager;
|
|
private RoomShareProvider $shareProvider;
|
|
private ParticipantService $participantService;
|
|
private PollService $pollService;
|
|
private Notifier $notifier;
|
|
protected ITimeFactory $timeFactory;
|
|
protected ICache $cache;
|
|
protected ICache $unreadCountCache;
|
|
protected AttachmentService $attachmentService;
|
|
|
|
public function __construct(CommentsManager $commentsManager,
|
|
IEventDispatcher $dispatcher,
|
|
IDBConnection $connection,
|
|
INotificationManager $notificationManager,
|
|
IManager $shareManager,
|
|
RoomShareProvider $shareProvider,
|
|
ParticipantService $participantService,
|
|
PollService $pollService,
|
|
Notifier $notifier,
|
|
ICacheFactory $cacheFactory,
|
|
ITimeFactory $timeFactory,
|
|
AttachmentService $attachmentService) {
|
|
$this->commentsManager = $commentsManager;
|
|
$this->dispatcher = $dispatcher;
|
|
$this->connection = $connection;
|
|
$this->notificationManager = $notificationManager;
|
|
$this->shareManager = $shareManager;
|
|
$this->shareProvider = $shareProvider;
|
|
$this->participantService = $participantService;
|
|
$this->pollService = $pollService;
|
|
$this->notifier = $notifier;
|
|
$this->cache = $cacheFactory->createDistributed('talk/lastmsgid');
|
|
$this->unreadCountCache = $cacheFactory->createDistributed('talk/unreadcount');
|
|
$this->timeFactory = $timeFactory;
|
|
$this->attachmentService = $attachmentService;
|
|
}
|
|
|
|
/**
|
|
* Sends a new message to the given chat.
|
|
*
|
|
* @param Room $chat
|
|
* @param string $actorType
|
|
* @param string $actorId
|
|
* @param string $message
|
|
* @param \DateTime $creationDateTime
|
|
* @param bool $sendNotifications
|
|
* @param string|null $referenceId
|
|
* @param int|null $parentId
|
|
* @param bool $shouldSkipLastMessageUpdate If multiple messages will be posted
|
|
* (e.g. when adding multiple users to a room) we can skip the last
|
|
* message and last activity update until the last entry was created
|
|
* and then update with those values.
|
|
* This will replace O(n) with 1 database update.
|
|
* @return IComment
|
|
*/
|
|
public function addSystemMessage(
|
|
Room $chat,
|
|
string $actorType,
|
|
string $actorId,
|
|
string $message,
|
|
\DateTime $creationDateTime,
|
|
bool $sendNotifications,
|
|
?string $referenceId = null,
|
|
?int $parentId = null,
|
|
bool $shouldSkipLastMessageUpdate = false
|
|
): IComment {
|
|
$comment = $this->commentsManager->create($actorType, $actorId, 'chat', (string) $chat->getId());
|
|
$comment->setMessage($message, self::MAX_CHAT_LENGTH);
|
|
$comment->setCreationDateTime($creationDateTime);
|
|
if ($referenceId !== null) {
|
|
$referenceId = trim(substr($referenceId, 0, 64));
|
|
if ($referenceId !== '') {
|
|
$comment->setReferenceId($referenceId);
|
|
}
|
|
}
|
|
if ($parentId !== null) {
|
|
$comment->setParentId((string) $parentId);
|
|
}
|
|
|
|
$messageDecoded = json_decode($message, true);
|
|
$messageType = $messageDecoded['message'] ?? '';
|
|
|
|
if ($messageType === 'object_shared' || $messageType === 'file_shared') {
|
|
$comment->setVerb(self::VERB_OBJECT_SHARED);
|
|
} else {
|
|
$comment->setVerb(self::VERB_SYSTEM);
|
|
}
|
|
|
|
$event = new ChatEvent($chat, $comment, $shouldSkipLastMessageUpdate);
|
|
$this->dispatcher->dispatch(self::EVENT_BEFORE_SYSTEM_MESSAGE_SEND, $event);
|
|
try {
|
|
$this->commentsManager->save($comment);
|
|
|
|
if (!$shouldSkipLastMessageUpdate) {
|
|
// Update last_message
|
|
$chat->setLastMessage($comment);
|
|
$this->unreadCountCache->clear($chat->getId() . '-');
|
|
}
|
|
|
|
if ($sendNotifications) {
|
|
$this->notifier->notifyOtherParticipant($chat, $comment, [], false);
|
|
}
|
|
|
|
$this->dispatcher->dispatch(self::EVENT_AFTER_SYSTEM_MESSAGE_SEND, $event);
|
|
} catch (NotFoundException $e) {
|
|
}
|
|
$this->cache->remove($chat->getToken());
|
|
|
|
if ($messageType === 'object_shared' || $messageType === 'file_shared') {
|
|
$this->attachmentService->createAttachmentEntry($chat, $comment, $messageType, $messageDecoded['parameters'] ?? []);
|
|
}
|
|
|
|
return $comment;
|
|
}
|
|
|
|
/**
|
|
* Sends a new message to the given chat.
|
|
*
|
|
* @param Room $chat
|
|
* @param string $message
|
|
* @return IComment
|
|
*/
|
|
public function addChangelogMessage(Room $chat, string $message): IComment {
|
|
$comment = $this->commentsManager->create(Attendee::ACTOR_GUESTS, 'changelog', 'chat', (string) $chat->getId());
|
|
|
|
$comment->setMessage($message, self::MAX_CHAT_LENGTH);
|
|
$comment->setCreationDateTime($this->timeFactory->getDateTime());
|
|
$comment->setVerb(self::VERB_MESSAGE); // Has to be comment, so it counts as unread message
|
|
|
|
$event = new ChatEvent($chat, $comment);
|
|
$this->dispatcher->dispatch(self::EVENT_BEFORE_SYSTEM_MESSAGE_SEND, $event);
|
|
try {
|
|
$this->commentsManager->save($comment);
|
|
|
|
// Update last_message
|
|
$chat->setLastMessage($comment);
|
|
$this->unreadCountCache->clear($chat->getId() . '-');
|
|
|
|
$this->dispatcher->dispatch(self::EVENT_AFTER_SYSTEM_MESSAGE_SEND, $event);
|
|
} catch (NotFoundException $e) {
|
|
}
|
|
$this->cache->remove($chat->getToken());
|
|
|
|
return $comment;
|
|
}
|
|
|
|
/**
|
|
* Sends a new message to the given chat.
|
|
*
|
|
* @param Room $chat
|
|
* @param Participant $participant
|
|
* @param string $actorType
|
|
* @param string $actorId
|
|
* @param string $message
|
|
* @param \DateTime $creationDateTime
|
|
* @param IComment|null $replyTo
|
|
* @param string $referenceId
|
|
* @return IComment
|
|
*/
|
|
public function sendMessage(Room $chat, Participant $participant, string $actorType, string $actorId, string $message, \DateTime $creationDateTime, ?IComment $replyTo, string $referenceId, bool $silent): IComment {
|
|
$comment = $this->commentsManager->create($actorType, $actorId, 'chat', (string) $chat->getId());
|
|
$comment->setMessage($message, self::MAX_CHAT_LENGTH);
|
|
$comment->setCreationDateTime($creationDateTime);
|
|
// A verb ('comment', 'like'...) must be provided to be able to save a
|
|
// comment
|
|
$comment->setVerb(self::VERB_MESSAGE);
|
|
|
|
if ($replyTo instanceof IComment) {
|
|
$comment->setParentId($replyTo->getId());
|
|
}
|
|
|
|
$referenceId = trim(substr($referenceId, 0, 64));
|
|
if ($referenceId !== '') {
|
|
$comment->setReferenceId($referenceId);
|
|
}
|
|
|
|
$event = new ChatParticipantEvent($chat, $comment, $participant, $silent);
|
|
$this->dispatcher->dispatch(self::EVENT_BEFORE_MESSAGE_SEND, $event);
|
|
|
|
$shouldFlush = $this->notificationManager->defer();
|
|
try {
|
|
$this->commentsManager->save($comment);
|
|
|
|
// Update last_message
|
|
if ($comment->getActorType() !== 'bots' || $comment->getActorId() === 'changelog') {
|
|
$chat->setLastMessage($comment);
|
|
$this->unreadCountCache->clear($chat->getId() . '-');
|
|
} else {
|
|
$chat->setLastActivity($comment->getCreationDateTime());
|
|
}
|
|
|
|
$alreadyNotifiedUsers = [];
|
|
$usersDirectlyMentioned = $this->notifier->getMentionedUserIds($comment);
|
|
if ($replyTo instanceof IComment) {
|
|
$alreadyNotifiedUsers = $this->notifier->notifyReplyToAuthor($chat, $comment, $replyTo, $silent);
|
|
if ($replyTo->getActorType() === Attendee::ACTOR_USERS) {
|
|
$usersDirectlyMentioned[] = $replyTo->getActorId();
|
|
}
|
|
}
|
|
|
|
$alreadyNotifiedUsers = $this->notifier->notifyMentionedUsers($chat, $comment, $alreadyNotifiedUsers, $silent);
|
|
if (!empty($alreadyNotifiedUsers)) {
|
|
$userIds = array_column($alreadyNotifiedUsers, 'id');
|
|
$this->participantService->markUsersAsMentioned($chat, $userIds, (int) $comment->getId(), $usersDirectlyMentioned);
|
|
}
|
|
|
|
// User was not mentioned, send a normal notification
|
|
$this->notifier->notifyOtherParticipant($chat, $comment, $alreadyNotifiedUsers, $silent);
|
|
|
|
$this->dispatcher->dispatch(self::EVENT_AFTER_MESSAGE_SEND, $event);
|
|
} catch (NotFoundException $e) {
|
|
}
|
|
$this->cache->remove($chat->getToken());
|
|
if ($shouldFlush) {
|
|
$this->notificationManager->flush();
|
|
}
|
|
|
|
return $comment;
|
|
}
|
|
|
|
/**
|
|
* @param Room $room
|
|
* @param Participant $participant
|
|
* @param array $messageData
|
|
* @throws ShareNotFound
|
|
*/
|
|
public function unshareFileOnMessageDelete(Room $room, Participant $participant, array $messageData): void {
|
|
if (!isset($messageData['message'], $messageData['parameters']['share']) || $messageData['message'] !== 'file_shared') {
|
|
// Not a file share
|
|
return;
|
|
}
|
|
|
|
$share = $this->shareManager->getShareById('ocRoomShare:' . $messageData['parameters']['share']);
|
|
|
|
if ($share->getShareType() !== IShare::TYPE_ROOM || $share->getSharedWith() !== $room->getToken()) {
|
|
// Share does not match the correct room
|
|
throw new ShareNotFound();
|
|
}
|
|
|
|
$attendee = $participant->getAttendee();
|
|
|
|
if (!$participant->hasModeratorPermissions() &&
|
|
!($attendee->getActorType() === Attendee::ACTOR_USERS && $attendee->getActorId() === $share->getShareOwner())) {
|
|
// Only moderators or the share owner can delete the share
|
|
return;
|
|
}
|
|
|
|
$this->shareManager->deleteShare($share);
|
|
}
|
|
|
|
/**
|
|
* @param Room $chat
|
|
* @param IComment $comment
|
|
* @param Participant $participant
|
|
* @param \DateTime $deletionTime
|
|
* @return IComment
|
|
* @throws ShareNotFound
|
|
*/
|
|
public function deleteMessage(Room $chat, IComment $comment, Participant $participant, \DateTime $deletionTime): IComment {
|
|
if ($comment->getVerb() === self::VERB_OBJECT_SHARED) {
|
|
$messageData = json_decode($comment->getMessage(), true);
|
|
$this->unshareFileOnMessageDelete($chat, $participant, $messageData);
|
|
}
|
|
|
|
$comment->setMessage(
|
|
json_encode([
|
|
'deleted_by_type' => $participant->getAttendee()->getActorType(),
|
|
'deleted_by_id' => $participant->getAttendee()->getActorId(),
|
|
'deleted_on' => $deletionTime->getTimestamp(),
|
|
])
|
|
);
|
|
$comment->setVerb(self::VERB_MESSAGE_DELETED);
|
|
$this->commentsManager->save($comment);
|
|
|
|
$this->attachmentService->deleteAttachmentByMessageId((int) $comment->getId());
|
|
|
|
return $this->addSystemMessage(
|
|
$chat,
|
|
$participant->getAttendee()->getActorType(),
|
|
$participant->getAttendee()->getActorId(),
|
|
json_encode(['message' => 'message_deleted', 'parameters' => ['message' => $comment->getId()]]),
|
|
$this->timeFactory->getDateTime(),
|
|
false,
|
|
null,
|
|
(int) $comment->getId(),
|
|
true
|
|
);
|
|
}
|
|
|
|
public function clearHistory(Room $chat, string $actorType, string $actorId): IComment {
|
|
$this->commentsManager->deleteCommentsAtObject('chat', (string) $chat->getId());
|
|
|
|
$this->shareProvider->deleteInRoom($chat->getToken());
|
|
|
|
$this->notifier->removePendingNotificationsForRoom($chat, true);
|
|
|
|
$this->participantService->resetChatDetails($chat);
|
|
|
|
return $this->addSystemMessage(
|
|
$chat,
|
|
$actorType,
|
|
$actorId,
|
|
json_encode(['message' => 'history_cleared', 'parameters' => []]),
|
|
$this->timeFactory->getDateTime(),
|
|
false
|
|
);
|
|
}
|
|
|
|
/**
|
|
* @param Room $chat
|
|
* @param string $parentId
|
|
* @return IComment
|
|
* @throws NotFoundException
|
|
*/
|
|
public function getParentComment(Room $chat, string $parentId): IComment {
|
|
$comment = $this->commentsManager->get($parentId);
|
|
|
|
if ($comment->getObjectType() !== 'chat' || $comment->getObjectId() !== (string) $chat->getId()) {
|
|
throw new NotFoundException('Parent not found in the right context');
|
|
}
|
|
|
|
return $comment;
|
|
}
|
|
|
|
/**
|
|
* @param Room $chat
|
|
* @param string $messageId
|
|
* @return IComment
|
|
* @throws NotFoundException
|
|
*/
|
|
public function getComment(Room $chat, string $messageId): IComment {
|
|
$comment = $this->commentsManager->get($messageId);
|
|
|
|
if ($comment->getObjectType() !== 'chat' || $comment->getObjectId() !== (string) $chat->getId()) {
|
|
throw new NotFoundException('Message not found in the right context');
|
|
}
|
|
|
|
return $comment;
|
|
}
|
|
|
|
public function getLastReadMessageFromLegacy(Room $chat, IUser $user): int {
|
|
$marker = $this->commentsManager->getReadMark('chat', $chat->getId(), $user);
|
|
if ($marker === null) {
|
|
return 0;
|
|
}
|
|
|
|
return $this->commentsManager->getLastCommentBeforeDate('chat', (string) $chat->getId(), $marker, self::VERB_MESSAGE);
|
|
}
|
|
|
|
public function getUnreadCount(Room $chat, int $lastReadMessage): int {
|
|
/**
|
|
* for a given message id $lastReadMessage we cache the number of messages
|
|
* that exist past that message, which happen to also be the number of
|
|
* unread messages, because this is expensive to query per room and user repeatedly
|
|
*/
|
|
$key = $chat->getId() . '-' . $lastReadMessage;
|
|
$unreadCount = $this->unreadCountCache->get($key);
|
|
if ($unreadCount === null) {
|
|
$unreadCount = $this->commentsManager->getNumberOfCommentsWithVerbsForObjectSinceComment('chat', (string) $chat->getId(), $lastReadMessage, [self::VERB_MESSAGE, 'object_shared']);
|
|
$this->unreadCountCache->set($key, $unreadCount, 1800);
|
|
}
|
|
return $unreadCount;
|
|
}
|
|
|
|
/**
|
|
* Returns the ID of the last chat message, that was read by everyone
|
|
* sharing their read status.
|
|
*
|
|
* @param Room $chat
|
|
* @return int
|
|
*/
|
|
public function getLastCommonReadMessage(Room $chat): int {
|
|
return $this->participantService->getLastCommonReadChatMessage($chat);
|
|
}
|
|
|
|
/**
|
|
* Receive the history of a chat
|
|
*
|
|
* @param Room $chat
|
|
* @param int $offset Last known message id
|
|
* @param int $limit
|
|
* @param bool $includeLastKnown
|
|
* @return IComment[] the messages found (only the id, actor type and id,
|
|
* creation date and message are relevant), or an empty array if the
|
|
* timeout expired.
|
|
*/
|
|
public function getHistory(Room $chat, int $offset, int $limit, bool $includeLastKnown): array {
|
|
return $this->commentsManager->getForObjectSince('chat', (string) $chat->getId(), $offset, 'desc', $limit, $includeLastKnown);
|
|
}
|
|
|
|
/**
|
|
* @param Room $chat
|
|
* @param int $offset Last known message id
|
|
* @param array $verbs
|
|
* @param bool $offsetIsVerbMatch
|
|
* @return IComment
|
|
* @throws NotFoundException
|
|
*/
|
|
public function getPreviousMessageWithVerb(Room $chat, int $offset, array $verbs, bool $offsetIsVerbMatch): IComment {
|
|
$messages = $this->commentsManager->getCommentsWithVerbForObjectSinceComment(
|
|
'chat',
|
|
(string) $chat->getId(),
|
|
$verbs,
|
|
$offset,
|
|
'desc',
|
|
!$offsetIsVerbMatch ? 2 : 1
|
|
);
|
|
|
|
if (empty($messages)) {
|
|
throw new NotFoundException('No comment with verb found');
|
|
}
|
|
|
|
return array_pop($messages);
|
|
}
|
|
|
|
/**
|
|
* If there are currently no messages the response will not be sent
|
|
* immediately. Instead, HTTP connection will be kept open waiting for new
|
|
* messages to arrive and, when they do, then the response will be sent. The
|
|
* connection will not be kept open indefinitely, though; the number of
|
|
* seconds to wait for new messages to arrive can be set using the timeout
|
|
* parameter; the default timeout is 30 seconds, maximum timeout is 60
|
|
* seconds. If the timeout ends a successful but empty response will be
|
|
* sent.
|
|
*
|
|
* @param Room $chat
|
|
* @param int $offset Last known message id
|
|
* @param int $limit
|
|
* @param int $timeout
|
|
* @param IUser|null $user
|
|
* @param bool $includeLastKnown
|
|
* @return IComment[] the messages found (only the id, actor type and id,
|
|
* creation date and message are relevant), or an empty array if the
|
|
* timeout expired.
|
|
*/
|
|
public function waitForNewMessages(Room $chat, int $offset, int $limit, int $timeout, ?IUser $user, bool $includeLastKnown): array {
|
|
if ($user instanceof IUser) {
|
|
$this->notifier->markMentionNotificationsRead($chat, $user->getUID());
|
|
}
|
|
|
|
if ($this->cache instanceof NullCache
|
|
|| $this->cache instanceof ArrayCache) {
|
|
return $this->waitForNewMessagesWithDatabase($chat, $offset, $limit, $timeout, $includeLastKnown);
|
|
}
|
|
|
|
return $this->waitForNewMessagesWithCache($chat, $offset, $limit, $timeout, $includeLastKnown);
|
|
}
|
|
|
|
/**
|
|
* Check the cache until we found new messages, or the timeout was reached
|
|
*
|
|
* @param Room $chat
|
|
* @param int $offset
|
|
* @param int $limit
|
|
* @param int $timeout
|
|
* @param bool $includeLastKnown
|
|
* @return IComment[]
|
|
*/
|
|
protected function waitForNewMessagesWithCache(Room $chat, int $offset, int $limit, int $timeout, bool $includeLastKnown): array {
|
|
$elapsedTime = 0;
|
|
|
|
$comments = $this->checkCacheOrDatabase($chat, $offset, $limit, $includeLastKnown);
|
|
|
|
while (empty($comments) && $elapsedTime < $timeout) {
|
|
$this->connection->close();
|
|
sleep(1);
|
|
$elapsedTime++;
|
|
|
|
$comments = $this->checkCacheOrDatabase($chat, $offset, $limit, $includeLastKnown);
|
|
}
|
|
|
|
return $comments;
|
|
}
|
|
|
|
/**
|
|
* Check the cache for the last message id or check the database for updates
|
|
*
|
|
* @param Room $chat
|
|
* @param int $offset
|
|
* @param int $limit
|
|
* @param bool $includeLastKnown
|
|
* @return IComment[]
|
|
*/
|
|
protected function checkCacheOrDatabase(Room $chat, int $offset, int $limit, bool $includeLastKnown): array {
|
|
$cachedId = $this->cache->get($chat->getToken());
|
|
if ($offset === $cachedId) {
|
|
// Cache hit, nothing new ¯\_(ツ)_/¯
|
|
return [];
|
|
}
|
|
|
|
// Load data from the database
|
|
$comments = $this->commentsManager->getForObjectSince('chat', (string) $chat->getId(), $offset, 'asc', $limit, $includeLastKnown);
|
|
|
|
if (empty($comments)) {
|
|
// We only write the cache when there were no new comments,
|
|
// otherwise it could happen that this is not the last message,
|
|
// but the last within $limit
|
|
$this->cache->set($chat->getToken(), $offset, 30);
|
|
return [];
|
|
}
|
|
|
|
return $comments;
|
|
}
|
|
|
|
/**
|
|
* Check the database for new messages until there a new messages or we exceeded the timeout
|
|
*
|
|
* @param Room $chat
|
|
* @param int $offset
|
|
* @param int $limit
|
|
* @param int $timeout
|
|
* @param bool $includeLastKnown
|
|
* @return array
|
|
*/
|
|
protected function waitForNewMessagesWithDatabase(Room $chat, int $offset, int $limit, int $timeout, bool $includeLastKnown): array {
|
|
$elapsedTime = 0;
|
|
|
|
$comments = $this->commentsManager->getForObjectSince('chat', (string) $chat->getId(), $offset, 'asc', $limit, $includeLastKnown);
|
|
|
|
while (empty($comments) && $elapsedTime < $timeout) {
|
|
sleep(1);
|
|
$elapsedTime++;
|
|
|
|
$comments = $this->commentsManager->getForObjectSince('chat', (string) $chat->getId(), $offset, 'asc', $limit, $includeLastKnown);
|
|
}
|
|
|
|
return $comments;
|
|
}
|
|
|
|
/**
|
|
* Deletes all the messages for the given chat.
|
|
*
|
|
* @param Room $chat
|
|
*/
|
|
public function deleteMessages(Room $chat): void {
|
|
$this->commentsManager->deleteCommentsAtObject('chat', (string) $chat->getId());
|
|
|
|
$this->shareProvider->deleteInRoom($chat->getToken());
|
|
|
|
$this->notifier->removePendingNotificationsForRoom($chat);
|
|
|
|
$this->attachmentService->deleteAttachmentsForRoom($chat);
|
|
|
|
$this->pollService->deleteByRoomId($chat->getId());
|
|
}
|
|
|
|
/**
|
|
* Search for comments with a given content
|
|
*
|
|
* @param Room $chat
|
|
* @param int[] $commentIds
|
|
* @return IComment[]
|
|
*/
|
|
public function getMessagesById(Room $chat, array $commentIds): array {
|
|
$comments = $this->commentsManager->getCommentsById($commentIds);
|
|
|
|
$comments = array_filter($comments, static function (IComment $comment) use ($chat) {
|
|
return $comment->getObjectType() === 'chat'
|
|
&& (int)$comment->getObjectId() === $chat->getId();
|
|
});
|
|
|
|
return $comments;
|
|
}
|
|
|
|
/**
|
|
* Search for comments with a given content
|
|
*
|
|
* @param string $search content to search for
|
|
* @param array $objectIds Limit the search by object ids
|
|
* @param string $verb Limit the verb of the comment
|
|
* @param int $offset
|
|
* @param int $limit
|
|
* @return IComment[]
|
|
*/
|
|
public function searchForObjects(string $search, array $objectIds, string $verb = '', int $offset = 0, int $limit = 50): array {
|
|
return $this->commentsManager->searchForObjects($search, 'chat', $objectIds, $verb, $offset, $limit);
|
|
}
|
|
|
|
public function addConversationNotify(array $results, string $search, Room $room, Participant $participant): array {
|
|
if ($room->getType() === Room::TYPE_ONE_TO_ONE) {
|
|
return $results;
|
|
}
|
|
$attendee = $participant->getAttendee();
|
|
if ($attendee->getActorType() === Attendee::ACTOR_USERS) {
|
|
$roomDisplayName = $room->getDisplayName($attendee->getActorId());
|
|
} else {
|
|
$roomDisplayName = $room->getDisplayName('');
|
|
}
|
|
if ($search === '' || $this->searchIsPartOfConversationNameOrAtAll($search, $roomDisplayName)) {
|
|
array_unshift($results, [
|
|
'id' => 'all',
|
|
'label' => $roomDisplayName,
|
|
'source' => 'calls'
|
|
]);
|
|
}
|
|
return $results;
|
|
}
|
|
|
|
private function searchIsPartOfConversationNameOrAtAll(string $search, string $roomDisplayName): bool {
|
|
if (stripos($roomDisplayName, $search) === 0) {
|
|
return true;
|
|
}
|
|
if (strpos('all', $search) === 0) {
|
|
return true;
|
|
}
|
|
if (strpos('here', $search) === 0) {
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
}
|