698 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
		
		
			
		
	
	
			698 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
|  | // Copyright 2011 Timothy J Fontaine <tjfontaine@gmail.com>
 | ||
|  | //
 | ||
|  | // 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'; | ||
|  | 
 | ||
|  | var ipaddr = require('ipaddr.js'), | ||
|  |     net = require('net'), | ||
|  |     util = require('util'), | ||
|  |     EventEmitter = require('events').EventEmitter, | ||
|  |     PendingRequests = require('./pending'), | ||
|  |     Packet = require('./packet'), | ||
|  |     consts = require('native-dns-packet').consts, | ||
|  |     utils = require('./utils'), | ||
|  |     platform = require('./platform'); | ||
|  | 
 | ||
|  | var A = consts.NAME_TO_QTYPE.A, | ||
|  |     AAAA = consts.NAME_TO_QTYPE.AAAA, | ||
|  |     MX = consts.NAME_TO_QTYPE.MX, | ||
|  |     TXT = consts.NAME_TO_QTYPE.TXT, | ||
|  |     NS = consts.NAME_TO_QTYPE.NS, | ||
|  |     CNAME = consts.NAME_TO_QTYPE.CNAME, | ||
|  |     SRV = consts.NAME_TO_QTYPE.SRV, | ||
|  |     PTR = consts.NAME_TO_QTYPE.PTR, | ||
|  |     TLSA = consts.NAME_TO_QTYPE.TLSA; | ||
|  | 
 | ||
|  | var debug = function() {}; | ||
|  | 
 | ||
|  | if (process.env.NODE_DEBUG && process.env.NODE_DEBUG.match(/dns/)) { | ||
|  | debug = function debug() { | ||
|  |   var args = Array.prototype.slice.call(arguments); | ||
|  |   console.error.apply(this, ['client', Date.now().toString()].concat(args)); | ||
|  | }; | ||
|  | } | ||
|  | 
 | ||
|  | var Request = exports.Request = function(opts) { | ||
|  |   if (!(this instanceof Request)) return new Request(opts); | ||
|  | 
 | ||
|  |   this.question = opts.question; | ||
|  |   this.server = opts.server; | ||
|  | 
 | ||
|  |   if (typeof(this.server) === 'string' || this.server instanceof String) | ||
|  |     this.server = { address: this.server, port: 53, type: 'udp'}; | ||
|  | 
 | ||
|  |   if (!this.server || !this.server.address || !net.isIP(this.server.address)) | ||
|  |     throw new Error('Server object must be supplied with at least address'); | ||
|  | 
 | ||
|  |   if (!this.server.type || ['udp', 'tcp'].indexOf(this.server.type) === -1) | ||
|  |     this.server.type = 'udp'; | ||
|  | 
 | ||
|  |   if (!this.server.port) | ||
|  |     this.server.port = 53; | ||
|  | 
 | ||
|  |   this.timeout = opts.timeout || 4 * 1000; | ||
|  |   this.try_edns = opts.try_edns || false; | ||
|  | 
 | ||
|  |   this.fired = false; | ||
|  |   this.id = undefined; | ||
|  | 
 | ||
|  |   if (opts.cache || opts.cache === false) { | ||
|  |     this.cache = opts.cache; | ||
|  |   } else { | ||
|  |     this.cache = platform.cache; | ||
|  |   } | ||
|  |   debug('request created', this.question); | ||
|  | }; | ||
|  | util.inherits(Request, EventEmitter); | ||
|  | 
 | ||
|  | Request.prototype.handle = function(err, answer, cached) { | ||
|  |   if (!this.fired) { | ||
|  |     debug('request handled', this.id, this.question); | ||
|  | 
 | ||
|  |     if (!cached && this.cache && this.cache.store && answer) { | ||
|  |       this.cache.store(answer); | ||
|  |     } | ||
|  | 
 | ||
|  |     this.emit('message', err, answer); | ||
|  |     this.done(); | ||
|  |   } | ||
|  | }; | ||
|  | 
 | ||
