2022-01-02 20:04:52 +00:00
|
|
|
<?php
|
|
|
|
|
|
|
|
declare(strict_types = 1);
|
|
|
|
|
|
|
|
namespace Component\Collection;
|
|
|
|
|
2022-03-27 15:19:09 +01:00
|
|
|
use App\Core\DB;
|
2022-01-08 15:10:39 +00:00
|
|
|
use App\Core\Event;
|
2022-01-02 20:04:52 +00:00
|
|
|
use App\Core\Modules\Component;
|
2022-01-08 15:10:39 +00:00
|
|
|
use App\Entity\Actor;
|
|
|
|
use App\Util\Formatting;
|
|
|
|
use Component\Collection\Util\Parser;
|
|
|
|
use Component\Subscription\Entity\ActorSubscription;
|
|
|
|
use Doctrine\Common\Collections\ExpressionBuilder;
|
|
|
|
use Doctrine\ORM\Query\Expr;
|
|
|
|
use Doctrine\ORM\QueryBuilder;
|
2022-04-03 21:40:32 +01:00
|
|
|
use EventResult;
|
2022-01-02 20:04:52 +00:00
|
|
|
|
|
|
|
class Collection extends Component
|
|
|
|
{
|
2022-01-08 15:10:39 +00:00
|
|
|
/**
|
|
|
|
* Perform a high level query on notes or actors
|
|
|
|
*
|
|
|
|
* Supports a variety of query terms and is used both in feeds and
|
|
|
|
* in search. Uses query builders to allow for extension
|
2022-10-19 22:39:17 +01:00
|
|
|
*
|
|
|
|
* @param array<string, OrderByType> $note_order_by
|
|
|
|
* @param array<string, OrderByType> $actor_order_by
|
2022-10-19 22:39:17 +01:00
|
|
|
*
|
|
|
|
* @return array{notes: null|Note[], actors: null|Actor[]}
|
2022-01-08 15:10:39 +00:00
|
|
|
*/
|
2022-02-27 21:12:51 +00:00
|
|
|
public static function query(string $query, int $page, ?string $locale = null, ?Actor $actor = null, array $note_order_by = [], array $actor_order_by = []): array
|
2022-01-08 15:10:39 +00:00
|
|
|
{
|
|
|
|
$note_criteria = null;
|
|
|
|
$actor_criteria = null;
|
|
|
|
if (!empty($query = trim($query))) {
|
2022-01-11 20:28:15 +00:00
|
|
|
[$note_criteria, $actor_criteria] = Parser::parse($query, $locale, $actor);
|
2022-01-08 15:10:39 +00:00
|
|
|
}
|
2022-02-27 21:12:51 +00:00
|
|
|
|
2022-01-08 15:10:39 +00:00
|
|
|
$note_qb = DB::createQueryBuilder();
|
|
|
|
$actor_qb = DB::createQueryBuilder();
|
2022-01-08 17:15:00 +00:00
|
|
|
// TODO consider selecting note related stuff, to avoid separate queries (though they're cached, so maybe it's okay)
|
2022-02-27 21:12:51 +00:00
|
|
|
$note_qb->select('note')->from('App\Entity\Note', 'note');
|
|
|
|
$actor_qb->select('actor')->from('App\Entity\Actor', 'actor');
|
2022-01-08 15:10:39 +00:00
|
|
|
Event::handle('CollectionQueryAddJoins', [&$note_qb, &$actor_qb, $note_criteria, $actor_criteria]);
|
|
|
|
|
2022-02-27 21:12:51 +00:00
|
|
|
// Handle ordering
|
|
|
|
$note_order_by = !empty($note_order_by) ? $note_order_by : ['note.created' => 'DESC', 'note.id' => 'DESC'];
|
|
|
|
$actor_order_by = !empty($actor_order_by) ? $actor_order_by : ['actor.created' => 'DESC', 'actor.id' => 'DESC'];
|
|
|
|
foreach ($note_order_by as $field => $order) {
|
|
|
|
$note_qb->addOrderBy($field, $order);
|
|
|
|
}
|
|
|
|
foreach ($actor_order_by as $field => $order) {
|
|
|
|
$actor_qb->addOrderBy($field, $order);
|
|
|
|
}
|
|
|
|
|
2022-01-08 15:10:39 +00:00
|
|
|
$notes = [];
|
|
|
|
$actors = [];
|
|
|
|
if (!\is_null($note_criteria)) {
|
|
|
|
$note_qb->addCriteria($note_criteria);
|
2022-01-08 15:17:13 +00:00
|
|
|
$notes = $note_qb->getQuery()->execute();
|
2022-01-08 15:10:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!\is_null($actor_criteria)) {
|
|
|
|
$actor_qb->addCriteria($actor_criteria);
|
2022-01-08 15:17:13 +00:00
|
|
|
$actors = $actor_qb->getQuery()->execute();
|
2022-01-08 15:10:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// N.B.: Scope is only enforced at FeedController level
|
|
|
|
return ['notes' => $notes ?? null, 'actors' => $actors ?? null];
|
|
|
|
}
|
|
|
|
|
2022-04-03 21:40:32 +01:00
|
|
|
public function onCollectionQueryAddJoins(QueryBuilder &$note_qb, QueryBuilder &$actor_qb): EventResult
|
2022-01-08 15:10:39 +00:00
|
|
|
{
|
2022-03-07 14:08:07 +00:00
|
|
|
$note_aliases = $note_qb->getAllAliases();
|
|
|
|
if (!\in_array('subscription', $note_aliases)) {
|
|
|
|
$note_qb->leftJoin(ActorSubscription::class, 'subscription', Expr\Join::WITH, 'note.actor_id = subscription.subscribed_id');
|
|
|
|
}
|
|
|
|
if (!\in_array('note_actor', $note_aliases)) {
|
|
|
|
$note_qb->leftJoin(Actor::class, 'note_actor', Expr\Join::WITH, 'note.actor_id = note_actor.id');
|
|
|
|
}
|
2022-01-08 15:10:39 +00:00
|
|
|
return Event::next;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Convert $term to $note_expr and $actor_expr, search criteria. Handles searching for text
|
|
|
|
* notes, for different types of actors and for the content of text notes
|
2022-10-19 22:39:17 +01:00
|
|
|
*
|
|
|
|
* @param mixed $note_expr
|
|
|
|
* @param mixed $actor_expr
|
2022-01-08 15:10:39 +00:00
|
|
|
*/
|
2022-04-03 21:40:32 +01:00
|
|
|
public function onCollectionQueryCreateExpression(ExpressionBuilder $eb, string $term, ?string $locale, ?Actor $actor, &$note_expr, &$actor_expr): EventResult
|
2022-01-08 15:10:39 +00:00
|
|
|
{
|
|
|
|
if (str_contains($term, ':')) {
|
|
|
|
$term = explode(':', $term);
|
2022-01-10 10:17:05 +00:00
|
|
|
if (Formatting::startsWith($term[0], 'note')) {
|
2022-01-08 15:10:39 +00:00
|
|
|
switch ($term[0]) {
|
2022-10-19 22:39:17 +01:00
|
|
|
case 'notes-all':
|
|
|
|
$note_expr = $eb->neq('note.created', null);
|
|
|
|
break;
|
|
|
|
case 'note-local':
|
|
|
|
$note_expr = $eb->eq('note.is_local', filter_var($term[1], \FILTER_VALIDATE_BOOLEAN));
|
|
|
|
break;
|
|
|
|
case 'note-types':
|
|
|
|
case 'notes-include':
|
|
|
|
case 'note-filter':
|
|
|
|
if (\is_null($note_expr)) {
|
|
|
|
$note_expr = [];
|
|
|
|
}
|
|
|
|
if (array_intersect(explode(',', $term[1]), ['text', 'words']) !== []) {
|
|
|
|
$note_expr[] = $eb->neq('note.content', null);
|
|
|
|
} else {
|
|
|
|
$note_expr[] = $eb->eq('note.content', null);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'note-conversation':
|
|
|
|
$note_expr = $eb->eq('note.conversation_id', (int) trim($term[1]));
|
|
|
|
break;
|
|
|
|
case 'note-from':
|
|
|
|
case 'notes-from':
|
|
|
|
$subscribed_expr = $eb->eq('subscription.subscriber_id', $actor->getId());
|
|
|
|
$type_consts = [];
|
|
|
|
if ($term[1] === 'subscribed') {
|
|
|
|
$type_consts = null;
|
|
|
|
}
|
|
|
|
foreach (explode(',', $term[1]) as $from) {
|
|
|
|
if (str_starts_with($from, 'subscribed-')) {
|
|
|
|
[, $type] = explode('-', $from);
|
|
|
|
if (\in_array($type, ['actor', 'actors'])) {
|
|
|
|
$type_consts = null;
|
|
|
|
} else {
|
|
|
|
$type_consts[] = \constant(Actor::class . '::' . mb_strtoupper($type === 'organisation' ? 'group' : $type));
|
|
|
|
}
|
2022-01-08 15:10:39 +00:00
|
|
|
}
|
|
|
|
}
|
2022-10-19 22:39:17 +01:00
|
|
|
if (\is_null($type_consts)) {
|
|
|
|
$note_expr = $subscribed_expr;
|
|
|
|
} elseif (!empty($type_consts)) {
|
|
|
|
$note_expr = $eb->andX($subscribed_expr, $eb->in('note_actor.type', $type_consts));
|
|
|
|
}
|
|
|
|
break;
|
2022-01-08 15:10:39 +00:00
|
|
|
}
|
|
|
|
} elseif (Formatting::startsWith($term, 'actor-')) {
|
|
|
|
switch ($term[0]) {
|
|
|
|
case 'actor-types':
|
|
|
|
case 'actors-include':
|
|
|
|
case 'actor-filter':
|
|
|
|
case 'actor-local':
|
|
|
|
if (\is_null($actor_expr)) {
|
|
|
|
$actor_expr = [];
|
|
|
|
}
|
|
|
|
foreach (
|
|
|
|
[
|
|
|
|
Actor::PERSON => ['person', 'people'],
|
2022-10-19 22:39:17 +01:00
|
|
|
Actor::GROUP => ['group', 'groups', 'org', 'orgs', 'organisation', 'organisations', 'organization', 'organizations'],
|
|
|
|
Actor::BOT => ['bot', 'bots'],
|
2022-01-08 15:10:39 +00:00
|
|
|
] as $type => $match) {
|
|
|
|
if (array_intersect(explode(',', $term[1]), $match) !== []) {
|
|
|
|
$actor_expr[] = $eb->eq('actor.type', $type);
|
|
|
|
} else {
|
|
|
|
$actor_expr[] = $eb->neq('actor.type', $type);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
$note_expr = $eb->contains('note.content', $term);
|
|
|
|
}
|
|
|
|
return Event::next;
|
|
|
|
}
|
2022-01-02 20:04:52 +00:00
|
|
|
}
|