import cls from "@triliumnext/server/src/services/cls.js"; import TaskContext from "@triliumnext/server/src/services/task_context.js"; import windowService from "@triliumnext/server/src/services/window.js"; import archiver, { type Archiver } from "archiver"; import electron from "electron"; import type { WriteStream } from "fs"; import fs from "fs/promises"; import fsExtra from "fs-extra"; import path from "path"; import { deferred, type DeferredPromise } from "../../../packages/commons/src/index.js"; export function initializeDatabase(skipDemoDb: boolean): Promise { return new Promise((resolve) => { import("@triliumnext/server/src/services/sql_init.js").then((m) => { const sqlInit = m.default; cls.init(async () => { if (!sqlInit.isDbInitialized()) { sqlInit.createInitialDatabase(skipDemoDb).then(() => resolve()); } else { sqlInit.dbReady.resolve(); resolve(); } }); }); }); } /** * Electron has a behaviour in which the "ready" event must have a listener attached before it gets to initialize. * If async tasks are awaited before the "ready" event is bound, then the window will never shown. * This method works around by creating a deferred promise. It will immediately bind to the "ready" event and wait for that promise to be resolved externally. * * @param callback a method to be called after the server and Electron is initialized. * @returns the deferred promise that must be resolved externally before the Electron app is started. */ export function startElectron(callback: () => void): DeferredPromise { const initializedPromise = deferred(); const readyHandler = async () => { await initializedPromise; // Start the server. const startTriliumServer = (await import("@triliumnext/server/src/www.js")).default; await startTriliumServer(); // Create the main window. await windowService.createMainWindow(electron.app); callback(); }; // Handle race condition: Electron ready event may have already fired if (electron.app.isReady()) { readyHandler(); } else { electron.app.on("ready", readyHandler); } return initializedPromise; } export async function importData(path: string) { const buffer = await createImportZip(path); const importService = (await import("@triliumnext/server/src/services/import/zip.js")).default; const context = new TaskContext("no-progress-reporting", "importNotes", null); const becca = (await import("@triliumnext/server/src/becca/becca.js")).default; const rootNote = becca.getRoot(); if (!rootNote) { throw new Error("Missing root note for import."); } await importService.importZip(context, buffer, rootNote, { preserveIds: true }); } async function createImportZip(path: string) { const inputFile = "input.zip"; const archive = archiver("zip", { zlib: { level: 0 } }); archive.directory(path, "/"); const outputStream = fsExtra.createWriteStream(inputFile); archive.pipe(outputStream); await waitForEnd(archive, outputStream); try { return await fsExtra.readFile(inputFile); } finally { await fsExtra.rm(inputFile); } } function waitForEnd(archive: Archiver, stream: WriteStream) { return new Promise((res, rej) => { stream.on("finish", res); stream.on("error", rej); archive.on("error", rej); archive.finalize().catch(rej); }); } export async function extractZip(zipFilePath: string, outputPath: string, ignoredFiles?: Set) { const promise = deferred(); setTimeout(async () => { // Then extract the zip. const { readZipFile, readContent } = (await import("@triliumnext/server/src/services/import/zip.js")); await readZipFile(await fs.readFile(zipFilePath), async (zip, entry) => { // We ignore directories since they can appear out of order anyway. if (!entry.fileName.endsWith("/") && !ignoredFiles?.has(entry.fileName)) { const destPath = path.join(outputPath, entry.fileName); const fileContent = await readContent(zip, entry); await fsExtra.mkdirs(path.dirname(destPath)); await fs.writeFile(destPath, fileContent); } zip.readEntry(); }); promise.resolve(); }, 1000); await promise; }