mirror of
https://iceshrimp.dev/limepotato/jormungandr-bite.git
synced 2025-01-25 06:41:36 -07:00
[backend] Implement filters for postgres FTS
This commit is contained in:
parent
9b2e966c19
commit
a88d581413
3 changed files with 178 additions and 10 deletions
167
packages/backend/src/server/api/common/generate-fts-query.ts
Normal file
167
packages/backend/src/server/api/common/generate-fts-query.ts
Normal file
|
@ -0,0 +1,167 @@
|
||||||
|
import { Brackets, IsNull, Not, SelectQueryBuilder } from "typeorm";
|
||||||
|
import { sqlLikeEscape } from "@/misc/sql-like-escape.js";
|
||||||
|
import { Followings, Users } from "@/models/index.js";
|
||||||
|
import { FILE_TYPE_BROWSERSAFE } from "@/const.js";
|
||||||
|
|
||||||
|
const filters = {
|
||||||
|
"from": fromFilter,
|
||||||
|
"-from": fromFilterInverse,
|
||||||
|
"mention": mentionFilter,
|
||||||
|
"-mention": mentionFilterInverse,
|
||||||
|
"reply": replyFilter,
|
||||||
|
"-reply": replyFilterInverse,
|
||||||
|
"replyto": replyFilter,
|
||||||
|
"-replyto": replyFilterInverse,
|
||||||
|
"to": replyFilter,
|
||||||
|
"-to": replyFilterInverse,
|
||||||
|
"before": beforeFilter,
|
||||||
|
"until": beforeFilter,
|
||||||
|
"after": afterFilter,
|
||||||
|
"since": afterFilter,
|
||||||
|
"domain": domainFilter,
|
||||||
|
"host": domainFilter,
|
||||||
|
"filter": miscFilter,
|
||||||
|
"-filter": miscFilterInverse,
|
||||||
|
"has": attachmentFilter,
|
||||||
|
} as Record<string, (query: SelectQueryBuilder<any>, search: string) => any>
|
||||||
|
|
||||||
|
//TODO: (phrase OR phrase2) should be treated as an OR part of the query
|
||||||
|
//TODO: "phrase with multiple words" should be treated as one term
|
||||||
|
//TODO: editing the query should be possible, clicking search again resets it (it should be a twitter-like top of the page kind of deal)
|
||||||
|
|
||||||
|
export function generateFtsQuery(query: SelectQueryBuilder<any>, q: string): void {
|
||||||
|
const components = q.split(" ");
|
||||||
|
const terms: string[] = [];
|
||||||
|
|
||||||
|
for (const component of components) {
|
||||||
|
const split = component.split(":");
|
||||||
|
if (split.length > 1 && filters[split[0]] !== undefined)
|
||||||
|
filters[split[0]](query, split.slice(1).join(":"));
|
||||||
|
else terms.push(component);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const term of terms) {
|
||||||
|
if (term.startsWith('-')) query.andWhere("note.text NOT ILIKE :q", { q: `%${sqlLikeEscape(term.substring(1))}%` });
|
||||||
|
else query.andWhere("note.text ILIKE :q", { q: `%${sqlLikeEscape(term)}%` });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function fromFilter(query: SelectQueryBuilder<any>, filter: string) {
|
||||||
|
const userQuery = generateUserSubquery(filter);
|
||||||
|
query.andWhere(`note.userId = (${userQuery.getQuery()})`);
|
||||||
|
query.setParameters(userQuery.getParameters());
|
||||||
|
}
|
||||||
|
|
||||||
|
function fromFilterInverse(query: SelectQueryBuilder<any>, filter: string) {
|
||||||
|
const userQuery = generateUserSubquery(filter);
|
||||||
|
query.andWhere(`note.userId <> (${userQuery.getQuery()})`);
|
||||||
|
query.setParameters(userQuery.getParameters());
|
||||||
|
}
|
||||||
|
|
||||||
|
function mentionFilter(query: SelectQueryBuilder<any>, filter: string) {
|
||||||
|
const userQuery = generateUserSubquery(filter);
|
||||||
|
query.andWhere(`note.mentions @> array[(${userQuery.getQuery()})]`);
|
||||||
|
query.setParameters(userQuery.getParameters());
|
||||||
|
}
|
||||||
|
|
||||||
|
function mentionFilterInverse(query: SelectQueryBuilder<any>, filter: string) {
|
||||||
|
const userQuery = generateUserSubquery(filter);
|
||||||
|
query.andWhere(`NOT (note.mentions @> array[(${userQuery.getQuery()})])`);
|
||||||
|
query.setParameters(userQuery.getParameters());
|
||||||
|
}
|
||||||
|
|
||||||
|
function replyFilter(query: SelectQueryBuilder<any>, filter: string) {
|
||||||
|
const userQuery = generateUserSubquery(filter);
|
||||||
|
query.andWhere(`note.replyUserId = (${userQuery.getQuery()})`);
|
||||||
|
query.setParameters(userQuery.getParameters());
|
||||||
|
}
|
||||||
|
|
||||||
|
function replyFilterInverse(query: SelectQueryBuilder<any>, filter: string) {
|
||||||
|
const userQuery = generateUserSubquery(filter);
|
||||||
|
query.andWhere(`note.replyUserId <> (${userQuery.getQuery()})`);
|
||||||
|
query.setParameters(userQuery.getParameters());
|
||||||
|
}
|
||||||
|
|
||||||
|
function beforeFilter(query: SelectQueryBuilder<any>, filter: string) {
|
||||||
|
query.andWhere('note.createdAt < :before', { before: filter });
|
||||||
|
}
|
||||||
|
|
||||||
|
function afterFilter(query: SelectQueryBuilder<any>, filter: string) {
|
||||||
|
query.andWhere('note.createdAt > :after', { after: filter });
|
||||||
|
}
|
||||||
|
|
||||||
|
function domainFilter(query: SelectQueryBuilder<any>, filter: string) {
|
||||||
|
query.andWhere('note.userHost = :domain', { domain: filter });
|
||||||
|
}
|
||||||
|
|
||||||
|
function miscFilter(query: SelectQueryBuilder<any>, filter: string) {
|
||||||
|
let subQuery: SelectQueryBuilder<any> | null = null;
|
||||||
|
if (filter === 'followers') {
|
||||||
|
subQuery = Followings.createQueryBuilder('following')
|
||||||
|
.select('following.followerId')
|
||||||
|
.where('following.followeeId = :meId')
|
||||||
|
} else if (filter === 'following') {
|
||||||
|
subQuery = Followings.createQueryBuilder('following')
|
||||||
|
.select('following.followeeId')
|
||||||
|
.where('following.followerId = :meId')
|
||||||
|
} else if (filter === 'replies') {
|
||||||
|
query.andWhere('note.replyId IS NOT NULL');
|
||||||
|
} else if (filter === 'boosts') {
|
||||||
|
query.andWhere('note.renoteId IS NOT NULL');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (subQuery !== null) query.andWhere(`note.userId IN (${subQuery.getQuery()})`);
|
||||||
|
}
|
||||||
|
|
||||||
|
function miscFilterInverse(query: SelectQueryBuilder<any>, filter: string) {
|
||||||
|
let subQuery: SelectQueryBuilder<any> | null = null;
|
||||||
|
if (filter === 'followers') {
|
||||||
|
subQuery = Followings.createQueryBuilder('following')
|
||||||
|
.select('following.followerId')
|
||||||
|
.where('following.followeeId = :meId')
|
||||||
|
} else if (filter === 'following') {
|
||||||
|
subQuery = Followings.createQueryBuilder('following')
|
||||||
|
.select('following.followeeId')
|
||||||
|
.where('following.followerId = :meId')
|
||||||
|
} else if (filter === 'replies') {
|
||||||
|
query.andWhere('note.replyId IS NULL');
|
||||||
|
} else if (filter === 'boosts' || filter === 'renotes') {
|
||||||
|
query.andWhere('note.renoteId IS NULL');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (subQuery !== null) query.andWhere(`note.userId NOT IN (${subQuery.getQuery()})`);
|
||||||
|
}
|
||||||
|
|
||||||
|
function attachmentFilter(query: SelectQueryBuilder<any>, filter: string) {
|
||||||
|
switch(filter) {
|
||||||
|
case 'image':
|
||||||
|
case 'video':
|
||||||
|
case 'audio':
|
||||||
|
query.andWhere(`note.attachedFileTypes && array[:...types]::varchar[]`, { types: FILE_TYPE_BROWSERSAFE.filter(t => t.startsWith(`${filter}/`)) });
|
||||||
|
break;
|
||||||
|
case 'file':
|
||||||
|
query.andWhere(`note.attachedFileTypes <> '{}'`);
|
||||||
|
query.andWhere(`NOT (note.attachedFileTypes && array[:...types]::varchar[])`, { types: FILE_TYPE_BROWSERSAFE });
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function generateUserSubquery(filter: string) {
|
||||||
|
if (filter.startsWith('@')) filter = filter.substring(1);
|
||||||
|
const split = filter.split('@');
|
||||||
|
const id = Buffer.from(filter).toString('hex');
|
||||||
|
|
||||||
|
const query = Users.createQueryBuilder('user')
|
||||||
|
.select('user.id')
|
||||||
|
.where(`user.usernameLower = :user_${id}`)
|
||||||
|
.andWhere(`user.host ${split[1] !== undefined ? `= :host_${id}` : 'IS NULL'}`);
|
||||||
|
|
||||||
|
query.setParameter(`user_${id}`, split[0].toLowerCase());
|
||||||
|
|
||||||
|
if (split[1] !== undefined)
|
||||||
|
query.setParameter(`host_${id}`, split[1].toLowerCase());
|
||||||
|
|
||||||
|
return query;
|
||||||
|
}
|
|
@ -6,6 +6,7 @@ import { generateVisibilityQuery } from "../../common/generate-visibility-query.
|
||||||
import { generateMutedUserQuery } from "../../common/generate-muted-user-query.js";
|
import { generateMutedUserQuery } from "../../common/generate-muted-user-query.js";
|
||||||
import { generateBlockedUserQuery } from "../../common/generate-block-query.js";
|
import { generateBlockedUserQuery } from "../../common/generate-block-query.js";
|
||||||
import { sqlLikeEscape } from "@/misc/sql-like-escape.js";
|
import { sqlLikeEscape } from "@/misc/sql-like-escape.js";
|
||||||
|
import { generateFtsQuery } from "@/server/api/common/generate-fts-query.js";
|
||||||
|
|
||||||
export const meta = {
|
export const meta = {
|
||||||
tags: ["notes"],
|
tags: ["notes"],
|
||||||
|
@ -73,7 +74,6 @@ export default define(meta, paramDef, async (ps, me) => {
|
||||||
}
|
}
|
||||||
|
|
||||||
query
|
query
|
||||||
.andWhere("note.text ILIKE :q", { q: `%${sqlLikeEscape(ps.query)}%` })
|
|
||||||
.innerJoinAndSelect("note.user", "user")
|
.innerJoinAndSelect("note.user", "user")
|
||||||
.leftJoinAndSelect("user.avatar", "avatar")
|
.leftJoinAndSelect("user.avatar", "avatar")
|
||||||
.leftJoinAndSelect("user.banner", "banner")
|
.leftJoinAndSelect("user.banner", "banner")
|
||||||
|
@ -86,11 +86,11 @@ export default define(meta, paramDef, async (ps, me) => {
|
||||||
.leftJoinAndSelect("renoteUser.avatar", "renoteUserAvatar")
|
.leftJoinAndSelect("renoteUser.avatar", "renoteUserAvatar")
|
||||||
.leftJoinAndSelect("renoteUser.banner", "renoteUserBanner");
|
.leftJoinAndSelect("renoteUser.banner", "renoteUserBanner");
|
||||||
|
|
||||||
|
generateFtsQuery(query, ps.query);
|
||||||
generateVisibilityQuery(query, me);
|
generateVisibilityQuery(query, me);
|
||||||
if (me) generateMutedUserQuery(query, me);
|
generateMutedUserQuery(query, me);
|
||||||
if (me) generateBlockedUserQuery(query, me);
|
generateBlockedUserQuery(query, me);
|
||||||
|
query.setParameter("meId", me.id);
|
||||||
|
|
||||||
const notes: Note[] = await query.take(ps.limit).getMany();
|
return await query.take(ps.limit).getMany().then(notes => Notes.packMany(notes, me));
|
||||||
|
|
||||||
return await Notes.packMany(notes, me);
|
|
||||||
});
|
});
|
||||||
|
|
|
@ -18,6 +18,7 @@ import { resolveUser } from "@/remote/resolve-user.js";
|
||||||
import { createNote } from "@/remote/activitypub/models/note.js";
|
import { createNote } from "@/remote/activitypub/models/note.js";
|
||||||
import config from "@/config/index.js";
|
import config from "@/config/index.js";
|
||||||
import { logger, MastoContext } from "@/server/api/mastodon/index.js";
|
import { logger, MastoContext } from "@/server/api/mastodon/index.js";
|
||||||
|
import { generateFtsQuery } from "@/server/api/common/generate-fts-query.js";
|
||||||
|
|
||||||
export class SearchHelpers {
|
export class SearchHelpers {
|
||||||
public static async search(q: string | undefined, type: string | undefined, resolve: boolean = false, following: boolean = false, accountId: string | undefined, excludeUnreviewed: boolean = false, maxId: string | undefined, minId: string | undefined, limit: number = 20, offset: number | undefined, ctx: MastoContext): Promise<MastodonEntity.Search> {
|
public static async search(q: string | undefined, type: string | undefined, resolve: boolean = false, following: boolean = false, accountId: string | undefined, excludeUnreviewed: boolean = false, maxId: string | undefined, minId: string | undefined, limit: number = 20, offset: number | undefined, ctx: MastoContext): Promise<MastodonEntity.Search> {
|
||||||
|
@ -163,11 +164,9 @@ export class SearchHelpers {
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
query
|
query.leftJoinAndSelect("note.renote", "renote");
|
||||||
.andWhere("note.text ILIKE :q", { q: `%${sqlLikeEscape(q)}%` })
|
|
||||||
.leftJoinAndSelect("note.renote", "renote");
|
|
||||||
|
|
||||||
|
|
||||||
|
generateFtsQuery(query, q);
|
||||||
generateVisibilityQuery(query, user);
|
generateVisibilityQuery(query, user);
|
||||||
|
|
||||||
if (!accountId) {
|
if (!accountId) {
|
||||||
|
@ -175,6 +174,8 @@ export class SearchHelpers {
|
||||||
generateBlockedUserQuery(query, user);
|
generateBlockedUserQuery(query, user);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
query.setParameter("meId", user);
|
||||||
|
|
||||||
return query.skip(offset ?? 0).take(limit).getMany().then(p => minId ? p.reverse() : p);
|
return query.skip(offset ?? 0).take(limit).getMany().then(p => minId ? p.reverse() : p);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue