javascript - Node.js + 请求 - 保存远程文件并在响应中提供它

标签 javascript node.js request

我了解如何使用 Node.js + 请求加载远程文件,然后我可以读取它并返回 png 二进制 blob。有没有一种优雅的方法可以通过一个请求(甚至是一行)来做到这一点

类似于:

http.createServer(function(req, res) {
    res.writeHead(200, {
        'Content-Type': 'image/png'
    });
    var picWrite = fs.createWriteStream(local);
    var picFetch = fs.createReadStream(local);
    picStream.on('close', function() {
        console.log("file loaded");
    });
    request(remote).pipe(picWrite).pipe(picFetch).pipe(res);
})

明确一点:我的目标是从 CDN 加载远程文件,将其缓存到服务器本地,然后在原始请求中返回文件。在以后的请求中,我首先使用 fs.exists() 检查它是否存在。


这是我迄今为止的最大努力:

http.createServer(function(req, res) {
    var file = fs.createWriteStream(local);
    request.get(remote).pipe(file).on('close', function() {
        res.end(fs.readFileSync(local), 'binary');
    });
})

最佳答案

由于请求将返回一个可读流,我们可以监听它的 dataend 事件以写入 HTTP 响应和可写流。

var http = require('http');
var request = require('request');

http.createServer(function(req, res) {
  res.writeHead(200, { 'Content-Type': 'image/png' });
  var file = fs.createWriteStream(local);

  // request the file from a remote server
  var rem = request(remote);
  rem.on('data', function(chunk) {
    // instead of loading the file into memory
    // after the download, we can just pipe
    // the data as it's being downloaded
    file.write(chunk);
    res.write(chunk);
  });
  rem.on('end', function() {
    res.end();
  });
});

您展示的方法首先将数据写入磁盘,然后再次将其读入内存。这是毫无意义的,因为数据在写入磁盘时就已经可以访问了。

如果您使用事件处理程序,您可以同时写入 HTTP 响应和文件流,而无需毫无意义地将文件再次加载到内存中。这也解决了使用 pipe() 的问题,因为 pipe() 会消耗可读流中的数据,并且只能执行一次。

这也解决了内存不足的问题,因为如果你要下载一个大文件,那么它会有效地运行你的 Node.js 进程内存不足。使用流时,一次只会将文件 block 加载到内存中,因此您不会遇到这个问题。

关于javascript - Node.js + 请求 - 保存远程文件并在响应中提供它,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19237460/

相关文章:

node.js - 如何使用 Telegram bot API 请求用户的实时位置?

java - 如何访问我正在定义的 java grpc 服务的请求元数据?

php - 使用 PHP 进行 USPS 费率计算器的 XML 请求和响应

javascript - CamanJS:将图像动态加载到 Canvas 上

javascript - 如何在 CodeceptJS 中尝试捕捉

android - 谷歌云功能 Node.js Firebase 网址

Android启动画面与服务器通信

javascript - Angular : How to disable an anchor tag out of an array (ng-repeat) of anchor tags

javascript - 将 div 移动到位于 iframe 内的另一个 div

javascript - 2014 年 RequireJS 与 Backbone、Underscore 和 jQuery 的依赖关系