1

我有一个客户端应用程序,用户可以在其中上传图像。我在我的 Node JS 应用程序中收到此图像作为可读数据,然后在保存之前对其进行操作,如下所示:

uploadPhoto: async (server, request) => {  

        try {         

            const randomString = `${uuidv4()}.jpg`;
            
            const stream = Fse.createWriteStream(`${rootUploadPath}/${userId}/${randomString}`);

            const resizer = Sharp()
                                .resize({
                                    width: 450
                                });

            await data.file
                    .pipe(resizer)
                    .pipe(stream);

这工作正常,并将文件写入项目本地目录。当我尝试在同一个异步函数中再次使用相同的可读数据时,问题就出现了。请注意,所有这些代码都在 try 块中。

        const stream2 = Fse.createWriteStream(`${rootUploadPath}/${userId}/thumb_${randomString}`);

        const resizer2 = Sharp()
                            .resize({
                                width: 45
                            });

        await data.file
                .pipe(resizer2)
                .pipe(stream2);

第二个文件已写入,但当我检查文件时,它似乎已损坏或未成功写入数据。第一张图总是好的。

我尝试了几件事,发现了一种似乎有效的方法,但我不明白为什么。我在创建第二个写入流之前添加了这段代码:

    data.file.on('end', () => {
        console.log('There will be no more data.');
    });

将第二个写入流的代码放在 on-end 回调块中并没有什么区别,但是,如果我将代码留在块之外,在第一个写入流代码和第二个写入流代码之间,那么它可以工作,并且两个文件都已成功写入。

让代码保持原样感觉不对。有没有更好的方法可以写第二个缩略图?在第一个写入流写入数据后,我尝试使用 Sharp 模块读取文件,然后创建它的较小版本,但它不起作用。该文件似乎从未准备好使用。

4

1 回答 1

1

您有 2 种选择,这取决于您的软件是如何设计的。

如果可能的话,我会避免在同一个“上下文”中对同一个流执行两个转换操作,例如:一个 API 端点。我宁愿将这两个不同的转换分开,这样它们就不能在同一个输入流上工作。

如果这不可能或需要太多更改,则解决方案是将输入流分叉并将其通过管道传输到两个不同的 Writable 中。我通常使用Highland.js fork来完成这些任务。

另请参阅我关于如何使用 async/await 正确处理流以检查写入操作何时完成的评论。

于 2020-07-10T13:38:51.893 回答