Skip to content

Commit 15ba19b

Browse files
oyydtargos
authored andcommittedSep 4, 2021
dns: allow --dns-result-order to change default dns verbatim
Allow the `--dns-result-order` option to change the default value of verbatim in `dns.lookup()`. This is useful when running Node.js in ipv6-only environments to avoid possible ENETUNREACH errors. PR-URL: #38099 Refs: #31566 Reviewed-By: Matteo Collina <matteo.collina@gmail.com> Reviewed-By: James M Snell <jasnell@gmail.com> Reviewed-By: Antoine du Hamel <duhamelantoine1995@gmail.com> Reviewed-By: Anna Henningsen <anna@addaleax.net>
1 parent 587deac commit 15ba19b

10 files changed

+298
-9
lines changed
 

‎doc/api/cli.md

+17
Original file line numberDiff line numberDiff line change
@@ -183,6 +183,19 @@ Make built-in language features like `eval` and `new Function` that generate
183183
code from strings throw an exception instead. This does not affect the Node.js
184184
`vm` module.
185185

186+
### `--dns-result-order=order`
187+
<!-- YAML
188+
added: REPLACEME
189+
-->
190+
191+
Set the default value of `verbatim` in [`dns.lookup()`][] and
192+
[`dnsPromises.lookup()`][]. The value could be:
193+
* `ipv4first`: sets default `verbatim` `false`.
194+
* `verbatim`: sets default `verbatim` `true`.
195+
196+
The default is `ipv4first` and [`dns.setDefaultResultOrder()`][] have higher
197+
priority than `--dns-result-order`.
198+
186199
### `--enable-fips`
187200
<!-- YAML
188201
added: v6.0.0
@@ -1314,6 +1327,7 @@ Node.js options that are allowed are:
13141327
* `--conditions`, `-C`
13151328
* `--diagnostic-dir`
13161329
* `--disable-proto`
1330+
* `--dns-result-order`
13171331
* `--enable-fips`
13181332
* `--enable-source-maps`
13191333
* `--experimental-abortcontroller`
@@ -1668,6 +1682,9 @@ $ node --max-old-space-size=1536 index.js
16681682
[`NODE_OPTIONS`]: #cli_node_options_options
16691683
[`NO_COLOR`]: https://no-color.org
16701684
[`SlowBuffer`]: buffer.md#buffer_class_slowbuffer
1685+
[`dns.lookup()`]: dns.md#dns_dns_lookup_hostname_options_callback
1686+
[`dns.setDefaultResultOrder()`]: dns.md#dns_dns_setdefaultresultorder_order
1687+
[`dnsPromises.lookup()`]: dns.md#dns_dnspromises_lookup_hostname_options
16711688
[`process.setUncaughtExceptionCaptureCallback()`]: process.md#process_process_setuncaughtexceptioncapturecallback_fn
16721689
[`tls.DEFAULT_MAX_VERSION`]: tls.md#tls_tls_default_max_version
16731690
[`tls.DEFAULT_MIN_VERSION`]: tls.md#tls_tls_default_min_version

‎doc/api/dns.md

+44-4
Original file line numberDiff line numberDiff line change
@@ -184,8 +184,9 @@ changes:
184184
addresses in the order the DNS resolver returned them. When `false`,
185185
IPv4 addresses are placed before IPv6 addresses.
186186
**Default:** currently `false` (addresses are reordered) but this is
187-
expected to change in the not too distant future.
188-
New code should use `{ verbatim: true }`.
187+
expected to change in the not too distant future. Default value is
188+
configurable using [`dns.setDefaultResultOrder()`][] or
189+
[`--dns-result-order`][]. New code should use `{ verbatim: true }`.
189190
* `callback` {Function}
190191
* `err` {Error}
191192
* `address` {string} A string representation of an IPv4 or IPv6 address.
@@ -627,6 +628,23 @@ array of host names.
627628
On error, `err` is an [`Error`][] object, where `err.code` is
628629
one of the [DNS error codes][].
629630

