| 
									
										
										
										
											2017-01-21 03:15:57 -07:00
										 |  |  | 'use strict'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var pdns = module.exports; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Order http://www.zytrax.com/books/dns/ch15/
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | pdns.unpackHeader = function (i) { | 
					
						
							|  |  |  |   // i is one element from a Uint16Array (as a 16-bit unsigned integer)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   var header = { | 
					
						
							| 
									
										
										
										
											2017-01-21 03:38:25 -07:00
										 |  |  |     id:     0                   // added here to preserve console.log order
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   , qr:     (i & 0x8000) >> 15  // Query Response (0 is question, 1 is response)
 | 
					
						
							| 
									
										
										
										
											2017-01-21 03:15:57 -07:00
										 |  |  |   , opcode: (i & 0x7800) >> 11  // 0 is question
 | 
					
						
							|  |  |  |   , aa:     (i &  0x400) >> 10  // Authoritative Answer (response-only)
 | 
					
						
							| 
									
										
										
										
											2017-01-21 10:45:45 -07:00
										 |  |  |   , tc:     (i &  0x200) >>  9  // TrunCated - expect another packet with same (?) id
 | 
					
						
							|  |  |  |   , rd:     (i &  0x100) >>  8  // Recursion Desired
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   , ra:     (i &   0x80) >>  7 | 
					
						
							|  |  |  |   , res1:   (i &   0x40) >>  6  // z
 | 
					
						
							|  |  |  |   , res2:   (i &   0x20) >>  5  // ad
 | 
					
						
							|  |  |  |   , res3:   (i &   0x10) >>  4  // cd
 | 
					
						
							|  |  |  |   , rcode:  (i &    0xF)        // Error Code (response-only)
 | 
					
						
							| 
									
										
										
										
											2017-01-21 03:15:57 -07:00
										 |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return header; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-25 14:41:30 -07:00
										 |  |  | pdns.unpackQname = function (ab, ptr, q) { | 
					
						
							|  |  |  |   var ui8 = new Uint8Array(ab); | 
					
						
							|  |  |  |   var total = ptr; | 
					
						
							| 
									
										
										
										
											2017-01-21 03:15:57 -07:00
										 |  |  |   var i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   var len; | 
					
						
							| 
									
										
										
										
											2017-01-25 14:41:30 -07:00
										 |  |  |   var label = []; | 
					
						
							|  |  |  |   q = q || { | 
					
						
							| 
									
										
										
										
											2017-01-21 03:38:25 -07:00
										 |  |  |     name: '' | 
					
						
							| 
									
										
										
										
											2017-01-21 03:15:57 -07:00
										 |  |  |   , type: 0 | 
					
						
							|  |  |  |   , class: 0 | 
					
						
							| 
									
										
										
										
											2017-01-25 14:41:30 -07:00
										 |  |  |   , byteLength: 0 | 
					
						
							|  |  |  |   , labels: [] | 
					
						
							|  |  |  |   , cpcount: 0 | 
					
						
							| 
									
										
										
										
											2017-01-21 03:15:57 -07:00
										 |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   do { | 
					
						
							| 
									
										
										
										
											2017-01-25 14:41:30 -07:00
										 |  |  |     label.length = 0; | 
					
						
							| 
									
										
										
										
											2017-01-21 03:15:57 -07:00
										 |  |  |     len = ui8[total]; | 
					
						
							| 
									
										
										
										
											2017-01-21 11:53:07 -07:00
										 |  |  |     if (0xc0 === len) { | 
					
						
							| 
									
										
										
										
											2017-01-25 14:41:30 -07:00
										 |  |  |       console.log('byteLength', ui8.byteLength); | 
					
						
							|  |  |  |       var cpptr = ui8[total + 1]; | 
					
						
							|  |  |  |       var cplen = ui8[ui8[total + 1]]; | 
					
						
							|  |  |  |       console.log('cp at', total); | 
					
						
							|  |  |  |       console.log('ptr to', cpptr); | 
					
						
							|  |  |  |       console.log('label len', cplen); | 
					
						
							|  |  |  |       console.log('label', Buffer.from(ui8.slice(cpptr + 1, cpptr + 1 + cplen)).toString('ascii')); | 
					
						
							| 
									
										
										
										
											2017-01-21 11:53:07 -07:00
										 |  |  |       throw new Error("discovered a compression pointer at byte " + total | 
					
						
							|  |  |  |         + " pointing to byte " + ui8[total + 1] | 
					
						
							|  |  |  |         + ", but compression pointer support is not yet implemented" | 
					
						
							|  |  |  |       ); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-01-21 03:15:57 -07:00
										 |  |  |     //str.length = 0; // fast empty array
 | 
					
						
							|  |  |  |     if (ui8.byteLength - total < len) { | 
					
						
							|  |  |  |       throw new Error( | 
					
						
							|  |  |  |         "Expected a string of length " + len | 
					
						
							|  |  |  |           + " but packet itself has " + (ui8.byteLength - total) + " bytes remaining" | 
					
						
							|  |  |  |       ); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     for (i = 0; i < len; i += 1) { | 
					
						
							|  |  |  |       total += 1; | 
					
						
							|  |  |  |       // TODO check url-allowable characters
 | 
					
						
							| 
									
										
										
										
											2017-01-25 14:41:30 -07:00
										 |  |  |       label.push(String.fromCharCode(ui8[total])); | 
					
						
							| 
									
										
										
										
											2017-01-21 03:15:57 -07:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-01-25 14:41:30 -07:00
										 |  |  |     if (label.length) { | 
					
						
							|  |  |  |       q.labels.push(label.join('')); | 
					
						
							| 
									
										
										
										
											2017-01-21 03:15:57 -07:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-01-25 14:41:30 -07:00
										 |  |  |     total += 1; | 
					
						
							| 
									
										
										
										
											2017-01-21 03:15:57 -07:00
										 |  |  |     //console.log('total', total, ui8[total], String.fromCharCode(ui8[total]));
 | 
					
						
							|  |  |  |   } while (len); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   //str.pop(); // remove trailing '.'
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-25 14:41:30 -07:00
										 |  |  |   q.name = q.labels.join('.'); | 
					
						
							|  |  |  |   if (0 === q.cpcount) { | 
					
						
							|  |  |  |     q.byteLength = total - ptr; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2017-01-21 03:15:57 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return q; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | pdns.unpack = function (ab) { | 
					
						
							|  |  |  |   if (ab.buffer) { | 
					
						
							|  |  |  |     ab = ab.buffer; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // SANITY Check
 | 
					
						
							|  |  |  |   if (ab.byteLength < 12) { | 
					
						
							|  |  |  |     throw new Error( | 
					
						
							|  |  |  |       "A DNS header has a minimum length of 12 bytes but this packet has only " + ab.byteLength + "bytes." | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // DO: new Uint8Array(arrayBuffer);
 | 
					
						
							|  |  |  |   // DO NOT: Uint8Array.from(arrayBuffer); // WILL NOT WORK
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // DO: new DataView(arrayBuffer).getUint16(7);
 | 
					
						
							|  |  |  |   // DO NOT: arrayBuffer.slice();
 | 
					
						
							|  |  |  |   //
 | 
					
						
							|  |  |  |   // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/ArrayBuffer
 | 
					
						
							|  |  |  |   // https://developer.mozilla.org/en-US/docs/Web/API/ArrayBufferView
 | 
					
						
							|  |  |  |   // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Uint8Array
 | 
					
						
							|  |  |  |   // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/TypedArray/slice
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/DataView
 | 
					
						
							|  |  |  |   var dv = new DataView(ab); | 
					
						
							|  |  |  |   var id = dv.getUint16(0); | 
					
						
							|  |  |  |   var header = pdns.unpackHeader(dv.getUint16(2)); | 
					
						
							|  |  |  |   var qdcount = dv.getUint16(4);  // query count
 | 
					
						
							|  |  |  |   var ancount = dv.getUint16(6);  // answer count
 | 
					
						
							|  |  |  |   var nscount = dv.getUint16(8);  // authority count
 | 
					
						
							|  |  |  |   var arcount = dv.getUint16(10); // additional count
 | 
					
						
							|  |  |  |   var total = 12; | 
					
						
							|  |  |  |   var i; | 
					
						
							| 
									
										
										
										
											2017-01-21 10:45:45 -07:00
										 |  |  |   var rec; | 
					
						
							| 
									
										
										
										
											2017-01-21 03:15:57 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // TODO move to pdns.unpackQuestion to make testable
 | 
					
						
							| 
									
										
										
										
											2017-01-21 10:45:45 -07:00
										 |  |  |   function unpackQuestion(ab, dv, total) { | 
					
						
							|  |  |  |     var ototal = total; | 
					
						
							|  |  |  |     var data = new Uint8Array(ab).slice(total); | 
					
						
							| 
									
										
										
										
											2017-01-25 14:41:30 -07:00
										 |  |  |     var q = pdns.unpackQname(ab, total); | 
					
						
							|  |  |  |     total += q.byteLength; | 
					
						
							| 
									
										
										
										
											2017-01-21 03:15:57 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (ab.byteLength - total < 4) { | 
					
						
							|  |  |  |       // console.error(str.join(''));
 | 
					
						
							|  |  |  |       throw new Error( | 
					
						
							|  |  |  |         "Expected a 2-byte QTYPE and 2-byte QCLASS following " + total + "-byte QNAME" | 
					
						
							|  |  |  |           + " but packet itself has " + (ab.byteLength - total) + " bytes remaining" | 
					
						
							|  |  |  |       ); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     q.type = dv.getUint16(total); | 
					
						
							|  |  |  |     total += 2; | 
					
						
							|  |  |  |     q.class = dv.getUint16(total); | 
					
						
							|  |  |  |     total += 2; | 
					
						
							| 
									
										
										
										
											2017-01-21 15:11:39 -07:00
										 |  |  |     q.byteLength = total - ototal; | 
					
						
							| 
									
										
										
										
											2017-01-21 10:45:45 -07:00
										 |  |  |     return q; | 
					
						
							| 
									
										
										
										
											2017-01-21 03:15:57 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-21 10:45:45 -07:00
										 |  |  |   function unpackAnswer(ab, dv, total) { | 
					
						
							|  |  |  |     var ototal = total; | 
					
						
							|  |  |  |     var data = new Uint8Array(ab).slice(total); | 
					
						
							| 
									
										
										
										
											2017-01-25 14:41:30 -07:00
										 |  |  |     var q = pdns.unpackQname(ab, total); | 
					
						
							|  |  |  |     total += q.byteLength; | 
					
						
							| 
									
										
										
										
											2017-01-21 03:15:57 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (ab.byteLength - total < 10) { | 
					
						
							|  |  |  |       // console.error(str.join(''));
 | 
					
						
							|  |  |  |       throw new Error( | 
					
						
							|  |  |  |         "Expected a 2-byte QTYPE, 2-byte QCLASS, 4-byte TTL, and 2-byte RDLENGTH following " + total + "-byte QNAME" | 
					
						
							|  |  |  |           + " but packet itself has " + (ab.byteLength - total) + " bytes remaining" | 
					
						
							|  |  |  |       ); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     q.type = dv.getUint16(total); | 
					
						
							|  |  |  |     total += 2; | 
					
						
							|  |  |  |     q.class = dv.getUint16(total); | 
					
						
							|  |  |  |     total += 2; | 
					
						
							|  |  |  |     q.ttl = dv.getUint32(total); | 
					
						
							|  |  |  |     total += 4; | 
					
						
							|  |  |  |     q.rdlength = dv.getUint16(total); | 
					
						
							|  |  |  |     total += 2; | 
					
						
							| 
									
										
										
										
											2017-01-21 11:02:31 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // TODO actually parse RDATA
 | 
					
						
							| 
									
										
										
										
											2017-01-21 03:32:50 -07:00
										 |  |  |     console.log(ab.byteLength, ab.byteLength - total, ab.byteLength + -total + q.rdlength); | 
					
						
							|  |  |  |     q.rdata = new Uint8Array(ab).slice(total, total + q.rdlength); | 
					
						
							| 
									
										
										
										
											2017-01-21 11:13:42 -07:00
										 |  |  |     console.log('q.rdata', q.rdata.byteLength, 'bytes:'); | 
					
						
							|  |  |  |     q.rdata = Array.prototype.slice.apply(q.rdata); | 
					
						
							| 
									
										
										
										
											2017-01-21 03:32:50 -07:00
										 |  |  |     console.log(q.rdata); | 
					
						
							| 
									
										
										
										
											2017-01-21 03:15:57 -07:00
										 |  |  |     console.log('total', total); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     total += q.rdlength; | 
					
						
							|  |  |  |     console.log('total', total); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-21 15:11:39 -07:00
										 |  |  |     q.byteLength = total - ototal; | 
					
						
							| 
									
										
										
										
											2017-01-21 10:45:45 -07:00
										 |  |  |     return q; | 
					
						
							| 
									
										
										
										
											2017-01-21 03:15:57 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-21 03:38:25 -07:00
										 |  |  |   header.id = id; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-21 03:15:57 -07:00
										 |  |  |   console.log('qdcount', qdcount); | 
					
						
							|  |  |  |   header.questions = []; | 
					
						
							|  |  |  |   for (i = 0; i < qdcount; i += 1) { | 
					
						
							| 
									
										
										
										
											2017-01-21 10:45:45 -07:00
										 |  |  |     rec = unpackQuestion(ab, dv, total); | 
					
						
							| 
									
										
										
										
											2017-01-21 15:11:39 -07:00
										 |  |  |     total += rec.byteLength; | 
					
						
							| 
									
										
										
										
											2017-01-21 10:45:45 -07:00
										 |  |  |     header.questions.push(rec); | 
					
						
							| 
									
										
										
										
											2017-01-21 03:15:57 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   console.log('ancount', ancount); | 
					
						
							|  |  |  |   header.answers = []; | 
					
						
							|  |  |  |   for (i = 0; i < ancount; i += 1) { | 
					
						
							| 
									
										
										
										
											2017-01-21 10:45:45 -07:00
										 |  |  |     rec = unpackAnswer(ab, dv, total); | 
					
						
							| 
									
										
										
										
											2017-01-21 15:11:39 -07:00
										 |  |  |     total += rec.byteLength; | 
					
						
							| 
									
										
										
										
											2017-01-21 10:45:45 -07:00
										 |  |  |     header.answers.push(rec); | 
					
						
							| 
									
										
										
										
											2017-01-21 03:15:57 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   console.log('nscount', nscount); | 
					
						
							|  |  |  |   header.authority = []; | 
					
						
							|  |  |  |   for (i = 0; i < nscount; i += 1) { | 
					
						
							| 
									
										
										
										
											2017-01-21 11:13:42 -07:00
										 |  |  |     rec = unpackAnswer(ab, dv, total); | 
					
						
							| 
									
										
										
										
											2017-01-21 15:11:39 -07:00
										 |  |  |     total += rec.byteLength; | 
					
						
							| 
									
										
										
										
											2017-01-21 11:13:42 -07:00
										 |  |  |     header.authority.push(rec); | 
					
						
							| 
									
										
										
										
											2017-01-21 03:15:57 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   console.log('arcount', arcount); | 
					
						
							|  |  |  |   header.additional = []; | 
					
						
							|  |  |  |   for (i = 0; i < arcount; i += 1) { | 
					
						
							| 
									
										
										
										
											2017-01-21 11:13:42 -07:00
										 |  |  |     rec = unpackAnswer(ab, dv, total); | 
					
						
							| 
									
										
										
										
											2017-01-21 15:11:39 -07:00
										 |  |  |     total += rec.byteLength; | 
					
						
							| 
									
										
										
										
											2017-01-21 11:13:42 -07:00
										 |  |  |     header.additional.push(rec); | 
					
						
							| 
									
										
										
										
											2017-01-21 03:15:57 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-21 11:02:31 -07:00
										 |  |  |   if (ab.byteLength !== total) { | 
					
						
							|  |  |  |     throw new Error( | 
					
						
							|  |  |  |       "Parsed " + total + " bytes, but packet is " + ab.byteLength + " bytes." | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2017-01-21 15:11:39 -07:00
										 |  |  |   header.byteLength = total; | 
					
						
							| 
									
										
										
										
											2017-01-21 10:45:45 -07:00
										 |  |  |   return header; | 
					
						
							| 
									
										
										
										
											2017-01-21 03:15:57 -07:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2017-01-21 14:33:51 -07:00
										 |  |  | pdns.unpackRdata = require('./dns.rdata.parse.js').DNS_RDATA_PARSE; |