123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771 |
- 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
- }
|