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.
289 lines
8.1 KiB
289 lines
8.1 KiB
<?php
|
|
|
|
declare(strict_types=1);
|
|
/**
|
|
* @copyright Copyright (c) 2020 Joas Schilling <coding@schilljs.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\Search;
|
|
|
|
use OCA\Talk\AppInfo\Application;
|
|
use OCA\Talk\Chat\ChatManager;
|
|
use OCA\Talk\Chat\MessageParser;
|
|
use OCA\Talk\Exceptions\ParticipantNotFoundException;
|
|
use OCA\Talk\Exceptions\UnauthorizedException;
|
|
use OCA\Talk\Manager as RoomManager;
|
|
use OCA\Talk\Model\Attendee;
|
|
use OCA\Talk\Room;
|
|
use OCA\Talk\Service\ParticipantService;
|
|
use OCA\Talk\Webinary;
|
|
use OCP\AppFramework\Utility\ITimeFactory;
|
|
use OCP\Comments\IComment;
|
|
use OCP\IL10N;
|
|
use OCP\IURLGenerator;
|
|
use OCP\IUser;
|
|
use OCP\Search\IFilter;
|
|
use OCP\Search\IFilteringProvider;
|
|
use OCP\Search\IProvider;
|
|
use OCP\Search\ISearchQuery;
|
|
use OCP\Search\SearchResult;
|
|
use OCP\Search\SearchResultEntry;
|
|
|
|
class MessageSearch implements IProvider, IFilteringProvider {
|
|
|
|
public function __construct(
|
|
protected RoomManager $roomManager,
|
|
protected ParticipantService $participantService,
|
|
protected ChatManager $chatManager,
|
|
protected MessageParser $messageParser,
|
|
protected ITimeFactory $timeFactory,
|
|
protected IURLGenerator $url,
|
|
protected IL10N $l,
|
|
) {
|
|
}
|
|
|
|
/**
|
|
* @inheritDoc
|
|
*/
|
|
public function getId(): string {
|
|
return 'talk-message';
|
|
}
|
|
|
|
/**
|
|
* @inheritDoc
|
|
*/
|
|
public function getName(): string {
|
|
return $this->l->t('Messages');
|
|
}
|
|
|
|
/**
|
|
* @inheritDoc
|
|
*/
|
|
public function getOrder(string $route, array $routeParameters): ?int {
|
|
if (str_starts_with($route, Application::APP_ID . '.')) {
|
|
// Active app, prefer Talk results
|
|
return -2;
|
|
}
|
|
|
|
return 15;
|
|
}
|
|
|
|
protected function getCurrentConversationToken(ISearchQuery $query): string {
|
|
if ($query->getRoute() === 'spreed.Page.showCall') {
|
|
return $query->getRouteParameters()['token'];
|
|
}
|
|
return '';
|
|
}
|
|
|
|
protected function getSublineTemplate(): string {
|
|
return $this->l->t('{user} in {conversation}');
|
|
}
|
|
|
|
/**
|
|
* @inheritDoc
|
|
*/
|
|
public function search(IUser $user, ISearchQuery $query): SearchResult {
|
|
$title = $this->l->t('Messages');
|
|
if ($this->getCurrentConversationToken($query) !== '') {
|
|
$title = $this->l->t('Messages in other conversations');
|
|
}
|
|
|
|
$rooms = $this->roomManager->getRoomsForUser($user->getUID());
|
|
return $this->performSearch($user, $query, $title, $rooms);
|
|
}
|
|
|
|
/**
|
|
* @param Room[] $rooms
|
|
*/
|
|
public function performSearch(IUser $user, ISearchQuery $query, string $title, array $rooms, bool $isCurrentMessageSearch = false): SearchResult {
|
|
$roomMap = [];
|
|
foreach ($rooms as $room) {
|
|
if (!$isCurrentMessageSearch &&
|
|
$room->getType() === Room::TYPE_CHANGELOG) {
|
|
continue;
|
|
}
|
|
|
|
if (!$isCurrentMessageSearch &&
|
|
$this->getCurrentConversationToken($query) === $room->getToken()) {
|
|
// No search result from current conversation
|
|
continue;
|
|
}
|
|
|
|
if ($room->getLobbyState() !== Webinary::LOBBY_NONE) {
|
|
$participant = $this->participantService->getParticipant($room, $user->getUID(), false);
|
|
if (!($participant->getPermissions() & Attendee::PERMISSIONS_LOBBY_IGNORE)) {
|
|
continue;
|
|
}
|
|
}
|
|
|
|
$roomMap[(string) $room->getId()] = $room;
|
|
}
|
|
|
|
if (empty($roomMap)) {
|
|
return SearchResult::complete($title, []);
|
|
}
|
|
|
|
// Apply filters when available
|
|
$lowerTimeBoundary = $upperTimeBoundary = $actorType = $actorId = null;
|
|
if ($since = $query->getFilter(IFilter::BUILTIN_SINCE)?->get()) {
|
|
if ($since instanceof \DateTimeImmutable) {
|
|
$lowerTimeBoundary = $since;
|
|
}
|
|
}
|
|
|
|
if ($until = $query->getFilter(IFilter::BUILTIN_UNTIL)?->get()) {
|
|
if ($until instanceof \DateTimeImmutable) {
|
|
$upperTimeBoundary = $until;
|
|
}
|
|
}
|
|
|
|
if ($person = $query->getFilter(IFilter::BUILTIN_PERSON)?->get()) {
|
|
if ($person instanceof IUser) {
|
|
$actorType = Attendee::ACTOR_USERS;
|
|
$actorId = $person->getUID();
|
|
}
|
|
}
|
|
|
|
$offset = (int) $query->getCursor();
|
|
$comments = $this->chatManager->searchForObjectsWithFilters(
|
|
$query->getTerm(),
|
|
array_keys($roomMap),
|
|
ChatManager::VERB_MESSAGE,
|
|
$lowerTimeBoundary,
|
|
$upperTimeBoundary,
|
|
$actorType,
|
|
$actorId,
|
|
$offset,
|
|
$query->getLimit()
|
|
);
|
|
|
|
$result = [];
|
|
foreach ($comments as $comment) {
|
|
$room = $roomMap[$comment->getObjectId()];
|
|
try {
|
|
$result[] = $this->commentToSearchResultEntry($room, $user, $comment, $query);
|
|
} catch (UnauthorizedException $e) {
|
|
} catch (ParticipantNotFoundException $e) {
|
|
}
|
|
}
|
|
|
|
return SearchResult::paginated(
|
|
$title,
|
|
$result,
|
|
$offset + $query->getLimit()
|
|
);
|
|
}
|
|
|
|
protected function commentToSearchResultEntry(Room $room, IUser $user, IComment $comment, ISearchQuery $query): SearchResultEntry {
|
|
$participant = $this->participantService->getParticipant($room, $user->getUID(), false);
|
|
|
|
$id = (int) $comment->getId();
|
|
$message = $this->messageParser->createMessage($room, $participant, $comment, $this->l);
|
|
$this->messageParser->parseMessage($message);
|
|
|
|
$messageStr = $message->getMessage();
|
|
$search = $replace = [];
|
|
foreach ($message->getMessageParameters() as $key => $parameter) {
|
|
$search[] = '{' . $key . '}';
|
|
if ($parameter['type'] === 'user') {
|
|
$replace[] = '@' . $parameter['name'];
|
|
} else {
|
|
$replace[] = $parameter['name'];
|
|
}
|
|
}
|
|
$messageStr = str_replace($search, $replace, $messageStr);
|
|
|
|
$matchPosition = mb_stripos($messageStr, $query->getTerm());
|
|
if ($matchPosition > 30 && mb_strlen($messageStr) > 40) {
|
|
// Mostlikely the result is not visible from the beginning,
|
|
// so we cut of the message a bit.
|
|
$messageStr = '…' . mb_substr($messageStr, $matchPosition - 10);
|
|
}
|
|
|
|
$now = $this->timeFactory->getDateTime();
|
|
$expireDate = $message->getComment()->getExpireDate();
|
|
if ($expireDate instanceof \DateTime && $expireDate < $now) {
|
|
throw new UnauthorizedException('Expired');
|
|
}
|
|
|
|
if (!$message->getVisibility()) {
|
|
throw new UnauthorizedException('Not visible');
|
|
}
|
|
|
|
$iconUrl = '';
|
|
if ($message->getActorType() === Attendee::ACTOR_USERS) {
|
|
$iconUrl = $this->url->linkToRouteAbsolute('core.avatar.getAvatar', [
|
|
'userId' => $message->getActorId(),
|
|
'size' => 512,
|
|
]);
|
|
}
|
|
|
|
$subline = $this->getSublineTemplate();
|
|
if ($room->getType() === Room::TYPE_ONE_TO_ONE || $room->getType() === Room::TYPE_ONE_TO_ONE_FORMER) {
|
|
$subline = '{user}';
|
|
}
|
|
|
|
$displayName = $message->getActorDisplayName();
|
|
if ($message->getActorType() === Attendee::ACTOR_GUESTS) {
|
|
if ($displayName === '') {
|
|
$displayName = $this->l->t('Guest');
|
|
} else {
|
|
$displayName = $this->l->t('%s (guest)', $displayName);
|
|
}
|
|
}
|
|
|
|
$entry = new SearchResultEntry(
|
|
$iconUrl,
|
|
str_replace(
|
|
['{user}', '{conversation}'],
|
|
[$displayName, $room->getDisplayName($user->getUID())],
|
|
$subline
|
|
),
|
|
$messageStr,
|
|
$this->url->linkToRouteAbsolute('spreed.Page.showCall', ['token' => $room->getToken()]) . '#message_' . $comment->getId(),
|
|
'icon-talk', // $iconClass,
|
|
true
|
|
);
|
|
|
|
$entry->addAttribute('conversation', $room->getToken());
|
|
$entry->addAttribute('messageId', $comment->getId());
|
|
$entry->addAttribute('actorType', $comment->getActorType());
|
|
$entry->addAttribute('actorId', $comment->getActorId());
|
|
$entry->addAttribute('timestamp', '' . $comment->getCreationDateTime()->getTimestamp());
|
|
|
|
return $entry;
|
|
}
|
|
|
|
public function getSupportedFilters(): array {
|
|
return [
|
|
IFilter::BUILTIN_TERM,
|
|
IFilter::BUILTIN_SINCE,
|
|
IFilter::BUILTIN_UNTIL,
|
|
IFilter::BUILTIN_PERSON,
|
|
];
|
|
}
|
|
|
|
public function getAlternateIds(): array {
|
|
return ['talk-message'];
|
|
}
|
|
|
|
public function getCustomFilters(): array {
|
|
return [];
|
|
}
|
|
}
|