Skip to content

Commit be29823

Browse files
committed
lib: add UV_UDP_REUSEPORT for udp
1 parent 87da1f3 commit be29823

7 files changed

+131
-3
lines changed

doc/api/dgram.md

+14-2
Original file line numberDiff line numberDiff line change
@@ -343,7 +343,8 @@ used when using `dgram.Socket` objects with the [`cluster`][] module. When
343343
`exclusive` is set to `false` (the default), cluster workers will use the same
344344
underlying socket handle allowing connection handling duties to be shared.
345345
When `exclusive` is `true`, however, the handle is not shared and attempted
346-
port sharing results in an error.
346+
port sharing results in an error. If creating a `dgram.Socket` with `reusePort`
347+
flag, `exclusive` will always be `true` when call `socket.bind()` with a port.
347348

348349
A bound datagram socket keeps the Node.js process running to receive
349350
datagram messages.
@@ -916,6 +917,9 @@ chained.
916917
<!-- YAML
917918
added: v0.11.13
918919
changes:
920+
- version: REPLACEME
921+
pr-url: https://github.com/nodejs/node/pull/55403
922+
description: The `reusePort` option is supported.
919923
- version: v15.8.0
920924
pr-url: https://github.com/nodejs/node/pull/37026
921925
description: AbortSignal support was added.
@@ -935,7 +939,15 @@ changes:
935939
* `type` {string} The family of socket. Must be either `'udp4'` or `'udp6'`.
936940
Required.
937941
* `reuseAddr` {boolean} When `true` [`socket.bind()`][] will reuse the
938-
address, even if another process has already bound a socket on it.
942+
address, even if another process has already bound a socket on it, but
943+
only one socket can receive the data.
944+
**Default:** `false`.
945+
* `reusePort` {boolean} When `true` [`socket.bind()`][] will reuse the
946+
address, even if another process has already bound a socket on it. Incoming
947+
datagrams are distributed across the receiving sockets. The flag is available
948+
only on some platforms, such as Linux 3.9+, DragonFlyBSD 3.6+, FreeBSD 12.0+,
949+
Solaris 11.4, and AIX 7.2.5+. On unsupported platforms this function will
950+
return an error.
939951
**Default:** `false`.
940952
* `ipv6Only` {boolean} Setting `ipv6Only` to `true` will
941953
disable dual-stack support, i.e., binding to address `::` won't make

lib/dgram.js

