Fast, lightweight, easy-to-extend, easy-to-test, pure JavaScript (ES5.1) implementation for DNS / mDNS.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 

219 lines
7.2 KiB

#!/usr/bin/env node
// node bin/debug.js ../dig.js/blar.DOEsNteXiST.AJ.OnEal.DAplIE.me.any.0.bin
// node bin/debugd.js ../dig.js/blar.DOEsNteXiST.AJ.OnEal.DAplIE.me.any.0.bin
// dig @localhost -p 65053 ANY DOEsNteXiST.AJ.OnEal.DAplIE.me
// dd if=../dig.js/blar.DOEsNteXiST.AJ.OnEal.DAplIE.me.any.0.bin of=blar.DOEsNteXiST.AJ.OnEal.DAplIE.me.any.0.trunc.bin bs=1 count=78
(function () {
'use strict';
var input = process.argv[2];
var fs = require('fs');
var nb = fs.readFileSync(input, null);
var ab = nb.buffer.slice(nb.byteOffset, nb.byteOffset + nb.byteLength);
var dv = new DataView(ab);
//
// read header
//
console.log('');
console.log('//////////////////');
console.log('// HEADER //');
console.log('//////////////////');
console.log('');
console.log('');
console.log('byte index 0x00');
console.log('');
console.log('id (int)', dv.getUint16(0));
var flags = dv.getUint16(2, false);
console.log('header 0x', flags.toString(16));
var flagsstr = ''
+ 'qr ' + ((flags & 0x8000) >> 15) + ', '
+ 'opcode ' + ((flags & 0x7800) >> 11) + ', '
+ 'aa ' + ((flags & 0x400) >> 10) + ', '
+ 'tc ' + ((flags & 0x200) >> 9) + ', '
+ 'rd ' + ((flags & 0x100) >> 8) + ', '
+ 'ra ' + ((flags & 0x80) >> 7) + ', '
+ 'res1 ' + ((flags & 0x40) >> 6) + ', '
+ 'res2 ' + ((flags & 0x20) >> 5) + ', '
+ 'res3 ' + ((flags & 0x10) >> 4) + ', '
+ 'rcode ' + ((flags & 0xF));
console.log('(flags: ' + flagsstr);
var qdcount = dv.getUint16(4, false); // query count
var ancount = dv.getUint16(6, false); // answer count
var nscount = dv.getUint16(8, false); // authority count
var arcount = dv.getUint16(10, false); // additional count
var count;
console.log('qdcount', qdcount);
console.log('ancount', ancount);
console.log('nscount', nscount);
console.log('arcount', arcount);
function unpackQuestionLabels(opts) {
var dv = opts.dv;
//var origTotal = opts.total;
var total = opts.total;
var len = opts.len;
var qnames = [];
var labelLen;
while (true) {
if (total >= len) {
opts.trunc = true;
console.warn('');
console.warn('[WARNING] The label was truncated by byte length of message or rdata.');
console.warn('[WARNING] Depending on the Resource Record type, that may be a parse error.');
console.warn('');
break;
}
labelLen = dv.getUint8(total, false); // additional count
total += 1;
if (!labelLen) {
break;
}
if (labelLen >= 0xc0) {
// (11 000000 & whatever) signifies pointer
// (00 111111 & whatever) bitmask for potentially large pointer
// (00 000001 11111111) largest realistic pointer value (512 byte message size)
var pointer = ((labelLen & 0x3f) << 8) | dv.getUint8(total, false);
console.log('Found a pointer to'
+ ' 0x' + pointer.toString(16) + ' (' + pointer + ')'
+ ' at byte index'
+ ' 0x' + leftPad(total.toString(16)) + ' (' + total + ')'
//+ ' which back-references:'
);
/*
console.log('-4', dv.getUint8(pointer - 4, false));
console.log('-3', dv.getUint8(pointer - 3, false));
console.log('-2', dv.getUint8(pointer - 2, false));
console.log('-1', dv.getUint8(pointer - 1, false));
console.log(' 0', dv.getUint8(pointer + 0, false));
console.log('+1', dv.getUint8(pointer + 1, false));
console.log('+2', dv.getUint8(pointer + 2, false));
console.log('bytes near the pointer itself (not what it points to) are:');
console.log('-2', dv.getUint8(total - 2, false));
console.log('-1', dv.getUint8(total - 1, false));
console.log(' 0', dv.getUint8(total + 0, false));
console.log('+1', dv.getUint8(total + 1, false));
console.log('+2', dv.getUint8(total + 2, false));
console.log('+3', dv.getUint8(total + 3, false));
console.log('+4', dv.getUint8(total + 4, false));
*/
//total += 1;
opts.cp = true;
qnames.push(unpackQuestionLabels({ dv: dv, total: pointer, len: len }).qname);
total += 1;
break;
} else {
var i;
var label = '';
console.log('label len', labelLen);
for (i = 0; i < labelLen; i += 1) {
//console.log('total:', total, i);
label += String.fromCharCode(dv.getUint8(total + i, false));
}
total += label.length;
console.log('label:', label);
qnames.push(label);
}
}
return {
total: total
, qname: qnames.join('.')
};
}
function unpackQuestion(dv, total, len) {
var result = unpackQuestionLabels({ dv: dv, total: total, len: len });
// leading length and (potentially) trailing null
console.log('QNAME (len ' + (result.total - total) + '):', result.qname, result.cp ? '(pointer)' : '');
console.log('QTYPE (len 2):', dv.getUint16(result.total, false));
result.total += 2;
console.log('QCLASS (len 2):', dv.getUint16(result.total, false));
result.total += 2;
return result;
}
var q = { total: 12 };
function mapChar(ch) { return JSON.stringify(String.fromCharCode(ch)) + '(' + ch + ')'; }
function leftPad(ch) { ch = ch.toString(); while (ch.length < 2) { ch = '0' + ch; } return ch; }
console.log('');
console.log('//////////////////');
console.log('// QUESTION //');
console.log('//////////////////');
console.log('');
for (count = 0; count < qdcount; count += 1) {
console.log('');
console.log('Question ' + (count + 1) + ' of ' + qdcount + ' (byte index: 0x' + leftPad(q.total.toString(16)) + ' ' + q.total + ')');
q = unpackQuestion(dv, q.total, ab.byteLength);
console.log('');
}
function unpackResourceRecord(dv, q, ab) {
var _q = unpackQuestion(dv, q.total, ab.byteLength);
q.total = _q.total;
console.log('ttl (len 4):', dv.getUint32(q.total, false));
q.total += 4;
q.rdlength = dv.getUint16(q.total, false);
console.log('rdlen (len 2):', q.rdlength);
q.total += 2;
console.log('rrdata (len ' + q.rdlength + '):');
console.log([].slice.call(new Uint8Array(ab), q.total, q.total + q.rdlength).map(mapChar).join(' '));
q.total += q.rdlength;
console.log('');
}
console.log('');
console.log('//////////////////');
console.log('// ANSWER //');
console.log('//////////////////');
console.log('');
for (count = 0; count < ancount; count += 1) {
console.log('');
console.log('Answer', count + 1, 'of', ancount + ' (byte index: 0x' + q.total.toString(16) + ' ' + q.total + ')');
unpackResourceRecord(dv, q, ab);
console.log('');
}
console.log('');
console.log('//////////////////');
console.log('// AUTHORITY //');
console.log('//////////////////');
console.log('');
for (count = 0; count < nscount; count += 1) {
console.log('');
console.log('Authority', count + 1, 'of', nscount + ' (byte index: 0x' + q.total.toString(16) + ' ' + q.total + ')');
unpackResourceRecord(dv, q, ab);
console.log('');
}
console.log('');
console.log('//////////////////');
console.log('// ADDITIONAL //');
console.log('//////////////////');
console.log('');
for (count = 0; count < arcount; count += 1) {
console.log('');
console.log('Additional', count + 1, 'of', arcount + ' (byte index: 0x' + q.total.toString(16) + ' ' + q.total + ')');
unpackResourceRecord(dv, q, ab);
console.log('');
}
console.log('');
console.log('total bytes', dv.byteLength);
console.log('total bytes read', q.total);
console.log('');
}());