332 lines
		
	
	
		
			9.6 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
		
		
			
		
	
	
			332 lines
		
	
	
		
			9.6 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
|  | #!/usr/bin/env node
 | ||
|  | 'use strict'; | ||
|  | 
 | ||
|  | var dnsjs = require('dns-suite'); | ||
|  | var cli = require('cli'); | ||
|  | var hexdump = require('../hexdump'); | ||
|  | var crypto = require('crypto'); | ||
|  | cli.parse({ | ||
|  | //  'b': [ false, 'set source IP address (defaults to 0.0.0.0)', 'string' ]
 | ||
|  |   'debug': [ false, 'more verbose output', 'boolean', false ] | ||
|  | //, 'insecure': [ false, 'turn off RaNDOm cAPS required for securing queries']
 | ||
|  | //, 'ipv4': [ '4', 'use ipv4 exclusively (defaults to false)', 'boolean', false ]
 | ||
|  | //, 'ipv6': [ '6', 'use ipv6 exclusively (defaults to false)', 'boolean', false ]
 | ||
|  | //, 'json': [ false, 'output results as json', 'string' ]
 | ||
|  | //, 'lint': [ false, 'attack (in the metaphorical sense) a nameserver with all sorts of queries to test for correct responses', 'string', false ]
 | ||
|  | , 'mdns': [ false, "Alias for setting defaults to -p 5353 @224.0.0.251 -t PTR -q _services._dns-sd._udp.local and waiting for multiple responses", 'boolean', false ] | ||
|  | , 'output': [ 'o', 'output prefix to use for writing query and response(s) to disk', 'file' ] | ||
|  | , 'port': [ 'p', 'port (defaults to 53 for dns and 5353 for mdns)', 'int' ] | ||
|  | //, 'serve': [ 's', 'path to json file with array of responses to issue for given queries', 'string' ]
 | ||
|  | , 'type': [ 't', 'type (defaults to ANY for dns and PTR for mdns)', 'string' ] | ||
|  | , 'query': [ 'q', 'a superfluous explicit option to set the query as a command line flag' ] | ||
|  | }); | ||
|  | 
 | ||
|  | var fs = require('fs'); | ||
|  | var dgram = require('dgram'); | ||
|  | var commonTypes = [ 'A', 'AAAA', 'CNAME', 'MX', 'NS', 'PTR', 'SOA', 'SRV', 'TXT' ]; | ||
|  | var commonPrinters = { | ||
|  |   'ANY': function (q) { | ||
|  |     console.log(';' + q.name + '.', q.ttl, q.className, q.typeName, q.data || q.rdata || 'unknown record type'); | ||
|  |   } | ||
|  | 
 | ||
|  | , 'A': function (q) { | ||
|  |     console.log(';' + q.name + '.', q.ttl, q.className, q.typeName, q.address); | ||
|  |   } | ||
|  | , 'AAAA': function (q) { | ||
|  |     console.log(';' + q.name + '.', q.ttl, q.className, q.typeName, q.address); | ||
|  |   } | ||
|  | , 'CNAME': function (q) { | ||
|  |     console.log(';' + q.name + '.', q.ttl, q.className, q.typeName, q.data + '.'); | ||
|  |   } | ||
|  | , 'MX': function (q) { | ||
|  |     console.log(';' + q.name + '.', q.ttl, q.className, q.typeName, q.priority + ' ' + q.exchange + '.'); | ||
|  |   } | ||
|  | , 'NS': function (q) { | ||
|  |     console.log(';' + q.name + '.', q.ttl, q.className, q.typeName, q.data); | ||
|  |   } | ||
|  | , 'PTR': function (q) { | ||
|  |     console.log(';' + q.name + '.', q.ttl, q.className, q.typeName, q.data); | ||
|  |   } | ||
|  | /* | ||
|  | , 'SOA': function (q) { | ||
|  |     console.log(';' + q.name + '.', q.ttl, q.className, q.typeName, q.data); | ||
|  |   } | ||
|  | */ | ||
|  | , 'SRV': function (q) { | ||
|  |     console.log(';' + q.name + '.', q.ttl, q.className, q.typeName, q.priority + ' ' + q.weight + ' ' + q.port + ' ' + q.target); | ||
|  |   } | ||
|  | , 'TXT': function (q) { | ||
|  |     console.log(';' + q.name + '.', q.ttl, q.className, q.typeName, '"' + q.data.join('" "') + '"'); | ||
|  |   } | ||
|  | }; | ||
|  | 
 | ||
