258 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
		
		
			
		
	
	
			258 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
|  | // Copyright 2012 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 net = require('net'), | ||
|  |     util = require('util'), | ||
|  |     EventEmitter = require('events').EventEmitter, | ||
|  |     Packet = require('./packet'), | ||
|  |     consts = require('native-dns-packet').consts, | ||
|  |     UDPSocket = require('./utils').UDPSocket, | ||
|  |     TCPSocket = require('./utils').TCPSocket; | ||
|  | 
 | ||
|  | var debug = function() { | ||
|  |   //var args = Array.prototype.slice.call(arguments);
 | ||
|  |   //console.log.apply(this, ['pending', Date.now().toString()].concat(args));
 | ||
|  | }; | ||
|  | 
 | ||
|  | var SocketQueue = function(socket, server) { | ||
|  |   this._active = {}; | ||
|  |   this._active_count = 0; | ||
|  |   this._pending = []; | ||
|  | 
 | ||
|  |   debug('created', server); | ||
|  | 
 | ||
|  |   this._server = server; | ||
|  | 
 | ||
|  |   this._socket = socket; | ||
|  |   this._socket.on('ready', this._onlisten.bind(this)); | ||
|  |   this._socket.on('message', this._onmessage.bind(this)); | ||
|  |   this._socket.on('close', this._onclose.bind(this)); | ||
|  |   this._socket.bind(server); | ||
|  | 
 | ||
|  |   this._refd = true; | ||
|  | }; | ||
|  | util.inherits(SocketQueue, EventEmitter); | ||
|  | 
 | ||
|  | SocketQueue.prototype.send = function(request) { | ||
|  |   debug('added', request.question); | ||
|  |   this._pending.push(request); | ||
|  |   this._fill(); | ||
|  | }; | ||
|  | 
 | ||
|  | SocketQueue.prototype.remove = function(request) { | ||
|  |   var req = this._active[request.id]; | ||
|  |   var idx = this._pending.indexOf(request); | ||
|  | 
 | ||
|  |   if (req) { | ||
|  |     delete this._active[request.id]; | ||
|  |     this._active_count -= 1; | ||
|  |     this._fill(); | ||
|  |   } | ||
|  | 
 | ||
|  |   if (idx > -1) | ||
|  |     this._pending.splice(idx, 1); | ||
|  | 
 | ||
|  |   this._unref(); | ||
|  | }; | ||
|  | 
 | ||
|  | SocketQueue.prototype.close = function() { | ||
|  |   debug('closing', this._server); | ||
|  |   this._socket.close(); | ||
|  |   this._socket = undefined; | ||
|  |   this.emit('close'); | ||
|  | }; | ||
|  | 
 | ||
|  | SocketQueue.prototype._fill = function() { | ||
|  |   debug('pre fill, active:', this._active_count, 'pending:', | ||
|  |         this._pending.length); | ||
|  | 
 | ||
|  |   while (this._listening && this._pending.length && | ||
|  |          this._active_count < 100) { | ||
|  |     this._dequeue(); | ||
|  |   } | ||
|  | 
 | ||
|  |   debug('post fill, active:', this._active_count, 'pending:', | ||
|  |         this._pending.length); | ||
|  | }; | ||
|  | 
 | ||
|  | var random_integer = function() { | ||
|  |   return Math.floor(Math.random() * 50000 + 1); | ||
|  | }; | ||
|  | 
 | ||
|  | SocketQueue.prototype._dequeue = function() { | ||
|  |   var req = this._pending.pop(); | ||
|  |   var id, packet, dnssocket; | ||
|  | 
 | ||
|  |   if (req) { | ||
|  |     id = random_integer(); | ||
|  | 
 | ||
|  |     while (this._active[id]) | ||
|  |       id = random_integer(); | ||
|  | 
 | ||
|  |     debug('sending', req.question, id); | ||
|  | 
 | ||
|  |     req.id = id; | ||
|  |     this._active[id] = req; | ||
|  |     this._active_count += 1; | ||
|  | 
 | ||
|  |     try { | ||
|  |       packet = new Packet(this._socket.remote(req.server)); | ||
|  |       packet.header.id = id; | ||
|  |       packet.header.rd = 1; | ||
|  | 
 | ||
|  |       if (req.try_edns) { | ||
|  |         packet.edns_version = 0; | ||
|  |         //TODO when we support dnssec
 | ||
|  |         //packet.do = 1
 | ||
|  |       } | ||
|  | 
 | ||
|  |       packet.question.push(req.question); | ||
|  |       packet.send(); | ||
|  | 
 | ||
|  |       this._ref(); | ||
|  |     } catch (e) { | ||
|  |       req.error(e); | ||
|  |     } | ||
|  |   } | ||
|  | }; | ||
|  | 
 | ||
