mirror of
https://github.com/heyman/heynote.git
synced 2025-01-22 05:48:47 +01:00
344 lines
11 KiB
JavaScript
344 lines
11 KiB
JavaScript
import fs from "fs"
|
|
import os from "node:os"
|
|
import { join, dirname, 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 (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)
|
|
}
|
|
}
|
|
} |