Skip to content

Commit

Permalink
fix(nestjs-tools-fastify-upload): implement temporary file handling a…
Browse files Browse the repository at this point in the history
…nd improve stream processing
  • Loading branch information
getlarge committed Nov 6, 2024
1 parent 194a403 commit 07c2684
Showing 1 changed file with 27 additions and 8 deletions.
35 changes: 27 additions & 8 deletions packages/fastify-upload/src/lib/storage/stream-storage.ts
Original file line number Diff line number Diff line change
@@ -1,17 +1,35 @@
import { PassThrough, Readable } from 'node:stream';
import { finished } from 'node:stream/promises';
import { createReadStream, createWriteStream } from 'node:fs';
import { rm } from 'node:fs/promises';
import { tmpdir } from 'node:os';
import { normalize, resolve, sep } from 'node:path';
import { Readable } from 'node:stream';
import { finished, pipeline } from 'node:stream/promises';

import { MultipartFile, Storage, StorageFile } from './storage';

export interface StreamStorageFile extends StorageFile {
stream: Readable;
}

const temporaryFileOutput = (filename: string) => {
const safeFileName = normalize(filename).replace(/^(\.\.(\/|\\|$))+/, '');
const fullPath = resolve(tmpdir(), safeFileName);
// Ensure the resolved path starts with the intended storagePath to prevent path traversal
if (!fullPath.startsWith(resolve(tmpdir() + sep))) {
throw new Error('Invalid file path');
}
return fullPath;
};
export class StreamStorage extends Storage<StreamStorageFile> {
handleFile(file: MultipartFile): Promise<StreamStorageFile> {
async handleFile(file: MultipartFile): Promise<StreamStorageFile> {
const { encoding, mimetype, fieldname } = file;
const stream = file.file.pipe(new PassThrough());

/**
* force the stream to be consumed as required by Fastify and Busboy
* @see https://github.com/fastify/fastify-multipart?tab=readme-ov-file#usage
* */
const output = temporaryFileOutput(file.filename);
await pipeline(file.file, createWriteStream(output));
const stream = createReadStream(output);
return Promise.resolve({
size: file.file.readableLength,
stream,
Expand All @@ -26,10 +44,11 @@ export class StreamStorage extends Storage<StreamStorageFile> {
if (!force) {
await finished(file.stream);
file.stream.destroy();
return;
}
if (!file.stream.destroyed) {
} else if (!file.stream.destroyed) {
file.stream.destroy();
}
await rm(temporaryFileOutput(file.originalFilename)).catch(() => {
// ignore
});
}
}

0 comments on commit 07c2684

Please sign in to comment.