Skip to content

Commit 2982e75

Browse files
committed
set-cookies header is an array of values. always
1 parent 488aff0 commit 2982e75

2 files changed

Lines changed: 108 additions & 20 deletions

File tree

lib/http.js

Lines changed: 37 additions & 20 deletions
Original file line numberDiff line numberDiff line change
@@ -235,27 +235,44 @@ IncomingMessage.prototype.resume = function () {
235235
// and drop the second. Extended header fields (those beginning with 'x-') are
236236
// always joined.
237237
IncomingMessage.prototype._addHeaderLine = function (field, value) {
238-
if (!(field in this.headers)) {
239-
this.headers[field] = value;
240-
return;
241-
}
242-
243-
// If this field already exists in the request, use duplicate-resolution
244-
// logic from RFC2616.
245238
switch (field) {
246-
case 'accept':
247-
case 'accept-charset':
248-
case 'accept-encoding':
249-
case 'accept-language':
250-
case 'connection':
251-
case 'cookie':
252-
this.headers[field] += ', ' + value;
253-
break;
254-
255-
default:
256-
if (field[0] !== 'x' || field[1] !== '-') break;
257-
this.headers[field] += ', ' + value;
258-
break;
239+
// Array headers:
240+
case 'set-cookie':
241+
if (field in this.headers) {
242+
this.headers[field].push(value);
243+
} else {
244+
this.headers[field] = [value];
245+
}
246+
break;
247+
248+
// Comma separate. Maybe make these arrays?
249+
case 'accept':
250+
case 'accept-charset':
251+
case 'accept-encoding':
252+
case 'accept-language':
253+
case 'connection':
254+
case 'cookie':
255+
if (field in this.headers) {
256+
this.headers[field] += ', ' + value;
257+
} else {
258+
this.headers[field] = value;
259+
}
260+
break;
261+
262+
263+
default:
264+
if (field.slice(0,2) == 'x-') {
265+
// except for x-
266+
if (field in this.headers) {
267+
this.headers[field] += ', ' + value;
268+
} else {
269+
this.headers[field] = value;
270+
}
271+
} else {
272+
// drop duplicates
273+
if (!(field in this.headers)) this.headers[field] = value;
274+
}
275+
break;
259276
}
260277
};
261278

Lines changed: 71 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,71 @@
1+
common = require("../common");
2+
assert = common.assert;
3+
http = require("http");
4+
5+
nresponses = 0;
6+
7+
var server = http.createServer(function(req, res) {
8+
if (req.url == '/one') {
9+
res.writeHead(200, [ ['set-cookie', 'A'],
10+
['content-type', 'text/plain'] ]);
11+
res.end("one\n");
12+
} else {
13+
res.writeHead(200, [ ['set-cookie', 'A'],
14+
['set-cookie', 'B'],
15+
['content-type', 'text/plain'] ]);
16+
res.end("two\n");
17+
}
18+
});
19+
server.listen(common.PORT);
20+
21+
server.addListener("listening", function() {
22+
//
23+
// one set-cookie header
24+
//
25+
var client = http.createClient(common.PORT);
26+
var req = client.request('GET', '/one');
27+
req.end();
28+
29+
req.addListener('response', function(res) {
30+
// set-cookie headers are always return in an array.
31+
// even if there is only one.
32+
assert.deepEqual(['A'], res.headers['set-cookie']);
33+
assert.equal('text/plain', res.headers['content-type']);
34+
35+
res.addListener('data', function(chunk) {
36+
console.log(chunk.toString());
37+
});
38+
39+
res.addListener('end', function() {
40+
if (++nresponses == 2) {
41+
server.close();
42+
}
43+
});
44+
});
45+
46+
// two set-cookie headers
47+
48+
var client = http.createClient(common.PORT);
49+
var req = client.request('GET', '/two');
50+
req.end();
51+
52+
req.addListener('response', function(res) {
53+
assert.deepEqual(['A', 'B'], res.headers['set-cookie']);
54+
assert.equal('text/plain', res.headers['content-type']);
55+
56+
res.addListener('data', function(chunk) {
57+
console.log(chunk.toString());
58+
});
59+
60+
res.addListener('end', function() {
61+
if (++nresponses == 2) {
62+
server.close();
63+
}
64+
});
65+
});
66+
67+
});
68+
69+
process.addListener("exit", function () {
70+
assert.equal(2, nresponses);
71+
});

0 commit comments

Comments
 (0)