import fs from "fs/promises"; import fsExtra from "fs-extra"; import path from "path"; import type NoteMeta from "./src/services/meta/note_meta.js"; import type { NoteMetaFile } from "./src/services/meta/note_meta.js"; import cls from "./src/services/cls.js"; import { initializeTranslations } from "./src/services/i18n.js"; import archiver from "archiver"; const NOTE_ID_USER_GUIDE = "_help_user_guide"; const destRootPath = path.join("src", "public", "app", "doc_notes", "en", "User Guide"); async function startElectron() { await import("./electron-main.js"); } async function main() { await initializeTranslations(); await createImportZip(); await initializeDatabase(); cls.init(() => { importData(); }); await startElectron(); // await exportData(); } async function initializeDatabase() { const sqlInit = (await import("./src/services/sql_init.js")).default; cls.init(() => { if (!sqlInit.isDbInitialized()) { sqlInit.createInitialDatabase(); } }); } async function importData() { const beccaLoader = ((await import("./src/becca/becca_loader.js")).default); const notes = ((await import("./src/services/notes.js")).default); beccaLoader.load(); const becca = ((await import("./src/becca/becca.js")).default); notes.createNewNoteWithTarget("into", "none_root", { parentNoteId: "root", noteId: NOTE_ID_USER_GUIDE, title: "User Guide", content: "The sub-children of this note are automatically synced.", type: "text" }); } async function createImportZip() { const archive = archiver("zip", { zlib: { level: 0 } }); async function iterate(currentPath: string) { for (const entry of await fs.readdir(path.join(destRootPath, currentPath), { withFileTypes: true })) { const entryPath = path.join(currentPath, entry.name); if (entry.isDirectory()) { await iterate(entryPath); continue; } const source = fsExtra.createReadStream(path.join(destRootPath, entryPath)); archive.append(source, { name: path.join(currentPath, entry.name) }); } } await iterate("/"); const outputStream = fsExtra.createWriteStream("input.zip"); archive.pipe(outputStream); await archive.finalize(); } async function exportData() { const zipFilePath = "output.zip"; const deferred = (await import("./src/services/utils.js")).deferred; try { await fsExtra.remove(destRootPath); await fsExtra.mkdir(destRootPath); // First export as zip. const { exportToZipFile } = (await import("./src/services/export/zip.js")).default; await exportToZipFile(NOTE_ID_USER_GUIDE, "html", zipFilePath); const promise = deferred() setTimeout(async () => { // Then extract the zip. const { readZipFile, readContent } = (await import("./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("/")) { const destPath = path.join(destRootPath, 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; } finally { if (await fsExtra.exists(zipFilePath)) { await fsExtra.rm(zipFilePath); } } await cleanUpMeta(); } async function cleanUpMeta() { const metaPath = path.join(destRootPath, "!!!meta.json"); const meta = JSON.parse(await fs.readFile(metaPath, "utf-8")) as NoteMetaFile; for (const file of meta.files) { traverse(file); } function traverse(el: NoteMeta) { for (const child of el.children || []) { traverse(child); } el.isExpanded = false; } await fs.writeFile(metaPath, JSON.stringify(meta, null, 4)); } await main();