writer.js 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772
  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. return bufferDecoder.skipVarint(start) - start;
  412. case WireType.FIXED64:
  413. return 8;
  414. case WireType.DELIMITED:
  415. const {lowBits: dataLength, dataStart} = bufferDecoder.getVarint(start);
  416. return dataLength + dataStart - start;
  417. case WireType.START_GROUP:
  418. return this.getGroupLength_(bufferDecoder, start);
  419. case WireType.FIXED32:
  420. return 4;
  421. default:
  422. throw new Error(`Invalid wire type: ${wireType}`);
  423. }
  424. }
  425. /**
  426. * Skips over fields until it finds the end of a given group.
  427. * @param {!BufferDecoder} bufferDecoder
  428. * @param {number} start
  429. * @return {number}
  430. * @private
  431. */
  432. getGroupLength_(bufferDecoder, start) {
  433. // On a start group we need to keep skipping fields until we find a
  434. // corresponding stop group
  435. let cursor = start;
  436. while (cursor < bufferDecoder.endIndex()) {
  437. const {lowBits: tag, dataStart} = bufferDecoder.getVarint(cursor);
  438. const wireType = /** @type {!WireType} */ (tag & 0x07);
  439. if (wireType === WireType.END_GROUP) {
  440. return dataStart - start;
  441. }
  442. cursor = dataStart + this.getLength_(bufferDecoder, dataStart, wireType);
  443. }
  444. throw new Error('No end group found');
  445. }
  446. /**
  447. * Write the whole bytes as a length delimited field.
  448. * @param {number} fieldNumber
  449. * @param {!ArrayBuffer} arrayBuffer
  450. */
  451. writeDelimited(fieldNumber, arrayBuffer) {
  452. this.writeTag(fieldNumber, WireType.DELIMITED);
  453. this.writeUnsignedVarint32_(arrayBuffer.byteLength);
  454. this.writeRaw_(arrayBuffer);
  455. }
  456. /****************************************************************************
  457. * REPEATED METHODS
  458. ****************************************************************************/
  459. /**
  460. * Writes repeated boolean values to the buffer as unpacked varints.
  461. * @param {number} fieldNumber
  462. * @param {!Array<boolean>} values
  463. */
  464. writeRepeatedBool(fieldNumber, values) {
  465. values.forEach(val => this.writeBool(fieldNumber, val));
  466. }
  467. /**
  468. * Writes repeated boolean values to the buffer as packed varints.
  469. * @param {number} fieldNumber
  470. * @param {!Array<boolean>} values
  471. */
  472. writePackedBool(fieldNumber, values) {
  473. this.writeFixedPacked_(
  474. fieldNumber, values, val => this.writeBoolValue_(val), 1);
  475. }
  476. /**
  477. * Writes repeated double values to the buffer as unpacked fixed64.
  478. * @param {number} fieldNumber
  479. * @param {!Array<number>} values
  480. */
  481. writeRepeatedDouble(fieldNumber, values) {
  482. values.forEach(val => this.writeDouble(fieldNumber, val));
  483. }
  484. /**
  485. * Writes repeated double values to the buffer as packed fixed64.
  486. * @param {number} fieldNumber
  487. * @param {!Array<number>} values
  488. */
  489. writePackedDouble(fieldNumber, values) {
  490. this.writeFixedPacked_(
  491. fieldNumber, values, val => this.writeDoubleValue_(val), 8);
  492. }
  493. /**
  494. * Writes repeated fixed32 values to the buffer as unpacked fixed32.
  495. * @param {number} fieldNumber
  496. * @param {!Array<number>} values
  497. */
  498. writeRepeatedFixed32(fieldNumber, values) {
  499. values.forEach(val => this.writeFixed32(fieldNumber, val));
  500. }
  501. /**
  502. * Writes repeated fixed32 values to the buffer as packed fixed32.
  503. * @param {number} fieldNumber
  504. * @param {!Array<number>} values
  505. */
  506. writePackedFixed32(fieldNumber, values) {
  507. this.writeFixedPacked_(
  508. fieldNumber, values, val => this.writeFixed32Value_(val), 4);
  509. }
  510. /**
  511. * Writes repeated float values to the buffer as unpacked fixed64.
  512. * @param {number} fieldNumber
  513. * @param {!Array<number>} values
  514. */
  515. writeRepeatedFloat(fieldNumber, values) {
  516. values.forEach(val => this.writeFloat(fieldNumber, val));
  517. }
  518. /**
  519. * Writes repeated float values to the buffer as packed fixed64.
  520. * @param {number} fieldNumber
  521. * @param {!Array<number>} values
  522. */
  523. writePackedFloat(fieldNumber, values) {
  524. this.writeFixedPacked_(
  525. fieldNumber, values, val => this.writeFloatValue_(val), 4);
  526. }
  527. /**
  528. * Writes repeated int32 values to the buffer as unpacked int32.
  529. * @param {number} fieldNumber
  530. * @param {!Array<number>} values
  531. */
  532. writeRepeatedInt32(fieldNumber, values) {
  533. values.forEach(val => this.writeInt32(fieldNumber, val));
  534. }
  535. /**
  536. * Writes repeated int32 values to the buffer as packed int32.
  537. * @param {number} fieldNumber
  538. * @param {!Array<number>} values
  539. */
  540. writePackedInt32(fieldNumber, values) {
  541. this.writeVariablePacked_(
  542. fieldNumber, values, (writer, val) => writer.writeInt32Value_(val));
  543. }
  544. /**
  545. * Writes repeated int64 values to the buffer as unpacked varint.
  546. * @param {number} fieldNumber
  547. * @param {!Array<!Int64>} values
  548. */
  549. writeRepeatedInt64(fieldNumber, values) {
  550. values.forEach(val => this.writeInt64(fieldNumber, val));
  551. }
  552. /**
  553. * Writes repeated int64 values to the buffer as packed varint.
  554. * @param {number} fieldNumber
  555. * @param {!Array<!Int64>} values
  556. */
  557. writePackedInt64(fieldNumber, values) {
  558. this.writeVariablePacked_(
  559. fieldNumber, values,
  560. (writer, val) =>
  561. writer.writeVarint64_(val.getHighBits(), val.getLowBits()));
  562. }
  563. /**
  564. * Writes repeated sfixed32 values to the buffer as unpacked fixed32.
  565. * @param {number} fieldNumber
  566. * @param {!Array<number>} values
  567. */
  568. writeRepeatedSfixed32(fieldNumber, values) {
  569. values.forEach(val => this.writeSfixed32(fieldNumber, val));
  570. }
  571. /**
  572. * Writes repeated sfixed32 values to the buffer as packed fixed32.
  573. * @param {number} fieldNumber
  574. * @param {!Array<number>} values
  575. */
  576. writePackedSfixed32(fieldNumber, values) {
  577. this.writeFixedPacked_(
  578. fieldNumber, values, val => this.writeSfixed32Value_(val), 4);
  579. }
  580. /**
  581. * Writes repeated sfixed64 values to the buffer as unpacked fixed64.
  582. * @param {number} fieldNumber
  583. * @param {!Array<!Int64>} values
  584. */
  585. writeRepeatedSfixed64(fieldNumber, values) {
  586. values.forEach(val => this.writeSfixed64(fieldNumber, val));
  587. }
  588. /**
  589. * Writes repeated sfixed64 values to the buffer as packed fixed64.
  590. * @param {number} fieldNumber
  591. * @param {!Array<!Int64>} values
  592. */
  593. writePackedSfixed64(fieldNumber, values) {
  594. this.writeFixedPacked_(
  595. fieldNumber, values, val => this.writeSfixed64Value_(val), 8);
  596. }
  597. /**
  598. * Writes repeated sint32 values to the buffer as unpacked sint32.
  599. * @param {number} fieldNumber
  600. * @param {!Array<number>} values
  601. */
  602. writeRepeatedSint32(fieldNumber, values) {
  603. values.forEach(val => this.writeSint32(fieldNumber, val));
  604. }
  605. /**
  606. * Writes repeated sint32 values to the buffer as packed sint32.
  607. * @param {number} fieldNumber
  608. * @param {!Array<number>} values
  609. */
  610. writePackedSint32(fieldNumber, values) {
  611. this.writeVariablePacked_(
  612. fieldNumber, values, (writer, val) => writer.writeSint32Value_(val));
  613. }
  614. /**
  615. * Writes repeated sint64 values to the buffer as unpacked varint.
  616. * @param {number} fieldNumber
  617. * @param {!Array<!Int64>} values
  618. */
  619. writeRepeatedSint64(fieldNumber, values) {
  620. values.forEach(val => this.writeSint64(fieldNumber, val));
  621. }
  622. /**
  623. * Writes repeated sint64 values to the buffer as packed varint.
  624. * @param {number} fieldNumber
  625. * @param {!Array<!Int64>} values
  626. */
  627. writePackedSint64(fieldNumber, values) {
  628. this.writeVariablePacked_(
  629. fieldNumber, values, (writer, val) => writer.writeSint64Value_(val));
  630. }
  631. /**
  632. * Writes repeated uint32 values to the buffer as unpacked uint32.
  633. * @param {number} fieldNumber
  634. * @param {!Array<number>} values
  635. */
  636. writeRepeatedUint32(fieldNumber, values) {
  637. values.forEach(val => this.writeUint32(fieldNumber, val));
  638. }
  639. /**
  640. * Writes repeated uint32 values to the buffer as packed uint32.
  641. * @param {number} fieldNumber
  642. * @param {!Array<number>} values
  643. */
  644. writePackedUint32(fieldNumber, values) {
  645. this.writeVariablePacked_(
  646. fieldNumber, values, (writer, val) => writer.writeUint32Value_(val));
  647. }
  648. /**
  649. * Writes repeated bytes values to the buffer.
  650. * @param {number} fieldNumber
  651. * @param {!Array<!ByteString>} values
  652. */
  653. writeRepeatedBytes(fieldNumber, values) {
  654. values.forEach(val => this.writeBytes(fieldNumber, val));
  655. }
  656. /**
  657. * Writes packed fields with fixed length.
  658. * @param {number} fieldNumber
  659. * @param {!Array<T>} values
  660. * @param {function(T)} valueWriter
  661. * @param {number} entitySize
  662. * @template T
  663. * @private
  664. */
  665. writeFixedPacked_(fieldNumber, values, valueWriter, entitySize) {
  666. if (values.length === 0) {
  667. return;
  668. }
  669. this.writeTag(fieldNumber, WireType.DELIMITED);
  670. this.writeUnsignedVarint32_(values.length * entitySize);
  671. this.closeAndStartNewBuffer_();
  672. values.forEach(value => valueWriter(value));
  673. }
  674. /**
  675. * Writes packed fields with variable length.
  676. * @param {number} fieldNumber
  677. * @param {!Array<T>} values
  678. * @param {function(!Writer, T)} valueWriter
  679. * @template T
  680. * @private
  681. */
  682. writeVariablePacked_(fieldNumber, values, valueWriter) {
  683. if (values.length === 0) {
  684. return;
  685. }
  686. const writer = new Writer();
  687. values.forEach(val => valueWriter(writer, val));
  688. const bytes = writer.getAndResetResultBuffer();
  689. this.writeDelimited(fieldNumber, bytes);
  690. }
  691. /**
  692. * Writes repeated string values to the buffer.
  693. * @param {number} fieldNumber
  694. * @param {!Array<string>} values
  695. */
  696. writeRepeatedString(fieldNumber, values) {
  697. values.forEach(val => this.writeString(fieldNumber, val));
  698. }
  699. }
  700. exports = Writer;