Spaces:
Running
Running
; | |
const { EMPTY_BUFFER } = require('./constants'); | |
/** | |
* Merges an array of buffers into a new buffer. | |
* | |
* @param {Buffer[]} list The array of buffers to concat | |
* @param {Number} totalLength The total length of buffers in the list | |
* @return {Buffer} The resulting buffer | |
* @public | |
*/ | |
function concat(list, totalLength) { | |
if (list.length === 0) return EMPTY_BUFFER; | |
if (list.length === 1) return list[0]; | |
const target = Buffer.allocUnsafe(totalLength); | |
let offset = 0; | |
for (let i = 0; i < list.length; i++) { | |
const buf = list[i]; | |
target.set(buf, offset); | |
offset += buf.length; | |
} | |
return target; | |
} | |
/** | |
* Masks a buffer using the given mask. | |
* | |
* @param {Buffer} source The buffer to mask | |
* @param {Buffer} mask The mask to use | |
* @param {Buffer} output The buffer where to store the result | |
* @param {Number} offset The offset at which to start writing | |
* @param {Number} length The number of bytes to mask. | |
* @public | |
*/ | |
function _mask(source, mask, output, offset, length) { | |
for (let i = 0; i < length; i++) { | |
output[offset + i] = source[i] ^ mask[i & 3]; | |
} | |
} | |
/** | |
* Unmasks a buffer using the given mask. | |
* | |
* @param {Buffer} buffer The buffer to unmask | |
* @param {Buffer} mask The mask to use | |
* @public | |
*/ | |
function _unmask(buffer, mask) { | |
// Required until https://github.com/nodejs/node/issues/9006 is resolved. | |
const length = buffer.length; | |
for (let i = 0; i < length; i++) { | |
buffer[i] ^= mask[i & 3]; | |
} | |
} | |
/** | |
* Converts a buffer to an `ArrayBuffer`. | |
* | |
* @param {Buffer} buf The buffer to convert | |
* @return {ArrayBuffer} Converted buffer | |
* @public | |
*/ | |
function toArrayBuffer(buf) { | |
if (buf.byteLength === buf.buffer.byteLength) { | |
return buf.buffer; | |
} | |
return buf.buffer.slice(buf.byteOffset, buf.byteOffset + buf.byteLength); | |
} | |
/** | |
* Converts `data` to a `Buffer`. | |
* | |
* @param {*} data The data to convert | |
* @return {Buffer} The buffer | |
* @throws {TypeError} | |
* @public | |
*/ | |
function toBuffer(data) { | |
toBuffer.readOnly = true; | |
if (Buffer.isBuffer(data)) return data; | |
let buf; | |
if (data instanceof ArrayBuffer) { | |
buf = Buffer.from(data); | |
} else if (ArrayBuffer.isView(data)) { | |
buf = viewToBuffer(data); | |
} else { | |
buf = Buffer.from(data); | |
toBuffer.readOnly = false; | |
} | |
return buf; | |
} | |
/** | |
* Converts an `ArrayBuffer` view into a buffer. | |
* | |
* @param {(DataView|TypedArray)} view The view to convert | |
* @return {Buffer} Converted view | |
* @private | |
*/ | |
function viewToBuffer(view) { | |
const buf = Buffer.from(view.buffer); | |
if (view.byteLength !== view.buffer.byteLength) { | |
return buf.slice(view.byteOffset, view.byteOffset + view.byteLength); | |
} | |
return buf; | |
} | |
try { | |
const bufferUtil = require('bufferutil'); | |
const bu = bufferUtil.BufferUtil || bufferUtil; | |
module.exports = { | |
concat, | |
mask(source, mask, output, offset, length) { | |
if (length < 48) _mask(source, mask, output, offset, length); | |
else bu.mask(source, mask, output, offset, length); | |
}, | |
toArrayBuffer, | |
toBuffer, | |
unmask(buffer, mask) { | |
if (buffer.length < 32) _unmask(buffer, mask); | |
else bu.unmask(buffer, mask); | |
} | |
}; | |
} catch (e) /* istanbul ignore next */ { | |
module.exports = { | |
concat, | |
mask: _mask, | |
toArrayBuffer, | |
toBuffer, | |
unmask: _unmask | |
}; | |
} | |