node/test/sequential/test-http-econnrefused.js

158 lines
4.1 KiB
JavaScript

// Copyright Joyent, Inc. and other Node contributors.
//
// Permission is hereby granted, free of charge, to any person obtaining a
// copy of this software and associated documentation files (the
// "Software"), to deal in the Software without restriction, including
// without limitation the rights to use, copy, modify, merge, publish,
// distribute, sublicense, and/or sell copies of the Software, and to permit
// persons to whom the Software is furnished to do so, subject to the
// following conditions:
//
// The above copyright notice and this permission notice shall be included
// in all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
// USE OR OTHER DEALINGS IN THE SOFTWARE.
'use strict';
const common = require('../common');
// The test works by making a total of 8 requests to the server. The first
// two are made with the server off - they should come back as ECONNREFUSED.
// The next two are made with server on - they should come back successful.
// The next two are made with the server off - and so on. Without the fix
// we were experiencing parse errors instead of ECONNREFUSED.
// https://github.com/nodejs/node-v0.x-archive/issues/784
const http = require('http');
const assert = require('assert');
const server = http.createServer(function(req, res) {
let body = '';
req.setEncoding('utf8');
req.on('data', function(chunk) {
body += chunk;
});
req.on('end', function() {
assert.strictEqual(body, 'PING');
res.writeHead(200, { 'Connection': 'close' });
res.end('PONG');
});
});
server.on('listening', pingping);
function serverOn() {
console.error('Server ON');
server.listen(common.PORT);
}
function serverOff() {
console.error('Server OFF');
server.close();
pingping();
}
const responses = [];
function afterPing(result) {
responses.push(result);
console.error(`afterPing. responses.length = ${responses.length}`);
const ECONNREFUSED_RE = /ECONNREFUSED/;
const successRE = /success/;
switch (responses.length) {
case 2:
assert.match(responses[0], ECONNREFUSED_RE);
assert.match(responses[1], ECONNREFUSED_RE);
serverOn();
break;
case 4:
assert.match(responses[2], successRE);
assert.match(responses[3], successRE);
serverOff();
break;
case 6:
assert.match(responses[4], ECONNREFUSED_RE);
assert.match(responses[5], ECONNREFUSED_RE);
serverOn();
break;
case 8:
assert.match(responses[6], successRE);
assert.match(responses[7], successRE);
server.close();
// We should go to process.on('exit') from here.
break;
}
}
function ping() {
console.error('making req');
const opt = {
port: common.PORT,
path: '/ping',
method: 'POST'
};
const req = http.request(opt, function(res) {
let body = '';
res.setEncoding('utf8');
res.on('data', function(chunk) {
body += chunk;
});
res.on('end', function() {
assert.strictEqual(body, 'PONG');
assert.ok(!hadError);
gotEnd = true;
afterPing('success');
});
});
req.end('PING');
let gotEnd = false;
let hadError = false;
req.on('error', function(error) {
console.log(`Error making ping req: ${error}`);
hadError = true;
assert.ok(!gotEnd);
// Family autoselection might be skipped if only a single address is returned by DNS.
const actualError = Array.isArray(error.errors) ? error.errors[0] : error;
afterPing(actualError.message);
});
}
function pingping() {
ping();
ping();
}
pingping();
process.on('exit', function() {
console.error("process.on('exit')");
console.error(responses);
assert.strictEqual(responses.length, 8);
});