javascript - HTTP/2 : stream. 使用 For-Loop 结束

标签 javascript for-loop stream http2

如何在 for 循环 中关闭流?

我正在尝试通过读取 .json 配置文件在循环中使用 HTTP/2 资源 push-stream。如果resurce.file,type中只有一个资源,则推送成功。如果配置中有多个资源,则仅推送列出的第一个资源,不推送其余资源,客户端永远不会收到剩余的文件,也不会完成解析。

我在想只有第一个资源流结束,其余打开的流没有关闭。

我认为代码应该可以正常运行,所以我评论得最好:

// Verify if client can be pushed to:
if (res.push) {

  // Read what resources should be pushed via config.json5:
  for(var i = 0; i < confg.resource.length; i++) {

    // Compare what URL requested the resources & pass the relevant files:
    if (req.url == confg.resource[i].url) {

      // Request the resource(s) to stream:
      var ps  = fs.readFileSync('../build' + confg.resource[i].file)

          // Push the relevant resource stream(s):
        , stream = res.push(confg.resource[i].file
        , { // ..and set the content type;
            res: {'content-type': confg.resource[i].type},
          });

      // Close the resource stream(s):
      stream.end(ps, function() {
        // TODO(CelticParser): Remove for production ->v
        console.log("Pushed Sream");
      });
    }
  }
}

.config.json5:

resource: [ // HTTP/2 Push-Streams
  { file: '/js/theme.min.js',
     type: 'script'
  },
  { file: '/css/theme.min.css',
    type: 'style',
    url:  '/'
  }
]

使用上面的例子;

如果 /js/theme.min.js 在配置中排在第一位,而 /css/theme.min.css 在配置中排在第二位,则 /js/theme.min.js 会被浏览器加载,其他文件不会加载,客户端挂掉(不继续解析)。如果交换资源的列表顺序,则会发生同样的事情。如果配置中只列出一个文件,则一切正常。

任何帮助将不胜感激。

最佳答案

问题出在 config.json5 中。在将文件传递到此处之前,您正在检查请求的 url:

// Compare what URL requested the resources & pass the relevant files:
if (req.url == confg.resource[i].url) {
   ...
}

但在您的 json 中只有一项可以通过测试,另一项缺少 url 属性。将您的配置更改为此(将 url: '/' 添加到第一项),它将起作用:

{
   resource : [
      { 
         file : '/js/theme.min.js',
         type : 'script',
         url  :  '/'
      },
      { 
         file : '/css/theme.min.css',
         type : 'style',
         url  :  '/'
      }
   ]
}

http2 server example 之后使用这个小应用程序进行了测试,以及您的服务器代码。

server.js

var fs      = require("fs");
var path    = require("path");
var http2   = require("http2");
var config  = require("./config.json");

var options = {
    key  : fs.readFileSync("./localhost.key"),
    cert : fs.readFileSync("./localhost.crt")
};

//----------------------------------------------------
http2.createServer(options, function(req, res) {

    if (res.push) {

        config.resource.forEach(function(resource){

            if (req.url === resource.url) {

                var push = res.push(resource.file, { res: {"content-type": resource.type } });
                push.writeHead(200);
                fs.createReadStream(path.join("../build", resource.file)).pipe(push);
            }
        });
    }

    res.writeHead(200);
    res.end();

}).listen(8080);

config.json 文件包含上述更正的配置。

输出:

Receiving pushed resource: ./js/bootstrap.min.js -> /Developer/so/tests/push-0
Receiving pushed resource: ./css/bootstrap.min.css -> /Developer/so/tests/push-1

关于javascript - HTTP/2 : stream. 使用 For-Loop 结束,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34193374/

相关文章:

javascript - jQuery 初学者 - 解释这个 if 语句的逻辑?

返回自定义长度字符的 C 函数

Flutter 将多个 FireStore 流与 Rxdart 结合

arrays - 如何将循环参数解构为固定大小的数组?

c++ - 如何用 C++ 编写自定义流转换?

node.js - 我可以在结束前关闭 Node readStream 吗?

javascript - TinyMCE 更改文本区域背景色

javascript - 是否可以在 JavaScript 中的对象声明内执行循环

javascript - Function.prototype.softBind 中的 '!this'

linux - Bash for 循环 - 列出/etc/init.d 中的文件,然后对结果运行命令