Protocol Buffers - Google's data interchange format (grpc依赖)
https://developers.google.com/protocol-buffers/
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
144 lines
4.5 KiB
144 lines
4.5 KiB
/** |
|
* @fileoverview Utilities to index a binary proto by fieldnumbers without |
|
* relying on strutural proto information. |
|
*/ |
|
goog.module('protobuf.binary.indexer'); |
|
|
|
const BinaryStorage = goog.require('protobuf.runtime.BinaryStorage'); |
|
const BufferDecoder = goog.require('protobuf.binary.BufferDecoder'); |
|
const WireType = goog.require('protobuf.binary.WireType'); |
|
const {Field} = goog.require('protobuf.binary.field'); |
|
const {checkCriticalElementIndex, checkCriticalState} = goog.require('protobuf.internal.checks'); |
|
|
|
/** |
|
* Appends a new entry in the index array for the given field number. |
|
* @param {!BinaryStorage<!Field>} storage |
|
* @param {number} fieldNumber |
|
* @param {!WireType} wireType |
|
* @param {number} startIndex |
|
*/ |
|
function addIndexEntry(storage, fieldNumber, wireType, startIndex) { |
|
const field = storage.get(fieldNumber); |
|
if (field !== undefined) { |
|
field.addIndexEntry(wireType, startIndex); |
|
} else { |
|
storage.set(fieldNumber, Field.fromFirstIndexEntry(wireType, startIndex)); |
|
} |
|
} |
|
|
|
/** |
|
* Returns wire type stored in a tag. |
|
* Protos store the wire type as the first 3 bit of a tag. |
|
* @param {number} tag |
|
* @return {!WireType} |
|
*/ |
|
function tagToWireType(tag) { |
|
return /** @type {!WireType} */ (tag & 0x07); |
|
} |
|
|
|
/** |
|
* Returns the field number stored in a tag. |
|
* Protos store the field number in the upper 29 bits of a 32 bit number. |
|
* @param {number} tag |
|
* @return {number} |
|
*/ |
|
function tagToFieldNumber(tag) { |
|
return tag >>> 3; |
|
} |
|
|
|
/** |
|
* Creates an index of field locations in a given binary protobuf. |
|
* @param {!BufferDecoder} bufferDecoder |
|
* @param {number|undefined} pivot |
|
* @return {!BinaryStorage<!Field>} |
|
* @package |
|
*/ |
|
function buildIndex(bufferDecoder, pivot) { |
|
bufferDecoder.setCursor(bufferDecoder.startIndex()); |
|
|
|
const storage = new BinaryStorage(pivot); |
|
while (bufferDecoder.hasNext()) { |
|
const tag = bufferDecoder.getUnsignedVarint32(); |
|
const wireType = tagToWireType(tag); |
|
const fieldNumber = tagToFieldNumber(tag); |
|
checkCriticalState(fieldNumber > 0, `Invalid field number ${fieldNumber}`); |
|
|
|
addIndexEntry(storage, fieldNumber, wireType, bufferDecoder.cursor()); |
|
|
|
checkCriticalState( |
|
!skipField_(bufferDecoder, wireType, fieldNumber), |
|
'Found unmatched stop group.'); |
|
} |
|
return storage; |
|
} |
|
|
|
/** |
|
* Skips over fields until the next field of the message. |
|
* @param {!BufferDecoder} bufferDecoder |
|
* @param {!WireType} wireType |
|
* @param {number} fieldNumber |
|
* @return {boolean} Whether the field we skipped over was a stop group. |
|
* @private |
|
*/ |
|
function skipField_(bufferDecoder, wireType, fieldNumber) { |
|
switch (wireType) { |
|
case WireType.VARINT: |
|
checkCriticalElementIndex( |
|
bufferDecoder.cursor(), bufferDecoder.endIndex()); |
|
bufferDecoder.skipVarint(); |
|
return false; |
|
case WireType.FIXED64: |
|
bufferDecoder.skip(8); |
|
return false; |
|
case WireType.DELIMITED: |
|
checkCriticalElementIndex( |
|
bufferDecoder.cursor(), bufferDecoder.endIndex()); |
|
const length = bufferDecoder.getUnsignedVarint32(); |
|
bufferDecoder.skip(length); |
|
return false; |
|
case WireType.START_GROUP: |
|
checkCriticalState( |
|
skipGroup_(bufferDecoder, fieldNumber), 'No end group found.'); |
|
return false; |
|
case WireType.END_GROUP: |
|
// Signal that we found a stop group to the caller |
|
return true; |
|
case WireType.FIXED32: |
|
bufferDecoder.skip(4); |
|
return false; |
|
default: |
|
throw new Error(`Invalid wire type: ${wireType}`); |
|
} |
|
} |
|
|
|
/** |
|
* Skips over fields until it finds the end of a given group. |
|
* @param {!BufferDecoder} bufferDecoder |
|
* @param {number} groupFieldNumber |
|
* @return {boolean} Returns true if an end was found. |
|
* @private |
|
*/ |
|
function skipGroup_(bufferDecoder, groupFieldNumber) { |
|
// On a start group we need to keep skipping fields until we find a |
|
// corresponding stop group |
|
// Note: Since we are calling skipField from here nested groups will be |
|
// handled by recursion of this method and thus we will not see a nested |
|
// STOP GROUP here unless there is something wrong with the input data. |
|
while (bufferDecoder.hasNext()) { |
|
const tag = bufferDecoder.getUnsignedVarint32(); |
|
const wireType = tagToWireType(tag); |
|
const fieldNumber = tagToFieldNumber(tag); |
|
|
|
if (skipField_(bufferDecoder, wireType, fieldNumber)) { |
|
checkCriticalState( |
|
groupFieldNumber === fieldNumber, |
|
`Expected stop group for fieldnumber ${groupFieldNumber} not found.`); |
|
return true; |
|
} |
|
} |
|
return false; |
|
} |
|
|
|
exports = { |
|
buildIndex, |
|
};
|
|
|