mirror of https://github.com/laurent22/joplin.git
262 lines
8.3 KiB
TypeScript
262 lines
8.3 KiB
TypeScript
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 interface RemoveOptions {
|
|
recursive?: boolean;
|
|
}
|
|
|
|
|
|
export default class FsDriverBase {
|
|
|
|
public async stat(_path: string): Promise<Stat> {
|
|
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<any> {
|
|
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<any> {
|
|
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<string> {
|
|
throw new Error('Not implemented: readFileChunk');
|
|
}
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
public async readFileChunkAsBuffer(handle: any, length: number): Promise<Buffer> {
|
|
const chunk = await this.readFileChunk(handle, length, 'base64');
|
|
if (chunk) {
|
|
return Buffer.from(chunk, 'base64');
|
|
} else {
|
|
return null;
|
|
}
|
|
}
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
public async open(_path: string, _mode: any): Promise<any> {
|
|
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<any> {
|
|
throw new Error('Not implemented: close');
|
|
}
|
|
|
|
// Like .readFile, but returns a File object.
|
|
public async fileAtPath(_path: string): Promise<File> {
|
|
throw new Error('Not implemented: fileAtPath');
|
|
}
|
|
|
|
public async readDirStats(_path: string, _options: ReadDirStatsOptions = null): Promise<Stat[]> {
|
|
throw new Error('Not implemented: readDirStats');
|
|
}
|
|
|
|
public async exists(_path: string): Promise<boolean> {
|
|
throw new Error('Not implemented: exists');
|
|
}
|
|
|
|
public async remove(_path: string, _options: RemoveOptions = null): Promise<void> {
|
|
throw new Error('Not implemented: remove');
|
|
}
|
|
|
|
public async setTimestamp(_path: string, _timestampDate: Date): Promise<void> {
|
|
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<void> {
|
|
throw new Error('Not implemented');
|
|
}
|
|
|
|
public async md5File(_path: string): Promise<string> {
|
|
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<string | undefined> {
|
|
throw new Error('Not implemented: getExternalDirectoryPath');
|
|
}
|
|
|
|
public getCacheDirectoryPath(): string {
|
|
throw new Error('Not implemented: getCacheDirectoryPath');
|
|
}
|
|
|
|
public getAppDirectoryPath(): string {
|
|
throw new Error('Not implemented: getCacheDirectoryPath');
|
|
}
|
|
|
|
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<Stat[]> {
|
|
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<string> {
|
|
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');
|
|
}
|
|
|
|
}
|