631+
## `dns.setDefaultResultOrder(order)`
632+
<!-- YAML
633+
added: REPLACEME
634+
-->
635+
636+
* `order` {string} must be `'ipv4first'` or `'verbatim'`.
637+
638+
Set the default value of `verbatim` in [`dns.lookup()`][] and
639+
[`dnsPromises.lookup()`][]. The value could be:
640+
* `ipv4first`: sets default `verbatim` `false`.
641+
* `verbatim`: sets default `verbatim` `true`.
642+
643+
The default is `ipv4first` and [`dns.setDefaultResultOrder()`][] have higher
644+
priority than [`--dns-result-order`][]. When using [worker threads][],
645+
[`dns.setDefaultResultOrder()`][] from the main thread won't affect the default
646+
dns orders in workers.
647+
630648
## `dns.setServers(servers)`
631649
<!-- YAML
632650
added: v0.11.3
@@ -772,8 +790,9 @@ added: v10.6.0
772790
IPv6 addresses in the order the DNS resolver returned them. When `false`,
773791
IPv4 addresses are placed before IPv6 addresses.
774792
**Default:** currently `false` (addresses are reordered) but this is
775-
expected to change in the not too distant future.
776-
New code should use `{ verbatim: true }`.
793+
expected to change in the not too distant future. Default value is
794+
configurable using [`dns.setDefaultResultOrder()`][] or
795+
[`--dns-result-order`][]. New code should use `{ verbatim: true }`.
777796

