mirror of
https://github.com/TriliumNext/Notes.git
synced 2025-07-29 19:12:27 +08:00
117 lines
4.3 KiB
TypeScript
117 lines
4.3 KiB
TypeScript
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<void>(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<void> {
|
|
const initializedPromise = deferred<void>();
|
|
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<void>(async (res, rej) => {
|
|
stream.on("finish", () => res());
|
|
await archive.finalize();
|
|
});
|
|
}
|
|
|
|
export async function extractZip(zipFilePath: string, outputPath: string, ignoredFiles?: Set<string>) {
|
|
const deferred = (await import("@triliumnext/server/src/services/utils.js")).deferred;
|
|
|
|
const promise = deferred<void>()
|
|
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;
|
|
}
|