var types = require('./types') var rcodes = require('./rcodes') var opcodes = require('./opcodes') var ip = require('ip') var Buffer = require('safe-buffer').Buffer var QUERY_FLAG = 0 var RESPONSE_FLAG = 1 << 15 var FLUSH_MASK = 1 << 15 var NOT_FLUSH_MASK = ~FLUSH_MASK var QU_MASK = 1 << 15 var NOT_QU_MASK = ~QU_MASK var name = exports.txt = exports.name = {} name.encode = function (str, buf, offset) { if (!buf) buf = Buffer.alloc(name.encodingLength(str)) if (!offset) offset = 0 var oldOffset = offset // strip leading and trailing . var n = str.replace(/^\.|\.$/gm, '') if (n.length) { var list = n.split('.') for (var i = 0; i < list.length; i++) { var len = buf.write(list[i], offset + 1) buf[offset] = len offset += len + 1 } } buf[offset++] = 0 name.encode.bytes = offset - oldOffset return buf } name.encode.bytes = 0 name.decode = function (buf, offset) { if (!offset) offset = 0 var list = [] var oldOffset = offset var len = buf[offset++] if (len === 0) { name.decode.bytes = 1 return '.' } if (len >= 0xc0) { var res = name.decode(buf, buf.readUInt16BE(offset - 1) - 0xc000) name.decode.bytes = 2 return res } while (len) { if (len >= 0xc0) { list.push(name.decode(buf, buf.readUInt16BE(offset - 1) - 0xc000)) offset++ break } list.push(buf.toString('utf-8', offset, offset + len)) offset += len len = buf[offset++] } name.decode.bytes = offset - oldOffset return list.join('.') } name.decode.bytes = 0 name.encodingLength = function (n) { if (n === '.' || n === '..') return 1 return Buffer.byteLength(n.replace(/^\.|\.$/gm, '')) + 2 } var string = {} string.encode = function (s, buf, offset) { if (!buf) buf = Buffer.alloc(string.encodingLength(s)) if (!offset) offset = 0 var len = buf.write(s, offset + 1) buf[offset] = len string.encode.bytes = len + 1 return buf } string.encode.bytes = 0 string.decode = function (buf, offset) { if (!offset) offset = 0 var len = buf[offset] var s = buf.toString('utf-8', offset + 1, offset + 1 + len) string.decode.bytes = len + 1 return s } string.decode.bytes = 0 string.encodingLength = function (s) { return Buffer.byteLength(s) + 1 } var header = {} header.encode = function (h, buf, offset) { if (!buf) buf = header.encodingLength(h) if (!offset) offset = 0 var flags = (h.flags || 0) & 32767 var type = h.type === 'response' ? RESPONSE_FLAG : QUERY_FLAG buf.writeUInt16BE(h.id || 0, offset) buf.writeUInt16BE(flags | type, offset + 2) buf.writeUInt16BE(h.questions.length, offset + 4) buf.writeUInt16BE(h.answers.length, offset + 6) buf.writeUInt16BE(h.authorities.length, offset + 8) buf.writeUInt16BE(h.additionals.length, offset + 10) return buf } header.encode.bytes = 12 header.decode = function (buf, offset) { if (!offset) offset = 0 if (buf.length < 12) throw new Error('Header must be 12 bytes') var flags = buf.readUInt16BE(offset + 2) return { id: buf.readUInt16BE(offset), type: flags & RESPONSE_FLAG ? 'response' : 'query', flags: flags & 32767, flag_qr: ((flags >> 15) & 0x1) === 1, opcode: opcodes.toString((flags >> 11) & 0xf), flag_auth: ((flags >> 10) & 0x1) === 1, flag_trunc: ((flags >> 9) & 0x1) === 1, flag_rd: ((flags >> 8) & 0x1) === 1, flag_ra: ((flags >> 7) & 0x1) === 1, flag_z: ((flags >> 6) & 0x1) === 1, flag_ad: ((flags >> 5) & 0x1) === 1, flag_cd: ((flags >> 4) & 0x1) === 1, rcode: rcodes.toString(flags & 0xf), questions: new Array(buf.readUInt16BE(offset + 4)), answers: new Array(buf.readUInt16BE(offset + 6)), authorities: new Array(buf.readUInt16BE(offset + 8)), additionals: new Array(buf.readUInt16BE(offset + 10)) } } header.decode.bytes = 12 header.encodingLength = function () { return 12 } var runknown = exports.unknown = {} runknown.encode = function (data, buf, offset) { if (!buf) buf = Buffer.alloc(runknown.encodingLength(data)) if (!offset) offset = 0 buf.writeUInt16BE(data.length, offset) data.copy(buf, offset + 2) runknown.encode.bytes = data.length + 2 return buf } runknown.encode.bytes = 0 runknown.decode = function (buf, offset) { if (!offset) offset = 0 var len = buf.readUInt16BE(offset) var data = buf.slice(offset + 2, offset + 2 + len) runknown.decode.bytes = len + 2 return data } runknown.decode.bytes = 0 runknown.encodingLength = function (data) { return data.length + 2 } var rns = exports.ns = {} rns.encode = function (data, buf, offset) { if (!buf) buf = Buffer.alloc(rns.encodingLength(data)) if (!offset) offset = 0 name.encode(data, buf, offset + 2) buf.writeUInt16BE(name.encode.bytes, offset) rns.encode.bytes = name.encode.bytes + 2 return buf } rns.encode.bytes = 0 rns.decode = function (buf, offset) { if (!offset) offset = 0 var len = buf.readUInt16BE(offset) var dd = name.decode(buf, offset + 2) rns.decode.bytes = len + 2 return dd } rns.decode.bytes = 0 rns.encodingLength = function (data) { return name.encodingLength(data) + 2 } var rsoa = exports.soa = {} rsoa.encode = function (data, buf, offset) { if (!buf) buf = Buffer.alloc(rsoa.encodingLength(data)) if (!offset) offset = 0 var oldOffset = offset offset += 2 name.encode(data.mname, buf, offset) offset += name.encode.bytes name.encode(data.rname, buf, offset) offset += name.encode.bytes buf.writeUInt32BE(data.serial || 0, offset) offset += 4 buf.writeUInt32BE(data.refresh || 0, offset) offset += 4 buf.writeUInt32BE(data.retry || 0, offset) offset += 4 buf.writeUInt32BE(data.expire || 0, offset) offset += 4 buf.writeUInt32BE(data.minimum || 0, offset) offset += 4 buf.writeUInt16BE(offset - oldOffset - 2, oldOffset) rsoa.encode.bytes = offset - oldOffset return buf } rsoa.encode.bytes = 0 rsoa.decode = function (buf, offset) { if (!offset) offset = 0 var oldOffset = offset var data = {} offset += 2 data.mname = name.decode(buf, offset) offset += name.decode.bytes data.rname = name.decode(buf, offset) offset += name.decode.bytes data.serial = buf.readUInt32BE(offset) offset += 4 data.refresh = buf.readUInt32BE(offset) offset += 4 data.retry = buf.readUInt32BE(offset) offset += 4 data.expire = buf.readUInt32BE(offset) offset += 4 data.minimum = buf.readUInt32BE(offset) offset += 4 rsoa.decode.bytes = offset - oldOffset return data } rsoa.decode.bytes = 0 rsoa.encodingLength = function (data) { return 22 + name.encodingLength(data.mname) + name.encodingLength(data.rname) } var rtxt = exports.txt = exports.null = {} var rnull = rtxt rtxt.encode = function (data, buf, offset) { if (!buf) buf = Buffer.alloc(rtxt.encodingLength(data)) if (!offset) offset = 0 if (typeof data === 'string') data = Buffer.from(data) if (!data) data = Buffer.alloc(0) var oldOffset = offset offset += 2 var len = data.length data.copy(buf, offset, 0, len) offset += len buf.writeUInt16BE(offset - oldOffset - 2, oldOffset) rtxt.encode.bytes = offset - oldOffset return buf } rtxt.encode.bytes = 0 rtxt.decode = function (buf, offset) { if (!offset) offset = 0 var oldOffset = offset var len = buf.readUInt16BE(offset) offset += 2 var data = buf.slice(offset, offset + len) offset += len rtxt.decode.bytes = offset - oldOffset return data } rtxt.decode.bytes = 0 rtxt.encodingLength = function (data) { if (!data) return 2 return (Buffer.isBuffer(data) ? data.length : Buffer.byteLength(data)) + 2 } var rhinfo = exports.hinfo = {} rhinfo.encode = function (data, buf, offset) { if (!buf) buf = Buffer.alloc(rhinfo.encodingLength(data)) if (!offset) offset = 0 var oldOffset = offset offset += 2 string.encode(data.cpu, buf, offset) offset += string.encode.bytes string.encode(data.os, buf, offset) offset += string.encode.bytes buf.writeUInt16BE(offset - oldOffset - 2, oldOffset) rhinfo.encode.bytes = offset - oldOffset return buf } rhinfo.encode.bytes = 0 rhinfo.decode = function (buf, offset) { if (!offset) offset = 0 var oldOffset = offset var data = {} offset += 2 data.cpu = string.decode(buf, offset) offset += string.decode.bytes data.os = string.decode(buf, offset) offset += string.decode.bytes rhinfo.decode.bytes = offset - oldOffset return data } rhinfo.decode.bytes = 0 rhinfo.encodingLength = function (data) { return string.encodingLength(data.cpu) + string.encodingLength(data.os) + 2 } var rptr = exports.ptr = {} var rcname = exports.cname = rptr var rdname = exports.dname = rptr rptr.encode = function (data, buf, offset) { if (!buf) buf = Buffer.alloc(rptr.encodingLength(data)) if (!offset) offset = 0 name.encode(data, buf, offset + 2) buf.writeUInt16BE(name.encode.bytes, offset) rptr.encode.bytes = name.encode.bytes + 2 return buf } rptr.encode.bytes = 0 rptr.decode = function (buf, offset) { if (!offset) offset = 0 var data = name.decode(buf, offset + 2) rptr.decode.bytes = name.decode.bytes + 2 return data } rptr.decode.bytes = 0 rptr.encodingLength = function (data) { return name.encodingLength(data) + 2 } var rsrv = exports.srv = {} rsrv.encode = function (data, buf, offset) { if (!buf) buf = Buffer.alloc(rsrv.encodingLength(data)) if (!offset) offset = 0 buf.writeUInt16BE(data.priority || 0, offset + 2) buf.writeUInt16BE(data.weight || 0, offset + 4) buf.writeUInt16BE(data.port || 0, offset + 6) name.encode(data.target, buf, offset + 8) var len = name.encode.bytes + 6 buf.writeUInt16BE(len, offset) rsrv.encode.bytes = len + 2 return buf } rsrv.encode.bytes = 0 rsrv.decode = function (buf, offset) { if (!offset) offset = 0 var len = buf.readUInt16BE(offset) var data = {} data.priority = buf.readUInt16BE(offset + 2) data.weight = buf.readUInt16BE(offset + 4) data.port = buf.readUInt16BE(offset + 6) data.target = name.decode(buf, offset + 8) rsrv.decode.bytes = len + 2 return data } rsrv.decode.bytes = 0 rsrv.encodingLength = function (data) { return 8 + name.encodingLength(data.target) } var rcaa = exports.caa = {} rcaa.ISSUER_CRITICAL = 1 << 7 rcaa.encode = function (data, buf, offset) { var len = rcaa.encodingLength(data) if (!buf) buf = Buffer.alloc(rcaa.encodingLength(data)) if (!offset) offset = 0 if (data.issuerCritical) { data.flags = rcaa.ISSUER_CRITICAL } buf.writeUInt16BE(len - 2, offset) offset += 2 buf.writeUInt8(data.flags || 0, offset) offset += 1 string.encode(data.tag, buf, offset) offset += string.encode.bytes buf.write(data.value, offset) offset += Buffer.byteLength(data.value) rcaa.encode.bytes = len return buf } rcaa.encode.bytes = 0 rcaa.decode = function (buf, offset) { if (!offset) offset = 0 var len = buf.readUInt16BE(offset) offset += 2 var oldOffset = offset var data = {} data.flags = buf.readUInt8(offset) offset += 1 data.tag = string.decode(buf, offset) offset += string.decode.bytes data.value = buf.toString('utf-8', offset, oldOffset + len) data.issuerCritical = !!(data.flags & rcaa.ISSUER_CRITICAL) rcaa.decode.bytes = len + 2 return data } rcaa.decode.bytes = 0 rcaa.encodingLength = function (data) { return string.encodingLength(data.tag) + string.encodingLength(data.value) + 2 } var ra = exports.a = {} ra.encode = function (host, buf, offset) { if (!buf) buf = Buffer.alloc(ra.encodingLength(host)) if (!offset) offset = 0 buf.writeUInt16BE(4, offset) offset += 2 ip.toBuffer(host, buf, offset) ra.encode.bytes = 6 return buf } ra.encode.bytes = 0 ra.decode = function (buf, offset) { if (!offset) offset = 0 offset += 2 var host = ip.toString(buf, offset, 4) ra.decode.bytes = 6 return host } ra.decode.bytes = 0 ra.encodingLength = function () { return 6 } var raaaa = exports.aaaa = {} raaaa.encode = function (host, buf, offset) { if (!buf) buf = Buffer.alloc(raaaa.encodingLength(host)) if (!offset) offset = 0 buf.writeUInt16BE(16, offset) offset += 2 ip.toBuffer(host, buf, offset) raaaa.encode.bytes = 18 return buf } raaaa.encode.bytes = 0 raaaa.decode = function (buf, offset) { if (!offset) offset = 0 offset += 2 var host = ip.toString(buf, offset, 16) raaaa.decode.bytes = 18 return host } raaaa.decode.bytes = 0 raaaa.encodingLength = function () { return 18 } var renc = exports.record = function (type) { switch (type.toUpperCase()) { case 'A': return ra case 'PTR': return rptr case 'CNAME': return rcname case 'DNAME': return rdname case 'TXT': return rtxt case 'NULL': return rnull case 'AAAA': return raaaa case 'SRV': return rsrv case 'HINFO': return rhinfo case 'CAA': return rcaa case 'NS': return rns case 'SOA': return rsoa } return runknown } var answer = exports.answer = {} answer.encode = function (a, buf, offset) { if (!buf) buf = Buffer.alloc(answer.encodingLength(a)) if (!offset) offset = 0 var oldOffset = offset name.encode(a.name, buf, offset) offset += name.encode.bytes buf.writeUInt16BE(types.toType(a.type), offset) var klass = a.class === undefined ? 1 : a.class if (a.flush) klass |= FLUSH_MASK // the 1st bit of the class is the flush bit buf.writeUInt16BE(klass, offset + 2) buf.writeUInt32BE(a.ttl || 0, offset + 4) var enc = renc(a.type) enc.encode(a.data, buf, offset + 8) offset += 8 + enc.encode.bytes answer.encode.bytes = offset - oldOffset return buf } answer.encode.bytes = 0 answer.decode = function (buf, offset) { if (!offset) offset = 0 var a = {} var oldOffset = offset a.name = name.decode(buf, offset) offset += name.decode.bytes a.type = types.toString(buf.readUInt16BE(offset)) a.class = buf.readUInt16BE(offset + 2) a.ttl = buf.readUInt32BE(offset + 4) a.flush = !!(a.class & FLUSH_MASK) if (a.flush) a.class &= NOT_FLUSH_MASK var enc = renc(a.type) a.data = enc.decode(buf, offset + 8) offset += 8 + enc.decode.bytes answer.decode.bytes = offset - oldOffset return a } answer.decode.bytes = 0 answer.encodingLength = function (a) { return name.encodingLength(a.name) + 8 + renc(a.type).encodingLength(a.data) } var question = exports.question = {} question.encode = function (q, buf, offset) { if (!buf) buf = Buffer.alloc(question.encodingLength(q)) if (!offset) offset = 0 var oldOffset = offset name.encode(q.name, buf, offset) offset += name.encode.bytes buf.writeUInt16BE(types.toType(q.type), offset) offset += 2 buf.writeUInt16BE(q.class === undefined ? 1 : q.class, offset) offset += 2 question.encode.bytes = offset - oldOffset return q } question.encode.bytes = 0 question.decode = function (buf, offset) { if (!offset) offset = 0 var oldOffset = offset var q = {} q.name = name.decode(buf, offset) offset += name.decode.bytes q.type = types.toString(buf.readUInt16BE(offset)) offset += 2 q.class = buf.readUInt16BE(offset) offset += 2 var qu = !!(q.class & QU_MASK) if (qu) q.class &= NOT_QU_MASK question.decode.bytes = offset - oldOffset return q } question.decode.bytes = 0 question.encodingLength = function (q) { return name.encodingLength(q.name) + 4 } exports.AUTHORITATIVE_ANSWER = 1 << 10 exports.TRUNCATED_RESPONSE = 1 << 9 exports.RECURSION_DESIRED = 1 << 8 exports.RECURSION_AVAILABLE = 1 << 7 exports.AUTHENTIC_DATA = 1 << 5 exports.CHECKING_DISABLED = 1 << 4 exports.encode = function (result, buf, offset) { var allocing = !buf if (allocing) buf = Buffer.alloc(exports.encodingLength(result)) if (!offset) offset = 0 var oldOffset = offset if (!result.questions) result.questions = [] if (!result.answers) result.answers = [] if (!result.authorities) result.authorities = [] if (!result.additionals) result.additionals = [] header.encode(result, buf, offset) offset += header.encode.bytes offset = encodeList(result.questions, question, buf, offset) offset = encodeList(result.answers, answer, buf, offset) offset = encodeList(result.authorities, answer, buf, offset) offset = encodeList(result.additionals, answer, buf, offset) exports.encode.bytes = offset - oldOffset // just a quick sanity check if (allocing && exports.encode.bytes !== buf.length) { return buf.slice(0, exports.encode.bytes) } return buf } exports.encode.bytes = 0 exports.decode = function (buf, offset) { if (!offset) offset = 0 var oldOffset = offset var result = header.decode(buf, offset) offset += header.decode.bytes offset = decodeList(result.questions, question, buf, offset) offset = decodeList(result.answers, answer, buf, offset) offset = decodeList(result.authorities, answer, buf, offset) offset = decodeList(result.additionals, answer, buf, offset) exports.decode.bytes = offset - oldOffset return result } exports.decode.bytes = 0 exports.encodingLength = function (result) { return header.encodingLength(result) + encodingLengthList(result.questions || [], question) + encodingLengthList(result.answers || [], answer) + encodingLengthList(result.authorities || [], answer) + encodingLengthList(result.additionals || [], answer) } function encodingLengthList (list, enc) { var len = 0 for (var i = 0; i < list.length; i++) len += enc.encodingLength(list[i]) return len } function encodeList (list, enc, buf, offset) { for (var i = 0; i < list.length; i++) { enc.encode(list[i], buf, offset) offset += enc.encode.bytes } return offset } function decodeList (list, enc, buf, offset) { for (var i = 0; i < list.length; i++) { list[i] = enc.decode(buf, offset) offset += enc.decode.bytes } return offset }