line.js 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113
  1. "use strict";
  2. var _LineDecoder_buffer, _LineDecoder_carriageReturnIndex;
  3. Object.defineProperty(exports, "__esModule", { value: true });
  4. exports.LineDecoder = void 0;
  5. exports.findDoubleNewlineIndex = findDoubleNewlineIndex;
  6. const tslib_1 = require("../tslib.js");
  7. const bytes_1 = require("../utils/bytes.js");
  8. /**
  9. * A re-implementation of httpx's `LineDecoder` in Python that handles incrementally
  10. * reading lines from text.
  11. *
  12. * https://github.com/encode/httpx/blob/920333ea98118e9cf617f246905d7b202510941c/httpx/_decoders.py#L258
  13. */
  14. class LineDecoder {
  15. constructor() {
  16. _LineDecoder_buffer.set(this, void 0);
  17. _LineDecoder_carriageReturnIndex.set(this, void 0);
  18. tslib_1.__classPrivateFieldSet(this, _LineDecoder_buffer, new Uint8Array(), "f");
  19. tslib_1.__classPrivateFieldSet(this, _LineDecoder_carriageReturnIndex, null, "f");
  20. }
  21. decode(chunk) {
  22. if (chunk == null) {
  23. return [];
  24. }
  25. const binaryChunk = chunk instanceof ArrayBuffer ? new Uint8Array(chunk)
  26. : typeof chunk === 'string' ? (0, bytes_1.encodeUTF8)(chunk)
  27. : chunk;
  28. tslib_1.__classPrivateFieldSet(this, _LineDecoder_buffer, (0, bytes_1.concatBytes)([tslib_1.__classPrivateFieldGet(this, _LineDecoder_buffer, "f"), binaryChunk]), "f");
  29. const lines = [];
  30. let patternIndex;
  31. while ((patternIndex = findNewlineIndex(tslib_1.__classPrivateFieldGet(this, _LineDecoder_buffer, "f"), tslib_1.__classPrivateFieldGet(this, _LineDecoder_carriageReturnIndex, "f"))) != null) {
  32. if (patternIndex.carriage && tslib_1.__classPrivateFieldGet(this, _LineDecoder_carriageReturnIndex, "f") == null) {
  33. // skip until we either get a corresponding `\n`, a new `\r` or nothing
  34. tslib_1.__classPrivateFieldSet(this, _LineDecoder_carriageReturnIndex, patternIndex.index, "f");
  35. continue;
  36. }
  37. // we got double \r or \rtext\n
  38. if (tslib_1.__classPrivateFieldGet(this, _LineDecoder_carriageReturnIndex, "f") != null &&
  39. (patternIndex.index !== tslib_1.__classPrivateFieldGet(this, _LineDecoder_carriageReturnIndex, "f") + 1 || patternIndex.carriage)) {
  40. lines.push((0, bytes_1.decodeUTF8)(tslib_1.__classPrivateFieldGet(this, _LineDecoder_buffer, "f").subarray(0, tslib_1.__classPrivateFieldGet(this, _LineDecoder_carriageReturnIndex, "f") - 1)));
  41. tslib_1.__classPrivateFieldSet(this, _LineDecoder_buffer, tslib_1.__classPrivateFieldGet(this, _LineDecoder_buffer, "f").subarray(tslib_1.__classPrivateFieldGet(this, _LineDecoder_carriageReturnIndex, "f")), "f");
  42. tslib_1.__classPrivateFieldSet(this, _LineDecoder_carriageReturnIndex, null, "f");
  43. continue;
  44. }
  45. const endIndex = tslib_1.__classPrivateFieldGet(this, _LineDecoder_carriageReturnIndex, "f") !== null ? patternIndex.preceding - 1 : patternIndex.preceding;
  46. const line = (0, bytes_1.decodeUTF8)(tslib_1.__classPrivateFieldGet(this, _LineDecoder_buffer, "f").subarray(0, endIndex));
  47. lines.push(line);
  48. tslib_1.__classPrivateFieldSet(this, _LineDecoder_buffer, tslib_1.__classPrivateFieldGet(this, _LineDecoder_buffer, "f").subarray(patternIndex.index), "f");
  49. tslib_1.__classPrivateFieldSet(this, _LineDecoder_carriageReturnIndex, null, "f");
  50. }
  51. return lines;
  52. }
  53. flush() {
  54. if (!tslib_1.__classPrivateFieldGet(this, _LineDecoder_buffer, "f").length) {
  55. return [];
  56. }
  57. return this.decode('\n');
  58. }
  59. }
  60. exports.LineDecoder = LineDecoder;
  61. _LineDecoder_buffer = new WeakMap(), _LineDecoder_carriageReturnIndex = new WeakMap();
  62. // prettier-ignore
  63. LineDecoder.NEWLINE_CHARS = new Set(['\n', '\r']);
  64. LineDecoder.NEWLINE_REGEXP = /\r\n|[\n\r]/g;
  65. /**
  66. * This function searches the buffer for the end patterns, (\r or \n)
  67. * and returns an object with the index preceding the matched newline and the
  68. * index after the newline char. `null` is returned if no new line is found.
  69. *
  70. * ```ts
  71. * findNewLineIndex('abc\ndef') -> { preceding: 2, index: 3 }
  72. * ```
  73. */
  74. function findNewlineIndex(buffer, startIndex) {
  75. const newline = 0x0a; // \n
  76. const carriage = 0x0d; // \r
  77. for (let i = startIndex ?? 0; i < buffer.length; i++) {
  78. if (buffer[i] === newline) {
  79. return { preceding: i, index: i + 1, carriage: false };
  80. }
  81. if (buffer[i] === carriage) {
  82. return { preceding: i, index: i + 1, carriage: true };
  83. }
  84. }
  85. return null;
  86. }
  87. function findDoubleNewlineIndex(buffer) {
  88. // This function searches the buffer for the end patterns (\r\r, \n\n, \r\n\r\n)
  89. // and returns the index right after the first occurrence of any pattern,
  90. // or -1 if none of the patterns are found.
  91. const newline = 0x0a; // \n
  92. const carriage = 0x0d; // \r
  93. for (let i = 0; i < buffer.length - 1; i++) {
  94. if (buffer[i] === newline && buffer[i + 1] === newline) {
  95. // \n\n
  96. return i + 2;
  97. }
  98. if (buffer[i] === carriage && buffer[i + 1] === carriage) {
  99. // \r\r
  100. return i + 2;
  101. }
  102. if (buffer[i] === carriage &&
  103. buffer[i + 1] === newline &&
  104. i + 3 < buffer.length &&
  105. buffer[i + 2] === carriage &&
  106. buffer[i + 3] === newline) {
  107. // \r\n\r\n
  108. return i + 4;
  109. }
  110. }
  111. return -1;
  112. }
  113. //# sourceMappingURL=line.js.map