2022-10-23 02:23:54 +08:00
|
|
|
import { FullTextSearchOptions } from "../../plug-api/plugos-syscall/fulltext.ts";
|
2022-10-21 16:00:43 +08:00
|
|
|
import { AsyncSQLite } from "../../plugos/sqlite/async_sqlite.ts";
|
2022-10-10 20:50:21 +08:00
|
|
|
import { SysCallMapping } from "../system.ts";
|
|
|
|
|
2022-10-21 16:00:43 +08:00
|
|
|
export async function ensureFTSTable(
|
|
|
|
db: AsyncSQLite,
|
2022-10-10 20:50:21 +08:00
|
|
|
tableName: string,
|
|
|
|
) {
|
2022-10-21 16:00:43 +08:00
|
|
|
const result = await db.query(
|
2022-10-19 15:52:29 +08:00
|
|
|
`SELECT name FROM sqlite_master WHERE type='table' AND name=?`,
|
2022-10-21 16:00:43 +08:00
|
|
|
tableName,
|
2022-10-19 15:52:29 +08:00
|
|
|
);
|
|
|
|
if (result.length === 0) {
|
2022-10-21 16:00:43 +08:00
|
|
|
await db.execute(
|
2022-10-19 15:52:29 +08:00
|
|
|
`CREATE VIRTUAL TABLE ${tableName} USING fts5(key, value);`,
|
|
|
|
);
|
2022-10-10 20:50:21 +08:00
|
|
|
|
2022-11-25 20:05:41 +08:00
|
|
|
// console.log(`Created fts5 table ${tableName}`);
|
2022-10-19 15:52:29 +08:00
|
|
|
}
|
2022-10-10 20:50:21 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
export function fullTextSearchSyscalls(
|
2022-10-21 16:00:43 +08:00
|
|
|
db: AsyncSQLite,
|
2022-10-10 20:50:21 +08:00
|
|
|
tableName: string,
|
|
|
|
): SysCallMapping {
|
|
|
|
return {
|
|
|
|
"fulltext.index": async (_ctx, key: string, value: string) => {
|
2022-10-21 16:00:43 +08:00
|
|
|
await db.execute(`DELETE FROM ${tableName} WHERE key = ?`, key);
|
|
|
|
await db.execute(
|
2022-10-10 20:50:21 +08:00
|
|
|
`INSERT INTO ${tableName} (key, value) VALUES (?, ?)`,
|
|
|
|
key,
|
|
|
|
value,
|
|
|
|
);
|
|
|
|
},
|
|
|
|
"fulltext.delete": async (_ctx, key: string) => {
|
2022-10-21 16:00:43 +08:00
|
|
|
await db.execute(`DELETE FROM ${tableName} WHERE key = ?`, key);
|
2022-10-10 20:50:21 +08:00
|
|
|
},
|
2022-10-23 02:23:54 +08:00
|
|
|
"fulltext.search": async (
|
|
|
|
_ctx,
|
|
|
|
phrase: string,
|
|
|
|
options: FullTextSearchOptions,
|
|
|
|
) => {
|
2022-10-10 20:50:21 +08:00
|
|
|
return (
|
2022-10-21 16:00:43 +08:00
|
|
|
await db.query(
|
2022-10-23 02:23:54 +08:00
|
|
|
`SELECT key, bm25(fts) AS score, snippet(fts, 1, ?, ?, ?, ?) as snippet
|
|
|
|
FROM ${tableName}
|
|
|
|
WHERE value
|
|
|
|
MATCH ?
|
|
|
|
ORDER BY score LIMIT ?`,
|
|
|
|
options.highlightPrefix || "",
|
|
|
|
options.highlightPostfix || "",
|
|
|
|
options.highlightEllipsis || "...",
|
|
|
|
options.summaryMaxLength || 50,
|
2022-10-10 20:50:21 +08:00
|
|
|
phrase,
|
2022-10-23 02:23:54 +08:00
|
|
|
options.limit || 20,
|
2022-10-10 20:50:21 +08:00
|
|
|
)
|
2022-10-23 02:23:54 +08:00
|
|
|
).map((item) => ({
|
|
|
|
name: item.key,
|
|
|
|
score: item.score,
|
|
|
|
snippet: item.snippet,
|
|
|
|
}));
|
2022-10-10 20:50:21 +08:00
|
|
|
},
|
|
|
|
};
|
|
|
|
}
|