import time from './time'; import Setting from './models/Setting'; import { filename, fileExtension } from './path-utils'; const md5 = require('md5'); import resolvePathWithinDir from './utils/resolvePathWithinDir'; import { Buffer } from 'buffer'; export interface Stat { birthtime: Date; mtime: Date; isDirectory(): boolean; path: string; size: number; } export interface ReadDirStatsOptions { recursive: boolean; } export default class FsDriverBase { public async stat(_path: string): Promise { throw new Error('Not implemented: stat()'); } // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied public async readFile(_path: string, _encoding = 'utf8'): Promise { throw new Error('Not implemented: readFile'); } // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied public async appendFile(_path: string, _content: string, _encoding = 'base64'): Promise { throw new Error('Not implemented: appendFile'); } public async copy(_source: string, _dest: string) { throw new Error('Not implemented: copy'); } public async chmod(_source: string, _mode: string | number) { throw new Error('Not implemented: chmod'); } // Must also create parent directories public async mkdir(_path: string) { throw new Error('Not implemented: mkdir'); } public async unlink(_path: string) { throw new Error('Not implemented: unlink'); } public async move(_source: string, _dest: string) { throw new Error('Not implemented: move'); } public async rename(source: string, dest: string) { return this.move(source, dest); } // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied public async readFileChunk(_handle: any, _length: number, _encoding = 'base64'): Promise { throw new Error('Not implemented: readFileChunk'); } // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied public async open(_path: string, _mode: any): Promise { throw new Error('Not implemented: open'); } // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied public async close(_handle: any): Promise { throw new Error('Not implemented: close'); } public async readDirStats(_path: string, _options: ReadDirStatsOptions = null): Promise { throw new Error('Not implemented: readDirStats'); } public async exists(_path: string): Promise { throw new Error('Not implemented: exists'); } public async remove(_path: string): Promise { throw new Error('Not implemented: remove'); } public async setTimestamp(_path: string, _timestampDate: Date): Promise { throw new Error('Not implemented: setTimestamp'); } public async isDirectory(path: string) { const stat = await this.stat(path); return !stat ? false : stat.isDirectory(); } public async writeFile(_path: string, _content: string, _encoding = 'base64'): Promise { throw new Error('Not implemented'); } public async md5File(_path: string): Promise { throw new Error('Not implemented: md5File'); } public resolve(..._paths: string[]): string { throw new Error('Not implemented: resolve'); } // Resolves the provided relative path to an absolute path within baseDir. The function // also checks that the absolute path is within baseDir, to avoid security issues. // It is expected that baseDir is a safe path (not user-provided). public resolveRelativePathWithinDir(baseDir: string, relativePath: string) { const resolvedPath = resolvePathWithinDir(baseDir, relativePath); if (!resolvedPath) throw new Error(`Resolved path for relative path "${relativePath}" is not within base directory "${baseDir}" (Was resolved to ${resolvedPath})`); return resolvedPath; } public getExternalDirectoryPath(): Promise { throw new Error('Not implemented: getExternalDirectoryPath'); } public isUsingAndroidSAF() { return false; } public async appendBinaryReadableToFile(path: string, readable: { read(): number[]|null }) { let data: number[]|null = null; while ((data = readable.read()) !== null) { const buff = Buffer.from(data); const base64Data = buff.toString('base64'); await this.appendFile(path, base64Data, 'base64'); } } protected async readDirStatsHandleRecursion_(basePath: string, stat: Stat, output: Stat[], options: ReadDirStatsOptions): Promise { if (options.recursive && stat.isDirectory()) { const subPath = `${basePath}/${stat.path}`; const subStats = await this.readDirStats(subPath, options); for (let j = 0; j < subStats.length; j++) { const subStat = subStats[j]; subStat.path = `${stat.path}/${subStat.path}`; output.push(subStat); } } return output; } public async findUniqueFilename(name: string, reservedNames: string[] = null, markdownSafe = false): Promise { if (reservedNames === null) { reservedNames = []; } let counter = 1; // On Windows, ./FiLe.md and ./file.md are equivalent file paths. // As such, to avoid overwriting reserved names, comparisons need to be // case-insensitive. reservedNames = reservedNames.map(name => name.toLowerCase()); const isReserved = (testName: string) => { return reservedNames.includes(testName.toLowerCase()); }; const nameNoExt = filename(name, true); let extension = fileExtension(name); if (extension) extension = `.${extension}`; let nameToTry = nameNoExt + extension; while (true) { // Check if the filename does not exist in the filesystem and is not reserved const exists = await this.exists(nameToTry) || isReserved(nameToTry); if (!exists) return nameToTry; if (!markdownSafe) { nameToTry = `${nameNoExt} (${counter})${extension}`; } else { nameToTry = `${nameNoExt}-${counter}${extension}`; } counter++; if (counter >= 1000) { nameToTry = `${nameNoExt} (${new Date().getTime()})${extension}`; await time.msleep(10); } if (counter >= 1100) throw new Error('Cannot find unique filename'); } } public async removeAllThatStartWith(dirPath: string, filenameStart: string) { if (!filenameStart || !dirPath) throw new Error('dirPath and filenameStart cannot be empty'); const stats = await this.readDirStats(dirPath); for (const stat of stats) { if (stat.path.indexOf(filenameStart) === 0) { await this.remove(`${dirPath}/${stat.path}`); } } } public async waitTillExists(path: string, timeout = 10000) { const startTime = Date.now(); while (true) { const e = await this.exists(path); if (e) return true; if (Date.now() - startTime > timeout) return false; await time.msleep(100); } } // TODO: move out of here and make it part of joplin-renderer // or assign to option using .bind(fsDriver()) public async cacheCssToFile(cssStrings: string[]) { const cssString = Array.isArray(cssStrings) ? cssStrings.join('\n') : cssStrings; const cssFilePath = `${Setting.value('tempDir')}/${md5(escape(cssString))}.css`; if (!(await this.exists(cssFilePath))) { await this.writeFile(cssFilePath, cssString, 'utf8'); } return { path: cssFilePath, mime: 'text/css', }; } // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied public async tarExtract(_options: any) { throw new Error('Not implemented: tarExtract'); } // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied public async tarCreate(_options: any, _filePaths: string[]) { throw new Error('Not implemented: tarCreate'); } }