smartbucket/ts/classes.file.ts

274 lines
7.6 KiB
TypeScript
Raw Normal View History

2024-05-21 01:22:21 +02:00
import * as plugins from './plugins.js';
import * as helpers from './helpers.js';
import * as interfaces from './interfaces.js';
2024-05-21 01:22:21 +02:00
import { Directory } from './classes.directory.js';
import { MetaData } from './classes.metadata.js';
/**
* represents a file in a directory
*/
export class File {
// STATIC
/**
* creates a file in draft mode
* you need to call .save() to store it in s3
* @param optionsArg
*/
public static async create(optionsArg: {
directory: Directory;
name: string;
contents: Buffer | string | plugins.stream.Readable;
/**
* if contents are of type string, you can specify the encoding here
*/
encoding?: 'utf8' | 'binary';
}): Promise<File> {
const contents =
typeof optionsArg.contents === 'string'
? Buffer.from(optionsArg.contents, optionsArg.encoding)
: optionsArg.contents;
const file = new File({
directoryRefArg: optionsArg.directory,
fileName: optionsArg.name,
});
2024-06-03 21:35:08 +02:00
if (contents instanceof plugins.stream.Readable) {
2024-06-09 16:02:33 +02:00
await optionsArg.directory.fastPutStream({
path: optionsArg.name,
stream: contents,
});
2024-06-03 21:35:08 +02:00
} else {
2024-05-21 01:22:21 +02:00
await optionsArg.directory.fastPut({
path: optionsArg.name,
contents: contents,
});
}
return file;
}
// INSTANCE
public parentDirectoryRef: Directory;
public name: string;
/**
* get the full path to the file
* @returns the full path to the file
*/
2024-05-21 01:22:21 +02:00
public getBasePath(): string {
return plugins.path.join(this.parentDirectoryRef.getBasePath(), this.name);
2024-06-03 21:35:08 +02:00
}
2024-05-21 01:22:21 +02:00
constructor(optionsArg: { directoryRefArg: Directory; fileName: string }) {
this.parentDirectoryRef = optionsArg.directoryRefArg;
this.name = optionsArg.fileName;
}
public async getContentsAsString(): Promise<string> {
const fileBuffer = await this.getContents();
return fileBuffer.toString();
}
public async getContents(): Promise<Buffer> {
const resultBuffer = await this.parentDirectoryRef.bucketRef.fastGet({
path: this.getBasePath(),
});
return resultBuffer;
}
2024-06-03 21:35:08 +02:00
public async getReadStream(typeArg: 'webstream'): Promise<ReadableStream>;
public async getReadStream(typeArg: 'nodestream'): Promise<plugins.stream.Readable>;
public async getReadStream(
typeArg: 'nodestream' | 'webstream'
): Promise<ReadableStream | plugins.stream.Readable> {
const readStream = this.parentDirectoryRef.bucketRef.fastGetStream(
{
path: this.getBasePath(),
},
typeArg as any
);
return readStream;
2024-05-21 01:22:21 +02:00
}
/**
2024-06-03 21:35:08 +02:00
* deletes this file
2024-05-21 01:22:21 +02:00
*/
public async delete(optionsArg?: { mode: 'trash' | 'permanent' }) {
2024-06-03 21:35:08 +02:00
optionsArg = {
...{
2024-06-03 21:35:08 +02:00
mode: 'permanent',
},
...optionsArg,
};
2024-06-03 21:35:08 +02:00
if (optionsArg.mode === 'permanent') {
2024-05-21 01:22:21 +02:00
await this.parentDirectoryRef.bucketRef.fastRemove({
2024-06-03 21:35:08 +02:00
path: this.getBasePath(),
});
if (!this.name.endsWith('.metadata')) {
if (await this.hasMetaData()) {
const metadata = await this.getMetaData();
await metadata.metadataFile.delete(optionsArg);
}
2024-06-03 21:35:08 +02:00
}
} else if (optionsArg.mode === 'trash') {
const metadata = await this.getMetaData();
await metadata.storeCustomMetaData({
key: 'recycle',
value: {
deletedAt: Date.now(),
originalPath: this.getBasePath(),
},
});
2024-06-10 16:47:20 +02:00
const trash = await this.parentDirectoryRef.bucketRef.getTrash();
const trashDir = await trash.getTrashDir();
2024-06-03 21:35:08 +02:00
await this.move({
directory: trashDir,
2024-06-10 16:47:20 +02:00
path: await trash.getTrashKeyByOriginalBasePath(this.getBasePath()),
2024-05-21 01:22:21 +02:00
});
}
2024-05-21 01:22:21 +02:00
await this.parentDirectoryRef.listFiles();
}
/**
* allows locking the file
* @param optionsArg
*/
public async lock(optionsArg?: { timeoutMillis?: number }) {
const metadata = await this.getMetaData();
await metadata.setLock({
lock: 'locked',
2024-05-21 18:42:55 +02:00
expires: Date.now() + (optionsArg?.timeoutMillis || 1000),
2024-05-21 01:22:21 +02:00
});
}
/**
* actively unlocks a file
*
*/
public async unlock(optionsArg?: {
/**
* unlock the file even if not locked from this instance
*/
force?: boolean;
}) {
2024-05-21 18:42:55 +02:00
const metadata = await this.getMetaData();
await metadata.removeLock({
force: optionsArg?.force,
});
2024-05-21 01:22:21 +02:00
}
public async updateWithContents(optionsArg: {
2024-06-03 21:35:08 +02:00
contents: Buffer | string | plugins.stream.Readable | ReadableStream;
2024-05-21 01:22:21 +02:00
encoding?: 'utf8' | 'binary';
}) {
2024-06-03 21:35:08 +02:00
if (
optionsArg.contents instanceof plugins.stream.Readable ||
optionsArg.contents instanceof ReadableStream
) {
2024-05-21 01:22:21 +02:00
await this.parentDirectoryRef.bucketRef.fastPutStream({
path: this.getBasePath(),
2024-06-09 16:02:33 +02:00
readableStream: optionsArg.contents,
overwrite: true,
2024-05-21 01:22:21 +02:00
});
} else if (Buffer.isBuffer(optionsArg.contents)) {
await this.parentDirectoryRef.bucketRef.fastPut({
path: this.getBasePath(),
contents: optionsArg.contents,
overwrite: true,
2024-05-21 01:22:21 +02:00
});
} else if (typeof optionsArg.contents === 'string') {
await this.parentDirectoryRef.bucketRef.fastPut({
path: this.getBasePath(),
contents: Buffer.from(optionsArg.contents, optionsArg.encoding),
overwrite: true,
2024-05-21 01:22:21 +02:00
});
}
}
/**
* moves the file to another directory
*/
public async move(pathDescriptorArg: interfaces.IPathDecriptor) {
let moveToPath: string = '';
const isDirectory = await this.parentDirectoryRef.bucketRef.isDirectory(pathDescriptorArg);
if (isDirectory) {
moveToPath = await helpers.reducePathDescriptorToPath({
...pathDescriptorArg,
path: plugins.path.join(pathDescriptorArg.path!, this.name),
});
} else {
moveToPath = await helpers.reducePathDescriptorToPath(pathDescriptorArg);
}
// lets move the file
await this.parentDirectoryRef.bucketRef.fastMove({
sourcePath: this.getBasePath(),
destinationPath: moveToPath,
overwrite: true,
});
// lets move the metadatafile
if (!this.name.endsWith('.metadata')) {
const metadata = await this.getMetaData();
await this.parentDirectoryRef.bucketRef.fastMove({
sourcePath: metadata.metadataFile.getBasePath(),
destinationPath: moveToPath + '.metadata',
overwrite: true,
});
}
// lets update references of this
const baseDirectory = await this.parentDirectoryRef.bucketRef.getBaseDirectory();
this.parentDirectoryRef = await baseDirectory.getSubDirectoryByNameStrict(
pathDescriptorArg.directory?.getBasePath()!
);
this.name = pathDescriptorArg.path!;
}
public async hasMetaData(): Promise<boolean> {
if (!this.name.endsWith('.metadata')) {
const hasMetadataBool = MetaData.hasMetaData({
file: this,
});
return hasMetadataBool;
} else {
return false;
}
}
2024-05-21 01:22:21 +02:00
/**
* allows updating the metadata of a file
* @param updatedMetadata
*/
public async getMetaData() {
2024-05-21 18:42:55 +02:00
if (this.name.endsWith('.metadata')) {
throw new Error('metadata files cannot have metadata');
}
2024-05-21 01:22:21 +02:00
const metadata = await MetaData.createForFile({
file: this,
});
return metadata;
}
2024-05-21 18:42:55 +02:00
/**
* gets the contents as json
*/
public async getJsonData() {
const json = await this.getContentsAsString();
const parsed = await JSON.parse(json);
return parsed;
}
public async writeJsonData(dataArg: any) {
await this.updateWithContents({
contents: JSON.stringify(dataArg),
});
}
public async getMagicBytes(optionsArg: { length: number }): Promise<Buffer> {
return this.parentDirectoryRef.bucketRef.getMagicBytes({
path: this.getBasePath(),
length: optionsArg.length,
});
}
2024-05-21 01:22:21 +02:00
}