import fs from "fs" import os from "node:os" import { join, basename } from "path" import * as jetpack from "fs-jetpack"; import { app, ipcMain, dialog } from "electron" import CONFIG from "../config" import { SCRATCH_FILE_NAME } from "../../src/common/constants" import { NoteFormat } from "../../src/common/note-format" import { isDev } from '../detect-platform'; import { initialContent, initialDevContent } from '../initial-content' export const NOTES_DIR_NAME = isDev ? "notes-dev" : "notes" let library const untildify = (pathWithTilde) => { const homeDir = os.homedir() return homeDir ? pathWithTilde.replace(/^~(?=$|\/|\\)/, homeDir) : pathWithTilde } async function readNoteMetadata(filePath) { const chunks = [] for await (let chunk of fs.createReadStream(filePath, { start: 0, end:4000 })) { chunks.push(chunk) } const headContent = Buffer.concat(chunks).toString("utf8") const firstSeparator = headContent.indexOf("\nāˆžāˆžāˆž") if (firstSeparator === -1) { return null } try { const metadata = JSON.parse(headContent.slice(0, firstSeparator).trim()) return {"name": metadata.name, "tags": metadata.tags} } catch (e) { return {} } } export class FileLibrary { constructor(basePath) { basePath = untildify(basePath) if (jetpack.exists(basePath) !== "dir") { throw new Error(`Path directory does not exist: ${basePath}`) } this.basePath = fs.realpathSync(basePath) this.jetpack = jetpack.cwd(this.basePath) this.files = {}; this.watcher = null; this.contentSaved = false this.onChangeCallback = null // create scratch.txt if it doesn't exist if (!this.jetpack.exists(SCRATCH_FILE_NAME)) { this.jetpack.write(SCRATCH_FILE_NAME, isDev ? initialDevContent : initialContent) } } async exists(path) { return this.jetpack.exists(path) === "file" } async load(path) { if (this.files[path]) { return this.files[path].read() } const fullPath = fs.realpathSync(join(this.basePath, path)) this.files[path] = new NoteBuffer({fullPath, library:this}) return await this.files[path].read() } async save(path, content) { if (!this.files[path]) { throw new Error(`File not loaded: ${path}`) } return await this.files[path].save(content) } async create(path, content) { if (await this.exists(path)) { throw new Error(`File already exists: ${path}`) } const fullPath = join(this.basePath, path) await this.jetpack.writeAsync(fullPath, content) } async move(path, newPath) { if (await this.exists(newPath)) { throw new Error(`File already exists: ${newPath}`) } const fullOldPath = join(this.basePath, path) const fullNewPath = join(this.basePath, newPath) await this.jetpack.moveAsync(fullOldPath, fullNewPath) } async delete(path) { if (path === SCRATCH_FILE_NAME) { throw new Error("Can't delete scratch file") } const fullPath = join(this.basePath, path) await this.jetpack.removeAsync(fullPath) } async getList() { //console.log("Listing notes") const notes = {} const files = await this.jetpack.findAsync(".", { matching: "*.txt", recursive: true, }) const promises = [] for (const file of files) { promises.push(readNoteMetadata(join(this.basePath, file))) } const metadataList = await Promise.all(promises) metadataList.forEach((metadata, i) => { const path = files[i] notes[path] = metadata }) return notes } async getDirectoryList() { const directories = await this.jetpack.findAsync("", { files: false, directories: true, recursive: true, }) return directories } setupWatcher(win) { if (!this.watcher) { this.watcher = fs.watch( this.basePath, { persistent: true, recursive: true, encoding: "utf8", }, async (eventType, changedPath) => { //console.log("File changed", eventType, changedPath) //if (changedPath.toLowerCase().endsWith(".txt")) { // console.log("txt", this.notes) // if (await this.exists(changedPath)) { // console.log("file exists!") // const newMetadata = await readNoteMetadata(join(this.basePath, changedPath)) // if (!(changedPath in this.notes) || newMetadata.name !== this.notes[changedPath].name) { // this.notes[changedPath] = newMetadata // win.webContents.send("buffer:noteMetadataChanged", changedPath, newMetadata) // console.log("metadata changed") // } else { // console.log("no metadata change") // } // } else if (changedPath in this.notes) { // console.log("note removed", changedPath) // delete this.notes[changedPath] // win.webContents.send("buffer:noteRemoved", changedPath) // } //} for (const [path, buffer] of Object.entries(this.files)) { if (changedPath === basename(path)) { const content = await buffer.read() // if the file was removed (e.g. during a atomic save) the content will be undefined if (content !== undefined && buffer._lastSavedContent !== content) { win.webContents.send("buffer:change", path, content) } } } } ) } } closeFile(path) { if (this.files[path]) { delete this.files[path] } } close() { for (const buffer of Object.values(this.files)) { this.closeFile(buffer.filePath) } this.stopWatcher() } stopWatcher() { if (this.watcher) { this.watcher.close() this.watcher = null } } } export class NoteBuffer { constructor({fullPath, library}) { this.fullPath = fullPath this._lastSavedContent = null this.library = library } async read() { return await this.library.jetpack.read(this.fullPath, 'utf8') } async save(content) { this._lastSavedContent = content const saveResult = await this.library.jetpack.write(this.fullPath, content, { atomic: true, mode: '600', }) return saveResult } exists() { return jetpack.exists(this.fullPath) === "file" } } export function setCurrentFileLibrary(lib) { library = lib } export function setupFileLibraryEventHandlers(win) { ipcMain.handle('buffer:load', async (event, path) => { //console.log("buffer:load", path) return await library.load(path) }); ipcMain.handle('buffer:save', async (event, path, content) => { return await library.save(path, content) }); ipcMain.handle('buffer:create', async (event, path, content) => { return await library.create(path, content) }); ipcMain.handle('buffer:getList', async (event) => { return await library.getList() }); ipcMain.handle('buffer:getDirectoryList', async (event) => { return await library.getDirectoryList() }); ipcMain.handle('buffer:exists', async (event, path) => { return await library.exists(path) }); ipcMain.handle('buffer:close', async (event, path) => { return await library.closeFile(path) }); ipcMain.handle('buffer:saveAndQuit', async (event, contents) => { library.stopWatcher() for (const [path, content] of contents) { await library.save(path, content) } library.contentSaved = true app.quit() }) ipcMain.handle('buffer:move', async (event, path, newPath) => { return await library.move(path, newPath) }); ipcMain.handle('buffer:delete', async (event, path) => { return await library.delete(path) }); ipcMain.handle("library:selectLocation", async () => { let result = await dialog.showOpenDialog({ title: "Select directory to store buffer", properties: [ "openDirectory", "createDirectory", "noResolveAliases", ], }) if (result.canceled) { return } const filePath = result.filePaths[0] return filePath }) } export async function migrateBufferFileToLibrary(app) { async function ensureBufferFileMetadata(filePath) { const metadata = await readNoteMetadata(filePath) //console.log("Metadata", metadata) if (!metadata || !metadata.name) { console.log("Adding metadata to", filePath) const note = NoteFormat.load(jetpack.read(filePath)) note.metadata.name = "Scratch" jetpack.write(filePath, note.serialize()) } else { console.log("Metadata already exists for", filePath) } } const defaultLibraryPath = join(app.getPath("userData"), NOTES_DIR_NAME) const customBufferPath = CONFIG.get("settings.bufferPath") const oldBufferFile = isDev ? "buffer-dev.txt" : "buffer.txt" if (customBufferPath) { // if the new buffer file exists, no need to migrate if (jetpack.exists(join(customBufferPath, SCRATCH_FILE_NAME)) === "file") { return } const oldBufferFileFullPath = join(customBufferPath, oldBufferFile) if (jetpack.exists(oldBufferFileFullPath) === "file") { const newFileFullPath = join(customBufferPath, SCRATCH_FILE_NAME); console.log(`Migrating file ${oldBufferFileFullPath} to ${newFileFullPath}`) // rename buffer file to scratch.txt jetpack.move(oldBufferFileFullPath, newFileFullPath) // add metadata to scratch.txt (just to be sure, we'll double check that it's needed first) await ensureBufferFileMetadata(newFileFullPath) } } else { // if the new buffer file exists, no need to migrate if (jetpack.exists(join(defaultLibraryPath, SCRATCH_FILE_NAME)) === "file") { return } // check if the old buffer file exists, while the default *library* path doesn't exist const oldBufferFileFullPath = join(app.getPath("userData"), oldBufferFile) if (jetpack.exists(oldBufferFileFullPath) === "file" && jetpack.exists(defaultLibraryPath) !== "dir") { const newFileFullPath = join(defaultLibraryPath, SCRATCH_FILE_NAME); console.log(`Migrating buffer file ${oldBufferFileFullPath} to ${newFileFullPath}`) // create the default library path jetpack.dir(defaultLibraryPath) // move the buffer file to the library path jetpack.move(oldBufferFileFullPath, newFileFullPath) // add metadata to scratch.txt await ensureBufferFileMetadata(newFileFullPath) } } }