2022-07-01 21:17:40 +02:00
|
|
|
/*
|
|
|
|
Copyright 2022 New Vector Ltd
|
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
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
|
|
|
|
const seshatModule = require('matrix-seshat');
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
ipcMain.on('seshat', async function(_ev: IpcMainEvent, payload): Promise<void> {
|
|
|
|
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
|
|
|
|
const eventStorePath = path.join(app.getPath('userData'), 'EventStore');
|
|
|
|
|
2022-11-30 14:51:54 +01:00
|
|
|
const sendError = (id: string, e: Error) => {
|
2022-07-01 21:17:40 +02:00
|
|
|
const error = {
|
|
|
|
message: e.message,
|
|
|
|
};
|
|
|
|
|
2022-11-30 14:51:54 +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) {
|
|
|
|
case 'supportsEventIndexing':
|
|
|
|
ret = seshatSupported;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'initEventIndex':
|
|
|
|
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);
|
|
|
|
} catch (e) {
|
|
|
|
}
|
|
|
|
} 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;
|
|
|
|
|
|
|
|
case 'closeEventIndex':
|
|
|
|
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;
|
|
|
|
|
|
|
|
case 'deleteEventIndex': {
|
|
|
|
try {
|
|
|
|
await deleteContents(eventStorePath);
|
|
|
|
} catch (e) {
|
|
|
|
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case 'isEventIndexEmpty':
|
|
|
|
if (eventIndex === null) ret = true;
|
|
|
|
else ret = await eventIndex.isEmpty();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'isRoomIndexed':
|
|
|
|
if (eventIndex === null) ret = false;
|
|
|
|
else ret = await eventIndex.isRoomIndexed(args[0]);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'addEventToIndex':
|
|
|
|
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;
|
|
|
|
|
|
|
|
case 'deleteEvent':
|
|
|
|
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;
|
|
|
|
|
|
|
|
case 'commitLiveEvents':
|
|
|
|
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;
|
|
|
|
|
|
|
|
case 'searchEventIndex':
|
|
|
|
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;
|
|
|
|
|
|
|
|
case 'addHistoricEvents':
|
|
|
|
if (eventIndex === null) ret = false;
|
|
|
|
else {
|
|
|
|
try {
|
|
|
|
ret = await eventIndex.addHistoricEvents(
|
|
|
|
args[0], args[1], args[2]);
|
|
|
|
} 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;
|
|
|
|
|
|
|
|
case 'getStats':
|
|
|
|
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;
|
|
|
|
|
|
|
|
case 'removeCrawlerCheckpoint':
|
|
|
|
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;
|
|
|
|
|
|
|
|
case 'addCrawlerCheckpoint':
|
|
|
|
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;
|
|
|
|
|
|
|
|
case 'loadFileEvents':
|
|
|
|
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;
|
|
|
|
|
|
|
|
case 'loadCheckpoints':
|
|
|
|
if (eventIndex === null) ret = [];
|
|
|
|
else {
|
|
|
|
try {
|
|
|
|
ret = await eventIndex.loadCheckpoints();
|
|
|
|
} catch (e) {
|
|
|
|
ret = [];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'setUserVersion':
|
|
|
|
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;
|
|
|
|
|
|
|
|
case 'getUserVersion':
|
|
|
|
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:
|
|
|
|
global.mainWindow.webContents.send('seshatReply', {
|
|
|
|
id: payload.id,
|
|
|
|
error: "Unknown IPC Call: " + payload.name,
|
|
|
|
});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
global.mainWindow.webContents.send('seshatReply', {
|
|
|
|
id: payload.id,
|
|
|
|
reply: ret,
|
|
|
|
});
|
|
|
|
});
|