'use strict'; module.exports.create = function (opts) { // opts = { filepath }; var engine = { db: null }; function notDeleted(r) { return !r.revokedAt && !r.deletedAt; } var db = require(opts.filepath); var stat = require('fs').statSync(opts.filepath); var crypto = require('crypto'); // // Manual Migration // db.primaryNameservers.forEach(function (ns, i, arr) { if ('string' === typeof ns) { ns = { name: ns }; arr[i] = ns; } if (!ns.id) { ns.id = crypto.randomBytes(16).toString('hex'); } }); db.zones = db.zones || []; if (db.domains) { db.zones = db.zones.concat(db.domains); } db.zones.forEach(function (zone) { if (!zone.name) { zone.name = zone.id; zone.id = null; } if (!zone.id) { zone.id = crypto.randomBytes(16).toString('hex'); } if (!zone.createdAt) { zone.createdAt = stat.mtime.valueOf(); } if (!zone.updatedAt) { zone.updatedAt = stat.mtime.valueOf(); } }); db.records.forEach(function (record) { if (!record.id) { record.id = crypto.randomBytes(16).toString('hex'); } }); require('fs').writeFileSync(opts.filepath, JSON.stringify(db, null, 2)); // // End Migration // db.save = function (cb) { if (db.save._saving) { console.log('make pending'); db.save._pending.push(cb); return; } db.save._saving = true; require('fs').writeFile(opts.filepath, JSON.stringify(db, null, 2), function (err) { console.log('done writing'); var pending = db.save._pending.splice(0); db.save._saving = false; cb(err); if (!pending.length) { return; } db.save(function (err) { console.log('double save'); pending.forEach(function (cb) { cb(err); }); }); }); }; db.save._pending = []; engine.primaryNameservers = db.primaryNameservers; engine.peers = { all: function (cb) { var dns = require('dns'); var count = db.primaryNameservers.length; function gotRecord() { count -= 1; if (!count) { cb(null, db.primaryNameservers); } } function getRecord(ns) { dns.resolve4(ns.name, function (err, addresses) { console.log('ns addresses:'); console.log(addresses); if (err) { console.error(err); gotRecord(); return; } ns.type = 'A'; ns.address = addresses[0]; gotRecord(); }); } db.primaryNameservers.forEach(getRecord); } }; engine.zones = { _immutableKeys: [ 'id', 'name', 'primary', 'serial', 'revokedAt', 'changedAt', 'insertedAt', 'updatedAt', 'deletedAt' ] , _mutableKeys: [ 'admin', 'expiration', 'minimum', 'refresh', 'retry', 'ttl', 'vanity' ] , _dateToSerial: function (date) { // conventionally the format is YYYYMMDDxx, // but since it's an integer and I don't want to keep track of incrementing xx, // epoch in seconds will do return parseInt(Math.round(date/1000).toString().slice(-10), 10); } , _toSoa: function (domain) { var nameservers = domain.vanityNs || engine.primaryNameservers.map(function (n) { return n.name; }); var index = Math.floor(Math.random() * nameservers.length) % nameservers.length; var nameserver = nameservers[index]; return { id: domain.id , name: domain.name , typeName: 'SOA' , className: 'IN' , ttl: domain.ttl || 60 // nameserver -- select an NS at random if they're all in sync , primary: nameserver , name_server: nameserver // admin -- email address or domain for admin , admin: domain.admin || ('admin.' + domain.name) , email_addr: domain.admin || ('admin.' + domain.name) // serial -- the version, for cache-busting of secondary nameservers. suggested format: YYYYMMDDnn , serial: domain.serial || engine.zones._dateToSerial(domain.updatedAt || domain.createdAt || Date.now()) , sn: domain.serial || engine.zones._dateToSerial(domain.updatedAt || domain.createdAt || Date.now()) // refresh -- only used when nameservers following the DNS NOTIFY spec talk , refresh: domain.refresh || 1800 , ref: domain.refresh || 1800 // retry -- only used when nameservers following the DNS NOTIFY spec talk , retry: domain.retry || 600 , ret: domain.retry || 600 // expiration -- how long other nameservers should continue when the primary goes down , expiration: domain.expiration || 2419200 // 4 weeks , ex: domain.expiration || 2419200 // 4 weeks // minimum -- how long to cache a non-existent domain (also the default ttl for BIND) , minimum: domain.minimum || 5 , nx: domain.minimum || 5 }; } , all: function (cb) { process.nextTick(function () { cb(null, db.zones.slice(0).filter(notDeleted)); }); } , get: function (queries, cb) { if (!Array.isArray(queries)) { queries = queries.names.map(function (n) { return { name: n }; }); } var myDomains = db.zones.filter(function (d) { return queries.some(function (q) { return (d.name.toLowerCase() === q.name) && notDeleted(d); }); }); process.nextTick(function () { cb(null, myDomains); }); } , touch: function (zone, cb) { var existing; db.zones.some(function (z) { if (z.id && zone.id === z.id) { existing = z; return true; } if (z.name && zone.name === z.name) { existing = z; return true; } }); if (!existing) { cb(null, null); return; } existing.updatedAt = new Date().valueOf(); // toISOString(); console.log('touch saving...'); db.save(function (err) { cb(err, !err && existing || null); }); } , save: function (zone, cb) { if (zone.id) { console.log('update zone!'); engine.zones.update(zone, cb); } else { engine.zones.create(zone, cb); } } , update: function (zone, cb) { var existing; var dirty; db.zones.some(function (z) { if (z.id === zone.id) { existing = z; return true; } }); if (!existing) { console.log('no existing zone'); cb(new Error("zone for '" + zone.id + "' does not exist"), null); return; } console.log('found existing zone'); console.log(existing); console.log(zone); Object.keys(zone).forEach(function (key) { if (-1 !== engine.zones._immutableKeys.indexOf(key)) { return; } if (existing[key] !== zone[key]) { dirty = true; console.log('existing key', key, existing[key], zone[key]); existing[key] = zone[key]; } }); zone.updatedAt = new Date().valueOf(); // toISOString(); // Math.round(Date.now() / 1000); if (dirty) { zone.changedAt = zone.updatedAt; } console.log('saving...'); db.save(function (err) { cb(err, !err && existing || null); }); } , create: function (zone, cb) { var newZone = { id: crypto.randomBytes(16).toString('hex') }; var existing; var nss = []; zone.name = (zone.name||'').toLowerCase(); db.zones.some(function (z) { if (z.name === zone.name) { existing = z; return true; } }); if (existing) { cb(new Error("tried to create new zone, but '" + existing.name + "' already exists")); return; } newZone.name = zone.name; newZone.createdAt = Date.now(); newZone.updatedAt = newZone.createdAt; Object.keys(zone).forEach(function (key) { //if (-1 !== engine.zones._immutableKeys.indexOf(key)) { return; } if (-1 === engine.zones._mutableKeys.indexOf(key)) { return; } newZone[key] = zone[key]; }); // TODO create NS and A records for normal and vanity nameservers if (zone.vanity) { newZone.vanity = true; } else { newZone.vanity = false; } db.primaryNameservers.forEach(function (ns, i) { var nsx = 'ns' + (i + 1); var nsZone; var ttl = 43200; // 12h // TODO pick a well-reasoned number var now = Date.now(); if (zone.vanity) { nsZone = nsx + '.' + newZone.name; } else { nsZone = ns.name; } // NS example.com ns1.example.com 43200 nss.push({ id: crypto.randomBytes(16).toString('hex') , createdAt: Date.now() , updatedAt: Date.now() , changedAt: Date.now() , zone: newZone.name , soa: true , type: 'NS' , data: nsZone , name: newZone.name , ttl: ttl }); // A ns1.example.com 127.0.0.1 43200 nss.push({ id: crypto.randomBytes(16).toString('hex') , createdAt: now , updatedAt: now , changedAt: now , zone: newZone.name , soa: true , type: ns.type , name: nsZone , address: ns.address , ttl: 43200 // 12h // TODO pick a good number }); }); db.zones.push(newZone); nss.forEach(function (ns) { db.records.push(ns); }); console.log('[zone] [create] saving...'); db.save(function (err) { cb(err, !err && newZone || null); }); } , destroy: function (zoneId, cb) { var zone; var records; var now = Date.now(); db.zones.filter(notDeleted).some(function (z) { if (zoneId === z.id) { zone = z; z.deletedAt = now; return true; } }); if (!zone) { process.nextTick(function () { cb(null, null); }); return; } records = []; db.records.filter(notDeleted).forEach(function (r) { if (zone.name === r.zone) { r.deletedAt = now; records.push(r); } }); console.log('[zone] [destroy] saving...'); db.save(function (err) { zone.records = records; cb(err, !err && zone || null); }); } }; engine.records = { all: function (cb) { process.nextTick(function () { cb(null, db.records.slice(0).filter(notDeleted)); }); } , one: function (id, cb) { var myRecord; db.records.slice(0).some(function (r) { if (id && id === r.id) { if (notDeleted(r)) { myRecord = r; return true; } return false; } }); process.nextTick(function () { cb(null, myRecord); }); } , get: function (query, cb) { var myRecords = db.records.slice(0).filter(function (r) { if ('string' !== typeof r.name) { return false; } // TODO use IN in masterquest (or implement OR) // Only return single-level wildcard? if (query.name === r.name || ('*.' + query.name.split('.').slice(1).join('.')) === r.name) { if (notDeleted(r)) { return true; } } }); process.nextTick(function () { cb(null, myRecords); }); } , save: function (record, cb) { function touchZone(err, r) { if (err) { cb(err); } if (!r) { cb(null, null); } engine.zones.touch({ name: r.zone }, cb); } if (record.id) { console.log('update record!'); engine.records.update(record, touchZone); } else { engine.records.create(record, touchZone); } } , update: function (record, cb) { var existing; var dirty; db.records.some(function (r) { if (r.id === record.id) { existing = r; return true; } }); if (!existing) { console.log('no existing record'); cb(new Error("record for '" + record.id + "' does not exist"), null); return; } console.log('found existing record'); console.log(existing); console.log(record); Object.keys(record).forEach(function (key) { var keys = [ 'name', 'id', 'zone', 'revokedAt', 'changedAt', 'insertedAt', 'updatedAt', 'deletedAt' ]; if (-1 !== keys.indexOf(key)) { return; } if (existing[key] !== record[key]) { dirty = true; console.log(existing[key], record[key]); existing[key] = record[key]; } }); record.updatedAt = new Date().valueOf(); // toISOString(); // Math.round(Date.now() / 1000); if (dirty) { record.changedAt = record.updatedAt; } console.log('saving...'); db.save(function (err) { cb(err, !err && existing || null); }); } , create: function (record, cb) { var obj = { id: crypto.randomBytes(16).toString('hex') }; console.log('found existing record'); console.log(record); //var keys = [ 'name', 'id', 'zone', 'revokedAt', 'changedAt', 'insertedAt', 'updatedAt', 'deletedAt' ]; //var okeys = [ 'name', 'zone', 'admin', 'data', 'expiration', 'minimum', 'serial', 'retry', 'refresh', 'ttl', 'type' ]; // primary var okeys = [ 'name', 'zone', 'type', 'data', 'class', 'ttl', 'address' , 'exchange', 'priority', 'port', 'value', 'tag', 'flag', 'aname' ]; okeys.forEach(function (key) { if ('undefined' !== typeof record[key]) { obj[key] = record[key]; } }); record.updatedAt = new Date().valueOf(); // toISOString(); // Math.round(Date.now() / 1000); //record.changedAt = record.updatedAt; record.insertedAt = record.updatedAt; record.createdAt = record.updatedAt; console.log('saving new...'); db.records.push(record); db.save(function (err) { cb(err, record); }); } , destroy: function (id, cb) { var record; db.records.some(function (r/*, i*/) { if (id === r.id) { record = r; r.deletedAt = Date.now(); //record = db.records.splice(i, 1); return true; } }); process.nextTick(function () { db.save(function (err) { cb(err, record); }); }); } }; return engine; };