@itwin/core-common
Version:
iTwin.js components common to frontend and backend
379 lines (375 loc) • 14.8 kB
JavaScript
;
/*---------------------------------------------------------------------------------------------
* Copyright (c) Bentley Systems, Incorporated. All rights reserved.
* See LICENSE.md in the project root for license terms and full copyright notice.
*--------------------------------------------------------------------------------------------*/
/** @packageDocumentation
* @module RpcInterface
*/
Object.defineProperty(exports, "__esModule", { value: true });
exports.RpcMultipartParser = void 0;
/*
Adapted from https://github.com/pillarjs/multiparty. Multiparty license as follows:
(The MIT License)
Copyright (c) 2013 Felix Geisendörfer
Copyright (c) 2014 Andrew Kelley
Copyright (c) 2014 Douglas Christopher Wilson
Permission is hereby granted, free of charge, to any person obtaining
a copy of this software and associated documentation files (the
'Software'), to deal in the Software without restriction, including
without limitation the rights to use, copy, modify, merge, publish,
distribute, sublicense, and/or sell copies of the Software, and to
permit persons to whom the Software is furnished to do so, subject to
the following conditions:
The above copyright notice and this permission notice shall be
included in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED 'AS IS', WITHOUT WARRANTY OF ANY KIND,
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
*/
const RpcMarshaling_1 = require("../../core/RpcMarshaling");
const START = 0;
const START_BOUNDARY = 1;
const HEADER_FIELD_START = 2;
const HEADER_FIELD = 3;
const HEADER_VALUE_START = 4;
const HEADER_VALUE = 5;
const HEADER_VALUE_ALMOST_DONE = 6;
const HEADERS_ALMOST_DONE = 7;
const PART_DATA_START = 8;
const PART_DATA = 9;
const CLOSE_BOUNDARY = 10;
const END = 11;
const LF = 10;
const CR = 13;
const SPACE = 32;
const HYPHEN = 45;
const COLON = 58;
const A = 97;
const Z = 122;
const CONTENT_TYPE_RE = /^multipart\/(?:form-data|related)(?:;|$)/i;
const CONTENT_TYPE_PARAM_RE = /;\s*([^=]+)=(?:"([^"]+)"|([^;]+))/gi;
/** @internal */
class RpcMultipartParser {
_headerFieldMark;
_headerValueMark;
_partDataMark;
_partBoundaryFlag;
_headerField;
_partHeaders;
_partName;
_partChunks;
_headerValue;
_boundary;
_buffer;
_value;
constructor(contentType, buffer) {
let m = CONTENT_TYPE_RE.exec(contentType);
if (!m) {
throw new Error("unsupported content-type");
}
let boundary = "";
CONTENT_TYPE_PARAM_RE.lastIndex = m.index + m[0].length - 1;
while ((m = CONTENT_TYPE_PARAM_RE.exec(contentType))) {
if (m[1].toLowerCase() !== "boundary")
continue;
boundary = m[2] || m[3];
break;
}
if (!boundary) {
throw new Error("content-type missing boundary");
}
this._headerField = "";
this._partHeaders = {};
this._partName = null;
this._partChunks = [];
this._headerValue = "";
this._headerFieldMark = null;
this._headerValueMark = null;
this._partDataMark = null;
this._partBoundaryFlag = false;
this._boundary = boundary;
this._buffer = buffer;
this._value = RpcMarshaling_1.RpcSerializedValue.create();
}
parse() {
let i = 0;
const len = this._buffer.length;
let prevIndex = 0;
let index = 0;
let state = START;
const boundary = new Uint8Array(this._boundary.length + 4);
const boundaryEncoder = new TextEncoder(); // encodes utf8 only
boundaryEncoder.encodeInto("\r\n--", boundary.subarray(0));
boundaryEncoder.encodeInto(this._boundary, boundary.subarray(4));
const boundaryChars = {};
for (const char of boundary) {
boundaryChars[char] = true;
}
const boundaryLength = boundary.length;
const boundaryEnd = boundaryLength - 1;
const bufferLength = this._buffer.length;
const lookbehind = new Uint8Array(boundaryLength + 8);
let c;
let cl;
for (i = 0; i < len; i++) {
c = this._buffer[i];
switch (state) {
case START:
index = 0;
state = START_BOUNDARY;
/* falls through */
case START_BOUNDARY:
if (index === boundaryLength - 2 && c === HYPHEN) {
index = 1;
state = CLOSE_BOUNDARY;
break;
}
else if (index === boundaryLength - 2) {
if (c !== CR)
throw new Error(`Expected CR Received ${c}`);
index++;
break;
}
else if (index === boundaryLength - 1) {
if (c !== LF)
throw new Error(`Expected LF Received ${c}`);
index = 0;
this._onParsePartBegin();
state = HEADER_FIELD_START;
break;
}
if (c !== boundary[index + 2])
index = -2;
if (c === boundary[index + 2])
index++;
break;
case HEADER_FIELD_START:
state = HEADER_FIELD;
this._headerFieldMark = i;
index = 0;
/* falls through */
case HEADER_FIELD:
if (c === CR) {
this._headerFieldMark = null;
state = HEADERS_ALMOST_DONE;
break;
}
index++;
if (c === HYPHEN)
break;
if (c === COLON) {
if (index === 1) {
// empty header field
throw new Error("Empty header field");
}
this._onParseHeaderField(this._buffer.subarray(this._headerFieldMark, i));
this._headerFieldMark = null;
state = HEADER_VALUE_START;
break;
}
cl = c | 0x20;
if (cl < A || cl > Z) {
throw new Error(`Expected alphabetic character, received ${c}`);
}
break;
case HEADER_VALUE_START:
if (c === SPACE)
break;
this._headerValueMark = i;
state = HEADER_VALUE;
/* falls through */
case HEADER_VALUE:
if (c === CR) {
this._onParseHeaderValue(this._buffer.subarray(this._headerValueMark, i));
this._headerValueMark = null;
this._onParseHeaderEnd();
state = HEADER_VALUE_ALMOST_DONE;
}
break;
case HEADER_VALUE_ALMOST_DONE:
if (c !== LF)
throw new Error(`Expected LF Received ${c}`);
state = HEADER_FIELD_START;
break;
case HEADERS_ALMOST_DONE:
if (c !== LF)
throw new Error(`Expected LF Received ${c}`);
const err = this._onParseHeadersEnd(i + 1);
if (err)
throw err;
state = PART_DATA_START;
break;
case PART_DATA_START:
state = PART_DATA;
this._partDataMark = i;
/* falls through */
case PART_DATA:
prevIndex = index;
if (index === 0) {
// boyer-moore derived algorithm to safely skip non-boundary data
i += boundaryEnd;
while (i < bufferLength && !(this._buffer[i] in boundaryChars)) {
i += boundaryLength;
}
i -= boundaryEnd;
c = this._buffer[i];
}
if (index < boundaryLength) {
if (boundary[index] === c) {
if (index === 0) {
this._onParsePartData(this._buffer.subarray(this._partDataMark, i));
this._partDataMark = null;
}
index++;
}
else {
index = 0;
}
}
else if (index === boundaryLength) {
index++;
if (c === CR) {
// CR = part boundary
this._partBoundaryFlag = true;
}
else if (c === HYPHEN) {
index = 1;
state = CLOSE_BOUNDARY;
break;
}
else {
index = 0;
}
}
else if (index - 1 === boundaryLength) {
if (this._partBoundaryFlag) {
index = 0;
if (c === LF) {
this._partBoundaryFlag = false;
this._onParsePartEnd();
this._onParsePartBegin();
state = HEADER_FIELD_START;
break;
}
}
else {
index = 0;
}
}
if (index > 0) {
// when matching a possible boundary, keep a lookbehind reference
// in case it turns out to be a false lead
lookbehind[index - 1] = c;
}
else if (prevIndex > 0) {
// if our boundary turned out to be rubbish, the captured lookbehind
// belongs to partData
this._onParsePartData(lookbehind.subarray(0, prevIndex));
prevIndex = 0;
this._partDataMark = i;
// reconsider the current character even so it interrupted the sequence
// it could be the beginning of a new sequence
i--;
}
break;
case CLOSE_BOUNDARY:
if (c !== HYPHEN)
throw new Error(`Expected HYPHEN Received ${c}`);
if (index === 1) {
this._onParsePartEnd();
state = END;
}
else if (index > 1) {
throw new Error("Parser has invalid state.");
}
index++;
break;
case END:
break;
default:
throw new Error("Parser has invalid state.");
}
}
if (this._headerFieldMark != null) {
this._onParseHeaderField(this._buffer.subarray(this._headerFieldMark));
this._headerFieldMark = 0;
}
if (this._headerValueMark != null) {
this._onParseHeaderValue(this._buffer.subarray(this._headerValueMark));
this._headerValueMark = 0;
}
if (this._partDataMark != null) {
this._onParsePartData(this._buffer.subarray(this._partDataMark));
this._partDataMark = 0;
}
return this._value;
}
_onParsePartBegin() {
this._clearPartVars();
}
_clearPartVars() {
this._partHeaders = {};
this._partName = null;
this._partChunks.length = 0;
this._headerField = "";
this._headerValue = "";
}
_onParseHeaderField(b) {
this._headerField += new TextDecoder("utf8").decode(b);
}
_onParseHeaderValue(b) {
this._headerValue += new TextDecoder("utf8").decode(b);
}
_onParseHeaderEnd() {
this._headerField = this._headerField.toLowerCase();
this._partHeaders[this._headerField] = this._headerValue;
let m;
if (this._headerField === "content-disposition") {
if (m = this._headerValue.match(/\bname="([^"]+)"/i)) {
this._partName = m[1];
}
// this._partFilename = parseFilename(this._headerValue);
}
else if (this._headerField === "content-transfer-encoding") {
// this._partTransferEncoding = this._headerValue.toLowerCase();
}
this._headerField = "";
this._headerValue = "";
}
_onParsePartData(b) {
this._partChunks.push(b);
}
_concatParts() {
let totalSize = 0;
for (const chunk of this._partChunks) {
totalSize += chunk.length;
}
const combined = new Uint8Array(totalSize);
let offset = 0;
for (const chunk of this._partChunks) {
combined.set(chunk, offset);
offset += chunk.length;
}
return combined;
}
_onParsePartEnd() {
const partValue = this._partChunks.length === 1 ? this._partChunks[0] : this._concatParts();
if (this._partName === "objects") {
const partDecoder = new TextDecoder();
this._value.objects = partDecoder.decode(partValue);
}
else {
this._value.data.push(partValue);
}
this._clearPartVars();
}
_onParseHeadersEnd(_offset) {
}
}
exports.RpcMultipartParser = RpcMultipartParser;
//# sourceMappingURL=RpcMultipartParser.js.map