| 'use strict'; |
| // 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. |
| |
| const common = require('../common'); |
| const assert = require('assert'); |
| const cluster = require('cluster'); |
| const dgram = require('dgram'); |
| |
| // Without an explicit bind, send() causes an implicit bind, which always |
| // generate a unique per-socket ephemeral port. An explicit bind to a port |
| // number causes all sockets bound to that number to share a port. |
| // |
| // The 2 workers that call bind() will share a port, the two workers that do |
| // not will not share a port, so primary will see 3 unique source ports. |
| |
| // Note that on Windows, clustered dgram is not supported. Since explicit |
| // binding causes the dgram to be clustered, don't fork the workers that bind. |
| // This is a useful test, still, because it demonstrates that by avoiding |
| // clustering, client (ephemeral, implicitly bound) dgram sockets become |
| // supported while using cluster, though servers still cause the primary |
| // to error with ENOTSUP. |
| |
| if (cluster.isPrimary) { |
| let messages = 0; |
| const ports = {}; |
| const pids = []; |
| |
| const target = dgram.createSocket('udp4'); |
| |
| const done = common.mustCall(function() { |
| cluster.disconnect(); |
| target.close(); |
| }); |
| |
| target.on('message', common.mustCallAtLeast((buf, rinfo) => { |
| if (pids.includes(buf.toString())) |
| return; |
| pids.push(buf.toString()); |
| messages++; |
| ports[rinfo.port] = true; |
| |
| if (common.isWindows && messages === 2) { |
| assert.strictEqual(Object.keys(ports).length, 2); |
| done(); |
| } |
| |
| if (!common.isWindows && messages === 4) { |
| assert.strictEqual(Object.keys(ports).length, 3); |
| done(); |
| } |
| })); |
| |
| target.on('listening', function() { |
| cluster.fork({ PORT: target.address().port }); |
| cluster.fork({ PORT: target.address().port }); |
| if (!common.isWindows) { |
| cluster.fork({ BOUND: 'y', PORT: target.address().port }); |
| cluster.fork({ BOUND: 'y', PORT: target.address().port }); |
| } |
| }); |
| |
| target.bind({ port: 0, exclusive: true }); |
| |
| return; |
| } |
| |
| const source = dgram.createSocket('udp4'); |
| |
| source.on('close', function() { |
| clearInterval(interval); |
| }); |
| |
| if (process.env.BOUND === 'y') { |
| source.bind(0); |
| } else { |
| // Cluster doesn't know about exclusive sockets, so it won't close them. This |
| // is expected, its the same situation for timers, outgoing tcp connections, |
| // etc, which also keep workers alive after disconnect was requested. |
| source.unref(); |
| } |
| |
| assert(process.env.PORT); |
| const buf = Buffer.from(process.pid.toString()); |
| const interval = setInterval(() => { |
| source.send(buf, process.env.PORT, '127.0.0.1'); |
| }, 1).unref(); |