import cls from "@triliumnext/server/src/services/cls.js"; import fs from "fs/promises"; import fsExtra from "fs-extra"; import path from "path"; import electron from "electron"; import { deferred, type DeferredPromise } from "@triliumnext/server/src/services/utils.js"; import windowService from "@triliumnext/server/src/services/window.js"; import archiver, { type Archiver } from "archiver"; import type { WriteStream } from "fs"; import TaskContext from "@triliumnext/server/src/services/task_context.js"; import { resolve } from "path"; export function initializeDatabase(skipDemoDb: boolean) { return new Promise(async (resolve) => { const sqlInit = (await import("@triliumnext/server/src/services/sql_init.js")).default; cls.init(async () => { if (!sqlInit.isDbInitialized()) { await sqlInit.createInitialDatabase(skipDemoDb); } 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(); electron.app.on("ready", async () => { await initializedPromise; console.log("Electron is ready!"); // 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(); }); 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", "import", false); 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 } }); console.log("Archive path is ", resolve(path)) 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(async (res, rej) => { stream.on("finish", () => res()); await archive.finalize(); }); } export async function extractZip(zipFilePath: string, outputPath: string, ignoredFiles?: Set) { const deferred = (await import("@triliumnext/server/src/services/utils.js")).deferred; 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; }