-
-
Notifications
You must be signed in to change notification settings - Fork 4
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
new: Allow to search for tags and URL parts
- Loading branch information
1 parent
4cec5ba
commit 4258fe2
Showing
14 changed files
with
1,197 additions
and
23 deletions.
There are no files selected for viewing
Binary file not shown.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,8 +1,8 @@ | ||
msgid "" | ||
msgstr "" | ||
"Project-Id-Version: Flus\n" | ||
"POT-Creation-Date: 2024-10-03 14:41+0200\n" | ||
"PO-Revision-Date: 2024-10-03 14:41+0200\n" | ||
"POT-Creation-Date: 2024-10-04 19:18+0200\n" | ||
"PO-Revision-Date: 2024-10-04 19:18+0200\n" | ||
"Last-Translator: Marien Fressinaud <[email protected]>\n" | ||
"Language-Team: \n" | ||
"Language: fr_FR\n" | ||
|
@@ -48,7 +48,7 @@ msgstr "Afficher" | |
|
||
#: controllers/Collections.php:101 controllers/Collections.php:311 | ||
#: controllers/Feeds.php:107 controllers/Groups.php:90 | ||
#: controllers/Links.php:270 controllers/Links.php:442 | ||
#: controllers/Links.php:293 controllers/Links.php:465 | ||
#: controllers/Mastodon.php:211 controllers/Mastodon.php:297 | ||
#: controllers/News.php:77 controllers/Passwords.php:89 | ||
#: controllers/Passwords.php:192 controllers/Registrations.php:103 | ||
|
@@ -75,7 +75,7 @@ msgstr "L’une des thématiques associées n’existe pas." | |
|
||
#: controllers/Collections.php:386 controllers/Exportations.php:62 | ||
#: controllers/Groups.php:161 controllers/Importations.php:48 | ||
#: controllers/Links.php:499 controllers/Mastodon.php:87 | ||
#: controllers/Links.php:522 controllers/Mastodon.php:87 | ||
#: controllers/Messages.php:94 controllers/Messages.php:145 | ||
#: controllers/Sessions.php:206 controllers/collections/Read.php:87 | ||
#: controllers/collections/Read.php:168 controllers/collections/Read.php:243 | ||
|
@@ -100,11 +100,11 @@ msgstr "Aucun flux valide n’a été détecté à cette adresse." | |
msgid "You already have a group with this name." | ||
msgstr "Vous avez déjà un groupe portant le même nom." | ||
|
||
#: controllers/Links.php:303 | ||
#: controllers/Links.php:326 | ||
msgid "The link must be associated to a collection." | ||
msgstr "Le lien doit être associé à une collection." | ||
|
||
#: controllers/Links.php:320 controllers/links/Collections.php:154 | ||
#: controllers/Links.php:343 controllers/links/Collections.php:154 | ||
msgid "One of the associated collection doesn’t exist." | ||
msgstr "L’une des collections associées n’existe pas." | ||
|
||
|
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,29 @@ | ||
<?php | ||
|
||
namespace App\migrations; | ||
|
||
class Migration202410040001AddIndexLinksUrl | ||
{ | ||
public function migrate(): bool | ||
{ | ||
$database = \Minz\Database::get(); | ||
|
||
$database->exec(<<<'SQL' | ||
CREATE EXTENSION IF NOT EXISTS pg_trgm; | ||
CREATE INDEX idx_links_url ON links USING gin (url gin_trgm_ops); | ||
SQL); | ||
|
||
return true; | ||
} | ||
|
||
public function rollback(): bool | ||
{ | ||
$database = \Minz\Database::get(); | ||
|
||
$database->exec(<<<'SQL' | ||
DROP INDEX idx_links_url; | ||
SQL); | ||
|
||
return true; | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,210 @@ | ||
<?php | ||
|
||
namespace App\search_engine; | ||
|
||
use App\models; | ||
use Minz\Database; | ||
|
||
/** | ||
* @author Marien Fressinaud <[email protected]> | ||
* @license http://www.gnu.org/licenses/agpl-3.0.en.html AGPL | ||
*/ | ||
class LinksSearcher | ||
{ | ||
/** | ||
* @param array{ | ||
* 'offset'?: int, | ||
* 'limit'?: int|'ALL', | ||
* } $pagination | ||
* | ||
* @return models\Link[] | ||
*/ | ||
public static function getLinks( | ||
models\User $user, | ||
Query $query, | ||
array $pagination = [], | ||
): array { | ||
$default_pagination = [ | ||
'offset' => 0, | ||
'limit' => 'ALL', | ||
]; | ||
|
||
$pagination = array_merge($default_pagination, $pagination); | ||
|
||
$parameters = [ | ||
':user_id' => $user->id, | ||
':offset' => $pagination['offset'], | ||
]; | ||
|
||
$from_statement = 'links l'; | ||
if (self::includeTextCondition($query)) { | ||
$from_statement .= ", plainto_tsquery('french', :query) AS query"; | ||
$parameters[':query'] = ''; | ||
} | ||
|
||
$limit_statement = ''; | ||
if ($pagination['limit'] !== 'ALL') { | ||
$limit_statement = 'LIMIT :limit'; | ||
$parameters[':limit'] = $pagination['limit']; | ||
} | ||
|
||
list($query_statement, $query_parameters) = self::buildWhereQuery($query); | ||
$parameters = array_merge($parameters, $query_parameters); | ||
|
||
$sql = <<<SQL | ||
SELECT | ||
l.*, | ||
l.created_at AS published_at, | ||
( | ||
SELECT COUNT(*) FROM messages m | ||
WHERE m.link_id = l.id | ||
) AS number_comments | ||
FROM {$from_statement} | ||
WHERE l.user_id = :user_id | ||
{$query_statement} | ||
-- Exclude the links that are ONLY in the "never" collection | ||
AND NOT EXISTS ( | ||
SELECT 1 | ||
FROM links_to_collections lc, collections c | ||
WHERE lc.link_id = l.id | ||
AND lc.collection_id = c.id | ||
AND c.user_id = :user_id | ||
HAVING COUNT(CASE WHEN c.type='never' THEN 1 END) = 1 | ||
AND COUNT(c.*) = 1 | ||
) | ||
ORDER BY published_at DESC, l.id | ||
OFFSET :offset | ||
{$limit_statement} | ||
SQL; | ||
|
||
$database = Database::get(); | ||
$statement = $database->prepare($sql); | ||
$statement->execute($parameters); | ||
|
||
return models\Link::fromDatabaseRows($statement->fetchAll()); | ||
} | ||
|
||
public static function countLinks(models\User $user, Query $query): int | ||
{ | ||
$parameters = [ | ||
':user_id' => $user->id, | ||
]; | ||
|
||
$from_statement = 'links l'; | ||
if (self::includeTextCondition($query)) { | ||
$from_statement .= ", plainto_tsquery('french', :query) AS query"; | ||
$parameters[':query'] = ''; | ||
} | ||
|
||
list($query_statement, $query_parameters) = self::buildWhereQuery($query); | ||
$parameters = array_merge($parameters, $query_parameters); | ||
|
||
$sql = <<<SQL | ||
SELECT COUNT(l.id) | ||
FROM {$from_statement} | ||
WHERE l.user_id = :user_id | ||
{$query_statement} | ||
-- Exclude the links that are ONLY in the "never" collection | ||
AND NOT EXISTS ( | ||
SELECT 1 | ||
FROM links_to_collections lc, collections c | ||
WHERE lc.link_id = l.id | ||
AND lc.collection_id = c.id | ||
AND c.user_id = :user_id | ||
HAVING COUNT(CASE WHEN c.type='never' THEN 1 END) = 1 | ||
AND COUNT(c.*) = 1 | ||
) | ||
SQL; | ||
|
||
$database = Database::get(); | ||
$statement = $database->prepare($sql); | ||
$statement->execute($parameters); | ||
|
||
return intval($statement->fetchColumn()); | ||
} | ||
|
||
/** | ||
* @return array{string, array<string, mixed>} | ||
*/ | ||
private static function buildWhereQuery(Query $query): array | ||
{ | ||
$where_sql = ''; | ||
$parameters = []; | ||
|
||
$textConditions = $query->getConditions('text'); | ||
$textValues = array_map(function (Query\Condition $condition): string { | ||
return $condition->getValue(); | ||
}, $textConditions); | ||
$textQuery = implode(' ', $textValues); | ||
|
||
if ($textQuery !== '') { | ||
$where_sql .= ' AND search_index @@ query'; | ||
$parameters[':query'] = $textQuery; | ||
} | ||
|
||
$qualifierConditions = $query->getConditions('qualifier'); | ||
|
||
foreach ($qualifierConditions as $condition) { | ||
$qualifier = $condition->getQualifier(); | ||
if ($qualifier === 'url') { | ||
$value = $condition->getValue(); | ||
|
||
$parameter_name = ':url' . (count($parameters) + 1); | ||
|
||
$where_sql .= " AND l.url ILIKE {$parameter_name}"; | ||
|
||
$parameters[$parameter_name] = "%{$value}%"; | ||
} | ||
} | ||
|
||
$tagConditions = $query->getConditions('tag'); | ||
|
||
$tags_parameters = []; | ||
$not_tags_parameters = []; | ||
|
||
foreach ($tagConditions as $condition) { | ||
$value = $condition->getValue(); | ||
|
||
$parameter_name = ':tag' . (count($parameters) + 1); | ||
|
||
$parameters[$parameter_name] = $value; | ||
|
||
if ($condition->not()) { | ||
$not_tags_parameters[] = $parameter_name; | ||
} else { | ||
$tags_parameters[] = $parameter_name; | ||
} | ||
} | ||
|
||
if ($tags_parameters) { | ||
$tags_statement = implode(',', $tags_parameters); | ||
$where_sql .= " AND l.tags::jsonb ??& array[{$tags_statement}]"; | ||
} | ||
|
||
if ($not_tags_parameters) { | ||
$not_tags_statement = implode(',', $not_tags_parameters); | ||
$where_sql .= " AND NOT (l.tags::jsonb ??| array[{$not_tags_statement}])"; | ||
} | ||
|
||
return [$where_sql, $parameters]; | ||
} | ||
|
||
private static function includeTextCondition(Query $query): bool | ||
{ | ||
$textConditions = $query->getConditions('text'); | ||
return count($textConditions) > 0; | ||
} | ||
} |
Oops, something went wrong.