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;
 | |
|   }
 | |
| });
 |