繁体   English   中英

NodeJS - 请求文件并压缩它

[英]NodeJS - Request file and zip it

我目前正在为我的个人网站创建一个 REST API。 我想包括一些下载,并且我想提供选择多个下载并将它们作为 zip 文件下载的可能性。 我的第一个方法很简单:带有 url 的数组,请求每个 URL,压缩它,发送给用户,删除。 但是,我认为这种方法太肮脏了,考虑到周围有像流这样的东西似乎非常适合这个东西。

现在,我尝试了一下,目前正在努力解决在不同范围内处理流和事件的基本概念。

以下工作:

const r = request(url, options);
r.on('response', function(res) {
    res.pipe(fs.createWriteStream('./file.jpg'));
});

据我了解,在这种情况下,r 是一个传入流,我在其上侦听响应事件,一旦它发生,我将它通过管道传输到我用来写入文件系统的流中。

我的第一步是重构它,使其更适合我的情况,但我在这里已经失败了:

async function downloadFile(url) {
    return request({ method: 'GET', uri: url });
}

现在我想使用一个函数,它使用不同的 url 调用“downloadFile()”,并再次使用 createWriteStream() 将所有这些文件保存到磁盘:

const urls = ['https://download1', 'https://download2', 'https://download3'];
urls.forEach(element => {
    downloadFile(element).then(data => {
        data.pipe(fs.createWriteStream('file.jpg'));
    });
});

使用调试器,我发现数据对象中不存在“响应”事件——也许这已经是问题所在了? 此外,我认为 data.body 包含我下载的文档的字节(在本例中为 pdf),所以我想知道是否可以将其流式传输到其他地方?

在阅读了一些 stackoveflow 线程后,我发现了以下模块: archiver

阅读此线程: 动态创建和流式传输 zip 到客户端

@dankohn 提出了这样的方法:

archive
  .append(fs.createReadStream(file1), { name: 'file1.txt' })
  .append(fs.createReadStream(file2), { name: 'file2.txt' });

让我假设我需要能够从我的数据对象中提取流才能继续。

我是在错误的轨道上还是我从根本上搞错了?

编辑: lmao 感谢您解决我的问题我不知道发生了什么

使用归档器似乎是一种有效的方法,但是在将来自网络的大数据输入 zip 归档时,建议使用流。 否则,整个存档数据将需要保存在内存中。

archiver 不支持从流中添加文件,但zip-stream支持。 对于从 Web 读取流, request就派上用场了。

例子

// npm install -s express zip-stream request

const request = require('request');
const ZipStream = require('zip-stream');
const express = require('express');

const app = express();

app.get('/archive.zip', (req, res) => {
    var zip = new ZipStream()
    zip.pipe(res);

    var stream = request('https://loremflickr.com/640/480')
    zip.entry(stream, { name: 'picture.jpg' }, err => {
        if(err)
            throw err;          
    })

    zip.finalize()
});

app.listen(3000)

更新:使用多个文件的示例

zip.entry()的回调函数中添加一个递归处理下一个文件的示例。

app.get('/archive.zip', (req, res) => {
    var zip = new ZipStream()
    zip.pipe(res);

    var queue = [
        { name: 'one.jpg', url: 'https://loremflickr.com/640/480' },
        { name: 'two.jpg', url: 'https://loremflickr.com/640/480' },
        { name: 'three.jpg', url: 'https://loremflickr.com/640/480' }
    ]

    function addNextFile() {
        var elem = queue.shift()
        var stream = request(elem.url)
        zip.entry(stream, { name: elem.name }, err => {
            if(err)
                throw err;
            if(queue.length > 0)
                addNextFile()
            else
                zip.finalize()
        })
    }

    addNextFile()
})

使用异步/等待

您可以将其封装成使用 async/await 的承诺,例如:

await new Promise((resolve, reject) => {
  zip.entry(stream, { name: elem.name }, err => {
    if (err) reject(err)
    resolve()
  }) 
})
zip.finalize()

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM