154 lines
4.0 KiB
JavaScript
154 lines
4.0 KiB
JavaScript
|
'use strict';
|
||
|
|
||
|
const inherits = require('inherits');
|
||
|
const Reporter = require('../base/reporter').Reporter;
|
||
|
const Buffer = require('safer-buffer').Buffer;
|
||
|
|
||
|
function DecoderBuffer(base, options) {
|
||
|
Reporter.call(this, options);
|
||
|
if (!Buffer.isBuffer(base)) {
|
||
|
this.error('Input not Buffer');
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
this.base = base;
|
||
|
this.offset = 0;
|
||
|
this.length = base.length;
|
||
|
}
|
||
|
inherits(DecoderBuffer, Reporter);
|
||
|
exports.DecoderBuffer = DecoderBuffer;
|
||
|
|
||
|
DecoderBuffer.isDecoderBuffer = function isDecoderBuffer(data) {
|
||
|
if (data instanceof DecoderBuffer) {
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
// Or accept compatible API
|
||
|
const isCompatible = typeof data === 'object' &&
|
||
|
Buffer.isBuffer(data.base) &&
|
||
|
data.constructor.name === 'DecoderBuffer' &&
|
||
|
typeof data.offset === 'number' &&
|
||
|
typeof data.length === 'number' &&
|
||
|
typeof data.save === 'function' &&
|
||
|
typeof data.restore === 'function' &&
|
||
|
typeof data.isEmpty === 'function' &&
|
||
|
typeof data.readUInt8 === 'function' &&
|
||
|
typeof data.skip === 'function' &&
|
||
|
typeof data.raw === 'function';
|
||
|
|
||
|
return isCompatible;
|
||
|
};
|
||
|
|
||
|
DecoderBuffer.prototype.save = function save() {
|
||
|
return { offset: this.offset, reporter: Reporter.prototype.save.call(this) };
|
||
|
};
|
||
|
|
||
|
DecoderBuffer.prototype.restore = function restore(save) {
|
||
|
// Return skipped data
|
||
|
const res = new DecoderBuffer(this.base);
|
||
|
res.offset = save.offset;
|
||
|
res.length = this.offset;
|
||
|
|
||
|
this.offset = save.offset;
|
||
|
Reporter.prototype.restore.call(this, save.reporter);
|
||
|
|
||
|
return res;
|
||
|
};
|
||
|
|
||
|
DecoderBuffer.prototype.isEmpty = function isEmpty() {
|
||
|
return this.offset === this.length;
|
||
|
};
|
||
|
|
||
|
DecoderBuffer.prototype.readUInt8 = function readUInt8(fail) {
|
||
|
if (this.offset + 1 <= this.length)
|
||
|
return this.base.readUInt8(this.offset++, true);
|
||
|
else
|
||
|
return this.error(fail || 'DecoderBuffer overrun');
|
||
|
};
|
||
|
|
||
|
DecoderBuffer.prototype.skip = function skip(bytes, fail) {
|
||
|
if (!(this.offset + bytes <= this.length))
|
||
|
return this.error(fail || 'DecoderBuffer overrun');
|
||
|
|
||
|
const res = new DecoderBuffer(this.base);
|
||
|
|
||
|
// Share reporter state
|
||
|
res._reporterState = this._reporterState;
|
||
|
|
||
|
res.offset = this.offset;
|
||
|
res.length = this.offset + bytes;
|
||
|
this.offset += bytes;
|
||
|
return res;
|
||
|
};
|
||
|
|
||
|
DecoderBuffer.prototype.raw = function raw(save) {
|
||
|
return this.base.slice(save ? save.offset : this.offset, this.length);
|
||
|
};
|
||
|
|
||
|
function EncoderBuffer(value, reporter) {
|
||
|
if (Array.isArray(value)) {
|
||
|
this.length = 0;
|
||
|
this.value = value.map(function(item) {
|
||
|
if (!EncoderBuffer.isEncoderBuffer(item))
|
||
|
item = new EncoderBuffer(item, reporter);
|
||
|
this.length += item.length;
|
||
|
return item;
|
||
|
}, this);
|
||
|
} else if (typeof value === 'number') {
|
||
|
if (!(0 <= value && value <= 0xff))
|
||
|
return reporter.error('non-byte EncoderBuffer value');
|
||
|
this.value = value;
|
||
|
this.length = 1;
|
||
|
} else if (typeof value === 'string') {
|
||
|
this.value = value;
|
||
|
this.length = Buffer.byteLength(value);
|
||
|
} else if (Buffer.isBuffer(value)) {
|
||
|
this.value = value;
|
||
|
this.length = value.length;
|
||
|
} else {
|
||
|
return reporter.error('Unsupported type: ' + typeof value);
|
||
|
}
|
||
|
}
|
||
|
exports.EncoderBuffer = EncoderBuffer;
|
||
|
|
||
|
EncoderBuffer.isEncoderBuffer = function isEncoderBuffer(data) {
|
||
|
if (data instanceof EncoderBuffer) {
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
// Or accept compatible API
|
||
|
const isCompatible = typeof data === 'object' &&
|
||
|
data.constructor.name === 'EncoderBuffer' &&
|
||
|
typeof data.length === 'number' &&
|
||
|
typeof data.join === 'function';
|
||
|
|
||
|
return isCompatible;
|
||
|
};
|
||
|
|
||
|
EncoderBuffer.prototype.join = function join(out, offset) {
|
||
|
if (!out)
|
||
|
out = Buffer.alloc(this.length);
|
||
|
if (!offset)
|
||
|
offset = 0;
|
||
|
|
||
|
if (this.length === 0)
|
||
|
return out;
|
||
|
|
||
|
if (Array.isArray(this.value)) {
|
||
|
this.value.forEach(function(item) {
|
||
|
item.join(out, offset);
|
||
|
offset += item.length;
|
||
|
});
|
||
|
} else {
|
||
|
if (typeof this.value === 'number')
|
||
|
out[offset] = this.value;
|
||
|
else if (typeof this.value === 'string')
|
||
|
out.write(this.value, offset);
|
||
|
else if (Buffer.isBuffer(this.value))
|
||
|
this.value.copy(out, offset);
|
||
|
offset += this.length;
|
||
|
}
|
||
|
|
||
|
return out;
|
||
|
};
|