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, "to": replyFilter, "-to": replyFilterInverse, "before": beforeFilter, "until": beforeFilter, "after": afterFilter, "since": afterFilter, "domain": domainFilter, "-domain": domainFilterInverse, "host": domainFilter, "-host": domainFilterInverse, "filter": miscFilter, "-filter": miscFilterInverse, "has": attachmentFilter, } as Record, search: string, id: number) => any> //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, q: string): void { const components = q.trim().split(" "); const terms: string[] = []; let finalTerms: string[] = []; let counter = 0; 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(":"), counter++); else terms.push(component); } let idx = 0; let state: 'idle' | 'quote' | 'parenthesis' = 'idle'; for (let i = 0; i < terms.length; i++) { if (state === 'idle') { if ((terms[i].startsWith('"') && terms[i].endsWith('"')) || (terms[i].startsWith('(') && terms[i].endsWith(')'))) { finalTerms.push(trimStartAndEnd(terms[i])); } else if (terms[i].startsWith('"')) { idx = i; state = 'quote'; } else if (terms[i].startsWith('(')) { idx = i; state = 'parenthesis'; } else { finalTerms.push(terms[i]); } } else if (state === 'quote' && terms[i].endsWith('"')) { finalTerms.push(extractToken(terms, idx, i)); state = 'idle'; } else if (state === 'parenthesis' && terms[i].endsWith(')')) { query.andWhere(new Brackets(qb => { for (const term of extractToken(terms, idx, i).split(' OR ')) { const id = counter++; qb.orWhere(`note.text ILIKE :q_${id}`); query.setParameter(`q_${id}`, `%${sqlLikeEscape(term)}%`); } })); state = 'idle'; } } if (state != "idle") { finalTerms.push(...extractToken(terms, idx, terms.length - 1, false).substring(1).split(' ')); } for (const term of finalTerms) { const id = counter++; if (term.startsWith('-')) query.andWhere(`note.text NOT ILIKE :q_${id}`); else query.andWhere(`note.text ILIKE :q_${id}`); query.setParameter(`q_${id}`, `%${sqlLikeEscape(term.substring(term.startsWith('-') ? 1 : 0))}%`); } } function fromFilter(query: SelectQueryBuilder, filter: string, id: number) { const userQuery = generateUserSubquery(filter, id); query.andWhere(`note.userId = (${userQuery.getQuery()})`); query.setParameters(userQuery.getParameters()); } function fromFilterInverse(query: SelectQueryBuilder, filter: string, id: number) { const userQuery = generateUserSubquery(filter, id); query.andWhere(`note.userId <> (${userQuery.getQuery()})`); query.setParameters(userQuery.getParameters()); } function mentionFilter(query: SelectQueryBuilder, filter: string, id: number) { const userQuery = generateUserSubquery(filter, id); query.addCommonTableExpression(userQuery.getQuery(), `cte_${id}`, { materialized: true }) query.andWhere(`note.mentions @> array[(SELECT * FROM cte_${id})]::varchar[]`); query.setParameters(userQuery.getParameters()); } function mentionFilterInverse(query: SelectQueryBuilder, filter: string, id: number) { const userQuery = generateUserSubquery(filter, id); query.addCommonTableExpression(userQuery.getQuery(), `cte_${id}`, { materialized: true }) query.andWhere(`NOT (note.mentions @> array[(SELECT * FROM cte_${id})]::varchar[])`); query.setParameters(userQuery.getParameters()); } function replyFilter(query: SelectQueryBuilder, filter: string, id: number) { const userQuery = generateUserSubquery(filter, id); query.andWhere(`note.replyUserId = (${userQuery.getQuery()})`); query.setParameters(userQuery.getParameters()); } function replyFilterInverse(query: SelectQueryBuilder, filter: string, id: number) { const userQuery = generateUserSubquery(filter, id); query.andWhere(`note.replyUserId <> (${userQuery.getQuery()})`); query.setParameters(userQuery.getParameters()); } function beforeFilter(query: SelectQueryBuilder, filter: string) { query.andWhere('note.createdAt < :before', { before: filter }); } function afterFilter(query: SelectQueryBuilder, filter: string) { query.andWhere('note.createdAt > :after', { after: filter }); } function domainFilter(query: SelectQueryBuilder, filter: string) { query.andWhere('note.userHost = :domain', { domain: filter }); } function domainFilterInverse(query: SelectQueryBuilder, filter: string) { query.andWhere('note.userHost <> :domain', { domain: filter }); } function miscFilter(query: SelectQueryBuilder, filter: string) { let subQuery: SelectQueryBuilder | 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' || filter === 'reply') { query.andWhere('note.replyId IS NOT NULL'); } else if (filter === 'boosts' || filter === 'boost' || filter === 'renotes' || filter === 'renote') { query.andWhere('note.renoteId IS NOT NULL'); } if (subQuery !== null) query.andWhere(`note.userId IN (${subQuery.getQuery()})`); } function miscFilterInverse(query: SelectQueryBuilder, filter: string) { let subQuery: SelectQueryBuilder | 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 === 'boost' || filter === 'renotes' || filter === 'renote') { query.andWhere('note.renoteId IS NULL'); } if (subQuery !== null) query.andWhere(`note.userId NOT IN (${subQuery.getQuery()})`); } function attachmentFilter(query: SelectQueryBuilder, filter: string) { switch(filter) { case 'image': case 'video': case 'audio': query.andWhere(`note."attachedFileTypes"::varchar ILIKE :type`, { type: `%${sqlLikeEscape(filter)}/%` }); break; case 'file': query.andWhere(`note."attachedFileTypes" <> '{}'`); query.andWhere(`NOT (note."attachedFileTypes"::varchar ILIKE '%image/%')`); query.andWhere(`NOT (note."attachedFileTypes"::varchar ILIKE '%video/%')`); query.andWhere(`NOT (note."attachedFileTypes"::varchar ILIKE '%audio/%')`); break; default: break; } } function generateUserSubquery(filter: string, id: number) { if (filter.startsWith('@')) filter = filter.substring(1); const split = filter.split('@'); 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; } function extractToken(array: string[], start: number, end: number, trim: boolean = true) { const slice = array.slice(start, end+1).join(" "); return trim ? trimStartAndEnd(slice) : slice; } function trimStartAndEnd(str: string) { return str.substring(1, str.length - 1); }