Node.js:如何禁用分块传输编码?

12

我从一个Node服务器中管道传输另一个位置的.zip文件时,我的响应缺少content-length头。我通过下面的代码注入了一个content-length头,但似乎transfer-encoding: chunked以某种方式覆盖了它。

响应标头

HTTP/1.1 200 OK
access-control-allow-origin: *
connection: close
content-type: application/zip
date: Mon, 14 Jul 2014 03:47:00 GMT
etag: "\"eb939974703e14ee9f578642972ed984\""
last-modified: Sat, 12 Jul 2014 02:15:52 GMT
server: Apache-Coyote/1.1
set-cookie: rememberMe=deleteMe; Path=/; Max-Age=0; Expires=Sun, 13-Jul-2014 03:47:00 GMT
transfer-encoding: chunked
X-Powered-By: Express

代码

var request = require('request');
var express = require('express');
var async = require('async');

var app = express();

app.get('/:bundle_id?', function(req, res) {
    var bundle_id = req.params.bundle_id;
    bundle_id = bundle_id.replace(/\.zip$/, '');

    var url = "https://url....../bundles/" + bundle_id;

    async.waterfall([

        function(callback) {
            request.get(url, function(req, res, data) {
                callback(null, JSON.parse(data).entities[0]['file-metadata']['content-length']);
            });
        }
    ], function(err, contentLength) {

        request.get({
            url: url,
            headers: {
                "Accept": "application/zip"
            }
        }).pipe(res);

        res.oldWriteHead = res.writeHead;
        res.writeHead = function(statusCode, reasonPhrase, headers) {
            res.header('Content-Length', contentLength);
            res.oldWriteHead(statusCode, reasonPhrase, headers);
        }
    });
});

app.listen(9000);
2个回答

15

结果证明这实际上是一个相当简单的修复方法:在响应中设置transfer-encoding头为空字符串即可解决问题:

...
res.oldWriteHead = res.writeHead;
res.writeHead = function(statusCode, reasonPhrase, headers) {
    res.header('Content-Length', contentLength);
    res.header('transfer-encoding', ''); // <-- add this line
    res.oldWriteHead(statusCode, reasonPhrase, headers);
}
...
这个方法有效的原因是,在做了一些研究后,似乎transfer-encoding头部替换了content-length(因为两者不能共存)。碰巧我用来测试的客户端选择了分块传输编码而不是内容长度。

在响应头中仅将“transfer-encoding”设置为空字符串对我无效。我需要设置“content-length”,这就足以防止分块编码。我想知道为什么这种行为与您描述的不同。 - Flip
3
根据HTTP规范,如果您未指定内容长度,它将默认为分块传输,因此实际上并没有什么区别。 - brandonscript
我的意思是它与你的解决方案不同。显然,对于你来说,仅设置Content-Length头部是不够的。 - Flip

-2
如果您定义了Content-Length,则Transfer-Encoding将不再发送到“chunked”。

1
它与被接受的答案有何不同? - Akber Iqbal

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接