|  | Request.prototype.done = function() { | ||
|  |   debug('request finished', this.id, this.question); | ||
|  |   this.fired = true; | ||
|  |   clearTimeout(this.timer_); | ||
|  |   PendingRequests.remove(this); | ||
|  |   this.emit('end'); | ||
|  |   this.id = undefined; | ||
|  | }; | ||
|  | 
 | ||
|  | Request.prototype.handleTimeout = function() { | ||
|  |   if (!this.fired) { | ||
|  |     debug('request timedout', this.id, this.question); | ||
|  |     this.emit('timeout'); | ||
|  |     this.done(); | ||
|  |   } | ||
|  | }; | ||
|  | 
 | ||
|  | Request.prototype.error = function(err) { | ||
|  |   if (!this.fired) { | ||
|  |     debug('request error', err, this.id, this.question); | ||
|  |     this.emit('error', err); | ||
|  |     this.done(); | ||
|  |   } | ||
|  | }; | ||
|  | 
 | ||
|  | Request.prototype.send = function() { | ||
|  |   debug('request starting', this.question); | ||
|  |   var self = this; | ||
|  | 
 | ||
|  |   if (this.cache && this.cache.lookup) { | ||
|  |     this.cache.lookup(this.question, function(results) { | ||
|  |       var packet; | ||
|  | 
 | ||
|  |       if (!results) { | ||
|  |         self._send(); | ||
|  |       } else { | ||
|  |         packet = new Packet(); | ||
|  |         packet.answer = results.slice(); | ||
|  |         self.handle(null, packet, true); | ||
|  |       } | ||
|  |     }); | ||
|  |   } else { | ||
|  |     this._send(); | ||
|  |   } | ||
|  | }; | ||
|  | 
 | ||
|  | Request.prototype._send = function() { | ||
|  |   debug('request not in cache', this.question); | ||
|  |   var self = this; | ||
|  | 
 | ||
|  |   this.timer_ = setTimeout(function() { | ||
|  |     self.handleTimeout(); | ||
|  |   }, this.timeout); | ||
|  | 
 | ||
|  |   PendingRequests.send(self); | ||
|  | }; | ||
|  | 
 | ||
|  | Request.prototype.cancel = function() { | ||
|  |   debug('request cancelled', this.id, this.question); | ||
|  |   this.emit('cancelled'); | ||
|  |   this.done(); | ||
|  | }; | ||
|  | 
 | ||
|  | var _queue = []; | ||
|  | 
 | ||
|  | var sendQueued = function() { | ||
|  |   debug('platform ready sending queued requests'); | ||
|  |   _queue.forEach(function(request) { | ||
|  |     request.start(); | ||
|  |   }); | ||
|  |   _queue = []; | ||
|  | }; | ||
|  | 
 | ||
|  | platform.on('ready', function() { | ||
|  |   sendQueued(); | ||
|  | }); | ||
|  | 
 | ||
|  | if (platform.ready) { | ||
|  |   sendQueued(); | ||
|  | } | ||
|  | 
 | ||
