2022-07-01 21:17:40 +02:00
|
|
|
/*
|
2024-09-06 18:56:18 +02:00
|
|
|
Copyright 2022-2024 New Vector Ltd.
|
2022-07-01 21:17:40 +02:00
|
|
|
|
2024-09-06 18:56:18 +02:00
|
|
|
SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only
|
|
|
|
Please see LICENSE files in the repository root for full details.
|
2022-07-01 21:17:40 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
import { app, ipcMain } from "electron";
|
|
|
|
import { promises as afs } from "fs";
|
|
|
|
import path from "path";
|
|
|
|
|
|
|
|
import type {
|
|
|
|
Seshat as SeshatType,
|
|
|
|
SeshatRecovery as SeshatRecoveryType,
|
|
|
|
ReindexError as ReindexErrorType,
|
|
|
|
} from "matrix-seshat"; // Hak dependency type
|
|
|
|
import IpcMainEvent = Electron.IpcMainEvent;
|
|
|
|
import { randomArray } from "./utils";
|
|
|
|
import { keytar } from "./keytar";
|
|
|
|
|
|
|
|
let seshatSupported = false;
|
|
|
|
let Seshat: typeof SeshatType;
|
|
|
|
let SeshatRecovery: typeof SeshatRecoveryType;
|
|
|
|
let ReindexError: typeof ReindexErrorType;
|
|
|
|
|
|
|
|
try {
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-var-requires
|
2022-12-15 12:00:58 +01:00
|
|
|
const seshatModule = require("matrix-seshat");
|
2022-07-01 21:17:40 +02:00
|
|
|
Seshat = seshatModule.Seshat;
|
|
|
|
SeshatRecovery = seshatModule.SeshatRecovery;
|
|
|
|
ReindexError = seshatModule.ReindexError;
|
|
|
|
seshatSupported = true;
|
|
|
|
} catch (e) {
|
2022-10-13 13:42:33 +02:00
|
|
|
if ((<NodeJS.ErrnoException>e).code === "MODULE_NOT_FOUND") {
|
2022-07-01 21:17:40 +02:00
|
|
|
console.log("Seshat isn't installed, event indexing is disabled.");
|
|
|
|
} else {
|
|
|
|
console.warn("Seshat unexpected error:", e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-13 13:42:33 +02:00
|
|
|
let eventIndex: SeshatType | null = null;
|
2022-07-01 21:17:40 +02:00
|
|
|
|
|
|
|
const seshatDefaultPassphrase = "DEFAULT_PASSPHRASE";
|
|
|
|
async function getOrCreatePassphrase(key: string): Promise<string> {
|
|
|
|
if (keytar) {
|
|
|
|
try {
|
|
|
|
const storedPassphrase = await keytar.getPassword("element.io", key);
|
|
|
|
if (storedPassphrase !== null) {
|
|
|
|
return storedPassphrase;
|
|
|
|
} else {
|
|
|
|
const newPassphrase = await randomArray(32);
|
|
|
|
await keytar.setPassword("element.io", key, newPassphrase);
|
|
|
|
return newPassphrase;
|
|
|
|
}
|
|
|
|
} catch (e) {
|
|
|
|
console.log("Error getting the event index passphrase out of the secret store", e);
|
|
|
|
}
|
|
|
|
}
|
2022-10-13 13:42:33 +02:00
|
|
|
return seshatDefaultPassphrase;
|
2022-07-01 21:17:40 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
const deleteContents = async (p: string): Promise<void> => {
|
|
|
|
for (const entry of await afs.readdir(p)) {
|
|
|
|
const curPath = path.join(p, entry);
|
|
|
|
await afs.unlink(curPath);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2022-12-15 12:00:58 +01:00
|
|
|
ipcMain.on("seshat", async function (_ev: IpcMainEvent, payload): Promise<void> {
|
2022-07-01 21:17:40 +02:00
|
|
|
if (!global.mainWindow) return;
|
|
|
|
|
2022-11-04 15:58:23 +01:00
|
|
|
// We do this here to ensure we get the path after --profile has been resolved
|
2022-12-15 12:00:58 +01:00
|
|
|
const eventStorePath = path.join(app.getPath("userData"), "EventStore");
|
2022-11-04 15:58:23 +01:00
|
|
|
|
2022-12-01 08:20:55 +01:00
|
|
|
const sendError = (id: string, e: Error): void => {
|
2022-07-01 21:17:40 +02:00
|
|
|
const error = {
|
|
|
|
message: e.message,
|
|
|
|
};
|
|
|
|
|
2022-12-15 12:00:58 +01:00
|
|
|
global.mainWindow?.webContents.send("seshatReply", { id, error });
|
2022-07-01 21:17:40 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
const args = payload.args || [];
|
|
|
|
let ret: any;
|
|
|
|
|
|
|
|
switch (payload.name) {
|
2022-12-15 12:00:58 +01:00
|
|
|
case "supportsEventIndexing":
|
2022-07-01 21:17:40 +02:00
|
|
|
ret = seshatSupported;
|
|
|
|
break;
|
|
|
|
|
2022-12-15 12:00:58 +01:00
|
|
|
case "initEventIndex":
|
2022-07-01 21:17:40 +02:00
|
|
|
if (eventIndex === null) {
|
|
|
|
const userId = args[0];
|
|
|
|
const deviceId = args[1];
|
|
|
|
const passphraseKey = `seshat|${userId}|${deviceId}`;
|
|
|
|
|
|
|
|
const passphrase = await getOrCreatePassphrase(passphraseKey);
|
|
|
|
|
|
|
|
try {
|
|
|
|
await afs.mkdir(eventStorePath, { recursive: true });
|
|
|
|
eventIndex = new Seshat(eventStorePath, { passphrase });
|
|
|
|
} catch (e) {
|
|
|
|
if (e instanceof ReindexError) {
|
|
|
|
// If this is a reindex error, the index schema
|
|
|
|
// changed. Try to open the database in recovery mode,
|
|
|
|
// reindex the database and finally try to open the
|
|
|
|
// database again.
|
|
|
|
const recoveryIndex = new SeshatRecovery(eventStorePath, {
|
|
|
|
passphrase,
|
|
|
|
});
|
|
|
|
|
|
|
|
const userVersion = await recoveryIndex.getUserVersion();
|
|
|
|
|
|
|
|
// If our user version is 0 we'll delete the db
|
|
|
|
// anyways so reindexing it is a waste of time.
|
|
|
|
if (userVersion === 0) {
|
|
|
|
await recoveryIndex.shutdown();
|
|
|
|
|
|
|
|
try {
|
|
|
|
await deleteContents(eventStorePath);
|
2022-12-15 12:00:58 +01:00
|
|
|
} catch (e) {}
|
2022-07-01 21:17:40 +02:00
|
|
|
} else {
|
|
|
|
await recoveryIndex.reindex();
|
|
|
|
}
|
|
|
|
|
|
|
|
eventIndex = new Seshat(eventStorePath, { passphrase });
|
|
|
|
} else {
|
2022-11-30 14:51:54 +01:00
|
|
|
sendError(payload.id, <Error>e);
|
2022-07-01 21:17:40 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2022-12-15 12:00:58 +01:00
|
|
|
case "closeEventIndex":
|
2022-07-01 21:17:40 +02:00
|
|
|
if (eventIndex !== null) {
|
|
|
|
const index = eventIndex;
|
|
|
|
eventIndex = null;
|
|
|
|
|
|
|
|
try {
|
|
|
|
await index.shutdown();
|
|
|
|
} catch (e) {
|
2022-11-30 14:51:54 +01:00
|
|
|
sendError(payload.id, <Error>e);
|
2022-07-01 21:17:40 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2022-12-15 12:00:58 +01:00
|
|
|
case "deleteEventIndex": {
|
2022-07-01 21:17:40 +02:00
|
|
|
try {
|
|
|
|
await deleteContents(eventStorePath);
|
2022-12-15 12:00:58 +01:00
|
|
|
} catch (e) {}
|
2022-07-01 21:17:40 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2022-12-15 12:00:58 +01:00
|
|
|
case "isEventIndexEmpty":
|
2022-07-01 21:17:40 +02:00
|
|
|
if (eventIndex === null) ret = true;
|
|
|
|
else ret = await eventIndex.isEmpty();
|
|
|
|
break;
|
|
|
|
|
2022-12-15 12:00:58 +01:00
|
|
|
case "isRoomIndexed":
|
2022-07-01 21:17:40 +02:00
|
|
|
if (eventIndex === null) ret = false;
|
|
|
|
else ret = await eventIndex.isRoomIndexed(args[0]);
|
|
|
|
break;
|
|
|
|
|
2022-12-15 12:00:58 +01:00
|
|
|
case "addEventToIndex":
|
2022-07-01 21:17:40 +02:00
|
|
|
try {
|
2022-10-13 13:42:33 +02:00
|
|
|
eventIndex?.addEvent(args[0], args[1]);
|
2022-07-01 21:17:40 +02:00
|
|
|
} catch (e) {
|
2022-11-30 14:51:54 +01:00
|
|
|
sendError(payload.id, <Error>e);
|
2022-07-01 21:17:40 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2022-12-15 12:00:58 +01:00
|
|
|
case "deleteEvent":
|
2022-07-01 21:17:40 +02:00
|
|
|
try {
|
2022-10-13 13:42:33 +02:00
|
|
|
ret = await eventIndex?.deleteEvent(args[0]);
|
2022-07-01 21:17:40 +02:00
|
|
|
} catch (e) {
|
2022-11-30 14:51:54 +01:00
|
|
|
sendError(payload.id, <Error>e);
|
2022-07-01 21:17:40 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2022-12-15 12:00:58 +01:00
|
|
|
case "commitLiveEvents":
|
2022-07-01 21:17:40 +02:00
|
|
|
try {
|
2022-10-13 13:42:33 +02:00
|
|
|
ret = await eventIndex?.commit();
|
2022-07-01 21:17:40 +02:00
|
|
|
} catch (e) {
|
2022-11-30 14:51:54 +01:00
|
|
|
sendError(payload.id, <Error>e);
|
2022-07-01 21:17:40 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2022-12-15 12:00:58 +01:00
|
|
|
case "searchEventIndex":
|
2022-07-01 21:17:40 +02:00
|
|
|
try {
|
2022-10-13 13:42:33 +02:00
|
|
|
ret = await eventIndex?.search(args[0]);
|
2022-07-01 21:17:40 +02:00
|
|
|
} catch (e) {
|
2022-11-30 14:51:54 +01:00
|
|
|
sendError(payload.id, <Error>e);
|
2022-07-01 21:17:40 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2022-12-15 12:00:58 +01:00
|
|
|
case "addHistoricEvents":
|
2022-07-01 21:17:40 +02:00
|
|
|
if (eventIndex === null) ret = false;
|
|
|
|
else {
|
|
|
|
try {
|
2022-12-15 12:00:58 +01:00
|
|
|
ret = await eventIndex.addHistoricEvents(args[0], args[1], args[2]);
|
2022-07-01 21:17:40 +02:00
|
|
|
} catch (e) {
|
2022-11-30 14:51:54 +01:00
|
|
|
sendError(payload.id, <Error>e);
|
2022-07-01 21:17:40 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2022-12-15 12:00:58 +01:00
|
|
|
case "getStats":
|
2022-07-01 21:17:40 +02:00
|
|
|
if (eventIndex === null) ret = 0;
|
|
|
|
else {
|
|
|
|
try {
|
|
|
|
ret = await eventIndex.getStats();
|
|
|
|
} catch (e) {
|
2022-11-30 14:51:54 +01:00
|
|
|
sendError(payload.id, <Error>e);
|
2022-07-01 21:17:40 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2022-12-15 12:00:58 +01:00
|
|
|
case "removeCrawlerCheckpoint":
|
2022-07-01 21:17:40 +02:00
|
|
|
if (eventIndex === null) ret = false;
|
|
|
|
else {
|
|
|
|
try {
|
|
|
|
ret = await eventIndex.removeCrawlerCheckpoint(args[0]);
|
|
|
|
} catch (e) {
|
2022-11-30 14:51:54 +01:00
|
|
|
sendError(payload.id, <Error>e);
|
2022-07-01 21:17:40 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2022-12-15 12:00:58 +01:00
|
|
|
case "addCrawlerCheckpoint":
|
2022-07-01 21:17:40 +02:00
|
|
|
if (eventIndex === null) ret = false;
|
|
|
|
else {
|
|
|
|
try {
|
|
|
|
ret = await eventIndex.addCrawlerCheckpoint(args[0]);
|
|
|
|
} catch (e) {
|
2022-11-30 14:51:54 +01:00
|
|
|
sendError(payload.id, <Error>e);
|
2022-07-01 21:17:40 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2022-12-15 12:00:58 +01:00
|
|
|
case "loadFileEvents":
|
2022-07-01 21:17:40 +02:00
|
|
|
if (eventIndex === null) ret = [];
|
|
|
|
else {
|
|
|
|
try {
|
|
|
|
ret = await eventIndex.loadFileEvents(args[0]);
|
|
|
|
} catch (e) {
|
2022-11-30 14:51:54 +01:00
|
|
|
sendError(payload.id, <Error>e);
|
2022-07-01 21:17:40 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2022-12-15 12:00:58 +01:00
|
|
|
case "loadCheckpoints":
|
2022-07-01 21:17:40 +02:00
|
|
|
if (eventIndex === null) ret = [];
|
|
|
|
else {
|
|
|
|
try {
|
|
|
|
ret = await eventIndex.loadCheckpoints();
|
|
|
|
} catch (e) {
|
|
|
|
ret = [];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2022-12-15 12:00:58 +01:00
|
|
|
case "setUserVersion":
|
2022-07-01 21:17:40 +02:00
|
|
|
if (eventIndex === null) break;
|
|
|
|
else {
|
|
|
|
try {
|
|
|
|
await eventIndex.setUserVersion(args[0]);
|
|
|
|
} catch (e) {
|
2022-11-30 14:51:54 +01:00
|
|
|
sendError(payload.id, <Error>e);
|
2022-07-01 21:17:40 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2022-12-15 12:00:58 +01:00
|
|
|
case "getUserVersion":
|
2022-07-01 21:17:40 +02:00
|
|
|
if (eventIndex === null) ret = 0;
|
|
|
|
else {
|
|
|
|
try {
|
|
|
|
ret = await eventIndex.getUserVersion();
|
|
|
|
} catch (e) {
|
2022-11-30 14:51:54 +01:00
|
|
|
sendError(payload.id, <Error>e);
|
2022-07-01 21:17:40 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2023-05-09 10:23:01 +02:00
|
|
|
global.mainWindow?.webContents.send("seshatReply", {
|
2022-07-01 21:17:40 +02:00
|
|
|
id: payload.id,
|
|
|
|
error: "Unknown IPC Call: " + payload.name,
|
|
|
|
});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-05-09 10:23:01 +02:00
|
|
|
global.mainWindow?.webContents.send("seshatReply", {
|
2022-07-01 21:17:40 +02:00
|
|
|
id: payload.id,
|
|
|
|
reply: ret,
|
|
|
|
});
|
|
|
|
});
|