272 lines
7.4 KiB
TypeScript
272 lines
7.4 KiB
TypeScript
import { getFileType } from '@main/utils/file'
|
|
import { FileType } from '@types'
|
|
import * as crypto from 'crypto'
|
|
import {
|
|
app,
|
|
dialog,
|
|
OpenDialogOptions,
|
|
OpenDialogReturnValue,
|
|
SaveDialogOptions,
|
|
SaveDialogReturnValue
|
|
} from 'electron'
|
|
import logger from 'electron-log'
|
|
import * as fs from 'fs'
|
|
import { writeFileSync } from 'fs'
|
|
import { readFile } from 'fs/promises'
|
|
import * as path from 'path'
|
|
import { v4 as uuidv4 } from 'uuid'
|
|
|
|
class FileManager {
|
|
private storageDir: string
|
|
|
|
constructor() {
|
|
this.storageDir = path.join(app.getPath('userData'), 'Data', 'Files')
|
|
this.initStorageDir()
|
|
}
|
|
|
|
private initStorageDir(): void {
|
|
if (!fs.existsSync(this.storageDir)) {
|
|
fs.mkdirSync(this.storageDir, { recursive: true })
|
|
}
|
|
}
|
|
|
|
private async getFileHash(filePath: string): Promise<string> {
|
|
return new Promise((resolve, reject) => {
|
|
const hash = crypto.createHash('md5')
|
|
const stream = fs.createReadStream(filePath)
|
|
stream.on('data', (data) => hash.update(data))
|
|
stream.on('end', () => resolve(hash.digest('hex')))
|
|
stream.on('error', reject)
|
|
})
|
|
}
|
|
|
|
async findDuplicateFile(filePath: string): Promise<FileType | null> {
|
|
const stats = fs.statSync(filePath)
|
|
const fileSize = stats.size
|
|
|
|
const files = await fs.promises.readdir(this.storageDir)
|
|
for (const file of files) {
|
|
const storedFilePath = path.join(this.storageDir, file)
|
|
const storedStats = fs.statSync(storedFilePath)
|
|
|
|
if (storedStats.size === fileSize) {
|
|
const [originalHash, storedHash] = await Promise.all([
|
|
this.getFileHash(filePath),
|
|
this.getFileHash(storedFilePath)
|
|
])
|
|
|
|
if (originalHash === storedHash) {
|
|
const ext = path.extname(file)
|
|
const id = path.basename(file, ext)
|
|
return {
|
|
id,
|
|
origin_name: file,
|
|
name: file + ext,
|
|
path: storedFilePath,
|
|
created_at: storedStats.birthtime,
|
|
size: storedStats.size,
|
|
ext,
|
|
type: getFileType(ext),
|
|
count: 2
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return null
|
|
}
|
|
|
|
async selectFile(options?: OpenDialogOptions): Promise<FileType[] | null> {
|
|
const defaultOptions: OpenDialogOptions = {
|
|
properties: ['openFile']
|
|
}
|
|
|
|
const dialogOptions = { ...defaultOptions, ...options }
|
|
|
|
const result = await dialog.showOpenDialog(dialogOptions)
|
|
|
|
if (result.canceled || result.filePaths.length === 0) {
|
|
return null
|
|
}
|
|
|
|
const fileMetadataPromises = result.filePaths.map(async (filePath) => {
|
|
const stats = fs.statSync(filePath)
|
|
const ext = path.extname(filePath)
|
|
const fileType = getFileType(ext)
|
|
|
|
return {
|
|
id: uuidv4(),
|
|
origin_name: path.basename(filePath),
|
|
name: path.basename(filePath),
|
|
path: filePath,
|
|
created_at: stats.birthtime,
|
|
size: stats.size,
|
|
ext: ext,
|
|
type: fileType,
|
|
count: 1
|
|
}
|
|
})
|
|
|
|
return Promise.all(fileMetadataPromises)
|
|
}
|
|
|
|
async uploadFile(file: FileType): Promise<FileType> {
|
|
const duplicateFile = await this.findDuplicateFile(file.path)
|
|
|
|
if (duplicateFile) {
|
|
return duplicateFile
|
|
}
|
|
|
|
const uuid = uuidv4()
|
|
const origin_name = path.basename(file.path)
|
|
const ext = path.extname(origin_name)
|
|
const destPath = path.join(this.storageDir, uuid + ext)
|
|
|
|
await fs.promises.copyFile(file.path, destPath)
|
|
const stats = await fs.promises.stat(destPath)
|
|
const fileType = getFileType(ext)
|
|
|
|
const fileMetadata: FileType = {
|
|
id: uuid,
|
|
origin_name,
|
|
name: uuid + ext,
|
|
path: destPath,
|
|
created_at: stats.birthtime,
|
|
size: stats.size,
|
|
ext: ext,
|
|
type: fileType,
|
|
count: 1
|
|
}
|
|
|
|
return fileMetadata
|
|
}
|
|
|
|
async getFile(filePath: string): Promise<FileType | null> {
|
|
if (!fs.existsSync(filePath)) {
|
|
return null
|
|
}
|
|
|
|
const stats = fs.statSync(filePath)
|
|
const ext = path.extname(filePath)
|
|
const fileType = getFileType(ext)
|
|
|
|
const fileInfo: FileType = {
|
|
id: uuidv4(),
|
|
origin_name: path.basename(filePath),
|
|
name: path.basename(filePath),
|
|
path: filePath,
|
|
created_at: stats.birthtime,
|
|
size: stats.size,
|
|
ext: ext,
|
|
type: fileType,
|
|
count: 1
|
|
}
|
|
|
|
return fileInfo
|
|
}
|
|
|
|
async deleteFile(id: string): Promise<void> {
|
|
await fs.promises.unlink(path.join(this.storageDir, id))
|
|
}
|
|
|
|
async readFile(id: string): Promise<string> {
|
|
const filePath = path.join(this.storageDir, id)
|
|
return fs.readFileSync(filePath, 'utf8')
|
|
}
|
|
|
|
async createTempFile(fileName: string): Promise<string> {
|
|
const tempDir = path.join(app.getPath('temp'), 'CherryStudio')
|
|
if (!fs.existsSync(tempDir)) {
|
|
fs.mkdirSync(tempDir, { recursive: true })
|
|
}
|
|
const tempFilePath = path.join(tempDir, `temp_file_${uuidv4()}_${fileName}`)
|
|
return tempFilePath
|
|
}
|
|
|
|
async writeFile(filePath: string, data: Uint8Array | string): Promise<void> {
|
|
await fs.promises.writeFile(filePath, data)
|
|
}
|
|
|
|
async base64Image(id: string): Promise<{ mime: string; base64: string; data: string }> {
|
|
const filePath = path.join(this.storageDir, id)
|
|
const data = await fs.promises.readFile(filePath)
|
|
const base64 = data.toString('base64')
|
|
const mime = `image/${path.extname(filePath).slice(1)}`
|
|
return {
|
|
mime,
|
|
base64,
|
|
data: `data:${mime};base64,${base64}`
|
|
}
|
|
}
|
|
|
|
async clear(): Promise<void> {
|
|
await fs.promises.rmdir(this.storageDir, { recursive: true })
|
|
await this.initStorageDir()
|
|
}
|
|
|
|
async open(
|
|
_: Electron.IpcMainInvokeEvent,
|
|
options: OpenDialogOptions
|
|
): Promise<{ fileName: string; content: Buffer } | null> {
|
|
try {
|
|
const result: OpenDialogReturnValue = await dialog.showOpenDialog({
|
|
title: '打开文件',
|
|
properties: ['openFile'],
|
|
filters: [{ name: '所有文件', extensions: ['*'] }],
|
|
...options
|
|
})
|
|
|
|
if (!result.canceled && result.filePaths.length > 0) {
|
|
const filePath = result.filePaths[0]
|
|
const fileName = filePath.split('/').pop() || ''
|
|
const content = await readFile(filePath)
|
|
return { fileName, content }
|
|
}
|
|
|
|
return null
|
|
} catch (err) {
|
|
logger.error('[IPC - Error]', 'An error occurred opening the file:', err)
|
|
return null
|
|
}
|
|
}
|
|
|
|
async save(
|
|
_: Electron.IpcMainInvokeEvent,
|
|
fileName: string,
|
|
content: string,
|
|
options?: SaveDialogOptions
|
|
): Promise<void> {
|
|
try {
|
|
const result: SaveDialogReturnValue = await dialog.showSaveDialog({
|
|
title: '保存文件',
|
|
defaultPath: fileName,
|
|
...options
|
|
})
|
|
|
|
if (!result.canceled && result.filePath) {
|
|
await writeFileSync(result.filePath, content, { encoding: 'utf-8' })
|
|
}
|
|
} catch (err) {
|
|
logger.error('[IPC - Error]', 'An error occurred saving the file:', err)
|
|
}
|
|
}
|
|
|
|
async saveImage(_: Electron.IpcMainInvokeEvent, name: string, data: string): Promise<void> {
|
|
try {
|
|
const filePath = dialog.showSaveDialogSync({
|
|
defaultPath: `${name}.png`,
|
|
filters: [{ name: 'PNG Image', extensions: ['png'] }]
|
|
})
|
|
|
|
if (filePath) {
|
|
const base64Data = data.replace(/^data:image\/png;base64,/, '')
|
|
fs.writeFileSync(filePath, base64Data, 'base64')
|
|
}
|
|
} catch (error) {
|
|
logger.error('[IPC - Error]', 'An error occurred saving the image:', error)
|
|
}
|
|
}
|
|
}
|
|
|
|
export default FileManager
|