|  | SocketQueue.prototype._onmessage = function(msg, remote) { | ||
|  |   var req, packet; | ||
|  | 
 | ||
|  |   debug('got a message', this._server); | ||
|  | 
 | ||
|  |   try { | ||
|  |     packet = Packet.parse(msg, remote); | ||
|  |     req = this._active[packet.header.id]; | ||
|  |     debug('associated message', packet.header.id); | ||
|  |   } catch (e) { | ||
|  |     debug('error parsing packet', e); | ||
|  |   } | ||
|  | 
 | ||
|  |   if (req) { | ||
|  |     delete this._active[packet.header.id]; | ||
|  |     this._active_count -= 1; | ||
|  |     req.handle(null, packet); | ||
|  |     this._fill(); | ||
|  |   } | ||
|  | 
 | ||
|  |   this._unref(); | ||
|  | }; | ||
|  | 
 | ||
|  | SocketQueue.prototype._unref = function() { | ||
|  |   var self = this; | ||
|  |   this._refd = false; | ||
|  | 
 | ||
|  |   if (this._active_count <= 0) { | ||
|  |     if (this._socket.unref) { | ||
|  |       debug('unrefd socket'); | ||
|  |       this._socket.unref(); | ||
|  |     } else if (!this._timer) { | ||
|  |       this._timer = setTimeout(function() { | ||
|  |         self.close(); | ||
|  |       }, 300); | ||
|  |     } | ||
|  |   } | ||
|  | }; | ||
|  | 
 | ||
|  | SocketQueue.prototype._ref = function() { | ||
|  |   this._refd = true; | ||
|  |   if (this._socket.ref) { | ||
|  |     debug('refd socket'); | ||
|  |     this._socket.ref(); | ||
|  |   } else if (this._timer) { | ||
|  |     clearTimeout(this._timer); | ||
|  |     this._timer = null; | ||
|  |   } | ||
|  | }; | ||
|  | 
 | ||
|  | SocketQueue.prototype._onlisten = function() { | ||
|  |   this._unref(); | ||
|  |   this._listening = true; | ||
|  |   this._fill(); | ||
|  | }; | ||
|  | 
 | ||
|  | SocketQueue.prototype._onclose = function() { | ||
|  |   var req, err, self = this; | ||
|  | 
 | ||
|  |   debug('socket closed', this); | ||
|  | 
 | ||
|  |   this._listening = false; | ||
|  | 
 | ||
|  |   err = new Error('getHostByName ' + consts.TIMEOUT); | ||
|  |   err.errno = consts.TIMEOUT; | ||
|  | 
 | ||
|  |   while (this._pending.length) { | ||
|  |     req = this._pending.pop(); | ||
|  |     req.error(err); | ||
|  |   } | ||
|  | 
 | ||
|  |   Object.keys(this._active).forEach(function(key) { | ||
|  |     var req = self._active[key]; | ||
|  |     req.error(err); | ||
|  |     delete self._active[key]; | ||
|  |     self._active_count -= 1; | ||
|  |   }); | ||
|  | }; | ||
|  | 
 | ||
|  | var serverHash = function(server) { | ||
|  |   if (server.type === 'tcp') | ||
|  |     return server.address + ':' + server.port; | ||
|  |   else | ||
|  |     return 'udp' + net.isIP(server.address); | ||
|  | }; | ||
|  | 
 | ||
|  | var _sockets = {}; | ||
|  | 
 | ||
|  | exports.send = function(request) { | ||
|  |   var hash = serverHash(request.server); | ||
|  |   var socket = _sockets[hash]; | ||
|  | 
 | ||
|  |   if (!socket) { | ||
|  |     switch (hash) { | ||
|  |       case 'udp4': | ||
|  |       case 'udp6': | ||
|  |         socket = new SocketQueue(new UDPSocket(), hash); | ||
|  |         break; | ||
|  |       default: | ||
|  |         socket = new SocketQueue(new TCPSocket(), request.server); | ||
|  |         break; | ||
|  |     } | ||
|  | 
 | ||
|  |     socket.on('close', function() { | ||
|  |       delete _sockets[hash]; | ||
|  |     }); | ||
|  | 
 | ||
|  |     _sockets[hash] = socket; | ||
|  |   } | ||
|  | 
 | ||
|  |   socket.send(request); | ||
|  | }; | ||
|  | 
 | ||
|  | exports.remove = function(request) { | ||
|  |   var hash = serverHash(request.server); | ||
|  |   var socket = _sockets[hash]; | ||
|  |   if (socket) { | ||
|  |     socket.remove(request); | ||
|  |   } | ||
|  | }; |