778797
Resolves a host name (e.g. `'nodejs.org'`) into the first found A (IPv4) or
779798
AAAA (IPv6) record. All `option` properties are optional. If `options` is an
@@ -1127,6 +1146,23 @@ array of host names.
11271146
On error, the `Promise` is rejected with an [`Error`][] object, where `err.code`
11281147
is one of the [DNS error codes](#dns_error_codes).
11291148

1149+
### `dnsPromises.setDefaultResultOrder(order)`
1150+
<!-- YAML
1151+
added: REPLACEME
1152+
-->
1153+
1154+
* `order` {string} must be `'ipv4first'` or `'verbatim'`.
1155+
1156+
Set the default value of `verbatim` in [`dns.lookup()`][] and
1157+
[`dnsPromises.lookup()`][]. The value could be:
1158+
* `ipv4first`: sets default `verbatim` `false`.
1159+
* `verbatim`: sets default `verbatim` `true`.
1160+
1161+
The default is `ipv4first` and [`dnsPromises.setDefaultResultOrder()`][] have
1162+
higher priority than [`--dns-result-order`][]. When using [worker threads][],
1163+
[`dnsPromises.setDefaultResultOrder()`][] from the main thread won't affect the
1164+
default dns orders in workers.
1165+
11301166
### `dnsPromises.setServers(servers)`
11311167
<!-- YAML
11321168
added: v10.6.0
@@ -1236,6 +1272,7 @@ uses. For instance, _they do not use the configuration from `/etc/hosts`_.
12361272
[Implementation considerations section]: #dns_implementation_considerations
12371273
[RFC 5952]: https://tools.ietf.org/html/rfc5952#section-6
12381274
[RFC 8482]: https://tools.ietf.org/html/rfc8482
1275+
[`--dns-result-order`]: cli.md#cli_dns_result_order_order
12391276
[`Error`]: errors.md#errors_class_error
12401277
[`UV_THREADPOOL_SIZE`]: cli.md#cli_uv_threadpool_size_size
12411278
[`dgram.createSocket()`]: dgram.md#dgram_dgram_createsocket_options_callback
@@ -1255,6 +1292,7 @@ uses. For instance, _they do not use the configuration from `/etc/hosts`_.
12551292
[`dns.resolveSrv()`]: #dns_dns_resolvesrv_hostname_callback
12561293
[`dns.resolveTxt()`]: #dns_dns_resolvetxt_hostname_callback
12571294
[`dns.reverse()`]: #dns_dns_reverse_ip_callback
1295+
[`dns.setDefaultResultOrder()`]: #dns_dns_setdefaultresultorder_order
12581296
[`dns.setServers()`]: #dns_dns_setservers_servers
12591297
[`dnsPromises.getServers()`]: #dns_dnspromises_getservers
12601298
[`dnsPromises.lookup()`]: #dns_dnspromises_lookup_hostname_options
@@ -1272,7 +1310,9 @@ uses. For instance, _they do not use the configuration from `/etc/hosts`_.
12721310
[`dnsPromises.resolveSrv()`]: #dns_dnspromises_resolvesrv_hostname
12731311
[`dnsPromises.resolveTxt()`]: #dns_dnspromises_resolvetxt_hostname
12741312
[`dnsPromises.reverse()`]: #dns_dnspromises_reverse_ip
1313+
[`dnsPromises.setDefaultResultOrder()`]: #dns_dnspromises_setdefaultresultorder_order
12751314
[`dnsPromises.setServers()`]: #dns_dnspromises_setservers_servers
12761315
[`socket.connect()`]: net.md#net_socket_connect_options_connectlistener
12771316
[`util.promisify()`]: util.md#util_util_promisify_original
12781317
[supported `getaddrinfo` flags]: #dns_supported_getaddrinfo_flags
1318+
[worker threads]: worker_threads.md

‎lib/dns.js

+8-2
Original file line numberDiff line numberDiff line change
@@ -41,6 +41,8 @@ const {
4141
Resolver,
4242
validateHints,
4343
emitInvalidHostnameWarning,
44+
getDefaultVerbatim,
45+
setDefaultResultOrder,
4446
} = require('internal/dns/utils');
4547
const {
4648
ERR_INVALID_ARG_TYPE,
@@ -96,7 +98,7 @@ function lookup(hostname, options, callback) {
9698
let hints = 0;
9799
let family = -1;
98100
let all = false;
99-
let verbatim = false;
101+
let verbatim = getDefaultVerbatim();
100102

101103
// Parse arguments
102104
if (hostname && typeof hostname !== 'string') {
@@ -110,7 +112,9 @@ function lookup(hostname, options, callback) {
110112
hints = options.hints >>> 0;
111113
family = options.family >>> 0;
112114
all = options.all === true;
113-
verbatim = options.verbatim === true;
115+
if (typeof options.verbatim === 'boolean') {
116+
verbatim = options.verbatim === true;
117+
}
114118

115119
validateHints(hints);
116120
} else {
@@ -285,6 +289,7 @@ module.exports = {
285289
lookupService,
286290

287291
Resolver,
292+
setDefaultResultOrder,
288293
setServers: defaultResolverSetServers,
289294

290295
// uv_getaddrinfo flags
@@ -329,6 +334,7 @@ ObjectDefineProperties(module.exports, {
329334
if (promises === null) {
330335
promises = require('internal/dns/promises');
331336
promises.setServers = defaultResolverSetServers;
337+
promises.setDefaultResultOrder = setDefaultResultOrder;
332338
}
333339
return promises;
334340
}

‎lib/internal/dns/promises.js

+5-3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,4 @@
11
'use strict';
2-
32
const {
43
ArrayPrototypeMap,
54
ObjectCreate,
@@ -14,6 +13,7 @@ const {
1413
validateHints,
1514
validateTimeout,
1615
emitInvalidHostnameWarning,
16+
getDefaultVerbatim,
1717
} = require('internal/dns/utils');
1818
const { codes, dnsException } = require('internal/errors');
1919
const { toASCII } = require('internal/idna');
@@ -103,7 +103,7 @@ function lookup(hostname, options) {
103103
var hints = 0;
104104
var family = -1;
105105
var all = false;
106-
var verbatim = false;
106+
var verbatim = getDefaultVerbatim();
107107

108108
// Parse arguments
109109
if (hostname && typeof hostname !== 'string') {
@@ -112,7 +112,9 @@ function lookup(hostname, options) {
112112
hints = options.hints >>> 0;
113113
family = options.family >>> 0;
114114
all = options.all === true;
115-
verbatim = options.verbatim === true;
115+
if (typeof options.verbatim === 'boolean') {
116+
verbatim = options.verbatim === true;
117+
}
116118

117119
validateHints(hints);
118120
} else {

‎lib/internal/dns/utils.js

+21
Original file line numberDiff line numberDiff line change
@@ -14,8 +14,10 @@ const {
1414

1515
const errors = require('internal/errors');
1616
const { isIP } = require('internal/net');
17+
const { getOptionValue } = require('internal/options');
1718
const {
1819
validateInt32,
20+
validateOneOf,
1921
validateString,
2022
} = require('internal/validators');
2123
const {
@@ -186,6 +188,23 @@ function emitInvalidHostnameWarning(hostname) {
186188
);
187189
}
188190

191+
let dnsOrder = getOptionValue('--dns-result-order') || 'ipv4first';
192+
193+
function getDefaultVerbatim() {
194+
switch (dnsOrder) {
195+
case 'verbatim':
196+
return true;
197+
case 'ipv4first':
198+
default:
199+
return false;
200+
}
201+
}
202+
203+
function setDefaultResultOrder(value) {
204+
validateOneOf(value, 'dnsOrder', ['verbatim', 'ipv4first']);
205+
dnsOrder = value;
206+
}
207+
189208
module.exports = {
190209
bindDefaultResolver,
191210
getDefaultResolver,
@@ -194,4 +213,6 @@ module.exports = {
194213
validateTimeout,
195214
Resolver,
196215
emitInvalidHostnameWarning,
216+
getDefaultVerbatim,
217+
setDefaultResultOrder,
197218
};

‎src/node_options.cc

+7
Original file line numberDiff line numberDiff line change
@@ -282,6 +282,13 @@ EnvironmentOptionsParser::EnvironmentOptionsParser() {
282282
" (default: current working directory)",
283283
&EnvironmentOptions::diagnostic_dir,
284284
kAllowedInEnvironment);
285+
AddOption("--dns-result-order",
286+
"set default value of verbatim in dns.lookup. Options are "
287+
"'ipv4first' (IPv4 addresses are placed before IPv6 addresses) "
288+
"'verbatim' (addresses are in the order the DNS resolver "
289+
"returned)",
290+
&EnvironmentOptions::dns_result_order,
291+
kAllowedInEnvironment);
285292
AddOption("--enable-source-maps",
286293
"experimental Source Map V3 support",
287294
&EnvironmentOptions::enable_source_maps,

‎src/node_options.h

+1
Original file line numberDiff line numberDiff line change
@@ -101,6 +101,7 @@ class EnvironmentOptions : public Options {
101101
public:
102102
bool abort_on_uncaught_exception = false;
103103
std::vector<std::string> conditions;
104+
std::string dns_result_order;
104105
bool enable_source_maps = false;
105106
bool experimental_abortcontroller = false;
106107
bool experimental_json_modules = false;
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,51 @@
1+
// Flags: --expose-internals --dns-result-order=ipv4first
2+
'use strict';
3+
const common = require('../common');
4+
const assert = require('assert');
5+
const { internalBinding } = require('internal/test/binding');
6+
const cares = internalBinding('cares_wrap');
7+
const { promisify } = require('util');
8+
9+
// Test that --dns-result-order=ipv4first works as expected.
10+
11+
const originalGetaddrinfo = cares.getaddrinfo;
12+
const calls = [];
13+
cares.getaddrinfo = common.mustCallAtLeast((...args) => {
14+
calls.push(args);
15+
originalGetaddrinfo(...args);
16+
}, 1);
17+
18+
const dns = require('dns');
19+
const dnsPromises = dns.promises;
20+
21+
let verbatim;
22+
23+
// We want to test the parameter of verbatim only so that we
24+
// ignore possible errors here.
25+
function allowFailed(fn) {
26+
return fn.catch((_err) => {
27+
//
28+
});
29+
}
30+
31+
(async () => {
32+
let callsLength = 0;
33+
const checkParameter = (expected) => {
34+
assert.strictEqual(calls.length, callsLength + 1);
35+
verbatim = calls[callsLength][4];
36+
assert.strictEqual(verbatim, expected);
37+
callsLength += 1;
38+
};
39+
40+
await allowFailed(promisify(dns.lookup)('example.org'));
41+
checkParameter(false);
42+
43+
await allowFailed(dnsPromises.lookup('example.org'));
44+
checkParameter(false);
45+
46+
await allowFailed(promisify(dns.lookup)('example.org', {}));
47+
checkParameter(false);
48+
49+
await allowFailed(dnsPromises.lookup('example.org', {}));
50+
checkParameter(false);
51+
})().then(common.mustCall());
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,51 @@
1+
// Flags: --expose-internals --dns-result-order=verbatim
2+
'use strict';
3+
const common = require('../common');
4+
const assert = require('assert');
5+
const { internalBinding } = require('internal/test/binding');
6+
const cares = internalBinding('cares_wrap');
7+
const { promisify } = require('util');
8+
9+
// Test that --dns-result-order=verbatim works as expected.
10+
11+
const originalGetaddrinfo = cares.getaddrinfo;
12+
const calls = [];
13+
cares.getaddrinfo = common.mustCallAtLeast((...args) => {
14+
calls.push(args);
15+
originalGetaddrinfo(...args);
16+
}, 1);
17+
18+
const dns = require('dns');
19+
const dnsPromises = dns.promises;
20+
21+
let verbatim;
22+
23+
// We want to test the parameter of verbatim only so that we
24+
// ignore possible errors here.
25+
function allowFailed(fn) {
26+
return fn.catch((_err) => {
27+
//
28+
});
29+
}
30+
31+
(async () => {
32+
let callsLength = 0;
33+
const checkParameter = (expected) => {
34+
assert.strictEqual(calls.length, callsLength + 1);
35+
verbatim = calls[callsLength][4];
36+
assert.strictEqual(verbatim, expected);
37+
callsLength += 1;
38+
};
39+
40+
await allowFailed(promisify(dns.lookup)('example.org'));
41+
checkParameter(true);
42+
43+
await allowFailed(dnsPromises.lookup('example.org'));
44+
checkParameter(true);
45+
46+
await allowFailed(promisify(dns.lookup)('example.org', {}));
47+
checkParameter(true);
48+
49+
await allowFailed(dnsPromises.lookup('example.org', {}));
50+
checkParameter(true);
51+
})().then(common.mustCall());
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,93 @@
1+
// Flags: --expose-internals
2+
'use strict';
3+
const common = require('../common');
4+
const assert = require('assert');
5+
const { internalBinding } = require('internal/test/binding');
6+
const cares = internalBinding('cares_wrap');
7+
const { promisify } = require('util');
8+
9+
// Test that `dns.setDefaultResultOrder()` and
10+
// `dns.promises.setDefaultResultOrder()` works as expected.
11+
12+
const originalGetaddrinfo = cares.getaddrinfo;
13+
const calls = [];
14+
cares.getaddrinfo = common.mustCallAtLeast((...args) => {
15+
calls.push(args);
16+
originalGetaddrinfo(...args);
17+
}, 1);
18+
19+
const dns = require('dns');
20+
const dnsPromises = dns.promises;
21+
22+
let verbatim;
23+
24+
// We want to test the parameter of verbatim only so that we
25+
// ignore possible errors here.
26+
function allowFailed(fn) {
27+
return fn.catch((_err) => {
28+
//
29+
});
30+
}
31+
32+
assert.throws(() => dns.setDefaultResultOrder('my_order'), {
33+
code: 'ERR_INVALID_ARG_VALUE',
34+
});
35+
assert.throws(() => dns.promises.setDefaultResultOrder('my_order'), {
36+
code: 'ERR_INVALID_ARG_VALUE',
37+
});
38+
assert.throws(() => dns.setDefaultResultOrder(4), {
39+
code: 'ERR_INVALID_ARG_VALUE',
40+
});
41+
assert.throws(() => dns.promises.setDefaultResultOrder(4), {
42+
code: 'ERR_INVALID_ARG_VALUE',
43+
});
44+
45+
(async () => {
46+
let callsLength = 0;
47+
const checkParameter = (expected) => {
48+
assert.strictEqual(calls.length, callsLength + 1);
49+
verbatim = calls[callsLength][4];
50+
assert.strictEqual(verbatim, expected);
51+
callsLength += 1;
52+
};
53+
54+
dns.setDefaultResultOrder('verbatim');
55+
await allowFailed(promisify(dns.lookup)('example.org'));
56+
checkParameter(true);
57+
await allowFailed(dnsPromises.lookup('example.org'));
58+
checkParameter(true);
59+
await allowFailed(promisify(dns.lookup)('example.org', {}));
60+
checkParameter(true);
61+
await allowFailed(dnsPromises.lookup('example.org', {}));
62+
checkParameter(true);
63+
64+
dns.setDefaultResultOrder('ipv4first');
65+
await allowFailed(promisify(dns.lookup)('example.org'));
66+
checkParameter(false);
67+
await allowFailed(dnsPromises.lookup('example.org'));
68+
checkParameter(false);
69+
await allowFailed(promisify(dns.lookup)('example.org', {}));
70+
checkParameter(false);
71+
await allowFailed(dnsPromises.lookup('example.org', {}));
72+
checkParameter(false);
73+
74+
dns.promises.setDefaultResultOrder('verbatim');
75+
await allowFailed(promisify(dns.lookup)('example.org'));
76+
checkParameter(true);
77+
await allowFailed(dnsPromises.lookup('example.org'));
78+
checkParameter(true);
79+
await allowFailed(promisify(dns.lookup)('example.org', {}));
80+
checkParameter(true);
81+
await allowFailed(dnsPromises.lookup('example.org', {}));
82+
checkParameter(true);
83+
84+
dns.promises.setDefaultResultOrder('ipv4first');
85+
await allowFailed(promisify(dns.lookup)('example.org'));
86+
checkParameter(false);
87+
await allowFailed(dnsPromises.lookup('example.org'));
88+
checkParameter(false);
89+
await allowFailed(promisify(dns.lookup)('example.org', {}));
90+
checkParameter(false);
91+
await allowFailed(dnsPromises.lookup('example.org', {}));
92+
checkParameter(false);
93+
})().then(common.mustCall());

0 commit comments

Comments
 (0)
Please sign in to comment.