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.
528 lines
17 KiB
528 lines
17 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\ParticipantService;
|
|
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;
|
|
|
|
/**
|
|
* 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_BEFORE_MESSAGE_SEND = self::class . '::preSendMessage';
|
|
public const EVENT_AFTER_MESSAGE_SEND = self::class . '::postSendMessage';
|
|
|
|
public const MAX_CHAT_LENGTH = 32000;
|
|
|
|
/** @var ICommentsManager */
|
|
private $commentsManager;
|
|
/** @var IEventDispatcher */
|
|
private $dispatcher;
|
|
/** @var IDBConnection */
|
|
private $connection;
|
|
/** @var INotificationManager */
|
|
private $notificationManager;
|
|
/** @var RoomShareProvider */
|
|
private $shareProvider;
|
|
/** @var ParticipantService */
|
|
private $participantService;
|
|
/** @var Notifier */
|
|
private $notifier;
|
|
/** @var ITimeFactory */
|
|
protected $timeFactory;
|
|
/** @var ICache */
|
|
protected $cache;
|
|
/** @var ICache */
|
|
protected $unreadCountCache;
|
|
|
|
public function __construct(CommentsManager $commentsManager,
|
|
IEventDispatcher $dispatcher,
|
|
IDBConnection $connection,
|
|
INotificationManager $notificationManager,
|
|
RoomShareProvider $shareProvider,
|
|
ParticipantService $participantService,
|
|
Notifier $notifier,
|
|
ICacheFactory $cacheFactory,
|
|
ITimeFactory $timeFactory) {
|
|
$this->commentsManager = $commentsManager;
|
|
$this->dispatcher = $dispatcher;
|
|
$this->connection = $connection;
|
|
$this->notificationManager = $notificationManager;
|
|
$this->shareProvider = $shareProvider;
|
|
$this->participantService = $participantService;
|
|
$this->notifier = $notifier;
|
|
$this->cache = $cacheFactory->createDistributed('talk/lastmsgid');
|
|
$this->unreadCountCache = $cacheFactory->createDistributed('talk/unreadcount');
|
|
$this->timeFactory = $timeFactory;
|
|
}
|
|
|
|
/**
|
|
* 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
|
|
* @return IComment
|
|
*/
|
|
public function addSystemMessage(
|
|
Room $chat,
|
|
string $actorType,
|
|
string $actorId,
|
|
string $message,
|
|
\DateTime $creationDateTime,
|
|
bool $sendNotifications,
|
|
?string $referenceId = null,
|
|
?int $parentId = null
|
|
): 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, 40));
|
|
if ($referenceId !== '') {
|
|
$comment->setReferenceId($referenceId);
|
|
}
|
|
}
|
|
if ($parentId !== null) {
|
|
$comment->setParentId((string) $parentId);
|
|
}
|
|
$comment->setVerb('system');
|
|
|
|
$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() . '-');
|
|
|
|
if ($sendNotifications) {
|
|
$this->notifier->notifyOtherParticipant($chat, $comment, []);
|
|
}
|
|
|
|
$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 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('comment'); // 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): 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('comment');
|
|
|
|
if ($replyTo instanceof IComment) {
|
|
$comment->setParentId($replyTo->getId());
|
|
}
|
|
|
|
$referenceId = trim(substr($referenceId, 0, 40));
|
|
if ($referenceId !== '') {
|
|
$comment->setReferenceId($referenceId);
|
|
}
|
|
|
|
$event = new ChatParticipantEvent($chat, $comment, $participant);
|
|
$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() . '-');
|
|
}
|
|
|
|
$alreadyNotifiedUsers = [];
|
|
if ($replyTo instanceof IComment) {
|
|
$alreadyNotifiedUsers = $this->notifier->notifyReplyToAuthor($chat, $comment, $replyTo);
|
|
}
|
|
|
|
$alreadyNotifiedUsers = $this->notifier->notifyMentionedUsers($chat, $comment, $alreadyNotifiedUsers);
|
|
if (!empty($alreadyNotifiedUsers)) {
|
|
$this->participantService->markUsersAsMentioned($chat, $alreadyNotifiedUsers, (int) $comment->getId());
|
|
}
|
|
|
|
// User was not mentioned, send a normal notification
|
|
$this->notifier->notifyOtherParticipant($chat, $comment, $alreadyNotifiedUsers);
|
|
|
|
$this->dispatcher->dispatch(self::EVENT_AFTER_MESSAGE_SEND, $event);
|
|
} catch (NotFoundException $e) {
|
|
}
|
|
$this->cache->remove($chat->getToken());
|
|
if ($shouldFlush) {
|
|
$this->notificationManager->flush();
|
|
}
|
|
|
|
return $comment;
|
|
}
|
|
|
|
public function deleteMessage(Room $chat, int $messageId, string $actorType, string $actorId, \DateTime $deletionTime): IComment {
|
|
$comment = $this->getComment($chat, (string) $messageId);
|
|
$comment->setMessage(
|
|
json_encode([
|
|
'deleted_by_type' => $actorType,
|
|
'deleted_by_id' => $actorId,
|
|
'deleted_on' => $deletionTime->getTimestamp(),
|
|
])
|
|
);
|
|
$comment->setVerb('comment_deleted');
|
|
$this->commentsManager->save($comment);
|
|
|
|
return $this->addSystemMessage(
|
|
$chat,
|
|
$actorType,
|
|
$actorId,
|
|
json_encode(['message' => 'message_deleted', 'parameters' => ['message' => $messageId]]),
|
|
$this->timeFactory->getDateTime(),
|
|
false,
|
|
null,
|
|
$messageId
|
|
);
|
|
}
|
|
|
|
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, 'comment');
|
|
}
|
|
|
|
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->getNumberOfCommentsForObjectSinceComment('chat', (string) $chat->getId(), $lastReadMessage, 'comment');
|
|
$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);
|
|
}
|
|
|
|
/**
|
|
* 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);
|
|
}
|
|
|
|
/**
|
|
* 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);
|
|
}
|
|
}
|