|  | function writeQuery(opts, query, queryAb) { | ||
|  |   var path = require('path'); | ||
|  |   var binname = query.question[0].name + '.' + query.question[0].typeName.toLowerCase() + '.query.bin'; | ||
|  |   var jsonname = query.question[0].name + '.' + query.question[0].typeName.toLowerCase() + '.query.json'; | ||
|  |   var binpath = opts.output + '.' + binname; | ||
|  |   var jsonpath = opts.output + '.' + jsonname; | ||
|  |   var json = JSON.stringify(query, null, 2); | ||
|  |   if (-1 !== ['.', '/', '\\' ].indexOf(opts.output[opts.output.length -1])) { | ||
|  |     binpath = path.join(opts.output, binname); | ||
|  |     jsonpath = path.join(opts.output, jsonname); | ||
|  |   } | ||
|  | 
 | ||
|  |   fs.writeFile(binpath, Buffer.from(queryAb), null, function () { | ||
|  |     console.log('wrote ' + queryAb.byteLength + ' bytes to ' + binpath); | ||
|  |   }); | ||
|  |   fs.writeFile(jsonpath, json, null, function () { | ||
|  |     console.log('wrote ' + json.length + ' bytes to ' + jsonpath); | ||
|  |   }); | ||
|  | } | ||
|  | 
 | ||
|  | var count = 0; | ||
|  | function writeResponse(opts, query, nb, packet) { | ||
|  |   var path = require('path'); | ||
|  |   var binname = query.question[0].name + '.' + query.question[0].typeName.toLowerCase() + '.' + count + '.bin'; | ||
|  |   var jsonname = query.question[0].name + '.' + query.question[0].typeName.toLowerCase() + '.' + count + '.json'; | ||
|  |   var binpath = opts.output + '.' + binname; | ||
|  |   var jsonpath = opts.output + '.' + jsonname; | ||
|  |   var json = JSON.stringify(packet, null, 2); | ||
|  |   if (-1 !== ['.', '/', '\\' ].indexOf(opts.output[opts.output.length -1])) { | ||
|  |     binpath = path.join(opts.output, binname); | ||
|  |     jsonpath = path.join(opts.output, jsonname); | ||
|  |   } | ||
|  | 
 | ||
|  |   count += 1; | ||
|  | 
 | ||
|  |   fs.writeFile(binpath, nb, null, function () { | ||
|  |     console.log('wrote ' + nb.byteLength + ' bytes to ' + binpath); | ||
|  |   }); | ||
|  |   fs.writeFile(jsonpath, json, null, function () { | ||
|  |     console.log('wrote ' + json.length + ' bytes to ' + jsonpath); | ||
|  |   }); | ||
|  | } | ||
|  | 
 | ||
