Files
cloudron-box/src/dns/gcdns.js
T

205 lines
9.0 KiB
JavaScript
Raw Normal View History

2017-09-05 22:23:24 +02:00
'use strict';
exports = module.exports = {
upsert: upsert,
get: get,
del: del,
waitForDns: require('./waitfordns.js'),
verifyDnsConfig: verifyDnsConfig
};
var assert = require('assert'),
2017-09-12 16:29:07 +02:00
debug = require('debug')('box:dns/gcdns'),
2018-02-08 10:21:31 -08:00
dns = require('../native-dns.js'),
2018-04-29 11:20:12 -07:00
DomainsError = require('../domains.js').DomainsError,
2017-10-25 20:35:24 -07:00
GCDNS = require('@google-cloud/dns'),
2017-09-05 22:23:24 +02:00
util = require('util'),
_ = require('underscore');
function getDnsCredentials(dnsConfig) {
assert.strictEqual(typeof dnsConfig, 'object');
2018-06-17 21:44:08 -07:00
return {
2017-09-05 22:23:24 +02:00
projectId: dnsConfig.projectId,
2018-06-17 21:44:08 -07:00
credentials: {
2017-09-05 22:23:24 +02:00
client_email: dnsConfig.credentials.client_email,
private_key: dnsConfig.credentials.private_key
2018-06-17 21:44:08 -07:00
}
};
2017-09-05 22:23:24 +02:00
}
function getZoneByName(dnsConfig, zoneName, callback) {
assert.strictEqual(typeof dnsConfig, 'object');
assert.strictEqual(typeof zoneName, 'string');
assert.strictEqual(typeof callback, 'function');
var gcdns = GCDNS(getDnsCredentials(dnsConfig));
2017-09-14 23:11:17 -07:00
gcdns.getZones(function (error, zones) {
2018-04-29 11:20:12 -07:00
if (error && error.message === 'invalid_grant') return callback(new DomainsError(DomainsError.ACCESS_DENIED, 'The key was probably revoked'));
if (error && error.reason === 'No such domain') return callback(new DomainsError(DomainsError.NOT_FOUND, error.message));
if (error && error.code === 403) return callback(new DomainsError(DomainsError.ACCESS_DENIED, error.message));
if (error && error.code === 404) return callback(new DomainsError(DomainsError.NOT_FOUND, error.message));
2017-09-14 18:12:07 -07:00
if (error) {
debug('gcdns.getZones', error);
2018-04-29 11:20:12 -07:00
return callback(new DomainsError(DomainsError.EXTERNAL_ERROR, error));
2017-09-12 16:29:07 +02:00
}
2017-09-05 22:23:24 +02:00
var zone = zones.filter(function (zone) {
return zone.metadata.dnsName.slice(0, -1) === zoneName; // the zone name contains a '.' at the end
})[0];
2018-04-29 11:20:12 -07:00
if (!zone) return callback(new DomainsError(DomainsError.NOT_FOUND, 'no such zone'));
2017-09-05 22:23:24 +02:00
callback(null, zone); //zone.metadata ~= {name="", dnsName="", nameServers:[]}
});
}
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('add: %s for zone %s of type %s with values %j', subdomain, zoneName, type, values);
getZoneByName(getDnsCredentials(dnsConfig), zoneName, function (error, zone) {
if (error) return callback(error);
var domain = (subdomain ? subdomain + '.' : '') + zoneName + '.';
2017-09-05 22:23:24 +02:00
2017-09-14 23:11:17 -07:00
zone.getRecords({ type: type, name: domain }, function (error, oldRecords) {
2018-04-29 11:20:12 -07:00
if (error && error.code === 403) return callback(new DomainsError(DomainsError.ACCESS_DENIED, error.message));
2017-09-12 16:29:07 +02:00
if (error) {
debug('upsert->zone.getRecords', error);
2018-04-29 11:20:12 -07:00
return callback(new DomainsError(DomainsError.EXTERNAL_ERROR, error.message));
2017-09-12 16:29:07 +02:00
}
2017-09-05 22:23:24 +02:00
var newRecord = zone.record(type, {
name: domain,
2017-09-14 23:11:17 -07:00
data: values,
ttl: 1
});
2017-09-14 23:11:17 -07:00
zone.createChange({ delete: oldRecords, add: newRecord }, function(error, change) {
2018-04-29 11:20:12 -07:00
if (error && error.code === 403) return callback(new DomainsError(DomainsError.ACCESS_DENIED, error.message));
if (error && error.code === 412) return callback(new DomainsError(DomainsError.STILL_BUSY, error.message));
if (error) {
debug('upsert->zone.createChange', error);
2018-04-29 11:20:12 -07:00
return callback(new DomainsError(DomainsError.EXTERNAL_ERROR, error.message));
}
callback(null, change.id);
});
2017-09-05 22:23:24 +02:00
});
});
}
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');
getZoneByName(getDnsCredentials(dnsConfig), zoneName, function (error, zone) {
if (error) return callback(error);
var params = {
name: (subdomain ? subdomain + '.' : '') + zoneName + '.',
type: type
};
2017-09-14 23:11:17 -07:00
zone.getRecords(params, function (error, records) {
2018-04-29 11:20:12 -07:00
if (error && error.code === 403) return callback(new DomainsError(DomainsError.ACCESS_DENIED, error.message));
if (error) return callback(new DomainsError(DomainsError.EXTERNAL_ERROR, error));
2017-09-14 23:11:17 -07:00
if (records.length === 0) return callback(null, [ ]);
2017-09-14 23:11:17 -07:00
return callback(null, records[0].data);
});
2017-09-05 22:23:24 +02:00
});
}
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');
getZoneByName(getDnsCredentials(dnsConfig), zoneName, function (error, zone) {
if (error) return callback(error);
var domain = (subdomain ? subdomain + '.' : '') + zoneName + '.';
2017-09-05 22:23:24 +02:00
2017-09-14 23:11:17 -07:00
zone.getRecords({ type: type, name: domain }, function(error, oldRecords) {
2018-04-29 11:20:12 -07:00
if (error && error.code === 403) return callback(new DomainsError(DomainsError.ACCESS_DENIED, error.message));
2017-09-12 16:29:07 +02:00
if (error) {
debug('del->zone.getRecords', error);
2018-04-29 11:20:12 -07:00
return callback(new DomainsError(DomainsError.EXTERNAL_ERROR, error.message));
2017-09-12 16:29:07 +02:00
}
2017-09-14 23:11:17 -07:00
zone.deleteRecords(oldRecords, function (error, change) {
2018-04-29 11:20:12 -07:00
if (error && error.code === 403) return callback(new DomainsError(DomainsError.ACCESS_DENIED, error.message));
if (error && error.code === 412) return callback(new DomainsError(DomainsError.STILL_BUSY, error.message));
if (error) {
debug('del->zone.createChange', error);
2018-04-29 11:20:12 -07:00
return callback(new DomainsError(DomainsError.EXTERNAL_ERROR, error.message));
}
callback(null, change.id);
});
2017-09-05 22:23:24 +02:00
});
});
}
function verifyDnsConfig(dnsConfig, fqdn, zoneName, ip, callback) {
assert.strictEqual(typeof dnsConfig, 'object');
assert.strictEqual(typeof fqdn, 'string');
assert.strictEqual(typeof zoneName, 'string');
assert.strictEqual(typeof ip, 'string');
assert.strictEqual(typeof callback, 'function');
2018-06-17 21:44:08 -07:00
if (typeof dnsConfig.projectId !== 'string') return callback(new DomainsError(DomainsError.BAD_FIELD, 'projectId must be a string'));
if (!dnsConfig.credentials || typeof dnsConfig.credentials !== 'object') return callback(new DomainsError(DomainsError.BAD_FIELD, 'credentials must be an object'));
if (typeof dnsConfig.credentials.client_email !== 'string') return callback(new DomainsError(DomainsError.BAD_FIELD, 'credentials.client_email must be a string'));
if (typeof dnsConfig.credentials.private_key !== 'string') return callback(new DomainsError(DomainsError.BAD_FIELD, 'credentials.private_key must be a string'));
2017-09-05 22:23:24 +02:00
var credentials = getDnsCredentials(dnsConfig);
if (process.env.BOX_ENV === 'test') return callback(null, credentials); // this shouldn't be here
2018-02-08 14:39:35 -08:00
dns.resolve(zoneName, 'NS', { timeout: 5000 }, function (error, nameservers) {
2018-04-29 11:20:12 -07:00
if (error && error.code === 'ENOTFOUND') return callback(new DomainsError(DomainsError.BAD_FIELD, 'Unable to resolve nameservers for this domain'));
2018-05-10 10:05:35 -07:00
if (error || !nameservers) return callback(new DomainsError(DomainsError.BAD_FIELD, error ? error.message : 'Unable to get nameservers'));
2017-09-05 22:23:24 +02:00
2017-09-12 16:29:07 +02:00
getZoneByName(credentials, zoneName, function (error, zone) {
2017-09-05 22:23:24 +02:00
if (error) return callback(error);
2017-09-14 18:12:07 -07:00
var definedNS = zone.metadata.nameServers.sort().map(function(r) { return r.replace(/\.$/, ''); });
2018-05-10 10:05:35 -07:00
if (!_.isEqual(definedNS, nameservers.sort())) {
debug('verifyDnsConfig: %j and %j do not match', nameservers, definedNS);
2018-04-29 11:20:12 -07:00
return callback(new DomainsError(DomainsError.BAD_FIELD, 'Domain nameservers are not set to Google Cloud DNS'));
2017-09-05 22:23:24 +02:00
}
const testSubdomain = 'cloudrontestdns';
2017-09-05 22:23:24 +02:00
upsert(credentials, zoneName, testSubdomain, 'A', [ ip ], function (error, changeId) {
2017-09-05 22:23:24 +02:00
if (error) return callback(error);
debug('verifyDnsConfig: Test A record added with change id %s', changeId);
2017-09-05 22:23:24 +02:00
del(dnsConfig, zoneName, testSubdomain, 'A', [ ip ], function (error) {
if (error) return callback(error);
debug('verifyDnsConfig: Test A record removed again');
callback(null, credentials);
});
2017-09-05 22:23:24 +02:00
});
});
});
}