mirror of
https://github.com/denoland/deno.git
synced 2025-01-24 16:08:03 -05:00
472 lines
13 KiB
JavaScript
472 lines
13 KiB
JavaScript
|
// 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());
|
||
|
|
||
|
}));
|
||
|
}
|