2016-12-14 13:19:12 -08:00
|
|
|
'use strict';
|
|
|
|
|
|
|
|
|
|
exports = module.exports = {
|
|
|
|
|
upsert: upsert,
|
|
|
|
|
get: get,
|
|
|
|
|
del: del,
|
2017-01-10 11:12:38 +01:00
|
|
|
waitForDns: require('./waitfordns.js'),
|
|
|
|
|
verifyDnsConfig: verifyDnsConfig
|
2016-12-14 13:19:12 -08:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
var assert = require('assert'),
|
2017-01-10 11:12:38 +01:00
|
|
|
async = require('async'),
|
2016-12-14 13:19:12 -08:00
|
|
|
debug = require('debug')('box:dns/noop'),
|
2017-01-10 11:12:38 +01:00
|
|
|
dns = require('native-dns'),
|
2016-12-14 13:19:12 -08:00
|
|
|
SubdomainError = require('../subdomains.js').SubdomainError,
|
|
|
|
|
util = require('util');
|
|
|
|
|
|
|
|
|
|
function upsert(dnsConfig, zoneName, subdomain, type, values, callback) {
|
|
|
|
|
assert.strictEqual(typeof dnsConfig, 'object');
|
|
|
|
|
assert.strictEqual(typeof zoneName, 'string');
|
|
|
|
|
assert.strictEqual(typeof subdomain, 'string');
|
|
|
|
|
assert.strictEqual(typeof type, 'string');
|
|
|
|
|
assert(util.isArray(values));
|
|
|
|
|
assert.strictEqual(typeof callback, 'function');
|
|
|
|
|
|
|
|
|
|
debug('upsert: %s for zone %s of type %s with values %j', subdomain, zoneName, type, values);
|
|
|
|
|
|
|
|
|
|
return callback(null, 'noop-record-id');
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function get(dnsConfig, zoneName, subdomain, type, callback) {
|
|
|
|
|
assert.strictEqual(typeof dnsConfig, 'object');
|
|
|
|
|
assert.strictEqual(typeof zoneName, 'string');
|
|
|
|
|
assert.strictEqual(typeof subdomain, 'string');
|
|
|
|
|
assert.strictEqual(typeof type, 'string');
|
|
|
|
|
assert.strictEqual(typeof callback, 'function');
|
|
|
|
|
|
|
|
|
|
callback(null, [ ]); // returning ip confuses apptask into thinking the entry already exists
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function del(dnsConfig, zoneName, subdomain, type, values, callback) {
|
|
|
|
|
assert.strictEqual(typeof dnsConfig, 'object');
|
|
|
|
|
assert.strictEqual(typeof zoneName, 'string');
|
|
|
|
|
assert.strictEqual(typeof subdomain, 'string');
|
|
|
|
|
assert.strictEqual(typeof type, 'string');
|
|
|
|
|
assert(util.isArray(values));
|
|
|
|
|
assert.strictEqual(typeof callback, 'function');
|
|
|
|
|
|
|
|
|
|
return callback();
|
|
|
|
|
}
|
|
|
|
|
|
2017-01-10 11:12:38 +01:00
|
|
|
function verifyDnsConfig(dnsConfig, domain, ip, callback) {
|
|
|
|
|
assert.strictEqual(typeof dnsConfig, 'object');
|
|
|
|
|
assert.strictEqual(typeof domain, 'string');
|
|
|
|
|
assert.strictEqual(typeof ip, 'string');
|
|
|
|
|
assert.strictEqual(typeof callback, 'function');
|
|
|
|
|
|
2017-01-10 12:26:14 +01:00
|
|
|
var adminDomain = 'my.' + domain;
|
|
|
|
|
|
2017-01-10 11:12:38 +01:00
|
|
|
dns.resolveNs(domain, function (error, nameservers) {
|
2017-01-10 11:49:23 +01:00
|
|
|
if (error || !nameservers) return callback(new SubdomainError(SubdomainError.BAD_FIELD, 'Unable to get nameservers'));
|
|
|
|
|
|
|
|
|
|
// async.every only reports bools
|
|
|
|
|
var stashedError = null;
|
2017-01-10 11:12:38 +01:00
|
|
|
|
|
|
|
|
async.every(nameservers, function (nameserver, callback) {
|
|
|
|
|
// ns records cannot have cname
|
|
|
|
|
dns.resolve4(nameserver, function (error, nsIps) {
|
2017-01-10 11:49:23 +01:00
|
|
|
if (error || !nsIps || nsIps.length === 0) {
|
|
|
|
|
stashedError = new SubdomainError(SubdomainError.BAD_FIELD, 'Unable to resolve nameservers for this domain');
|
|
|
|
|
return callback(false);
|
|
|
|
|
}
|
2017-01-10 11:12:38 +01:00
|
|
|
|
2017-01-10 11:49:23 +01:00
|
|
|
async.every(nsIps, function (nsIp, callback) {
|
2017-01-10 11:12:38 +01:00
|
|
|
var req = dns.Request({
|
2017-01-10 12:26:14 +01:00
|
|
|
question: dns.Question({ name: adminDomain, type: 'A' }),
|
2017-01-10 11:12:38 +01:00
|
|
|
server: { address: nsIp },
|
|
|
|
|
timeout: 5000
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
req.on('timeout', function () {
|
2017-01-10 12:26:14 +01:00
|
|
|
debug('nameserver %s (%s) timed out when trying to resolve %s', nameserver, nsIp, adminDomain);
|
2017-01-10 11:49:23 +01:00
|
|
|
return callback(true); // should be ok if dns server is down
|
2017-01-10 11:12:38 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
req.on('message', function (error, message) {
|
|
|
|
|
if (error) {
|
2017-01-10 12:26:14 +01:00
|
|
|
debug('nameserver %s (%s) returned error trying to resolve %s: %s', nameserver, nsIp, adminDomain, error);
|
2017-01-10 11:49:23 +01:00
|
|
|
return callback(false);
|
2017-01-10 11:12:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var answer = message.answer;
|
|
|
|
|
|
|
|
|
|
if (!answer || answer.length === 0) {
|
2017-01-10 12:26:14 +01:00
|
|
|
debug('bad answer from nameserver %s (%s) resolving %s (%s): %j', nameserver, nsIp, adminDomain, 'A', message);
|
2017-01-10 11:49:23 +01:00
|
|
|
return callback(false);
|
2017-01-10 11:12:38 +01:00
|
|
|
}
|
|
|
|
|
|
2017-01-10 12:26:14 +01:00
|
|
|
debug('verifyDnsConfig: ns: %s (%s), name:%s Actual:%j Expecting:%s', nameserver, nsIp, adminDomain, answer, ip);
|
2017-01-10 11:12:38 +01:00
|
|
|
|
|
|
|
|
var match = answer.some(function (a) {
|
|
|
|
|
return a.address === ip;
|
|
|
|
|
});
|
|
|
|
|
|
2017-01-10 11:49:23 +01:00
|
|
|
if (match) return callback(true); // done!
|
2017-01-10 11:12:38 +01:00
|
|
|
|
2017-01-10 11:49:23 +01:00
|
|
|
callback(false);
|
2017-01-10 11:12:38 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
req.send();
|
|
|
|
|
}, callback);
|
|
|
|
|
});
|
2017-01-10 11:49:23 +01:00
|
|
|
}, function (success) {
|
|
|
|
|
if (stashedError) return callback(stashedError);
|
2017-01-10 12:26:14 +01:00
|
|
|
if (!success) return callback(new SubdomainError(SubdomainError.BAD_FIELD, 'The domain ' + adminDomain + ' does not resolve to the servers IP'));
|
2017-01-10 11:49:23 +01:00
|
|
|
|
2017-01-10 11:32:44 +01:00
|
|
|
callback(null, { provider: dnsConfig.provider, wildcard: !!dnsConfig.wildcard });
|
2017-01-10 11:12:38 +01:00
|
|
|
});
|
|
|
|
|
});
|
|
|
|
|
}
|