writer.js 22 KB

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