Skip to content

Commit 1a2a4da

Browse files
committed
net: allow port 0 in connect()
The added validation allows non-negative numbers and numeric strings. All other values result in a thrown exception. Fixes: nodejs/node-v0.x-archive#9194 PR-URL: nodejs/node-v0.x-archive#9268 Reviewed-By: Julien Gilli <julien.gilli@joyent.com> Reviewed-By: Trevor Norris <trev.norris@gmail.com> Reviewed-By: James M Snell <jasnell@users.noreply.github.com>
1 parent b0425f0 commit 1a2a4da

File tree

4 files changed

+97
-33
lines changed

4 files changed

+97
-33
lines changed

lib/net.js

+11-3
Original file line numberDiff line numberDiff line change
@@ -892,7 +892,7 @@ Socket.prototype.connect = function(options, cb) {
892892
} else {
893893
var dns = require('dns');
894894
var host = options.host || 'localhost';
895-
var port = options.port | 0;
895+
var port = 0;
896896
var localAddress = options.localAddress;
897897
var localPort = options.localPort;
898898
var dnsopts = {
@@ -906,8 +906,16 @@ Socket.prototype.connect = function(options, cb) {
906906
if (localPort && !util.isNumber(localPort))
907907
throw new TypeError('localPort should be a number: ' + localPort);
908908

909-
if (port <= 0 || port > 65535)
910-
throw new RangeError('port should be > 0 and < 65536: ' + port);
909+
if (typeof options.port === 'number')
910+
port = options.port;
911+
else if (typeof options.port === 'string')
912+
port = options.port.trim() === '' ? -1 : +options.port;
913+
else if (options.port !== undefined)
914+
throw new TypeError('port should be a number or string: ' + options.port);
915+
916+
if (port < 0 || port > 65535 || isNaN(port))
917+
throw new RangeError('port should be >= 0 and < 65536: ' +
918+
options.port);
911919

912920
if (dnsopts.family !== 4 && dnsopts.family !== 6)
913921
dnsopts.hints = dns.ADDRCONFIG | dns.V4MAPPED;

src/tcp_wrap.cc

+2-2
Original file line numberDiff line numberDiff line change
@@ -428,7 +428,7 @@ void TCPWrap::Connect(const FunctionCallbackInfo<Value>& args) {
428428

429429
assert(args[0]->IsObject());
430430
assert(args[1]->IsString());
431-
assert(args[2]->Uint32Value());
431+
assert(args[2]->IsUint32());
432432

433433
Local<Object> req_wrap_obj = args[0].As<Object>();
434434
node::Utf8Value ip_address(args[1]);
@@ -460,7 +460,7 @@ void TCPWrap::Connect6(const FunctionCallbackInfo<Value>& args) {
460460

461461
assert(args[0]->IsObject());
462462
assert(args[1]->IsString());
463-
assert(args[2]->Uint32Value());
463+
assert(args[2]->IsUint32());
464464

465465
Local<Object> req_wrap_obj = args[0].As<Object>();
466466
node::Utf8Value ip_address(args[1]);

test/simple/test-net-create-connection.js

+74-8
Original file line numberDiff line numberDiff line change
@@ -24,33 +24,99 @@ var assert = require('assert');
2424
var net = require('net');
2525

2626
var tcpPort = common.PORT;
27+
var expectedConnections = 7;
2728
var clientConnected = 0;
2829
var serverConnected = 0;
2930

3031
var server = net.createServer(function(socket) {
3132
socket.end();
32-
if (++serverConnected === 4) {
33+
if (++serverConnected === expectedConnections) {
3334
server.close();
3435
}
3536
});
37+
3638
server.listen(tcpPort, 'localhost', function() {
3739
function cb() {
3840
++clientConnected;
3941
}
4042

43+
function fail(opts, errtype, msg) {
44+
assert.throws(function() {
45+
var client = net.createConnection(opts, cb);
46+
}, function (err) {
47+
return err instanceof errtype && msg === err.message;
48+
});
49+
}
50+
4151
net.createConnection(tcpPort).on('connect', cb);
4252
net.createConnection(tcpPort, 'localhost').on('connect', cb);
4353
net.createConnection(tcpPort, cb);
4454
net.createConnection(tcpPort, 'localhost', cb);
55+
net.createConnection(tcpPort + '', 'localhost', cb);
56+
net.createConnection({port: tcpPort + ''}).on('connect', cb);
57+
net.createConnection({port: '0x' + tcpPort.toString(16)}, cb);
58+
59+
fail({
60+
port: true
61+
}, TypeError, 'port should be a number or string: true');
62+
63+
fail({
64+
port: false
65+
}, TypeError, 'port should be a number or string: false');
66+
67+
fail({
68+
port: []
69+
}, TypeError, 'port should be a number or string: ');
70+
71+
fail({
72+
port: {}
73+
}, TypeError, 'port should be a number or string: [object Object]');
74+
75+
fail({
76+
port: null
77+
}, TypeError, 'port should be a number or string: null');
78+
79+
fail({
80+
port: ''
81+
}, RangeError, 'port should be >= 0 and < 65536: ');
82+
83+
fail({
84+
port: ' '
85+
}, RangeError, 'port should be >= 0 and < 65536: ');
4586

46-
assert.throws(function () {
47-
net.createConnection({
48-
port: 'invalid!'
49-
}, cb);
50-
});
87+
fail({
88+
port: '0x'
89+
}, RangeError, 'port should be >= 0 and < 65536: 0x');
90+
91+
fail({
92+
port: '-0x1'
93+
}, RangeError, 'port should be >= 0 and < 65536: -0x1');
94+
95+
fail({
96+
port: NaN
97+
}, RangeError, 'port should be >= 0 and < 65536: NaN');
98+
99+
fail({
100+
port: Infinity
101+
}, RangeError, 'port should be >= 0 and < 65536: Infinity');
102+
103+
fail({
104+
port: -1
105+
}, RangeError, 'port should be >= 0 and < 65536: -1');
106+
107+
fail({
108+
port: 65536
109+
}, RangeError, 'port should be >= 0 and < 65536: 65536');
51110
});
52111

53-
process.on('exit', function() {
54-
assert.equal(clientConnected, 4);
112+
// Try connecting to random ports, but do so once the server is closed
113+
server.on('close', function() {
114+
function nop() {}
115+
116+
net.createConnection({port: 0}).on('error', nop);
117+
net.createConnection({port: undefined}).on('error', nop);
55118
});
56119

120+
process.on('exit', function() {
121+
assert.equal(clientConnected, expectedConnections);
122+
});

test/simple/test-net-localerror.js

+10-20
Original file line numberDiff line numberDiff line change
@@ -23,27 +23,17 @@ var common = require('../common');
2323
var assert = require('assert');
2424
var net = require('net');
2525

26-
connect({
27-
host: 'localhost',
28-
port: common.PORT,
29-
localPort: 'foobar',
30-
}, 'localPort should be a number: foobar');
26+
connect({
27+
host: 'localhost',
28+
port: common.PORT,
29+
localPort: 'foobar',
30+
}, 'localPort should be a number: foobar');
3131

32-
connect({
33-
host: 'localhost',
34-
port: common.PORT,
35-
localAddress: 'foobar',
36-
}, 'localAddress should be a valid IP: foobar');
37-
38-
connect({
39-
host: 'localhost',
40-
port: 65536
41-
}, 'port should be > 0 and < 65536: 65536');
42-
43-
connect({
44-
host: 'localhost',
45-
port: 0
46-
}, 'port should be > 0 and < 65536: 0');
32+
connect({
33+
host: 'localhost',
34+
port: common.PORT,
35+
localAddress: 'foobar',
36+
}, 'localAddress should be a valid IP: foobar');
4737

4838
function connect(opts, msg) {
4939
assert.throws(function() {

0 commit comments

Comments
 (0)