Files
smartfile/ts/smartfile.fs.ts

404 lines
11 KiB
TypeScript
Raw Permalink Normal View History

2022-06-07 15:43:28 +02:00
import * as plugins from './smartfile.plugins.js';
import * as interpreter from './smartfile.interpreter.js';
2022-06-07 15:43:28 +02:00
import { Smartfile } from './smartfile.classes.smartfile.js';
2022-06-07 15:43:28 +02:00
import * as memory from './smartfile.memory.js';
2016-06-24 03:36:51 +02:00
/*===============================================================
============================ Checks =============================
===============================================================*/
/**
*
* @param filePath
* @returns {boolean}
*/
2019-06-27 10:42:35 +02:00
export const fileExistsSync = (filePath): boolean => {
let fileExistsBool: boolean = false;
try {
plugins.fsExtra.readFileSync(filePath);
fileExistsBool = true;
} catch (err) {
fileExistsBool = false;
}
return fileExistsBool;
};
2016-06-24 03:36:51 +02:00
/**
*
* @param filePath
* @returns {any}
*/
2019-09-27 11:00:17 +02:00
export const fileExists = async (filePath): Promise<boolean> => {
const done = plugins.smartpromise.defer<boolean>();
plugins.fs.access(filePath, 4, (err) => {
err ? done.resolve(false) : done.resolve(true);
});
return done.promise;
};
2016-06-24 03:36:51 +02:00
/**
* Checks if given path points to an existing directory
*/
2023-08-23 10:58:38 +02:00
export const isDirectory = (pathArg: string): boolean => {
try {
return plugins.fsExtra.statSync(pathArg).isDirectory();
} catch (err) {
return false;
}
};
/**
* Checks if given path points to an existing directory
*/
export const isDirectorySync = (pathArg: string): boolean => {
try {
return plugins.fsExtra.statSync(pathArg).isDirectory();
} catch (err) {
return false;
}
};
2016-06-24 03:36:51 +02:00
/**
* Checks if a given path points to an existing file
*/
2019-06-27 10:42:35 +02:00
export const isFile = (pathArg): boolean => {
return plugins.fsExtra.statSync(pathArg).isFile();
};
2016-06-24 03:36:51 +02:00
2016-06-23 17:42:08 +02:00
/*===============================================================
============================ FS ACTIONS =========================
===============================================================*/
2016-09-24 21:42:45 +02:00
/**
* copies a file from A to B on the local disk
*/
2019-06-27 10:42:35 +02:00
export const copy = async (fromArg: string, toArg: string): Promise<boolean> => {
const done = plugins.smartpromise.defer<boolean>();
plugins.fsExtra.copy(fromArg, toArg, {}, (err) => {
2019-06-27 10:42:35 +02:00
if (err) {
throw new Error(`Could not copy from ${fromArg} to ${toArg}: ${err}`);
}
done.resolve(true);
});
return done.promise;
};
2016-09-24 21:42:45 +02:00
/**
* copies a file SYNCHRONOUSLY from A to B on the local disk
*/
2019-06-27 10:42:35 +02:00
export const copySync = (fromArg: string, toArg: string): boolean => {
plugins.fsExtra.copySync(fromArg, toArg);
return true;
};
2016-09-24 21:42:45 +02:00
2016-06-28 06:57:51 +02:00
/**
* ensures that a directory is in place
*/
2019-09-27 11:00:17 +02:00
export const ensureDir = async (dirPathArg: string) => {
await plugins.fsExtra.ensureDir(dirPathArg);
};
2016-06-28 06:57:51 +02:00
/**
* ensures that a directory is in place
*/
2019-09-27 11:00:17 +02:00
export const ensureDirSync = (dirPathArg: string) => {
plugins.fsExtra.ensureDirSync(dirPathArg);
};
2016-06-28 06:57:51 +02:00
/**
* ensure an empty directory
* @executes ASYNC
*/
2019-09-27 11:00:17 +02:00
export const ensureEmptyDir = async (dirPathArg: string) => {
await plugins.fsExtra.ensureDir(dirPathArg);
await plugins.fsExtra.emptyDir(dirPathArg);
};
/**
* ensure an empty directory
* @executes SYNC
*/
2019-09-27 11:00:17 +02:00
export const ensureEmptyDirSync = (dirPathArg: string) => {
plugins.fsExtra.ensureDirSync(dirPathArg);
plugins.fsExtra.emptyDirSync(dirPathArg);
};
2016-06-23 18:39:02 +02:00
/**
2016-09-24 21:42:45 +02:00
* ensures that a file is on disk
* @param filePath the filePath to ensureDir
* @param the fileContent to place into a new file in case it doesn't exist yet
* @returns Promise<void>
* @exec ASYNC
2016-06-23 18:39:02 +02:00
*/
2019-09-27 11:00:17 +02:00
export const ensureFile = async (filePathArg, initFileStringArg): Promise<void> => {
ensureFileSync(filePathArg, initFileStringArg);
};
2016-06-23 17:42:08 +02:00
2016-06-23 18:39:02 +02:00
/**
2016-09-24 21:42:45 +02:00
* ensures that a file is on disk
* @param filePath the filePath to ensureDir
* @param the fileContent to place into a new file in case it doesn't exist yet
* @returns Promise<void>
* @exec SYNC
2016-06-23 18:39:02 +02:00
*/
2019-09-27 11:00:17 +02:00
export const ensureFileSync = (filePathArg: string, initFileStringArg: string): void => {
if (fileExistsSync(filePathArg)) {
return null;
} else {
memory.toFsSync(initFileStringArg, filePathArg);
}
};
2016-09-20 17:56:49 +02:00
2016-09-24 21:42:45 +02:00
/**
* removes a file or folder from local disk
*/
2019-09-27 11:00:17 +02:00
export const remove = async (pathArg: string): Promise<void> => {
await plugins.fsExtra.remove(pathArg);
};
2016-06-23 17:42:08 +02:00
2016-06-23 18:39:02 +02:00
/**
* removes a file SYNCHRONOUSLY from local disk
*/
2019-09-27 11:00:17 +02:00
export const removeSync = (pathArg: string): void => {
plugins.fsExtra.removeSync(pathArg);
};
2016-06-23 17:42:08 +02:00
2016-09-29 14:17:46 +02:00
/**
* removes an array of filePaths from disk
*/
2019-09-27 11:00:17 +02:00
export const removeMany = async (filePathArrayArg: string[]) => {
const promiseArray: Array<Promise<void>> = [];
for (const filePath of filePathArrayArg) {
promiseArray.push(remove(filePath));
}
2019-09-27 11:00:17 +02:00
await Promise.all(promiseArray);
};
2016-09-29 14:17:46 +02:00
/**
* like removeFilePathArray but SYNCHRONOUSLY
*/
2019-09-27 11:00:17 +02:00
export const removeManySync = (filePathArrayArg: string[]): void => {
for (const filePath of filePathArrayArg) {
removeSync(filePath);
}
};
2016-09-29 14:17:46 +02:00
2016-06-23 17:42:08 +02:00
/*===============================================================
============================ Write/Read =========================
===============================================================*/
/**
*
* @param filePathArg
* @param fileTypeArg
* @returns {any}
*/
2019-09-27 11:00:17 +02:00
export const toObjectSync = (filePathArg, fileTypeArg?) => {
2019-01-27 03:11:10 +01:00
const fileString = plugins.fsExtra.readFileSync(filePathArg, 'utf8');
let fileType;
2020-10-05 16:20:57 +00:00
fileTypeArg ? (fileType = fileTypeArg) : (fileType = interpreter.filetype(filePathArg));
return interpreter.objectFile(fileString, fileType);
};
2016-06-23 17:42:08 +02:00
/**
* reads a file content to a String
*/
2019-06-27 10:42:35 +02:00
export const toStringSync = (filePath: string): string => {
2020-03-04 16:31:13 +00:00
const encoding = plugins.smartmime.getEncoding(filePath);
2020-10-05 16:20:57 +00:00
let fileString: string | Buffer = plugins.fsExtra.readFileSync(filePath, encoding);
if (Buffer.isBuffer(fileString)) {
fileString = fileString.toString('binary');
}
return fileString;
};
2021-11-30 16:34:35 +01:00
export const toBuffer = async (filePath: string): Promise<Buffer> => {
return plugins.fsExtra.readFile(filePath);
};
2020-03-04 16:31:13 +00:00
export const toBufferSync = (filePath: string): Buffer => {
return plugins.fsExtra.readFileSync(filePath);
2020-03-15 18:58:46 +00:00
};
2020-03-04 16:31:13 +00:00
2019-09-27 11:00:17 +02:00
export const fileTreeToHash = async (dirPathArg: string, miniMatchFilter: string) => {
const fileTreeObject = await fileTreeToObject(dirPathArg, miniMatchFilter);
let combinedString = '';
for (const smartfile of fileTreeObject) {
2023-01-09 15:32:37 +01:00
combinedString += await smartfile.getHash();
2019-09-27 11:00:17 +02:00
}
const hash = await plugins.smarthash.sha256FromString(combinedString);
return hash;
};
/**
* creates a smartfile array from a directory
* @param dirPathArg the directory to start from
* @param miniMatchFilter a minimatch filter of what files to include
*/
2019-06-26 21:17:58 +02:00
export const fileTreeToObject = async (dirPathArg: string, miniMatchFilter: string) => {
2017-04-30 15:37:34 +02:00
// handle absolute miniMatchFilter
let dirPath: string;
2017-04-30 15:37:34 +02:00
if (plugins.path.isAbsolute(miniMatchFilter)) {
dirPath = '/';
2017-04-30 15:37:34 +02:00
} else {
2022-06-07 15:43:28 +02:00
dirPath = plugins.smartpath.transform.toAbsolute(dirPathArg) as string;
2017-04-30 15:37:34 +02:00
}
2019-06-26 21:17:58 +02:00
const fileTree = await listFileTree(dirPath, miniMatchFilter);
const smartfileArray: Smartfile[] = [];
2019-09-27 11:00:17 +02:00
for (const filePath of fileTree) {
const readPath = ((): string => {
2017-08-02 13:10:30 +02:00
if (!plugins.path.isAbsolute(filePath)) {
return plugins.path.join(dirPath, filePath);
2017-08-02 13:10:30 +02:00
} else {
return filePath;
2017-08-02 13:10:30 +02:00
}
})();
2019-09-27 11:00:17 +02:00
const fileContentString = toStringSync(readPath);
2017-05-01 19:49:34 +02:00
2017-05-07 23:16:25 +02:00
// push a read file as Smartfile
smartfileArray.push(
new Smartfile({
2019-06-26 21:17:58 +02:00
contentBuffer: Buffer.from(fileContentString),
base: dirPath,
path: filePath,
})
);
}
return smartfileArray;
};
2016-06-23 17:42:08 +02:00
2016-06-23 18:39:02 +02:00
/**
* lists Folders in a directory on local disk
2018-11-22 23:23:26 +01:00
* @returns Promise with an array that contains the folder names
2016-06-23 18:39:02 +02:00
*/
2019-09-27 11:00:17 +02:00
export const listFolders = async (pathArg: string, regexFilter?: RegExp): Promise<string[]> => {
2019-01-27 03:11:10 +01:00
return listFoldersSync(pathArg, regexFilter);
};
2016-06-23 17:42:08 +02:00
2016-06-23 18:39:02 +02:00
/**
* lists Folders SYNCHRONOUSLY in a directory on local disk
2016-06-28 06:57:51 +02:00
* @returns an array with the folder names as strings
2016-06-23 18:39:02 +02:00
*/
2019-09-27 11:00:17 +02:00
export const listFoldersSync = (pathArg: string, regexFilter?: RegExp): string[] => {
let folderArray = plugins.fsExtra.readdirSync(pathArg).filter((file) => {
return plugins.fsExtra.statSync(plugins.path.join(pathArg, file)).isDirectory();
});
if (regexFilter) {
folderArray = folderArray.filter((fileItem) => {
return regexFilter.test(fileItem);
});
}
return folderArray;
};
/**
* lists Files in a directory on local disk
* @returns Promise
*/
2019-09-27 11:00:17 +02:00
export const listFiles = async (pathArg: string, regexFilter?: RegExp): Promise<string[]> => {
2019-01-27 03:11:10 +01:00
return listFilesSync(pathArg, regexFilter);
};
/**
* lists Files SYNCHRONOUSLY in a directory on local disk
* @returns an array with the folder names as strings
*/
2019-09-27 11:00:17 +02:00
export const listFilesSync = (pathArg: string, regexFilter?: RegExp): string[] => {
let fileArray = plugins.fsExtra.readdirSync(pathArg).filter((file) => {
return plugins.fsExtra.statSync(plugins.path.join(pathArg, file)).isFile();
});
if (regexFilter) {
fileArray = fileArray.filter((fileItem) => {
return regexFilter.test(fileItem);
});
}
return fileArray;
};
/**
* lists all items (folders AND files) in a directory on local disk
2016-09-17 23:11:44 +02:00
* @returns Promise<string[]>
*/
2019-09-27 11:00:17 +02:00
export const listAllItems = async (pathArg: string, regexFilter?: RegExp): Promise<string[]> => {
2019-01-27 03:11:10 +01:00
return listAllItemsSync(pathArg, regexFilter);
};
/**
2016-09-17 23:11:44 +02:00
* lists all items (folders AND files) in a directory on local disk
* @returns an array with the folder names as strings
2016-09-17 23:11:44 +02:00
* @executes SYNC
*/
2019-09-27 11:00:17 +02:00
export const listAllItemsSync = (pathArg: string, regexFilter?: RegExp): string[] => {
let allItmesArray = plugins.fsExtra.readdirSync(pathArg).filter((file) => {
return plugins.fsExtra.statSync(plugins.path.join(pathArg, file)).isFile();
});
if (regexFilter) {
allItmesArray = allItmesArray.filter((fileItem) => {
return regexFilter.test(fileItem);
});
}
return allItmesArray;
};
2016-09-17 23:11:44 +02:00
/**
* lists a file tree using a miniMatch filter
2016-09-30 16:16:11 +02:00
* note: if the miniMatch Filter is an absolute path, the cwdArg will be omitted
2016-09-17 23:11:44 +02:00
* @returns Promise<string[]> string array with the absolute paths of all matching files
*/
2019-09-27 11:00:17 +02:00
export const listFileTree = async (
dirPathArg: string,
miniMatchFilter: string,
absolutePathsBool: boolean = false
): Promise<string[]> => {
// handle absolute miniMatchFilter
let dirPath: string;
if (plugins.path.isAbsolute(miniMatchFilter)) {
dirPath = '/';
} else {
dirPath = dirPathArg;
}
2019-01-27 03:11:10 +01:00
const options = {
2017-05-27 01:07:32 +02:00
cwd: dirPath,
2017-05-27 02:54:22 +02:00
nodir: true,
dot: true,
};
2019-09-27 11:00:17 +02:00
2023-06-23 18:43:49 +02:00
let fileList = await plugins.glob.glob(miniMatchFilter, options);
2019-06-26 21:17:58 +02:00
if (absolutePathsBool) {
fileList = fileList.map((filePath) => {
2019-06-26 21:17:58 +02:00
return plugins.path.resolve(plugins.path.join(dirPath, filePath));
});
}
return fileList;
};
2021-12-03 00:24:10 +01:00
/**
* checks wether a file is ready for processing
*/
2021-12-20 15:11:21 +01:00
export const waitForFileToBeReady = async (filePathArg: string): Promise<void> => {
if (!plugins.path.isAbsolute(filePathArg)) {
2021-12-03 00:24:10 +01:00
filePathArg = plugins.path.resolve(filePathArg);
2021-12-20 15:11:21 +01:00
}
const limitedArray = new plugins.lik.LimitedArray<number>(3);
let fileReady = false;
while (!fileReady) {
const stats = await plugins.fsExtra.stat(filePathArg);
limitedArray.addOne(stats.size);
if (
limitedArray.array.length < 3 ||
2022-03-11 09:46:54 +01:00
!(
limitedArray.array[0] === limitedArray.array[1] &&
limitedArray.array[1] === limitedArray.array[2]
)
2021-12-20 15:11:21 +01:00
) {
await plugins.smartdelay.delayFor(5000);
} else {
fileReady = true;
}
}
};