summaryrefslogtreecommitdiff
path: root/tests/node_compat/test/parallel/test-dns.js
diff options
context:
space:
mode:
Diffstat (limited to 'tests/node_compat/test/parallel/test-dns.js')
-rw-r--r--tests/node_compat/test/parallel/test-dns.js471
1 files changed, 471 insertions, 0 deletions
diff --git a/tests/node_compat/test/parallel/test-dns.js b/tests/node_compat/test/parallel/test-dns.js
new file mode 100644
index 000000000..e56f7ca40
--- /dev/null
+++ b/tests/node_compat/test/parallel/test-dns.js
@@ -0,0 +1,471 @@
+// deno-fmt-ignore-file
+// deno-lint-ignore-file
+
+// Copyright Joyent and Node contributors. All rights reserved. MIT license.
+// Taken from Node 16.13.0
+// This file is automatically generated by "node/_tools/setup.ts". Do not modify this file manually
+
+// 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';
+
+// TODO: enable remaining tests once functionality is implemented.
+
+const common = require('../common');
+const dnstools = require('../common/dns');
+const assert = require('assert');
+
+const dns = require('dns');
+const dnsPromises = dns.promises;
+const dgram = require('dgram');
+
+// TODO(cmorten): currently don't expose defaults
+// const existing = dns.getServers();
+// assert(existing.length > 0);
+
+// Verify that setServers() handles arrays with holes and other oddities
+{
+ const servers = [];
+
+ servers[0] = '127.0.0.1';
+ servers[2] = '0.0.0.0';
+ dns.setServers(servers);
+
+ assert.deepStrictEqual(dns.getServers(), ['127.0.0.1', '0.0.0.0']);
+}
+
+{
+ const servers = ['127.0.0.1', '192.168.1.1'];
+
+ servers[3] = '127.1.0.1';
+ servers[4] = '127.1.0.1';
+ servers[5] = '127.1.1.1';
+
+ Object.defineProperty(servers, 2, {
+ enumerable: true,
+ get: () => {
+ servers.length = 3;
+ return '0.0.0.0';
+ }
+ });
+
+ dns.setServers(servers);
+ assert.deepStrictEqual(dns.getServers(), [
+ '127.0.0.1',
+ '192.168.1.1',
+ '0.0.0.0',
+ ]);
+}
+
+{
+ // Various invalidities, all of which should throw a clean error.
+ const invalidServers = [
+ ' ',
+ '\n',
+ '\0',
+ '1'.repeat(3 * 4),
+ // Check for REDOS issues.
+ ':'.repeat(100000),
+ '['.repeat(100000),
+ '['.repeat(100000) + ']'.repeat(100000) + 'a',
+ ];
+ invalidServers.forEach((serv) => {
+ assert.throws(
+ () => {
+ dns.setServers([serv]);
+ },
+ {
+ name: 'TypeError',
+ code: 'ERR_INVALID_IP_ADDRESS'
+ }
+ );
+ });
+}
+
+const goog = [
+ '8.8.8.8',
+ '8.8.4.4',
+];
+dns.setServers(goog);
+assert.deepStrictEqual(dns.getServers(), goog);
+assert.throws(() => dns.setServers(['foobar']), {
+ code: 'ERR_INVALID_IP_ADDRESS',
+ name: 'TypeError',
+ message: 'Invalid IP address: foobar'
+});
+assert.throws(() => dns.setServers(['127.0.0.1:va']), {
+ code: 'ERR_INVALID_IP_ADDRESS',
+ name: 'TypeError',
+ message: 'Invalid IP address: 127.0.0.1:va'
+});
+assert.deepStrictEqual(dns.getServers(), goog);
+
+const goog6 = [
+ '2001:4860:4860::8888',
+ '2001:4860:4860::8844',
+];
+dns.setServers(goog6);
+assert.deepStrictEqual(dns.getServers(), goog6);
+
+goog6.push('4.4.4.4');
+dns.setServers(goog6);
+assert.deepStrictEqual(dns.getServers(), goog6);
+
+const ports = [
+ '4.4.4.4:53',
+ '[2001:4860:4860::8888]:53',
+ '103.238.225.181:666',
+ '[fe80::483a:5aff:fee6:1f04]:666',
+ '[fe80::483a:5aff:fee6:1f04]',
+];
+const portsExpected = [
+ '4.4.4.4',
+ '2001:4860:4860::8888',
+ '103.238.225.181:666',
+ '[fe80::483a:5aff:fee6:1f04]:666',
+ 'fe80::483a:5aff:fee6:1f04',
+];
+dns.setServers(ports);
+assert.deepStrictEqual(dns.getServers(), portsExpected);
+
+dns.setServers([]);
+assert.deepStrictEqual(dns.getServers(), []);
+
+{
+ const errObj = {
+ code: 'ERR_INVALID_ARG_TYPE',
+ name: 'TypeError',
+ message: 'The "rrtype" argument must be of type string. ' +
+ 'Received an instance of Array'
+ };
+ assert.throws(() => {
+ dns.resolve('example.com', [], common.mustNotCall());
+ }, errObj);
+ assert.throws(() => {
+ dnsPromises.resolve('example.com', []);
+ }, errObj);
+}
+{
+ const errObj = {
+ code: 'ERR_INVALID_ARG_TYPE',
+ name: 'TypeError',
+ message: 'The "name" argument must be of type string. ' +
+ 'Received undefined'
+ };
+ assert.throws(() => {
+ dnsPromises.resolve();
+ }, errObj);
+}
+
+// dns.lookup should accept only falsey and string values
+{
+ const errorReg = {
+ code: 'ERR_INVALID_ARG_TYPE',
+ name: 'TypeError',
+ message: /^The "hostname" argument must be of type string\. Received .*/
+ };
+
+ assert.throws(() => dns.lookup({}, common.mustNotCall()), errorReg);
+
+ assert.throws(() => dns.lookup([], common.mustNotCall()), errorReg);
+
+ assert.throws(() => dns.lookup(true, common.mustNotCall()), errorReg);
+
+ assert.throws(() => dns.lookup(1, common.mustNotCall()), errorReg);
+
+ assert.throws(() => dns.lookup(common.mustNotCall(), common.mustNotCall()),
+ errorReg);
+
+ assert.throws(() => dnsPromises.lookup({}), errorReg);
+ assert.throws(() => dnsPromises.lookup([]), errorReg);
+ assert.throws(() => dnsPromises.lookup(true), errorReg);
+ assert.throws(() => dnsPromises.lookup(1), errorReg);
+ assert.throws(() => dnsPromises.lookup(common.mustNotCall()), errorReg);
+}
+
+// dns.lookup should accept falsey values
+{
+ const checkCallback = (err, address, family) => {
+ assert.ifError(err);
+ assert.strictEqual(address, null);
+ assert.strictEqual(family, 4);
+ };
+
+ ['', null, undefined, 0, NaN].forEach(async (value) => {
+ const res = await dnsPromises.lookup(value);
+ assert.deepStrictEqual(res, { address: null, family: 4 });
+ dns.lookup(value, common.mustCall(checkCallback));
+ });
+}
+
+{
+ // Make sure that dns.lookup throws if hints does not represent a valid flag.
+ // (dns.V4MAPPED | dns.ADDRCONFIG | dns.ALL) + 1 is invalid because:
+ // - it's different from dns.V4MAPPED and dns.ADDRCONFIG and dns.ALL.
+ // - it's different from any subset of them bitwise ored.
+ // - it's different from 0.
+ // - it's an odd number different than 1, and thus is invalid, because
+ // flags are either === 1 or even.
+ const hints = (dns.V4MAPPED | dns.ADDRCONFIG | dns.ALL) + 1;
+ const err = {
+ code: 'ERR_INVALID_ARG_VALUE',
+ name: 'TypeError',
+ message: /The argument 'hints' is invalid\. Received \d+/
+ };
+
+ assert.throws(() => {
+ dnsPromises.lookup('nodejs.org', { hints });
+ }, err);
+ assert.throws(() => {
+ dns.lookup('nodejs.org', { hints }, common.mustNotCall());
+ }, err);
+}
+
+assert.throws(() => dns.lookup("nodejs.org"), {
+ code: "ERR_INVALID_ARG_TYPE",
+ name: "TypeError",
+});
+
+assert.throws(() => dns.lookup("nodejs.org", 4), {
+ code: "ERR_INVALID_ARG_TYPE",
+ name: "TypeError",
+});
+
+dns.lookup('', { family: 4, hints: 0 }, common.mustCall());
+
+dns.lookup('', {
+ family: 6,
+ hints: dns.ADDRCONFIG
+}, common.mustCall());
+
+dns.lookup('', { hints: dns.V4MAPPED }, common.mustCall());
+
+dns.lookup('', {
+ hints: dns.ADDRCONFIG | dns.V4MAPPED
+}, common.mustCall());
+
+dns.lookup('', {
+ hints: dns.ALL
+}, common.mustCall());
+
+dns.lookup('', {
+ hints: dns.V4MAPPED | dns.ALL
+}, common.mustCall());
+
+dns.lookup('', {
+ hints: dns.ADDRCONFIG | dns.V4MAPPED | dns.ALL
+}, common.mustCall());
+
+(async function() {
+ await dnsPromises.lookup('', { family: 4, hints: 0 });
+ await dnsPromises.lookup('', { family: 6, hints: dns.ADDRCONFIG });
+ await dnsPromises.lookup('', { hints: dns.V4MAPPED });
+ await dnsPromises.lookup('', { hints: dns.ADDRCONFIG | dns.V4MAPPED });
+ await dnsPromises.lookup('', { hints: dns.ALL });
+ await dnsPromises.lookup('', { hints: dns.V4MAPPED | dns.ALL });
+ await dnsPromises.lookup('', {
+ hints: dns.ADDRCONFIG | dns.V4MAPPED | dns.ALL
+ });
+})().then(common.mustCall());
+
+// {
+// const err = {
+// code: 'ERR_MISSING_ARGS',
+// name: 'TypeError',
+// message: 'The "address", "port", and "callback" arguments must be ' +
+// 'specified'
+// };
+
+// assert.throws(() => dns.lookupService('0.0.0.0'), err);
+// err.message = 'The "address" and "port" arguments must be specified';
+// assert.throws(() => dnsPromises.lookupService('0.0.0.0'), err);
+// }
+
+// {
+// const invalidAddress = 'fasdfdsaf';
+// const err = {
+// code: 'ERR_INVALID_ARG_VALUE',
+// name: 'TypeError',
+// message: `The argument 'address' is invalid. Received '${invalidAddress}'`
+// };
+
+// assert.throws(() => {
+// dnsPromises.lookupService(invalidAddress, 0);
+// }, err);
+
+// assert.throws(() => {
+// dns.lookupService(invalidAddress, 0, common.mustNotCall());
+// }, err);
+// }
+
+// const portErr = (port) => {
+// const err = {
+// code: 'ERR_SOCKET_BAD_PORT',
+// message:
+// `Port should be >= 0 and < 65536. Received ${port}.`,
+// name: 'RangeError'
+// };
+
+// assert.throws(() => {
+// dnsPromises.lookupService('0.0.0.0', port);
+// }, err);
+
+// assert.throws(() => {
+// dns.lookupService('0.0.0.0', port, common.mustNotCall());
+// }, err);
+// };
+// portErr(null);
+// portErr(undefined);
+// portErr(65538);
+// portErr('test');
+
+// assert.throws(() => {
+// dns.lookupService('0.0.0.0', 80, null);
+// }, {
+// code: 'ERR_INVALID_ARG_TYPE',
+// name: 'TypeError'
+// });
+
+{
+ dns.resolveMx('foo.onion', function(err) {
+ assert.deepStrictEqual(err.code, 'ENOTFOUND');
+ assert.deepStrictEqual(err.syscall, 'queryMx');
+ assert.deepStrictEqual(err.hostname, 'foo.onion');
+ assert.deepStrictEqual(err.message, 'queryMx ENOTFOUND foo.onion');
+ });
+}
+
+{
+ const cases = [
+ {
+ method: "resolveAny",
+ answers: [
+ { type: "A", address: "1.2.3.4" /*ttl: 3333333333*/ },
+ { type: "AAAA", address: "::42" /*ttl: 3333333333*/ },
+ { type: "MX", priority: 42, exchange: "foobar.com", ttl: 3333333333 },
+ { type: "NS", value: "foobar.org", ttl: 3333333333 },
+ { type: "PTR", value: "baz.org", ttl: 3333333333 },
+ {
+ type: "SOA",
+ nsname: "ns1.example.com",
+ hostmaster: "admin.example.com",
+ serial: 3210987654,
+ refresh: 900,
+ retry: 900,
+ expire: 1800,
+ minttl: 3333333333,
+ },
+ ],
+ },
+
+ // TODO(cmorten): support ttl option
+ // {
+ // method: "resolve4",
+ // options: { ttl: true },
+ // answers: [{ type: "A", address: "1.2.3.4", ttl: 3333333333 }],
+ // },
+
+ // {
+ // method: "resolve6",
+ // options: { ttl: true },
+ // answers: [{ type: "AAAA", address: "::42", ttl: 3333333333 }],
+ // },
+
+ {
+ method: "resolveSoa",
+ answers: [
+ {
+ type: "SOA",
+ nsname: "ns1.example.com",
+ hostmaster: "admin.example.com",
+ serial: 3210987654,
+ refresh: 900,
+ retry: 900,
+ expire: 1800,
+ minttl: 3333333333,
+ },
+ ],
+ },
+ ];
+
+ const server = dgram.createSocket('udp4');
+
+ server.on('message', common.mustCall((msg, { address, port }) => {
+ const parsed = dnstools.parseDNSPacket(msg);
+ const domain = parsed.questions[0].domain;
+ assert.strictEqual(domain, 'example.org');
+
+ server.send(dnstools.writeDNSPacket({
+ id: parsed.id,
+ questions: parsed.questions,
+ answers: cases[0].answers.map(
+ (answer) => Object.assign({ domain }, answer)
+ ),
+ }), port, address);
+ // Don't have "ANY" query type available so calls greatly increased with
+ // polyfill method.
+ }, /*cases.length * 2*/ 32));
+
+ server.bind(0, common.mustCall(() => {
+ const address = server.address();
+ dns.setServers([`127.0.0.1:${address.port}`]);
+
+ function validateResults(res) {
+ if (!Array.isArray(res))
+ res = [res];
+
+ assert.deepStrictEqual(res.map(tweakEntry),
+ cases[0].answers.map(tweakEntry));
+ }
+
+ function tweakEntry(r) {
+ const ret = { ...r };
+
+ const { method } = cases[0];
+
+ // TTL values are only provided for A and AAAA entries.
+ if (!['A', 'AAAA'].includes(ret.type) && !/^resolve(4|6)?$/.test(method))
+ delete ret.ttl;
+
+ if (method !== 'resolveAny')
+ delete ret.type;
+
+ return ret;
+ }
+
+ (async function nextCase() {
+ if (cases.length === 0)
+ return server.close();
+
+ const { method, options } = cases[0];
+
+ validateResults(await dnsPromises[method]('example.org', options));
+
+ dns[method]('example.org', options, common.mustSucceed((res) => {
+ validateResults(res);
+ cases.shift();
+ nextCase();
+ }));
+ })().then(common.mustCall());
+
+ }));
+}