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

<?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 [];
}
}