|  | var Resolve = function Resolve(opts, cb) { | ||
|  |   if (!(this instanceof Resolve)) return new Resolve(opts, cb); | ||
|  | 
 | ||
|  |   this.opts = util._extend({ | ||
|  |     retryOnTruncate: true, | ||
|  |   }, opts); | ||
|  | 
 | ||
|  |   this._domain = opts.domain; | ||
|  |   this._rrtype = opts.rrtype; | ||
|  | 
 | ||
|  |   this._buildQuestion(this._domain); | ||
|  | 
 | ||
|  |   this._started = false; | ||
|  |   this._current_server = undefined; | ||
|  | 
 | ||
|  |   this._server_list = []; | ||
|  | 
 | ||
|  |   if (opts.remote) { | ||
|  |     this._server_list.push({ | ||
|  |       address: opts.remote, | ||
|  |       port: 53, | ||
|  |       type: 'tcp', | ||
|  |     }); | ||
|  |     this._server_list.push({ | ||
|  |       address: opts.remote, | ||
|  |       port: 53, | ||
|  |       type: 'udp', | ||
|  |     }); | ||
|  |   } | ||
|  | 
 | ||
|  |   this._request = undefined; | ||
|  |   this._type = 'getHostByName'; | ||
|  |   this._cb = cb; | ||
|  | 
 | ||
|  |   if (!platform.ready) { | ||
|  |     _queue.push(this); | ||
|  |   } else { | ||
|  |     this.start(); | ||
|  |   } | ||
|  | }; | ||
|  | util.inherits(Resolve, EventEmitter); | ||
|  | 
 | ||
|  | Resolve.prototype.cancel = function() { | ||
|  |   if (this._request) { | ||
|  |     this._request.cancel(); | ||
|  |   } | ||
|  | }; | ||
|  | 
 | ||
|  | Resolve.prototype._buildQuestion = function(name) { | ||
|  |   debug('building question', name); | ||
|  |   this.question = { | ||
|  |     type: this._rrtype, | ||
|  |     class: consts.NAME_TO_QCLASS.IN, | ||
|  |     name: name | ||
|  |   }; | ||
|  | }; | ||
|  | exports.Resolve = Resolve; | ||
|  | 
 | ||
|  | Resolve.prototype._emit = function(err, answer) { | ||
|  |   debug('resolve end', this._domain); | ||
|  |   var self = this; | ||
|  |   process.nextTick(function() { | ||
|  |     if (err) { | ||
|  |       err.syscall = self._type; | ||
|  |     } | ||
|  |     self._cb(err, answer); | ||
|  |   }); | ||
|  | }; | ||
|  | 
 | ||
|  | Resolve.prototype._fillServers = function() { | ||
|  |   debug('resolve filling servers', this._domain); | ||
|  |   var tries = 0, s, t, u, slist; | ||
|  | 
 | ||
|  |   slist = platform.name_servers; | ||
|  | 
 | ||
|  |   while (this._server_list.length < platform.attempts) { | ||
|  |     s = slist[tries % slist.length]; | ||
|  | 
 | ||
|  |     u = { | ||
|  |       address: s.address, | ||
|  |       port: s.port, | ||
|  |       type: 'udp' | ||
|  |     }; | ||
|  | 
 | ||
|  |     t = { | ||
|  |       address: s.address, | ||
|  |       port: s.port, | ||
|  |       type: 'tcp' | ||
|  |     }; | ||
|  | 
 | ||
|  |     this._server_list.push(u); | ||
|  |     this._server_list.push(t); | ||
|  | 
 | ||
|  |     tries += 1; | ||
|  |   } | ||
|  | 
 | ||
|  |   this._server_list.reverse(); | ||
|  | }; | ||
|  | 
 | ||
|  | Resolve.prototype._popServer = function() { | ||
|  |   debug('resolve pop server', this._current_server, this._domain); | ||
|  |   this._server_list.splice(0, 1, this._current_server); | ||
|  | }; | ||
|  | 
 | ||
|  | Resolve.prototype._preStart = function() { | ||
|  |   if (!this._started) { | ||
|  |     this._started = new Date().getTime(); | ||
|  |     this.try_edns = platform.edns; | ||
|  | 
 | ||
|  |     if (!this._server_list.length) | ||
|  |       this._fillServers(); | ||
|  |   } | ||
|  | }; | ||
|  | 
 | ||
|  | Resolve.prototype._shouldContinue = function() { | ||
|  |   debug('resolve should continue', this._server_list.length, this._domain); | ||
|  |   return this._server_list.length; | ||
|  | }; | ||
|  | 
 | ||