|  | function request(query, opts) { | ||
|  |   var queryAb = dnsjs.DNSPacket.write(query); | ||
|  | 
 | ||
|  |   if (opts.debug) { | ||
|  |     console.log(''); | ||
|  |     console.log('DNS Question:'); | ||
|  |     console.log(''); | ||
|  |     console.log(query); | ||
|  |     console.log(''); | ||
|  |     console.log(hexdump(queryAb)); | ||
|  |     console.log(''); | ||
|  |     console.log(dnsjs.DNSPacket.parse(queryAb)); | ||
|  |     console.log(''); | ||
|  |   } | ||
|  | 
 | ||
|  |   var handlers = {}; | ||
|  |   var server = dgram.createSocket({ | ||
|  |     type: 'udp4' | ||
|  |   , reuseAddr: true | ||
|  |   }); | ||
|  | 
 | ||
|  |   handlers.onError = function (err) { | ||
|  |     console.error("error:", err.stack); | ||
|  |     server.close(); | ||
|  |   }; | ||
|  |   handlers.onMessage = function (nb) { | ||
|  |     console.log("YOYOYO GOT MESSAGE"); | ||
|  |     var packet = dnsjs.DNSPacket.parse(nb.buffer.slice(nb.byteOffset, nb.byteOffset + nb.byteLength)); | ||
|  | 
 | ||
|  |     if (opts.debug) { | ||
|  |       console.log(''); | ||
|  |       console.log('DNS Request:'); | ||
|  |       console.log(packet); | ||
|  |     } | ||
|  | 
 | ||
|  |     console.log(''); | ||
|  |     console.log('; <<>> dig.js ' + 'v0.0.0' + ' <<>> ' + query.question[0].name); | ||
|  |     console.log(';; Got answer:'); | ||
|  |     console.log(';; ->>HEADER<<-'); | ||
|  |     console.log(JSON.stringify(packet.header)); | ||
|  |     console.log(''); | ||
|  |     console.log(';; QUESTION SECTION:'); | ||
|  |     packet.question.forEach(function (q) { | ||
|  |       console.log(';' + q.name + '.', ' ', q.className, q.typeName); | ||
|  |     }); | ||
|  |     /* | ||
|  |     function print(q) { | ||
|  |       var printer = commonPrinters[q.typeName] || commonPrinters.ANY; | ||
|  |       printer(q); | ||
|  |     } | ||
|  |     if (packet.answer.length) { | ||
|  |       console.log(''); | ||
|  |       console.log(';; ANSWER SECTION:'); | ||
|  |       packet.answer.forEach(print); | ||
|  |     } | ||
|  |     if (packet.authority.length) { | ||
|  |       console.log(''); | ||
|  |       console.log(';; AUTHORITY SECTION:'); | ||
|  |       packet.authority.forEach(print); | ||
|  |     } | ||
|  |     if (packet.additional.length) { | ||
|  |       console.log(''); | ||
|  |       console.log(';; ADDITIONAL SECTION:'); | ||
|  |       packet.additional.forEach(print); | ||
|  |     } | ||
|  |     console.log(''); | ||
|  |     console.log(';; MSG SIZE  rcvd: ' + nb.byteLength); | ||
|  |     console.log(''); | ||
|  |     */ | ||
|  | 
 | ||
|  |     if (opts.output) { | ||
|  |       console.log(''); | ||
|  |       writeQuery(opts, query, queryAb); | ||
|  |       //writeResponse(opts, query, nb, packet);
 | ||
|  |     } | ||
|  |   }; | ||
|  |   handlers.onListening = function () { | ||
|  |     console.log("YOYOYO ON LISTENING"); | ||
|  |     /*jshint validthis:true*/ | ||
|  |     var server = this; | ||
|  |     var nameserver = opts.nameserver; | ||
|  |     var nameservers; | ||
|  |     var index; | ||
|  | 
 | ||
|  |     if (!nameserver) { | ||
|  |       nameservers = require('dns').getServers(); | ||
|  |       index = crypto.randomBytes(2).readUInt16BE(0) % nameservers.length; | ||
|  |       nameserver = nameservers[index]; | ||
|  |       if (opts.debug) { | ||
|  |         console.log(index, nameservers); | ||
|  |       } | ||
|  |     } | ||
|  | 
 | ||
|  |     if (opts.mdns || '224.0.0.251' === opts.nameserver) { | ||
|  |       server.setBroadcast(true); | ||
|  |       server.addMembership(opts.nameserver); | ||
|  |     } | ||
|  | 
 | ||
|  |     if (opts.debug) { | ||
|  |       console.log(''); | ||
|  |       console.log('Bound and Listening:'); | ||
|  |       console.log(server.address()); | ||
|  |     } | ||
|  | 
 | ||
|  |     if (opts.debug) { | ||
|  |       console.log('querying ' + nameserver + ':' + opts.port); | ||
|  |     } | ||
|  |   }; | ||
|  | 
 | ||
|  | 
 | ||
|  |   server.on('error', handlers.onError); | ||
|  |   server.on('message', handlers.onMessage); | ||
|  |   server.on('listening', handlers.onListening); | ||
|  | 
 | ||
|  |   // 53 dns server
 | ||
|  |   // 5353 mdns
 | ||
|  |   console.log("YOYOYO BINDING ON", opts.port); | ||
|  |   server.bind(opts.port); | ||
|  | } | ||
|  | 
 | ||