+6-1
Original file line numberDiff line numberDiff line change
@@ -74,7 +74,7 @@ const {
7474
const { UV_UDP_REUSEADDR } = internalBinding('constants').os;
7575

7676
const {
77-
constants: { UV_UDP_IPV6ONLY },
77+
constants: { UV_UDP_IPV6ONLY, UV_UDP_REUSEPORT },
7878
UDP,
7979
SendWrap,
8080
} = internalBinding('udp_wrap');
@@ -130,6 +130,7 @@ function Socket(type, listener) {
130130
connectState: CONNECT_STATE_DISCONNECTED,
131131
queue: undefined,
132132
reuseAddr: options?.reuseAddr, // Use UV_UDP_REUSEADDR if true.
133+
reusePort: options?.reusePort,
133134
ipv6Only: options?.ipv6Only,
134135
recvBufferSize,
135136
sendBufferSize,
@@ -345,6 +346,10 @@ Socket.prototype.bind = function(port_, address_ /* , callback */) {
345346
flags |= UV_UDP_REUSEADDR;
346347
if (state.ipv6Only)
347348
flags |= UV_UDP_IPV6ONLY;
349+
if (state.reusePort) {
350+
exclusive = true;
351+
flags |= UV_UDP_REUSEPORT;
352+
}
348353

349354
if (cluster.isWorker && !exclusive) {
350355
bindServerHandle(this, {

src/udp_wrap.cc

+1
Original file line numberDiff line numberDiff line change
@@ -231,6 +231,7 @@ void UDPWrap::Initialize(Local<Object> target,
231231
Local<Object> constants = Object::New(isolate);
232232
NODE_DEFINE_CONSTANT(constants, UV_UDP_IPV6ONLY);
233233
NODE_DEFINE_CONSTANT(constants, UV_UDP_REUSEADDR);
234+
NODE_DEFINE_CONSTANT(constants, UV_UDP_REUSEPORT);
234235
target->Set(context,
235236
env->constants_string(),
236237
constants).Check();

test/common/udp.js

+25
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
'use strict';
2+
const dgram = require('dgram');
3+
4+
const options = { type: 'udp4', reuseAddr: true };
5+
6+
function checkSupportReusePort() {
7+
return new Promise((resolve, reject) => {
8+
const socket = dgram.createSocket(options);
9+
socket.bind(0);
10+
socket.on('listening', () => {
11+
socket.close();
12+
resolve();
13+
});
14+
socket.on('error', (err) => {
15+
console.log('don not support reusePort flag', err);
16+
socket.close();
17+
reject();
18+
});
19+
});
20+
}
21+
22+
module.exports = {
23+
checkSupportReusePort,
24+
options,
25+
};
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,35 @@
1+
'use strict';
2+
const common = require('../common');
3+
const { checkSupportReusePort, options } = require('../common/udp');
4+
const assert = require('assert');
5+
const child_process = require('child_process');
6+
const dgram = require('dgram');
7+
8+
if (!process.env.isWorker) {
9+
checkSupportReusePort().then(() => {
10+
const socket = dgram.createSocket(options);
11+
socket.bind(0, common.mustCall(() => {
12+
const port = socket.address().port;
13+
const workerOptions = { env: { 'isWorker': 1, port } };
14+
let count = 2;
15+
for (let i = 0; i < 2; i++) {
16+
const worker = child_process.fork(__filename, workerOptions);
17+
worker.on('exit', common.mustCall((code) => {
18+
assert.strictEqual(code, 0);
19+
if (--count === 0) {
20+
socket.close();
21+
}
22+
}));
23+
}
24+
}));
25+
}, process.exit);
26+
return;
27+
}
28+
29+
const socket = dgram.createSocket(options);
30+
31+
socket.bind(+process.env.port, common.mustCall(() => {
32+
socket.close(common.mustCall(() => {
33+
process.exit(0);
34+
}));
35+
})).on('error', common.mustNotCall());
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,31 @@
1+
'use strict';
2+
const common = require('../common');
3+
const { checkSupportReusePort, options } = require('../common/udp');
4+
const assert = require('assert');
5+
const cluster = require('cluster');
6+
const dgram = require('dgram');
7+
8+
if (cluster.isPrimary) {
9+
checkSupportReusePort().then(() => {
10+
cluster.fork().on('exit', common.mustCall((code) => {
11+
assert.strictEqual(code, 0);
12+
}));
13+
}, process.exit);
14+
return;
15+
}
16+
17+
let waiting = 2;
18+
function close() {
19+
if (--waiting === 0)
20+
cluster.worker.disconnect();
21+
}
22+
23+
const socket1 = dgram.createSocket(options);
24+
const socket2 = dgram.createSocket(options);
25+
26+
socket1.bind(0, () => {
27+
socket2.bind(socket1.address().port, () => {
28+
socket1.close(close);
29+
socket2.close(close);
30+
}).on('error', common.mustNotCall());
31+
}).on('error', common.mustNotCall());

test/parallel/test-dgram-reuseport.js

+19
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
'use strict';
2+
const common = require('../common');
3+
const { checkSupportReusePort, options } = require('../common/udp');
4+
const dgram = require('dgram');
5+
6+
function test() {
7+
const socket1 = dgram.createSocket(options);
8+
const socket2 = dgram.createSocket(options);
9+
socket1.bind(0, common.mustCall(() => {
10+
socket2.bind(socket1.address().port, common.mustCall(() => {
11+
socket1.close();
12+
socket2.close();
13+
}));
14+
}));
15+
socket1.on('error', common.mustNotCall());
16+
socket2.on('error', common.mustNotCall());
17+
}
18+
19+
checkSupportReusePort().then(test, process.exit);

0 commit comments

Comments
 (0)