| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141 | /*Copyright (c) 2014-2021, Matteo Collina <hello@matteocollina.com>Permission to use, copy, modify, and/or distribute this software for anypurpose with or without fee is hereby granted, provided that the abovecopyright notice and this permission notice appear in all copies.THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIESWITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OFMERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FORANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGESWHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN ANACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF ORIN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.*/'use strict'const { Transform } = require('stream')const { StringDecoder } = require('string_decoder')const kLast = Symbol('last')const kDecoder = Symbol('decoder')function transform (chunk, enc, cb) {  let list  if (this.overflow) { // Line buffer is full. Skip to start of next line.    const buf = this[kDecoder].write(chunk)    list = buf.split(this.matcher)    if (list.length === 1) return cb() // Line ending not found. Discard entire chunk.    // Line ending found. Discard trailing fragment of previous line and reset overflow state.    list.shift()    this.overflow = false  } else {    this[kLast] += this[kDecoder].write(chunk)    list = this[kLast].split(this.matcher)  }  this[kLast] = list.pop()  for (let i = 0; i < list.length; i++) {    try {      push(this, this.mapper(list[i]))    } catch (error) {      return cb(error)    }  }  this.overflow = this[kLast].length > this.maxLength  if (this.overflow && !this.skipOverflow) {    cb(new Error('maximum buffer reached'))    return  }  cb()}function flush (cb) {  // forward any gibberish left in there  this[kLast] += this[kDecoder].end()  if (this[kLast]) {    try {      push(this, this.mapper(this[kLast]))    } catch (error) {      return cb(error)    }  }  cb()}function push (self, val) {  if (val !== undefined) {    self.push(val)  }}function noop (incoming) {  return incoming}function split (matcher, mapper, options) {  // Set defaults for any arguments not supplied.  matcher = matcher || /\r?\n/  mapper = mapper || noop  options = options || {}  // Test arguments explicitly.  switch (arguments.length) {    case 1:      // If mapper is only argument.      if (typeof matcher === 'function') {        mapper = matcher        matcher = /\r?\n/      // If options is only argument.      } else if (typeof matcher === 'object' && !(matcher instanceof RegExp) && !matcher[Symbol.split]) {        options = matcher        matcher = /\r?\n/      }      break    case 2:      // If mapper and options are arguments.      if (typeof matcher === 'function') {        options = mapper        mapper = matcher        matcher = /\r?\n/      // If matcher and options are arguments.      } else if (typeof mapper === 'object') {        options = mapper        mapper = noop      }  }  options = Object.assign({}, options)  options.autoDestroy = true  options.transform = transform  options.flush = flush  options.readableObjectMode = true  const stream = new Transform(options)  stream[kLast] = ''  stream[kDecoder] = new StringDecoder('utf8')  stream.matcher = matcher  stream.mapper = mapper  stream.maxLength = options.maxLength  stream.skipOverflow = options.skipOverflow || false  stream.overflow = false  stream._destroy = function (err, cb) {    // Weird Node v12 bug that we need to work around    this._writableState.errorEmitted = false    cb(err)  }  return stream}module.exports = split
 |