|  | Resolve.prototype._nextQuestion = function() { | ||
|  |   debug('resolve next question', this._domain); | ||
|  | }; | ||
|  | 
 | ||
|  | Resolve.prototype.start = function() { | ||
|  |   if (!this._started) { | ||
|  |     this._preStart(); | ||
|  |   } | ||
|  | 
 | ||
|  |   if (this._server_list.length === 0) { | ||
|  |     debug('resolve no more servers', this._domain); | ||
|  |     this.handleTimeout(); | ||
|  |   } else { | ||
|  |     this._current_server = this._server_list.pop(); | ||
|  |     debug('resolve start', this._current_server, this._domain); | ||
|  | 
 | ||
|  |     this._request = Request({ | ||
|  |       question: this.question, | ||
|  |       server: this._current_server, | ||
|  |       timeout: platform.timeout, | ||
|  |       try_edns: this.try_edns | ||
|  |     }); | ||
|  | 
 | ||
|  |     this._request.on('timeout', this._handleTimeout.bind(this)); | ||
|  |     this._request.on('message', this._handle.bind(this)); | ||
|  |     this._request.on('error', this._handle.bind(this)); | ||
|  | 
 | ||
|  |     this._request.send(); | ||
|  |   } | ||
|  | }; | ||
|  | 
 | ||
|  | var NOERROR = consts.NAME_TO_RCODE.NOERROR, | ||
|  |     SERVFAIL = consts.NAME_TO_RCODE.SERVFAIL, | ||
|  |     NOTFOUND = consts.NAME_TO_RCODE.NOTFOUND, | ||
|  |     FORMERR = consts.NAME_TO_RCODE.FORMERR; | ||
|  | 
 | ||
