writer.js 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774
  1. /**
  2. * @fileoverview Implements Writer for writing data as the binary wire format
  3. * bytes array.
  4. */
  5. goog.module('protobuf.binary.Writer');
  6. const BufferDecoder = goog.require('protobuf.binary.BufferDecoder');
  7. const ByteString = goog.require('protobuf.ByteString');
  8. const Int64 = goog.require('protobuf.Int64');
  9. const WireType = goog.require('protobuf.binary.WireType');
  10. const {POLYFILL_TEXT_ENCODING, checkFieldNumber, checkTypeUnsignedInt32, checkWireType} = goog.require('protobuf.internal.checks');
  11. const {concatenateByteArrays} = goog.require('protobuf.binary.uint8arrays');
  12. const {encode} = goog.require('protobuf.binary.textencoding');
  13. /**
  14. * Returns a valid utf-8 encoder function based on TextEncoder if available or
  15. * a polyfill.
  16. * Some of the environments we run in do not have TextEncoder defined.
  17. * TextEncoder is faster than our polyfill so we prefer it over the polyfill.
  18. * @return {function(string):!Uint8Array}
  19. */
  20. function getEncoderFunction() {
  21. if (goog.global['TextEncoder']) {
  22. const textEncoder = new goog.global['TextEncoder']('utf-8');
  23. return s => s.length === 0 ? new Uint8Array(0) : textEncoder.encode(s);
  24. }
  25. if (POLYFILL_TEXT_ENCODING) {
  26. return encode;
  27. } else {
  28. throw new Error(
  29. 'TextEncoder is missing. ' +
  30. 'Enable protobuf.defines.POLYFILL_TEXT_ENCODING');
  31. }
  32. }
  33. /** @const {function(string): !Uint8Array} */
  34. const encoderFunction = getEncoderFunction();
  35. /**
  36. * Writer provides methods for encoding all protobuf supported type into a
  37. * binary format bytes array.
  38. * Check https://developers.google.com/protocol-buffers/docs/encoding for binary
  39. * format definition.
  40. * @final
  41. * @package
  42. */
  43. class Writer {
  44. constructor() {
  45. /**
  46. * Blocks of data that needs to be serialized. After writing all the data,
  47. * the blocks are concatenated into a single Uint8Array.
  48. * @private {!Array<!Uint8Array>}
  49. */
  50. this.blocks_ = [];
  51. /**
  52. * A buffer for writing varint data (tag number + field number for each
  53. * field, int32, uint32 etc.). Before writing a non-varint data block
  54. * (string, fixed32 etc.), the buffer is appended to the block array as a
  55. * new block, and a new buffer is started.
  56. *
  57. * We could've written each varint as a new block instead of writing
  58. * multiple varints in this buffer. But this will increase the number of
  59. * blocks, and concatenating many small blocks is slower than concatenating
  60. * few large blocks.
  61. *
  62. * TODO: Experiment with writing data in a fixed-length
  63. * Uint8Array instead of using a growing buffer.
  64. *
  65. * @private {!Array<number>}
  66. */
  67. this.currentBuffer_ = [];
  68. }
  69. /**
  70. * Converts the encoded data into a Uint8Array.
  71. * The writer is also reset.
  72. * @return {!ArrayBuffer}
  73. */
  74. getAndResetResultBuffer() {
  75. this.closeAndStartNewBuffer_();
  76. const result = concatenateByteArrays(this.blocks_);
  77. this.blocks_ = [];
  78. return result.buffer;
  79. }
  80. /**
  81. * Encodes a (field number, wire type) tuple into a wire-format field header.
  82. * @param {number} fieldNumber
  83. * @param {!WireType} wireType
  84. */
  85. writeTag(fieldNumber, wireType) {
  86. checkFieldNumber(fieldNumber);
  87. checkWireType(wireType);
  88. const tag = fieldNumber << 3 | wireType;
  89. this.writeUnsignedVarint32_(tag);
  90. }
  91. /**
  92. * Appends the current buffer into the blocks array and starts a new buffer.
  93. * @private
  94. */
  95. closeAndStartNewBuffer_() {
  96. this.blocks_.push(new Uint8Array(this.currentBuffer_));
  97. this.currentBuffer_ = [];
  98. }
  99. /**
  100. * Encodes a 32-bit integer into its wire-format varint representation and
  101. * stores it in the buffer.
  102. * @param {number} value
  103. * @private
  104. */
  105. writeUnsignedVarint32_(value) {
  106. checkTypeUnsignedInt32(value);
  107. while (value > 0x7f) {
  108. this.currentBuffer_.push((value & 0x7f) | 0x80);
  109. value = value >>> 7;
  110. }
  111. this.currentBuffer_.push(value);
  112. }
  113. /****************************************************************************
  114. * OPTIONAL METHODS
  115. ****************************************************************************/
  116. /**
  117. * Writes a boolean value field to the buffer as a varint.
  118. * @param {boolean} value
  119. * @private
  120. */
  121. writeBoolValue_(value) {
  122. this.currentBuffer_.push(value ? 1 : 0);
  123. }
  124. /**
  125. * Writes a boolean value field to the buffer as a varint.
  126. * @param {number} fieldNumber
  127. * @param {boolean} value
  128. */
  129. writeBool(fieldNumber, value) {
  130. this.writeTag(fieldNumber, WireType.VARINT);
  131. this.writeBoolValue_(value);
  132. }
  133. /**
  134. * Writes a bytes value field to the buffer as a length delimited field.
  135. * @param {number} fieldNumber
  136. * @param {!ByteString} value
  137. */
  138. writeBytes(fieldNumber, value) {
  139. this.writeTag(fieldNumber, WireType.DELIMITED);
  140. const buffer = value.toArrayBuffer();
  141. this.writeUnsignedVarint32_(buffer.byteLength);
  142. this.writeRaw_(buffer);
  143. }
  144. /**
  145. * Writes a double value field to the buffer without tag.
  146. * @param {number} value
  147. * @private
  148. */
  149. writeDoubleValue_(value) {
  150. const buffer = new ArrayBuffer(8);
  151. const view = new DataView(buffer);
  152. view.setFloat64(0, value, true);
  153. this.writeRaw_(buffer);
  154. }
  155. /**
  156. * Writes a double value field to the buffer.
  157. * @param {number} fieldNumber
  158. * @param {number} value
  159. */
  160. writeDouble(fieldNumber, value) {
  161. this.writeTag(fieldNumber, WireType.FIXED64);
  162. this.writeDoubleValue_(value);
  163. }
  164. /**
  165. * Writes a fixed32 value field to the buffer without tag.
  166. * @param {number} value
  167. * @private
  168. */
  169. writeFixed32Value_(value) {
  170. const buffer = new ArrayBuffer(4);
  171. const view = new DataView(buffer);
  172. view.setUint32(0, value, true);
  173. this.writeRaw_(buffer);
  174. }
  175. /**
  176. * Writes a fixed32 value field to the buffer.
  177. * @param {number} fieldNumber
  178. * @param {number} value
  179. */
  180. writeFixed32(fieldNumber, value) {
  181. this.writeTag(fieldNumber, WireType.FIXED32);
  182. this.writeFixed32Value_(value);
  183. }
  184. /**
  185. * Writes a float value field to the buffer without tag.
  186. * @param {number} value
  187. * @private
  188. */
  189. writeFloatValue_(value) {
  190. const buffer = new ArrayBuffer(4);
  191. const view = new DataView(buffer);
  192. view.setFloat32(0, value, true);
  193. this.writeRaw_(buffer);
  194. }
  195. /**
  196. * Writes a float value field to the buffer.
  197. * @param {number} fieldNumber
  198. * @param {number} value
  199. */
  200. writeFloat(fieldNumber, value) {
  201. this.writeTag(fieldNumber, WireType.FIXED32);
  202. this.writeFloatValue_(value);
  203. }
  204. /**
  205. * Writes a int32 value field to the buffer as a varint without tag.
  206. * @param {number} value
  207. * @private
  208. */
  209. writeInt32Value_(value) {
  210. if (value >= 0) {
  211. this.writeVarint64_(0, value);
  212. } else {
  213. this.writeVarint64_(0xFFFFFFFF, value);
  214. }
  215. }
  216. /**
  217. * Writes a int32 value field to the buffer as a varint.
  218. * @param {number} fieldNumber
  219. * @param {number} value
  220. */
  221. writeInt32(fieldNumber, value) {
  222. this.writeTag(fieldNumber, WireType.VARINT);
  223. this.writeInt32Value_(value);
  224. }
  225. /**
  226. * Writes a int64 value field to the buffer as a varint.
  227. * @param {number} fieldNumber
  228. * @param {!Int64} value
  229. */
  230. writeInt64(fieldNumber, value) {
  231. this.writeTag(fieldNumber, WireType.VARINT);
  232. this.writeVarint64_(value.getHighBits(), value.getLowBits());
  233. }
  234. /**
  235. * Writes a sfixed32 value field to the buffer.
  236. * @param {number} value
  237. * @private
  238. */
  239. writeSfixed32Value_(value) {
  240. const buffer = new ArrayBuffer(4);
  241. const view = new DataView(buffer);
  242. view.setInt32(0, value, true);
  243. this.writeRaw_(buffer);
  244. }
  245. /**
  246. * Writes a sfixed32 value field to the buffer.
  247. * @param {number} fieldNumber
  248. * @param {number} value
  249. */
  250. writeSfixed32(fieldNumber, value) {
  251. this.writeTag(fieldNumber, WireType.FIXED32);
  252. this.writeSfixed32Value_(value);
  253. }
  254. /**
  255. * Writes a sfixed64 value field to the buffer without tag.
  256. * @param {!Int64} value
  257. * @private
  258. */
  259. writeSfixed64Value_(value) {
  260. const buffer = new ArrayBuffer(8);
  261. const view = new DataView(buffer);
  262. view.setInt32(0, value.getLowBits(), true);
  263. view.setInt32(4, value.getHighBits(), true);
  264. this.writeRaw_(buffer);
  265. }
  266. /**
  267. * Writes a sfixed64 value field to the buffer.
  268. * @param {number} fieldNumber
  269. * @param {!Int64} value
  270. */
  271. writeSfixed64(fieldNumber, value) {
  272. this.writeTag(fieldNumber, WireType.FIXED64);
  273. this.writeSfixed64Value_(value);
  274. }
  275. /**
  276. * Writes a uint32 value field to the buffer as a varint without tag.
  277. * @param {number} value
  278. * @private
  279. */
  280. writeUint32Value_(value) {
  281. this.writeVarint64_(0, value);
  282. }
  283. /**
  284. * Writes a uint32 value field to the buffer as a varint.
  285. * @param {number} fieldNumber
  286. * @param {number} value
  287. */
  288. writeUint32(fieldNumber, value) {
  289. this.writeTag(fieldNumber, WireType.VARINT);
  290. this.writeUint32Value_(value);
  291. }
  292. /**
  293. * Writes the bits of a 64 bit number to the buffer as a varint.
  294. * @param {number} highBits
  295. * @param {number} lowBits
  296. * @private
  297. */
  298. writeVarint64_(highBits, lowBits) {
  299. for (let i = 0; i < 28; i = i + 7) {
  300. const shift = lowBits >>> i;
  301. const hasNext = !((shift >>> 7) === 0 && highBits === 0);
  302. const byte = (hasNext ? shift | 0x80 : shift) & 0xFF;
  303. this.currentBuffer_.push(byte);
  304. if (!hasNext) {
  305. return;
  306. }
  307. }
  308. const splitBits = ((lowBits >>> 28) & 0x0F) | ((highBits & 0x07) << 4);
  309. const hasMoreBits = !((highBits >> 3) === 0);
  310. this.currentBuffer_.push(
  311. (hasMoreBits ? splitBits | 0x80 : splitBits) & 0xFF);
  312. if (!hasMoreBits) {
  313. return;
  314. }
  315. for (let i = 3; i < 31; i = i + 7) {
  316. const shift = highBits >>> i;
  317. const hasNext = !((shift >>> 7) === 0);
  318. const byte = (hasNext ? shift | 0x80 : shift) & 0xFF;
  319. this.currentBuffer_.push(byte);
  320. if (!hasNext) {
  321. return;
  322. }
  323. }
  324. this.currentBuffer_.push((highBits >>> 31) & 0x01);
  325. }
  326. /**
  327. * Writes a sint32 value field to the buffer as a varint without tag.
  328. * @param {number} value
  329. * @private
  330. */
  331. writeSint32Value_(value) {
  332. value = (value << 1) ^ (value >> 31);
  333. this.writeVarint64_(0, value);
  334. }
  335. /**
  336. * Writes a sint32 value field to the buffer as a varint.
  337. * @param {number} fieldNumber
  338. * @param {number} value
  339. */
  340. writeSint32(fieldNumber, value) {
  341. this.writeTag(fieldNumber, WireType.VARINT);
  342. this.writeSint32Value_(value);
  343. }
  344. /**
  345. * Writes a sint64 value field to the buffer as a varint without tag.
  346. * @param {!Int64} value
  347. * @private
  348. */
  349. writeSint64Value_(value) {
  350. const highBits = value.getHighBits();
  351. const lowBits = value.getLowBits();
  352. const sign = highBits >> 31;
  353. const encodedLowBits = (lowBits << 1) ^ sign;
  354. const encodedHighBits = ((highBits << 1) | (lowBits >>> 31)) ^ sign;
  355. this.writeVarint64_(encodedHighBits, encodedLowBits);
  356. }
  357. /**
  358. * Writes a sint64 value field to the buffer as a varint.
  359. * @param {number} fieldNumber
  360. * @param {!Int64} value
  361. */
  362. writeSint64(fieldNumber, value) {
  363. this.writeTag(fieldNumber, WireType.VARINT);
  364. this.writeSint64Value_(value);
  365. }
  366. /**
  367. * Writes a string value field to the buffer as a varint.
  368. * @param {number} fieldNumber
  369. * @param {string} value
  370. */
  371. writeString(fieldNumber, value) {
  372. this.writeTag(fieldNumber, WireType.DELIMITED);
  373. const array = encoderFunction(value);
  374. this.writeUnsignedVarint32_(array.length);
  375. this.writeRaw_(array.buffer);
  376. }
  377. /**
  378. * Writes raw bytes to the buffer.
  379. * @param {!ArrayBuffer} arrayBuffer
  380. * @private
  381. */
  382. writeRaw_(arrayBuffer) {
  383. this.closeAndStartNewBuffer_();
  384. this.blocks_.push(new Uint8Array(arrayBuffer));
  385. }
  386. /**
  387. * Writes raw bytes to the buffer.
  388. * @param {!BufferDecoder} bufferDecoder
  389. * @param {number} start
  390. * @param {!WireType} wireType
  391. * @package
  392. */
  393. writeBufferDecoder(bufferDecoder, start, wireType) {
  394. this.closeAndStartNewBuffer_();
  395. const dataLength = this.getLength_(bufferDecoder, start, wireType);
  396. this.blocks_.push(
  397. bufferDecoder.subBufferDecoder(start, dataLength).asUint8Array());
  398. }
  399. /**
  400. * Returns the length of the data to serialize. Returns -1 when a STOP GROUP
  401. * is found.
  402. * @param {!BufferDecoder} bufferDecoder
  403. * @param {number} start
  404. * @param {!WireType} wireType
  405. * @return {number}
  406. * @private
  407. */
  408. getLength_(bufferDecoder, start, wireType) {
  409. switch (wireType) {
  410. case WireType.VARINT:
  411. bufferDecoder.skipVarint(start);
  412. return bufferDecoder.cursor() - start;
  413. case WireType.FIXED64:
  414. return 8;
  415. case WireType.DELIMITED:
  416. const dataLength = bufferDecoder.getUnsignedVarint32At(start);
  417. return dataLength + bufferDecoder.cursor() - start;
  418. case WireType.START_GROUP:
  419. return this.getGroupLength_(bufferDecoder, start);
  420. case WireType.FIXED32:
  421. return 4;
  422. default:
  423. throw new Error(`Invalid wire type: ${wireType}`);
  424. }
  425. }
  426. /**
  427. * Skips over fields until it finds the end of a given group.
  428. * @param {!BufferDecoder} bufferDecoder
  429. * @param {number} start
  430. * @return {number}
  431. * @private
  432. */
  433. getGroupLength_(bufferDecoder, start) {
  434. // On a start group we need to keep skipping fields until we find a
  435. // corresponding stop group
  436. let cursor = start;
  437. while (cursor < bufferDecoder.endIndex()) {
  438. const tag = bufferDecoder.getUnsignedVarint32At(cursor);
  439. const wireType = /** @type {!WireType} */ (tag & 0x07);
  440. if (wireType === WireType.END_GROUP) {
  441. return bufferDecoder.cursor() - start;
  442. }
  443. cursor = bufferDecoder.cursor() +
  444. this.getLength_(bufferDecoder, bufferDecoder.cursor(), wireType);
  445. }
  446. throw new Error('No end group found');
  447. }
  448. /**
  449. * Write the whole bytes as a length delimited field.
  450. * @param {number} fieldNumber
  451. * @param {!ArrayBuffer} arrayBuffer
  452. */
  453. writeDelimited(fieldNumber, arrayBuffer) {
  454. this.writeTag(fieldNumber, WireType.DELIMITED);
  455. this.writeUnsignedVarint32_(arrayBuffer.byteLength);
  456. this.writeRaw_(arrayBuffer);
  457. }
  458. /****************************************************************************
  459. * REPEATED METHODS
  460. ****************************************************************************/
  461. /**
  462. * Writes repeated boolean values to the buffer as unpacked varints.
  463. * @param {number} fieldNumber
  464. * @param {!Array<boolean>} values
  465. */
  466. writeRepeatedBool(fieldNumber, values) {
  467. values.forEach(val => this.writeBool(fieldNumber, val));
  468. }
  469. /**
  470. * Writes repeated boolean values to the buffer as packed varints.
  471. * @param {number} fieldNumber
  472. * @param {!Array<boolean>} values
  473. */
  474. writePackedBool(fieldNumber, values) {
  475. this.writeFixedPacked_(
  476. fieldNumber, values, val => this.writeBoolValue_(val), 1);
  477. }
  478. /**
  479. * Writes repeated double values to the buffer as unpacked fixed64.
  480. * @param {number} fieldNumber
  481. * @param {!Array<number>} values
  482. */
  483. writeRepeatedDouble(fieldNumber, values) {
  484. values.forEach(val => this.writeDouble(fieldNumber, val));
  485. }
  486. /**
  487. * Writes repeated double values to the buffer as packed fixed64.
  488. * @param {number} fieldNumber
  489. * @param {!Array<number>} values
  490. */
  491. writePackedDouble(fieldNumber, values) {
  492. this.writeFixedPacked_(
  493. fieldNumber, values, val => this.writeDoubleValue_(val), 8);
  494. }
  495. /**
  496. * Writes repeated fixed32 values to the buffer as unpacked fixed32.
  497. * @param {number} fieldNumber
  498. * @param {!Array<number>} values
  499. */
  500. writeRepeatedFixed32(fieldNumber, values) {
  501. values.forEach(val => this.writeFixed32(fieldNumber, val));
  502. }
  503. /**
  504. * Writes repeated fixed32 values to the buffer as packed fixed32.
  505. * @param {number} fieldNumber
  506. * @param {!Array<number>} values
  507. */
  508. writePackedFixed32(fieldNumber, values) {
  509. this.writeFixedPacked_(
  510. fieldNumber, values, val => this.writeFixed32Value_(val), 4);
  511. }
  512. /**
  513. * Writes repeated float values to the buffer as unpacked fixed64.
  514. * @param {number} fieldNumber
  515. * @param {!Array<number>} values
  516. */
  517. writeRepeatedFloat(fieldNumber, values) {
  518. values.forEach(val => this.writeFloat(fieldNumber, val));
  519. }
  520. /**
  521. * Writes repeated float values to the buffer as packed fixed64.
  522. * @param {number} fieldNumber
  523. * @param {!Array<number>} values
  524. */
  525. writePackedFloat(fieldNumber, values) {
  526. this.writeFixedPacked_(
  527. fieldNumber, values, val => this.writeFloatValue_(val), 4);
  528. }
  529. /**
  530. * Writes repeated int32 values to the buffer as unpacked int32.
  531. * @param {number} fieldNumber
  532. * @param {!Array<number>} values
  533. */
  534. writeRepeatedInt32(fieldNumber, values) {
  535. values.forEach(val => this.writeInt32(fieldNumber, val));
  536. }
  537. /**
  538. * Writes repeated int32 values to the buffer as packed int32.
  539. * @param {number} fieldNumber
  540. * @param {!Array<number>} values
  541. */
  542. writePackedInt32(fieldNumber, values) {
  543. this.writeVariablePacked_(
  544. fieldNumber, values, (writer, val) => writer.writeInt32Value_(val));
  545. }
  546. /**
  547. * Writes repeated int64 values to the buffer as unpacked varint.
  548. * @param {number} fieldNumber
  549. * @param {!Array<!Int64>} values
  550. */
  551. writeRepeatedInt64(fieldNumber, values) {
  552. values.forEach(val => this.writeInt64(fieldNumber, val));
  553. }
  554. /**
  555. * Writes repeated int64 values to the buffer as packed varint.
  556. * @param {number} fieldNumber
  557. * @param {!Array<!Int64>} values
  558. */
  559. writePackedInt64(fieldNumber, values) {
  560. this.writeVariablePacked_(
  561. fieldNumber, values,
  562. (writer, val) =>
  563. writer.writeVarint64_(val.getHighBits(), val.getLowBits()));
  564. }
  565. /**
  566. * Writes repeated sfixed32 values to the buffer as unpacked fixed32.
  567. * @param {number} fieldNumber
  568. * @param {!Array<number>} values
  569. */
  570. writeRepeatedSfixed32(fieldNumber, values) {
  571. values.forEach(val => this.writeSfixed32(fieldNumber, val));
  572. }
  573. /**
  574. * Writes repeated sfixed32 values to the buffer as packed fixed32.
  575. * @param {number} fieldNumber
  576. * @param {!Array<number>} values
  577. */
  578. writePackedSfixed32(fieldNumber, values) {
  579. this.writeFixedPacked_(
  580. fieldNumber, values, val => this.writeSfixed32Value_(val), 4);
  581. }
  582. /**
  583. * Writes repeated sfixed64 values to the buffer as unpacked fixed64.
  584. * @param {number} fieldNumber
  585. * @param {!Array<!Int64>} values
  586. */
  587. writeRepeatedSfixed64(fieldNumber, values) {
  588. values.forEach(val => this.writeSfixed64(fieldNumber, val));
  589. }
  590. /**
  591. * Writes repeated sfixed64 values to the buffer as packed fixed64.
  592. * @param {number} fieldNumber
  593. * @param {!Array<!Int64>} values
  594. */
  595. writePackedSfixed64(fieldNumber, values) {
  596. this.writeFixedPacked_(
  597. fieldNumber, values, val => this.writeSfixed64Value_(val), 8);
  598. }
  599. /**
  600. * Writes repeated sint32 values to the buffer as unpacked sint32.
  601. * @param {number} fieldNumber
  602. * @param {!Array<number>} values
  603. */
  604. writeRepeatedSint32(fieldNumber, values) {
  605. values.forEach(val => this.writeSint32(fieldNumber, val));
  606. }
  607. /**
  608. * Writes repeated sint32 values to the buffer as packed sint32.
  609. * @param {number} fieldNumber
  610. * @param {!Array<number>} values
  611. */
  612. writePackedSint32(fieldNumber, values) {
  613. this.writeVariablePacked_(
  614. fieldNumber, values, (writer, val) => writer.writeSint32Value_(val));
  615. }
  616. /**
  617. * Writes repeated sint64 values to the buffer as unpacked varint.
  618. * @param {number} fieldNumber
  619. * @param {!Array<!Int64>} values
  620. */
  621. writeRepeatedSint64(fieldNumber, values) {
  622. values.forEach(val => this.writeSint64(fieldNumber, val));
  623. }
  624. /**
  625. * Writes repeated sint64 values to the buffer as packed varint.
  626. * @param {number} fieldNumber
  627. * @param {!Array<!Int64>} values
  628. */
  629. writePackedSint64(fieldNumber, values) {
  630. this.writeVariablePacked_(
  631. fieldNumber, values, (writer, val) => writer.writeSint64Value_(val));
  632. }
  633. /**
  634. * Writes repeated uint32 values to the buffer as unpacked uint32.
  635. * @param {number} fieldNumber
  636. * @param {!Array<number>} values
  637. */
  638. writeRepeatedUint32(fieldNumber, values) {
  639. values.forEach(val => this.writeUint32(fieldNumber, val));
  640. }
  641. /**
  642. * Writes repeated uint32 values to the buffer as packed uint32.
  643. * @param {number} fieldNumber
  644. * @param {!Array<number>} values
  645. */
  646. writePackedUint32(fieldNumber, values) {
  647. this.writeVariablePacked_(
  648. fieldNumber, values, (writer, val) => writer.writeUint32Value_(val));
  649. }
  650. /**
  651. * Writes repeated bytes values to the buffer.
  652. * @param {number} fieldNumber
  653. * @param {!Array<!ByteString>} values
  654. */
  655. writeRepeatedBytes(fieldNumber, values) {
  656. values.forEach(val => this.writeBytes(fieldNumber, val));
  657. }
  658. /**
  659. * Writes packed fields with fixed length.
  660. * @param {number} fieldNumber
  661. * @param {!Array<T>} values
  662. * @param {function(T)} valueWriter
  663. * @param {number} entitySize
  664. * @template T
  665. * @private
  666. */
  667. writeFixedPacked_(fieldNumber, values, valueWriter, entitySize) {
  668. if (values.length === 0) {
  669. return;
  670. }
  671. this.writeTag(fieldNumber, WireType.DELIMITED);
  672. this.writeUnsignedVarint32_(values.length * entitySize);
  673. this.closeAndStartNewBuffer_();
  674. values.forEach(value => valueWriter(value));
  675. }
  676. /**
  677. * Writes packed fields with variable length.
  678. * @param {number} fieldNumber
  679. * @param {!Array<T>} values
  680. * @param {function(!Writer, T)} valueWriter
  681. * @template T
  682. * @private
  683. */
  684. writeVariablePacked_(fieldNumber, values, valueWriter) {
  685. if (values.length === 0) {
  686. return;
  687. }
  688. const writer = new Writer();
  689. values.forEach(val => valueWriter(writer, val));
  690. const bytes = writer.getAndResetResultBuffer();
  691. this.writeDelimited(fieldNumber, bytes);
  692. }
  693. /**
  694. * Writes repeated string values to the buffer.
  695. * @param {number} fieldNumber
  696. * @param {!Array<string>} values
  697. */
  698. writeRepeatedString(fieldNumber, values) {
  699. values.forEach(val => this.writeString(fieldNumber, val));
  700. }
  701. }
  702. exports = Writer;