2022-08-31 06:33:37 +05:00
|
|
|
/*
|
|
|
|
|
This file is part of the Notesnook project (https://notesnook.com/)
|
|
|
|
|
|
2023-01-16 13:44:52 +05:00
|
|
|
Copyright (C) 2023 Streetwriters (Private) Limited
|
2022-08-31 06:33:37 +05:00
|
|
|
|
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
|
|
|
it under the terms of the GNU 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 General Public License for more details.
|
|
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
*/
|
2022-08-30 16:13:11 +05:00
|
|
|
|
2023-11-11 13:05:03 +05:00
|
|
|
import { match } from "fuzzyjs";
|
2023-08-21 13:32:06 +05:00
|
|
|
import Database from ".";
|
2023-11-11 13:05:03 +05:00
|
|
|
import { Item, TrashItem } from "../types";
|
|
|
|
|
import { DatabaseSchema, DatabaseSchemaWithFTS, isFalse } from "../database";
|
|
|
|
|
import { AnyColumnWithTable, Kysely, sql } from "kysely";
|
|
|
|
|
import { FilteredSelector } from "../database/sql-collection";
|
|
|
|
|
import { VirtualizedGrouping } from "../utils/virtualized-grouping";
|
2020-03-09 12:39:49 +05:00
|
|
|
|
2023-11-21 13:16:49 +05:00
|
|
|
type SearchResults<T> = {
|
|
|
|
|
sorted: (limit?: number) => Promise<VirtualizedGrouping<T>>;
|
|
|
|
|
items: (limit?: number) => Promise<T[]>;
|
|
|
|
|
ids: () => Promise<string[]>;
|
|
|
|
|
};
|
|
|
|
|
|
2023-11-11 13:05:03 +05:00
|
|
|
type FuzzySearchField<T> = {
|
|
|
|
|
weight?: number;
|
|
|
|
|
name: keyof T;
|
|
|
|
|
column: AnyColumnWithTable<DatabaseSchema, keyof DatabaseSchema>;
|
|
|
|
|
};
|
2020-03-09 12:39:49 +05:00
|
|
|
export default class Lookup {
|
2023-08-21 13:32:06 +05:00
|
|
|
constructor(private readonly db: Database) {}
|
2020-03-19 12:38:33 +05:00
|
|
|
|
2023-11-21 13:16:49 +05:00
|
|
|
notes(query: string, noteIds?: string[]) {
|
|
|
|
|
return this.toSearchResults(async (limit) => {
|
|
|
|
|
const db = this.db.sql() as Kysely<DatabaseSchemaWithFTS>;
|
2023-11-22 08:48:16 +05:00
|
|
|
query = query.replace(/"/, '""');
|
2023-11-21 13:16:49 +05:00
|
|
|
const result = await db
|
|
|
|
|
.with("matching", (eb) =>
|
|
|
|
|
eb
|
|
|
|
|
.selectFrom("content_fts")
|
2023-11-22 08:48:16 +05:00
|
|
|
.where("data", "match", `"${query}"`)
|
2023-11-21 13:16:49 +05:00
|
|
|
.select(["noteId as id", "rank"])
|
|
|
|
|
.unionAll(
|
|
|
|
|
eb
|
|
|
|
|
.selectFrom("notes_fts")
|
2023-11-22 08:48:16 +05:00
|
|
|
.where("title", "match", `"${query}"`)
|
2023-11-21 13:16:49 +05:00
|
|
|
// add 10 weight to title
|
|
|
|
|
.select(["id", sql.raw<number>(`rank * 10`).as("rank")])
|
|
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
.selectFrom("notes")
|
|
|
|
|
.$if(!!noteIds && noteIds.length > 0, (eb) =>
|
2023-11-24 15:11:38 +05:00
|
|
|
eb.where("notes.id", "in", noteIds!)
|
2023-11-21 13:16:49 +05:00
|
|
|
)
|
|
|
|
|
.$if(!!limit, (eb) => eb.limit(limit!))
|
|
|
|
|
.where(isFalse("notes.deleted"))
|
|
|
|
|
.where(isFalse("notes.dateDeleted"))
|
|
|
|
|
.innerJoin("matching", (eb) =>
|
|
|
|
|
eb.onRef("notes.id", "==", "matching.id")
|
|
|
|
|
)
|
|
|
|
|
.orderBy("matching.rank")
|
|
|
|
|
.select(["notes.id"])
|
|
|
|
|
.execute();
|
|
|
|
|
return result.map((id) => id.id);
|
|
|
|
|
}, this.db.notes.all);
|
2020-03-09 12:39:49 +05:00
|
|
|
}
|
|
|
|
|
|
2023-11-11 13:05:03 +05:00
|
|
|
notebooks(query: string) {
|
|
|
|
|
return this.search(this.db.notebooks.all, query, [
|
|
|
|
|
{ name: "id", column: "notebooks.id", weight: -100 },
|
|
|
|
|
{ name: "title", column: "notebooks.title", weight: 10 },
|
|
|
|
|
{ name: "description", column: "notebooks.description" }
|
|
|
|
|
]);
|
2020-03-09 12:39:49 +05:00
|
|
|
}
|
|
|
|
|
|
2023-11-11 13:05:03 +05:00
|
|
|
tags(query: string) {
|
|
|
|
|
return this.search(this.db.tags.all, query, [
|
|
|
|
|
{ name: "id", column: "tags.id", weight: -100 },
|
|
|
|
|
{ name: "title", column: "tags.title" }
|
|
|
|
|
]);
|
2022-11-17 15:41:20 +05:00
|
|
|
}
|
|
|
|
|
|
2023-11-11 13:05:03 +05:00
|
|
|
reminders(query: string) {
|
|
|
|
|
return this.search(this.db.reminders.all, query, [
|
|
|
|
|
{ name: "id", column: "reminders.id", weight: -100 },
|
|
|
|
|
{ name: "title", column: "reminders.title", weight: 10 },
|
|
|
|
|
{ name: "description", column: "reminders.description" }
|
|
|
|
|
]);
|
2020-03-09 12:39:49 +05:00
|
|
|
}
|
|
|
|
|
|
2023-11-21 13:16:49 +05:00
|
|
|
trash(query: string): SearchResults<TrashItem> {
|
|
|
|
|
return {
|
|
|
|
|
sorted: async (limit?: number) => {
|
2023-12-05 15:34:18 +05:00
|
|
|
const { ids, items } = await this.filterTrash(query, limit);
|
2023-11-21 13:16:49 +05:00
|
|
|
return new VirtualizedGrouping<TrashItem>(
|
2023-12-05 15:34:18 +05:00
|
|
|
ids.length,
|
2023-11-21 13:16:49 +05:00
|
|
|
this.db.options.batchSize,
|
2023-12-05 15:34:18 +05:00
|
|
|
async (start, end) => {
|
|
|
|
|
return {
|
|
|
|
|
ids: ids.slice(start, end),
|
|
|
|
|
items: items.slice(start, end)
|
|
|
|
|
};
|
|
|
|
|
}
|
2023-11-21 13:16:49 +05:00
|
|
|
);
|
|
|
|
|
},
|
|
|
|
|
items: async (limit?: number) => {
|
2023-12-05 15:34:18 +05:00
|
|
|
const { items } = await this.filterTrash(query, limit);
|
|
|
|
|
return items;
|
2023-11-21 13:16:49 +05:00
|
|
|
},
|
|
|
|
|
ids: () => this.filterTrash(query).then(({ ids }) => ids)
|
|
|
|
|
};
|
2022-02-28 13:02:16 +05:00
|
|
|
}
|
|
|
|
|
|
2023-11-11 13:05:03 +05:00
|
|
|
attachments(query: string) {
|
|
|
|
|
return this.search(this.db.attachments.all, query, [
|
|
|
|
|
{ name: "id", column: "attachments.id", weight: -100 },
|
|
|
|
|
{ name: "filename", column: "attachments.filename", weight: 5 },
|
|
|
|
|
{ name: "mimeType", column: "attachments.mimeType" },
|
|
|
|
|
{ name: "hash", column: "attachments.hash" }
|
|
|
|
|
]);
|
2021-12-21 11:24:45 +05:00
|
|
|
}
|
|
|
|
|
|
2023-11-21 13:16:49 +05:00
|
|
|
private search<T extends Item>(
|
2023-11-11 13:05:03 +05:00
|
|
|
selector: FilteredSelector<T>,
|
|
|
|
|
query: string,
|
|
|
|
|
fields: FuzzySearchField<T>[]
|
|
|
|
|
) {
|
2023-11-21 13:16:49 +05:00
|
|
|
return this.toSearchResults(
|
|
|
|
|
(limit) => this.filter(selector, query, fields, limit),
|
|
|
|
|
selector
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private async filter<T extends Item>(
|
|
|
|
|
selector: FilteredSelector<T>,
|
|
|
|
|
query: string,
|
|
|
|
|
fields: FuzzySearchField<T>[],
|
|
|
|
|
limit?: number
|
|
|
|
|
) {
|
|
|
|
|
const results: Map<string, number> = new Map();
|
2023-11-11 13:05:03 +05:00
|
|
|
const columns = fields.map((f) => f.column);
|
|
|
|
|
for await (const item of selector.fields(columns)) {
|
2023-11-21 13:16:49 +05:00
|
|
|
if (limit && results.size >= limit) break;
|
|
|
|
|
|
2023-11-11 13:05:03 +05:00
|
|
|
for (const field of fields) {
|
|
|
|
|
const result = match(query, `${item[field.name]}`);
|
|
|
|
|
if (result.match) {
|
2023-11-21 13:16:49 +05:00
|
|
|
const oldScore = results.get(item.id) || 0;
|
|
|
|
|
results.set(item.id, oldScore + result.score * (field.weight || 1));
|
2023-11-11 13:05:03 +05:00
|
|
|
}
|
|
|
|
|
}
|
2023-10-07 13:26:03 +05:00
|
|
|
}
|
2023-11-11 13:05:03 +05:00
|
|
|
selector.fields([]);
|
|
|
|
|
|
2023-11-21 13:16:49 +05:00
|
|
|
return Array.from(results.entries())
|
|
|
|
|
.sort((a, b) => a[1] - b[1])
|
|
|
|
|
.map((a) => a[0]);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private toSearchResults<T extends Item>(
|
|
|
|
|
ids: (limit?: number) => Promise<string[]>,
|
|
|
|
|
selector: FilteredSelector<T>
|
|
|
|
|
): SearchResults<T> {
|
|
|
|
|
return {
|
|
|
|
|
sorted: async (limit?: number) =>
|
|
|
|
|
this.toVirtualizedGrouping(await ids(limit), selector),
|
|
|
|
|
items: async (limit?: number) => this.toItems(await ids(limit), selector),
|
|
|
|
|
ids
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private async filterTrash(query: string, limit?: number) {
|
|
|
|
|
const items = await this.db.trash.all();
|
|
|
|
|
|
2023-12-05 15:34:18 +05:00
|
|
|
const results: Map<string, { rank: number; item: TrashItem }> = new Map();
|
2023-11-21 13:16:49 +05:00
|
|
|
for (const item of items) {
|
|
|
|
|
if (limit && results.size >= limit) break;
|
|
|
|
|
|
|
|
|
|
const result = match(query, item.title);
|
|
|
|
|
if (result.match) {
|
2023-12-05 15:34:18 +05:00
|
|
|
results.set(item.id, { rank: result.score, item });
|
2023-11-21 13:16:49 +05:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-12-05 15:34:18 +05:00
|
|
|
const sorted = Array.from(results.entries()).sort(
|
|
|
|
|
(a, b) => a[1].rank - b[1].rank
|
|
|
|
|
);
|
|
|
|
|
return {
|
|
|
|
|
ids: sorted.map((a) => a[0]),
|
|
|
|
|
items: sorted.map((a) => a[1].item)
|
|
|
|
|
};
|
2023-11-21 13:16:49 +05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private toVirtualizedGrouping<T extends Item>(
|
|
|
|
|
ids: string[],
|
|
|
|
|
selector: FilteredSelector<T>
|
|
|
|
|
) {
|
2023-11-11 13:05:03 +05:00
|
|
|
return new VirtualizedGrouping<T>(
|
2023-12-05 15:34:18 +05:00
|
|
|
ids.length,
|
2023-11-11 13:05:03 +05:00
|
|
|
this.db.options.batchSize,
|
2023-12-05 15:34:18 +05:00
|
|
|
async (start, end) => {
|
|
|
|
|
const items = await selector.items(ids);
|
|
|
|
|
return {
|
|
|
|
|
ids: ids.slice(start, end),
|
|
|
|
|
items: items.slice(start, end)
|
|
|
|
|
};
|
|
|
|
|
}
|
2023-11-11 13:05:03 +05:00
|
|
|
);
|
2020-03-09 12:39:49 +05:00
|
|
|
}
|
2023-11-21 13:16:49 +05:00
|
|
|
|
|
|
|
|
private toItems<T extends Item>(
|
|
|
|
|
ids: string[],
|
|
|
|
|
selector: FilteredSelector<T>
|
|
|
|
|
) {
|
|
|
|
|
if (!ids.length) return [];
|
|
|
|
|
return selector.items(ids);
|
|
|
|
|
}
|
2020-03-09 12:39:49 +05:00
|
|
|
}
|