|  | Resolve.prototype._handle = function(err, answer) { | ||
|  |   var rcode, errno; | ||
|  | 
 | ||
|  |   if (answer) { | ||
|  |     rcode = answer.header.rcode; | ||
|  |   } | ||
|  | 
 | ||
|  |   debug('resolve handle', rcode, this._domain); | ||
|  | 
 | ||
|  |   switch (rcode) { | ||
|  |     case NOERROR: | ||
|  |       // answer trucated retry with tcp
 | ||
|  |       //console.log(answer);
 | ||
|  |       if (answer.header.tc && | ||
|  |           this.opts.retryOnTruncate && | ||
|  |           this._shouldContinue()) { | ||
|  |         debug('truncated', this._domain, answer); | ||
|  |         this.emit('truncated', err, answer); | ||
|  |          | ||
|  |         // remove udp servers
 | ||
|  |         this._server_list = this._server_list.filter(function(server) { | ||
|  |           return server.type === 'tcp'; | ||
|  |         }); | ||
|  |         answer = undefined; | ||
|  |       } | ||
|  |       break; | ||
|  |     case SERVFAIL: | ||
|  |       if (this._shouldContinue()) { | ||
|  |         this._nextQuestion(); | ||
|  |         //this._popServer();
 | ||
|  |       } else { | ||
|  |         errno = consts.SERVFAIL; | ||
|  |       } | ||
|  |       answer = undefined; | ||
|  |       break; | ||
|  |     case NOTFOUND: | ||
|  |       if (this._shouldContinue()) { | ||
|  |         this._nextQuestion(); | ||
|  |       } else { | ||
|  |         errno = consts.NOTFOUND; | ||
|  |       } | ||
|  |       answer = undefined; | ||
|  |       break; | ||
|  |     case FORMERR: | ||
|  |       if (this.try_edns) { | ||
|  |         this.try_edns = false; | ||
|  |         //this._popServer();
 | ||
|  |       } else { | ||
|  |         errno = consts.FORMERR; | ||
|  |       } | ||
|  |       answer = undefined; | ||
|  |       break; | ||
|  |     default: | ||
|  |       if (!err) { | ||
|  |         errno = consts.RCODE_TO_NAME[rcode]; | ||
|  |         answer = undefined; | ||
|  |       } else { | ||
|  |         errno = consts.NOTFOUND; | ||
|  |       } | ||
|  |       break; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (errno || answer) { | ||
|  |     if (errno) { | ||
|  |       err = new Error(this._type + ' ' + errno); | ||
|  |       err.errno = err.code = errno; | ||
|  |     } | ||
|  |     this._emit(err, answer); | ||
|  |   } else { | ||
|  |     this.start(); | ||
|  |   } | ||
|  | }; | ||
|  | 
 | ||
|  | Resolve.prototype._handleTimeout = function() { | ||
|  |   var err; | ||
|  | 
 | ||
|  |   if (this._server_list.length === 0) { | ||
|  |     debug('resolve timeout no more servers', this._domain); | ||
|  |     err = new Error(this._type + ' ' + consts.TIMEOUT); | ||
|  |     err.errno = consts.TIMEOUT; | ||
|  |     this._emit(err, undefined); | ||
|  |   } else { | ||
|  |     debug('resolve timeout continue', this._domain); | ||
|  |     this.start(); | ||
|  |   } | ||
|  | }; | ||
|  | 
 | ||
|  | var resolve = function(domain, rrtype, ip, callback) { | ||
|  |   var res; | ||
|  | 
 | ||
|  |   if (!callback) { | ||
|  |     callback = ip; | ||
|  |     ip = undefined; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (!callback) { | ||
|  |     callback = rrtype; | ||
|  |     rrtype = undefined; | ||
|  |   } | ||
|  | 
 | ||
|  |   rrtype = consts.NAME_TO_QTYPE[rrtype || 'A']; | ||
|  | 
 | ||
|  |   if (rrtype === PTR) { | ||
|  |     return reverse(domain, callback); | ||
|  |   } | ||
|  | 
 | ||
|  |   var opts = { | ||
|  |     domain: domain, | ||
|  |     rrtype: rrtype, | ||
|  |     remote: ip, | ||
|  |   }; | ||
|  | 
 | ||
|  |   res = new Resolve(opts); | ||
|  | 
 | ||
|  |   res._cb = function(err, response) { | ||
|  |     var ret = [], i, a; | ||
|  | 
 | ||
|  |     if (err) { | ||
|  |       callback(err, response); | ||
|  |       return; | ||
|  |     } | ||
|  | 
 | ||
|  |     for (i = 0; i < response.answer.length; i++) { | ||
|  |       a = response.answer[i]; | ||
|  |       if (a.type === rrtype) { | ||
|  |         switch (rrtype) { | ||
|  |           case A: | ||
|  |           case AAAA: | ||
|  |             ret.push(a.address); | ||
|  |             break; | ||
|  |           case consts.NAME_TO_QTYPE.MX: | ||
|  |             ret.push({ | ||
|  |               priority: a.priority, | ||
|  |               exchange: a.exchange | ||
|  |             }); | ||
|  |             break; | ||
|  |           case TXT: | ||
|  |           case NS: | ||
|  |           case CNAME: | ||
|  |           case PTR: | ||
|  |             ret.push(a.data); | ||
|  |             break; | ||
|  |           case SRV: | ||
|  |             ret.push({ | ||
|  |               priority: a.priority, | ||
|  |               weight: a.weight, | ||
|  |               port: a.port, | ||
|  |               name: a.target | ||
|  |             }); | ||
|  |             break; | ||
|  |           default: | ||
|  |             ret.push(a); | ||
|  |             break; | ||
|  |         } | ||
|  |       } | ||
|  |     } | ||
|  | 
 | ||
|  |     if (ret.length === 0) { | ||
|  |       ret = undefined; | ||
|  |     } | ||
|  | 
 | ||
|  |     callback(err, ret); | ||
|  |   }; | ||
|  | 
 | ||
|  |   return res; | ||
|  | }; | ||
|  | exports.resolve = resolve; | ||
|  | 
 | ||
|  | var resolve4 = function(domain, callback) { | ||
|  |   return resolve(domain, 'A', function(err, results) { | ||
|  |     callback(err, results); | ||
|  |   }); | ||
|  | }; | ||
|  | exports.resolve4 = resolve4; | ||
|  | 
 | ||
|  | var resolve6 = function(domain, callback) { | ||
|  |   return resolve(domain, 'AAAA', function(err, results) { | ||
|  |     callback(err, results); | ||
|  |   }); | ||
|  | }; | ||
|  | exports.resolve6 = resolve6; | ||
|  | 
 | ||
|  | var resolveMx = function(domain, callback) { | ||
|  |   return resolve(domain, 'MX', function(err, results) { | ||
|  |     callback(err, results); | ||
|  |   }); | ||
|  | }; | ||
|  | exports.resolveMx = resolveMx; | ||
|  | 
 | ||
|  | var resolveTxt = function(domain, callback) { | ||
|  |   return resolve(domain, 'TXT', function(err, results) { | ||
|  |     callback(err, results); | ||
|  |   }); | ||
|  | }; | ||
|  | exports.resolveTxt = resolveTxt; | ||
|  | 
 | ||
|  | var resolveSrv = function(domain, callback) { | ||
|  |   return resolve(domain, 'SRV', function(err, results) { | ||
|  |     callback(err, results); | ||
|  |   }); | ||
|  | }; | ||
|  | exports.resolveSrv = resolveSrv; | ||
|  | 
 | ||
|  | var resolveNs = function(domain, callback) { | ||
|  |   return resolve(domain, 'NS', function(err, results) { | ||
|  |     callback(err, results); | ||
|  |   }); | ||
|  | }; | ||
|  | exports.resolveNs = resolveNs; | ||
|  | 
 | ||
|  | var resolveCname = function(domain, callback) { | ||
|  |   return resolve(domain, 'CNAME', function(err, results) { | ||
|  |     callback(err, results); | ||
|  |   }); | ||
|  | }; | ||
|  | exports.resolveCname = resolveCname; | ||
|  | 
 | ||
|  | var resolveTlsa = function(domain, callback) { | ||
|  |   return resolve(domain, 'TLSA', function(err, results) { | ||
|  |     callback(err, results); | ||
|  |   }); | ||
|  | }; | ||
|  | exports.resolveTlsa = resolveTlsa; | ||
|  | 
 | ||
|  | var reverse = function(ip, callback) { | ||
|  |   var error, opts, res; | ||
|  | 
 | ||
|  |   if (!net.isIP(ip)) { | ||
|  |     error = new Error('getHostByAddr ENOTIMP'); | ||
|  |     error.errno = error.code = 'ENOTIMP'; | ||
|  |     throw error; | ||
|  |   } | ||
|  | 
 | ||
|  |   opts = { | ||
|  |     domain: utils.reverseIP(ip), | ||
|  |     rrtype: PTR | ||
|  |   }; | ||
|  | 
 | ||
|  |   res = new Lookup(opts); | ||
|  | 
 | ||
|  |   res._cb = function(err, response) { | ||
|  |     var results = []; | ||
|  | 
 | ||
|  |     if (response) { | ||
|  |       response.answer.forEach(function(a) { | ||
|  |         if (a.type === PTR) { | ||
|  |           results.push(a.data); | ||
|  |         } | ||
|  |       }); | ||
|  |     } | ||
|  | 
 | ||
|  |     if (results.length === 0) { | ||
|  |       results = undefined; | ||
|  |     } | ||
|  | 
 | ||
|  |     callback(err, results); | ||
|  |   }; | ||
|  | 
 | ||
|  |   return res; | ||
|  | }; | ||
|  | exports.reverse = reverse; | ||
|  | 
 | ||
|  | var Lookup = function(opts) { | ||
|  |   Resolve.call(this, opts); | ||
|  |   this._type = 'getaddrinfo'; | ||
|  | }; | ||
|  | util.inherits(Lookup, Resolve); | ||
|  | 
 | ||
|  | Lookup.prototype.start = function() { | ||
|  |   var self = this; | ||
|  | 
 | ||
|  |   if (!this._started) { | ||
|  |     this._search_path = platform.search_path.slice(0); | ||
|  |     this._preStart(); | ||
|  |   } | ||
|  | 
 | ||
|  |   platform.hosts.lookup(this.question, function(results) { | ||
|  |     var packet; | ||
|  |     if (results && results.length) { | ||
|  |       debug('Lookup in hosts', results); | ||
|  |       packet = new Packet(); | ||
|  |       packet.answer = results.slice(); | ||
|  |       self._emit(null, packet); | ||
|  |     } else { | ||
|  |       debug('Lookup not in hosts'); | ||
|  |       Resolve.prototype.start.call(self); | ||
|  |     } | ||
|  |   }); | ||
|  | }; | ||
|  | 
 | ||
|  | Lookup.prototype._shouldContinue = function() { | ||
|  |   debug('Lookup should continue', this._server_list.length, | ||
|  |         this._search_path.length); | ||
|  |   return this._server_list.length && this._search_path.length; | ||
|  | }; | ||
|  | 
 | ||
|  | Lookup.prototype._nextQuestion = function() { | ||
|  |   debug('Lookup next question'); | ||
|  |   this._buildQuestion([this._domain, this._search_path.pop()].join('.')); | ||
|  | }; | ||
|  | 
 | ||
|  | var lookup = function(domain, family, callback) { | ||
|  |   var rrtype, revip, res; | ||
|  | 
 | ||
|  |   if (!callback) { | ||
|  |     callback = family; | ||
|  |     family = undefined; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (!family) { | ||
|  |     family = 4; | ||
|  |   } | ||
|  | 
 | ||
|  |   revip = net.isIP(domain); | ||
|  | 
 | ||
|  |   if (revip === 4 || revip === 6) { | ||
|  |     process.nextTick(function() { | ||
|  |       callback(null, domain, revip); | ||
|  |     }); | ||
|  |     return {}; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (!domain) { | ||
|  |     process.nextTick(function() { | ||
|  |       callback(null, null, family); | ||
|  |     }); | ||
|  |     return {}; | ||
|  |   } | ||
|  | 
 | ||
|  |   rrtype = consts.FAMILY_TO_QTYPE[family]; | ||
|  | 
 | ||
|  |   var opts = { | ||
|  |     domain: domain, | ||
|  |     rrtype: rrtype | ||
|  |   }; | ||
|  | 
 | ||
|  |   res = new Lookup(opts); | ||
|  | 
 | ||
|  |   res._cb = function(err, response) { | ||
|  |     var i, afamily, address, a, all; | ||
|  | 
 | ||
|  |     if (err) { | ||
|  |       callback(err, undefined, undefined); | ||
|  |       return; | ||
|  |     } | ||
|  | 
 | ||
|  |     all = response.answer.concat(response.additional); | ||
|  | 
 | ||
|  |     for (i = 0; i < all.length; i++) { | ||
|  |       a = all[i]; | ||
|  | 
 | ||
|  |       if (a.type === A || a.type === AAAA) { | ||
|  |         afamily = consts.QTYPE_TO_FAMILY[a.type]; | ||
|  |         address = a.address; | ||
|  |         break; | ||
|  |       } | ||
|  |     } | ||
|  | 
 | ||
|  |     callback(err, address, afamily); | ||
|  |   }; | ||
|  | 
 | ||
|  |   return res; | ||
|  | }; | ||
|  | exports.lookup = lookup; |