2017-04-17 12:55:09 -04:00
|
|
|
class ArrayBufferStream {
|
|
|
|
/**
|
|
|
|
* ArrayBufferStream wraps the built-in javascript ArrayBuffer, adding the ability to access
|
|
|
|
* data in it like a stream, tracking its position.
|
|
|
|
* You can request to read a value from the front of the array, and it will keep track of the position
|
|
|
|
* within the byte array, so that successive reads are consecutive.
|
|
|
|
* The available types to read include:
|
|
|
|
* Uint8, Uint8String, Int16, Uint16, Int32, Uint32
|
|
|
|
* @param {ArrayBuffer} arrayBuffer - array to use as a stream
|
2018-10-22 15:59:26 -04:00
|
|
|
* @param {number} start - the start position in the raw buffer. position
|
|
|
|
* will be relative to the start value.
|
|
|
|
* @param {number} end - the end position in the raw buffer. length and
|
|
|
|
* bytes available will be relative to the end value.
|
|
|
|
* @param {ArrayBufferStream} parent - if passed reuses the parent's
|
|
|
|
* internal objects
|
2017-04-17 12:55:09 -04:00
|
|
|
* @constructor
|
|
|
|
*/
|
2018-10-22 15:59:26 -04:00
|
|
|
constructor (
|
|
|
|
arrayBuffer, start = 0, end = arrayBuffer.byteLength,
|
|
|
|
{
|
|
|
|
_uint8View = new Uint8Array(arrayBuffer)
|
|
|
|
} = {}
|
|
|
|
) {
|
|
|
|
/**
|
|
|
|
* Raw data buffer for stream to read.
|
|
|
|
* @type {ArrayBufferStream}
|
|
|
|
*/
|
2017-04-17 12:55:09 -04:00
|
|
|
this.arrayBuffer = arrayBuffer;
|
2018-10-22 15:59:26 -04:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Start position in arrayBuffer. Read values are relative to the start
|
|
|
|
* in the arrayBuffer.
|
|
|
|
* @type {number}
|
|
|
|
*/
|
|
|
|
this.start = start;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* End position in arrayBuffer. Length and bytes available are relative
|
|
|
|
* to the start, end, and _position in the arrayBuffer;
|
|
|
|
* @type {number};
|
|
|
|
*/
|
|
|
|
this.end = end;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Cached Uint8Array view of the arrayBuffer. Heavily used for reading
|
|
|
|
* Uint8 values and Strings from the stream.
|
|
|
|
* @type {Uint8Array}
|
|
|
|
*/
|
|
|
|
this._uint8View = _uint8View;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Raw position in the arrayBuffer relative to the beginning of the
|
|
|
|
* arrayBuffer.
|
|
|
|
* @type {number}
|
|
|
|
*/
|
|
|
|
this._position = start;
|
2017-04-17 12:55:09 -04:00
|
|
|
}
|
2016-11-16 17:06:34 -05:00
|
|
|
|
2017-04-17 12:55:09 -04:00
|
|
|
/**
|
|
|
|
* Return a new ArrayBufferStream that is a slice of the existing one
|
|
|
|
* @param {number} length - the number of bytes of extract
|
|
|
|
* @return {ArrayBufferStream} the extracted stream
|
|
|
|
*/
|
|
|
|
extract (length) {
|
2018-10-22 15:59:26 -04:00
|
|
|
return new ArrayBufferStream(this.arrayBuffer, this._position, this._position + length, this);
|
2017-04-17 12:55:09 -04:00
|
|
|
}
|
2016-11-16 17:06:34 -05:00
|
|
|
|
2017-04-17 12:55:09 -04:00
|
|
|
/**
|
|
|
|
* @return {number} the length of the stream in bytes
|
|
|
|
*/
|
|
|
|
getLength () {
|
2018-10-22 15:59:26 -04:00
|
|
|
return this.end - this.start;
|
2017-04-17 12:55:09 -04:00
|
|
|
}
|
2016-11-16 17:06:34 -05:00
|
|
|
|
2017-04-17 12:55:09 -04:00
|
|
|
/**
|
|
|
|
* @return {number} the number of bytes available after the current position in the stream
|
|
|
|
*/
|
|
|
|
getBytesAvailable () {
|
2018-10-22 15:59:26 -04:00
|
|
|
return this.end - this._position;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Position relative to the start value in the arrayBuffer of this
|
|
|
|
* ArrayBufferStream.
|
|
|
|
* @type {number}
|
|
|
|
*/
|
|
|
|
get position () {
|
|
|
|
return this._position - this.start;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set the position to read from in the arrayBuffer.
|
|
|
|
* @type {number}
|
|
|
|
* @param {number} value - new value to set position to
|
|
|
|
*/
|
|
|
|
set position (value) {
|
|
|
|
this._position = value + this.start;
|
2017-04-17 12:55:09 -04:00
|
|
|
}
|
2016-11-16 17:06:34 -05:00
|
|
|
|
2017-04-17 12:55:09 -04:00
|
|
|
/**
|
|
|
|
* Read an unsigned 8 bit integer from the stream
|
|
|
|
* @return {number} the next 8 bit integer in the stream
|
|
|
|
*/
|
|
|
|
readUint8 () {
|
2018-10-22 15:59:26 -04:00
|
|
|
const val = this._uint8View[this._position];
|
|
|
|
this._position += 1;
|
2017-04-17 12:55:09 -04:00
|
|
|
return val;
|
|
|
|
}
|
2016-11-16 17:06:34 -05:00
|
|
|
|
2017-04-17 12:55:09 -04:00
|
|
|
/**
|
|
|
|
* Read a sequence of bytes of the given length and convert to a string.
|
|
|
|
* This is a convenience method for use with short strings.
|
|
|
|
* @param {number} length - the number of bytes to convert
|
|
|
|
* @return {string} a String made by concatenating the chars in the input
|
|
|
|
*/
|
|
|
|
readUint8String (length) {
|
2018-10-22 15:59:26 -04:00
|
|
|
const arr = this._uint8View;
|
2017-04-17 12:55:09 -04:00
|
|
|
let str = '';
|
2018-10-22 15:59:26 -04:00
|
|
|
const end = this._position + length;
|
|
|
|
for (let i = this._position; i < end; i++) {
|
2017-04-17 12:55:09 -04:00
|
|
|
str += String.fromCharCode(arr[i]);
|
|
|
|
}
|
2018-10-22 15:59:26 -04:00
|
|
|
this._position += length;
|
2017-04-17 12:55:09 -04:00
|
|
|
return str;
|
2016-11-16 17:06:34 -05:00
|
|
|
}
|
|
|
|
|
2017-04-17 12:55:09 -04:00
|
|
|
/**
|
|
|
|
* Read a 16 bit integer from the stream
|
|
|
|
* @return {number} the next 16 bit integer in the stream
|
|
|
|
*/
|
|
|
|
readInt16 () {
|
2018-10-22 15:59:26 -04:00
|
|
|
const val = new Int16Array(this.arrayBuffer, this._position, 1)[0];
|
|
|
|
this._position += 2; // one 16 bit int is 2 bytes
|
2017-04-17 12:55:09 -04:00
|
|
|
return val;
|
|
|
|
}
|
2016-11-16 17:06:34 -05:00
|
|
|
|
2017-04-17 12:55:09 -04:00
|
|
|
/**
|
|
|
|
* Read an unsigned 16 bit integer from the stream
|
|
|
|
* @return {number} the next unsigned 16 bit integer in the stream
|
|
|
|
*/
|
|
|
|
readUint16 () {
|
2018-10-22 15:59:26 -04:00
|
|
|
const val = new Uint16Array(this.arrayBuffer, this._position, 1)[0];
|
|
|
|
this._position += 2; // one 16 bit int is 2 bytes
|
2017-04-17 12:55:09 -04:00
|
|
|
return val;
|
|
|
|
}
|
2016-11-16 17:06:34 -05:00
|
|
|
|
2017-04-17 12:55:09 -04:00
|
|
|
/**
|
|
|
|
* Read a 32 bit integer from the stream
|
|
|
|
* @return {number} the next 32 bit integer in the stream
|
|
|
|
*/
|
|
|
|
readInt32 () {
|
2019-01-08 12:11:53 -05:00
|
|
|
let val;
|
|
|
|
if (this._position % 4 === 0) {
|
|
|
|
val = new Int32Array(this.arrayBuffer, this._position, 1)[0];
|
|
|
|
} else {
|
|
|
|
// Cannot read Int32 directly out because offset is not multiple of 4
|
|
|
|
// Need to slice out the values first
|
|
|
|
val = new Int32Array(
|
|
|
|
this.arrayBuffer.slice(this._position, this._position + 4)
|
|
|
|
)[0];
|
|
|
|
}
|
2018-10-22 15:59:26 -04:00
|
|
|
this._position += 4; // one 32 bit int is 4 bytes
|
2017-04-17 12:55:09 -04:00
|
|
|
return val;
|
|
|
|
}
|
2016-11-16 17:06:34 -05:00
|
|
|
|
2017-04-17 12:55:09 -04:00
|
|
|
/**
|
|
|
|
* Read an unsigned 32 bit integer from the stream
|
|
|
|
* @return {number} the next unsigned 32 bit integer in the stream
|
|
|
|
*/
|
|
|
|
readUint32 () {
|
2018-10-22 15:59:26 -04:00
|
|
|
const val = new Uint32Array(this.arrayBuffer, this._position, 1)[0];
|
|
|
|
this._position += 4; // one 32 bit int is 4 bytes
|
2017-04-17 12:55:09 -04:00
|
|
|
return val;
|
|
|
|
}
|
|
|
|
}
|
2016-11-16 17:06:34 -05:00
|
|
|
|
|
|
|
module.exports = ArrayBufferStream;
|