|  | cli.main(function (args, cli) { | ||
|  |   args.forEach(function (arg) { | ||
|  |     if (-1 !== commonTypes.indexOf(arg.toUpperCase())) { | ||
|  |       if (cli.type) { | ||
|  |         console.error("'type' was specified more than once"); | ||
|  |         process.exit(1); | ||
|  |         return; | ||
|  |       } | ||
|  |       cli.type = cli.t = arg.toUpperCase(); | ||
|  |       return; | ||
|  |     } | ||
|  | 
 | ||
|  |     if (/^\+time=/.test(arg)) { | ||
|  |       if (cli.timeout) { | ||
|  |         console.error("'+time=' was specified more than once"); | ||
|  |         process.exit(1); | ||
|  |         return; | ||
|  |       } | ||
|  |       cli.timeout = Math.round(parseInt(arg.replace(/\+time=/, ''), 10) * 1000); | ||
|  |       return; | ||
|  |     } | ||
|  | 
 | ||
|  |     if (/^@/.test(arg)) { | ||
|  |       if (cli.nameserver) { | ||
|  |         console.error("'@server' was specified more than once"); | ||
|  |         process.exit(1); | ||
|  |         return; | ||
|  |       } | ||
|  |       cli.nameserver = cli.n = arg.substr(1); | ||
|  |       return; | ||
|  |     } | ||
|  | 
 | ||
|  |     if (cli.query) { | ||
|  |       console.error("'query' was specified more than once"); | ||
|  |       process.exit(1); | ||
|  |       return; | ||
|  |     } | ||
|  |     cli.query = cli.q = arg; | ||
|  | 
 | ||
|  |   }); | ||
|  | 
 | ||
|  |   if (cli.mdns) { | ||
|  |     if (!cli.type) { | ||
|  |       cli.type = cli.t = 'PTR'; | ||
|  |     } | ||
|  |     if (!cli.port) { | ||
|  |       cli.port = cli.p = 5353; | ||
|  |     } | ||
|  |     if (!cli.nameserver) { | ||
|  |       cli.nameserver = '224.0.0.251'; | ||
|  |     } | ||
|  |     if (!cli.query) { | ||
|  |       cli.query = '_services._dns-sd._udp.local'; | ||
|  |     } | ||
|  |     if (!cli.timeout) { | ||
|  |       cli.timeout = 3000; | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   if (!cli.type) { | ||
|  |     cli.type = cli.t = 'A'; | ||
|  |   } | ||
|  |   if (!cli.port) { | ||
|  |     cli.port = cli.p = 53; | ||
|  |   } | ||
|  |   if (!cli.class) { | ||
|  |     cli.class = cli.c = 'IN'; | ||
|  |   } | ||
|  |   if (!cli.query) { | ||
|  |     cli.query = 'example.com'; | ||
|  |     /* | ||
|  |     console.error(''); | ||
|  |     console.error('Usage:'); | ||
|  |     console.error('digd.js [@server] [TYPE] [domain]'); | ||
|  |     console.error(''); | ||
|  |     console.error('Example:'); | ||
|  |     console.error('digd.js daplie.com'); | ||
|  |     console.error(''); | ||
|  |     process.exit(1); | ||
|  |     */ | ||
|  |   } | ||
|  | 
 | ||
|  |   var query = { | ||
|  |     header: { | ||
|  |       id: crypto.randomBytes(2).readUInt16BE(0) | ||
|  |     , qr: 0 | ||
|  |     , opcode: 0 | ||
|  |     , aa: 0     // NA
 | ||
|  |     , tc: 0     // NA
 | ||
|  |     , rd: 1 | ||
|  |     , ra: 0     // NA
 | ||
|  |     , rcode: 0  // NA
 | ||
|  |     } | ||
|  |   , question: [ | ||
|  |       { name: cli.query | ||
|  |       , typeName: cli.type | ||
|  |       , className: cli.class | ||
|  |       } | ||
|  |     ] | ||
|  |   }; | ||
|  | 
 | ||
|  |   if (!cli.daemon) { | ||
|  |     request(query, cli); | ||
|  |     return; | ||
|  |   } | ||
|  | }); |