smartfile/ts/classes.smartfile.ts

354 lines
9.2 KiB
TypeScript
Raw Permalink Normal View History

2024-04-02 20:53:02 +02:00
import * as plugins from './plugins.js';
2023-11-04 20:07:43 +01:00
import * as fs from './fs.js';
import * as memory from './memory.js';
export interface ISmartfileConstructorOptions {
path: string;
contentBuffer: Buffer;
base: string;
}
2016-09-20 17:56:49 +02:00
2017-04-27 16:48:08 +02:00
/**
2023-11-04 20:07:43 +01:00
* an vinyl file compatible in memory file class
2017-04-27 16:48:08 +02:00
*/
2023-11-04 20:07:43 +01:00
export class SmartFile extends plugins.smartjson.Smartjson {
2019-01-27 03:11:10 +01:00
// ======
// STATIC
// ======
/**
* creates a Smartfile from a filePath
* @param filePath
*/
public static async fromFilePath(filePath: string, baseArg: string = process.cwd()) {
2019-01-27 03:11:10 +01:00
filePath = plugins.path.resolve(filePath);
const fileBuffer = fs.toBufferSync(filePath);
2023-11-04 20:07:43 +01:00
const smartfile = new SmartFile({
contentBuffer: fileBuffer,
base: baseArg,
path: plugins.path.relative(baseArg, filePath),
2019-01-27 03:11:10 +01:00
});
return smartfile;
}
public static async fromBuffer(
filePath: string,
contentBufferArg: Buffer,
baseArg: string = process.cwd()
) {
2023-11-04 20:07:43 +01:00
const smartfile = new SmartFile({
contentBuffer: contentBufferArg,
base: baseArg,
path: plugins.path.relative(baseArg, filePath),
});
return smartfile;
}
public static async fromString(
filePath: string,
contentStringArg: string,
encodingArg: 'utf8' | 'binary',
baseArg = process.cwd()
) {
2023-11-04 20:07:43 +01:00
const smartfile = new SmartFile({
contentBuffer: Buffer.from(contentStringArg, encodingArg),
base: baseArg,
path: plugins.path.relative(baseArg, filePath),
});
return smartfile;
}
2021-04-26 08:24:36 +00:00
public static async fromFoldedJson(foldedJsonArg: string) {
2023-11-04 20:07:43 +01:00
return new SmartFile(plugins.smartjson.parse(foldedJsonArg));
2021-04-26 08:24:36 +00:00
}
2023-11-03 01:25:37 +01:00
/**
* creates a Smartfile from a ReadableStream
* @param stream a readable stream that provides file content
* @param filePath the file path to associate with the content
* @param baseArg the base path to use for the file
*/
public static async fromStream(
stream: plugins.stream.Readable,
filePath: string,
baseArg: string = process.cwd()
2023-11-04 20:07:43 +01:00
): Promise<SmartFile> {
return new Promise<SmartFile>((resolve, reject) => {
2023-11-03 01:25:37 +01:00
const chunks: Buffer[] = [];
stream.on('data', (chunk) => chunks.push(Buffer.from(chunk)));
stream.on('error', (error) => reject(error));
stream.on('end', () => {
const contentBuffer = Buffer.concat(chunks);
2023-11-04 20:07:43 +01:00
const smartfile = new SmartFile({
2023-11-03 01:25:37 +01:00
contentBuffer: contentBuffer,
base: baseArg,
path: plugins.path.relative(baseArg, filePath),
});
resolve(smartfile);
});
});
}
public static async fromUrl(urlArg: string) {
2023-11-24 19:15:41 +01:00
const response = await plugins.smartrequest.getBinary(urlArg);
const smartfile = await SmartFile.fromBuffer(urlArg, response.body);
return smartfile;
}
2019-01-27 03:11:10 +01:00
// ========
// INSTANCE
// ========
2017-04-27 16:48:08 +02:00
/**
* the relative path of the file
2017-04-27 16:48:08 +02:00
*/
2020-10-05 16:20:57 +00:00
@plugins.smartjson.foldDec()
public path: string;
2017-04-27 16:48:08 +02:00
2017-05-07 23:00:56 +02:00
/**
2020-10-06 00:57:47 +00:00
* a parsed path
2017-05-07 23:00:56 +02:00
*/
2022-03-11 09:46:54 +01:00
public get parsedPath(): plugins.path.ParsedPath {
2021-12-01 10:47:29 +01:00
return plugins.path.parse(this.path);
2022-03-11 09:46:54 +01:00
}
public get absolutePath() {
2021-12-01 10:47:29 +01:00
return plugins.path.join(this.base, this.path);
}
public get absoluteParsedPath() {
return plugins.path.parse(this.absolutePath);
}
2017-05-07 23:00:56 +02:00
2017-04-29 17:20:09 +02:00
/**
* the content of the file as Buffer
*/
2020-10-06 00:57:47 +00:00
@plugins.smartjson.foldDec()
public contentBuffer: Buffer;
2017-04-29 17:20:09 +02:00
2017-04-27 16:48:08 +02:00
/**
* The current working directory of the file
* Note:this is similar to gulp and different from native node path base
2017-04-27 16:48:08 +02:00
*/
2020-10-06 00:57:47 +00:00
@plugins.smartjson.foldDec()
public base: string;
2017-04-27 16:48:08 +02:00
/**
* sync the file with disk
*/
2020-10-06 00:57:47 +00:00
@plugins.smartjson.foldDec()
public sync: boolean;
2017-04-27 16:48:08 +02:00
/**
* the constructor of Smartfile
* @param optionsArg
*/
2017-05-07 23:00:56 +02:00
constructor(optionsArg: ISmartfileConstructorOptions) {
2020-10-05 16:20:57 +00:00
super();
if (optionsArg.contentBuffer) {
this.contentBuffer = optionsArg.contentBuffer;
2017-04-29 17:20:09 +02:00
} else {
console.log('created empty Smartfile?');
2016-09-20 17:56:49 +02:00
}
this.path = optionsArg.path;
this.base = optionsArg.base;
2017-04-27 16:48:08 +02:00
}
/**
* set contents from string
* @param contentString
*/
public setContentsFromString(contentString: string, encodingArg: 'utf8' | 'binary' = 'utf8') {
2023-11-03 02:31:57 +01:00
this.contents = Buffer.from(contentString, encodingArg);
}
2017-04-27 16:48:08 +02:00
/**
2020-10-09 15:15:47 +00:00
* write file to disk at its original location
2018-02-16 21:57:44 +01:00
* Behaviours:
* - no argument write to exactly where the file was picked up
2017-04-27 16:48:08 +02:00
*/
2020-10-09 15:15:47 +00:00
public async write() {
2024-04-12 14:51:23 +02:00
let writePath = plugins.smartpath.transform.makeAbsolute(this.path, this.base);
2024-04-12 15:00:55 +02:00
console.log(`writing to ${writePath}`);
2024-04-12 14:44:12 +02:00
await memory.toFs(this.contentBuffer, writePath);
2020-10-09 15:15:47 +00:00
}
/**
* writes the file to path given as argument
* note: if the path is not absolute, takes process.cwd() as base
2020-10-09 15:15:47 +00:00
* @param filePathArg
*/
public async writeToDiskAtPath(filePathArg: string) {
if (!plugins.path.isAbsolute(filePathArg)) {
filePathArg = plugins.path.join(process.cwd(), filePathArg);
}
await memory.toFs(this.contentBuffer, filePathArg);
}
2021-11-30 16:34:35 +01:00
/**
* writes the file to a directory combined with the relative path portion
* @param dirPathArg
* @returns
*/
2020-10-09 15:15:47 +00:00
public async writeToDir(dirPathArg: string) {
2022-06-07 15:43:28 +02:00
dirPathArg = plugins.smartpath.transform.toAbsolute(dirPathArg) as string;
2020-10-11 15:34:24 +00:00
const filePath = plugins.path.join(dirPathArg, this.path);
2020-10-09 15:15:47 +00:00
await memory.toFs(this.contentBuffer, filePath);
2020-10-11 15:34:24 +00:00
return filePath;
2017-04-27 16:48:08 +02:00
}
/**
* read file from disk
*/
2021-11-30 16:34:35 +01:00
public async read() {
this.contentBuffer = await fs.toBuffer(plugins.path.join(this.base, this.path));
2021-11-30 16:34:35 +01:00
}
/**
* deletes the file from disk at its original location
*/
public async delete() {
await fs.remove(plugins.path.join(this.base, this.path));
2021-11-30 16:34:35 +01:00
}
2017-04-30 18:13:17 +02:00
/**
* Renames the file to the specified new name.
* - Updates the `path` property with the new name.
* - Writes the file to the new location if it exists on disk.
* @param newName The new name of the file (including extension if applicable).
* @param writeToDisk (optional) If true, also renames the file on the disk.
* @returns The updated file path after renaming.
*/
public async rename(newName: string, writeToDisk: boolean = false): Promise<string> {
// Validate the new name
if (!newName || typeof newName !== 'string') {
throw new Error('Invalid new name provided.');
}
// Extract the directory path
const oldFilePath = this.path;
const dirPath = plugins.path.dirname(this.path);
// Create the new file path
const newFilePath = plugins.path.join(dirPath, newName);
// Update the `path` property
this.path = newFilePath;
// Optionally write the renamed file to disk
if (writeToDisk) {
const oldAbsolutePath = plugins.smartpath.transform.makeAbsolute(oldFilePath, this.base);
const newAbsolutePath = plugins.smartpath.transform.makeAbsolute(newFilePath, this.base);
// Rename the file on disk
await plugins.fsExtra.rename(oldAbsolutePath, newAbsolutePath);
}
// Return the new path
return this.path;
}
2017-04-30 18:13:17 +02:00
// -----------------------------------------------
// vinyl compatibility
// -----------------------------------------------
/**
* vinyl-compatibility: alias of this.contentBuffer
*/
get contents(): Buffer {
return this.contentBuffer;
2017-04-30 18:13:17 +02:00
}
set contents(contentsArg) {
this.contentBuffer = contentsArg;
2017-04-30 18:13:17 +02:00
}
/**
* vinyl-compatibility
*/
public get cwd() {
return process.cwd();
2017-04-30 18:13:17 +02:00
}
/**
* return relative path of file
*/
public get relative(): string {
return this.path;
2017-04-30 18:13:17 +02:00
}
/**
* return truw when the file has content
*/
public isNull(): boolean {
2017-04-30 18:13:17 +02:00
if (!this.contentBuffer) {
return true;
2017-04-30 18:13:17 +02:00
}
return false;
2017-04-30 18:13:17 +02:00
}
/**
* return true if contents are Buffer
*/
public isBuffer(): boolean {
2017-04-30 18:13:17 +02:00
if (this.contents instanceof Buffer) {
return true;
2017-04-30 18:13:17 +02:00
}
return false;
2017-04-30 18:13:17 +02:00
}
public isDirectory() {
return false;
2017-04-30 18:13:17 +02:00
}
public isStream() {
return false;
2017-04-30 18:13:17 +02:00
}
public isSymbolic() {
return false;
}
2017-05-27 23:47:39 +02:00
2023-01-09 15:32:37 +01:00
public async getHash(typeArg: 'path' | 'content' | 'all' = 'all') {
const pathHash = await plugins.smarthash.sha256FromString(this.path);
const contentHash = await plugins.smarthash.sha256FromBuffer(this.contentBuffer);
const combinedHash = await plugins.smarthash.sha256FromString(pathHash + contentHash);
2023-01-09 15:34:05 +01:00
switch (typeArg) {
2023-01-09 15:32:37 +01:00
case 'path':
return pathHash;
case 'content':
return contentHash;
case 'all':
default:
return combinedHash;
}
}
2017-05-27 23:47:39 +02:00
// update things
public updateFileName(fileNameArg: string) {
const oldFileName = this.parsedPath.base;
this.path = this.path.replace(new RegExp(oldFileName + '$'), fileNameArg);
2017-05-27 23:47:39 +02:00
}
2022-07-24 23:04:51 +02:00
public async editContentAsString(editFuncArg: (fileStringArg: string) => Promise<string>) {
const newFileString = await editFuncArg(this.contentBuffer.toString());
this.contentBuffer = Buffer.from(newFileString);
}
2023-11-03 02:24:36 +01:00
/**
* Returns a ReadableStream from the file's content buffer
*/
public getStream(): plugins.stream.Readable {
const stream = new plugins.stream.Readable();
stream.push(this.contentBuffer); // Push the content buffer to the stream
stream.push(null); // Push null to signify the end of the stream (EOF)
return stream;
}
2024-06-07 17:13:07 +02:00
/**
* Returns the size of the file in bytes
*/
public async getSize(): Promise<number> {
return this.contentBuffer.length;
}
2016-09-20 17:56:49 +02:00
}