You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							433 lines
						
					
					
						
							9.8 KiB
						
					
					
				
			
		
		
	
	
							433 lines
						
					
					
						
							9.8 KiB
						
					
					
				'use strict'
 | 
						|
 | 
						|
// A readable tar stream creator
 | 
						|
// Technically, this is a transform stream that you write paths into,
 | 
						|
// and tar format comes out of.
 | 
						|
// The `add()` method is like `write()` but returns this,
 | 
						|
// and end() return `this` as well, so you can
 | 
						|
// do `new Pack(opt).add('files').add('dir').end().pipe(output)
 | 
						|
// You could also do something like:
 | 
						|
// streamOfPaths().pipe(new Pack()).pipe(new fs.WriteStream('out.tar'))
 | 
						|
 | 
						|
class PackJob {
 | 
						|
  constructor (path, absolute) {
 | 
						|
    this.path = path || './'
 | 
						|
    this.absolute = absolute
 | 
						|
    this.entry = null
 | 
						|
    this.stat = null
 | 
						|
    this.readdir = null
 | 
						|
    this.pending = false
 | 
						|
    this.ignore = false
 | 
						|
    this.piped = false
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
const { Minipass } = require('minipass')
 | 
						|
const zlib = require('minizlib')
 | 
						|
const ReadEntry = require('./read-entry.js')
 | 
						|
const WriteEntry = require('./write-entry.js')
 | 
						|
const WriteEntrySync = WriteEntry.Sync
 | 
						|
const WriteEntryTar = WriteEntry.Tar
 | 
						|
const Yallist = require('yallist')
 | 
						|
const EOF = Buffer.alloc(1024)
 | 
						|
const ONSTAT = Symbol('onStat')
 | 
						|
const ENDED = Symbol('ended')
 | 
						|
const QUEUE = Symbol('queue')
 | 
						|
const CURRENT = Symbol('current')
 | 
						|
const PROCESS = Symbol('process')
 | 
						|
const PROCESSING = Symbol('processing')
 | 
						|
const PROCESSJOB = Symbol('processJob')
 | 
						|
const JOBS = Symbol('jobs')
 | 
						|
const JOBDONE = Symbol('jobDone')
 | 
						|
const ADDFSENTRY = Symbol('addFSEntry')
 | 
						|
const ADDTARENTRY = Symbol('addTarEntry')
 | 
						|
const STAT = Symbol('stat')
 | 
						|
const READDIR = Symbol('readdir')
 | 
						|
const ONREADDIR = Symbol('onreaddir')
 | 
						|
const PIPE = Symbol('pipe')
 | 
						|
const ENTRY = Symbol('entry')
 | 
						|
const ENTRYOPT = Symbol('entryOpt')
 | 
						|
const WRITEENTRYCLASS = Symbol('writeEntryClass')
 | 
						|
const WRITE = Symbol('write')
 | 
						|
const ONDRAIN = Symbol('ondrain')
 | 
						|
 | 
						|
const fs = require('fs')
 | 
						|
const path = require('path')
 | 
						|
const warner = require('./warn-mixin.js')
 | 
						|
const normPath = require('./normalize-windows-path.js')
 | 
						|
 | 
						|
const Pack = warner(class Pack extends Minipass {
 | 
						|
  constructor (opt) {
 | 
						|
    super(opt)
 | 
						|
    opt = opt || Object.create(null)
 | 
						|
    this.opt = opt
 | 
						|
    this.file = opt.file || ''
 | 
						|
    this.cwd = opt.cwd || process.cwd()
 | 
						|
    this.maxReadSize = opt.maxReadSize
 | 
						|
    this.preservePaths = !!opt.preservePaths
 | 
						|
    this.strict = !!opt.strict
 | 
						|
    this.noPax = !!opt.noPax
 | 
						|
    this.prefix = normPath(opt.prefix || '')
 | 
						|
    this.linkCache = opt.linkCache || new Map()
 | 
						|
    this.statCache = opt.statCache || new Map()
 | 
						|
    this.readdirCache = opt.readdirCache || new Map()
 | 
						|
 | 
						|
    this[WRITEENTRYCLASS] = WriteEntry
 | 
						|
    if (typeof opt.onwarn === 'function') {
 | 
						|
      this.on('warn', opt.onwarn)
 | 
						|
    }
 | 
						|
 | 
						|
    this.portable = !!opt.portable
 | 
						|
    this.zip = null
 | 
						|
 | 
						|
    if (opt.gzip || opt.brotli) {
 | 
						|
      if (opt.gzip && opt.brotli) {
 | 
						|
        throw new TypeError('gzip and brotli are mutually exclusive')
 | 
						|
      }
 | 
						|
      if (opt.gzip) {
 | 
						|
        if (typeof opt.gzip !== 'object') {
 | 
						|
          opt.gzip = {}
 | 
						|
        }
 | 
						|
        if (this.portable) {
 | 
						|
          opt.gzip.portable = true
 | 
						|
        }
 | 
						|
        this.zip = new zlib.Gzip(opt.gzip)
 | 
						|
      }
 | 
						|
      if (opt.brotli) {
 | 
						|
        if (typeof opt.brotli !== 'object') {
 | 
						|
          opt.brotli = {}
 | 
						|
        }
 | 
						|
        this.zip = new zlib.BrotliCompress(opt.brotli)
 | 
						|
      }
 | 
						|
      this.zip.on('data', chunk => super.write(chunk))
 | 
						|
      this.zip.on('end', _ => super.end())
 | 
						|
      this.zip.on('drain', _ => this[ONDRAIN]())
 | 
						|
      this.on('resume', _ => this.zip.resume())
 | 
						|
    } else {
 | 
						|
      this.on('drain', this[ONDRAIN])
 | 
						|
    }
 | 
						|
 | 
						|
    this.noDirRecurse = !!opt.noDirRecurse
 | 
						|
    this.follow = !!opt.follow
 | 
						|
    this.noMtime = !!opt.noMtime
 | 
						|
    this.mtime = opt.mtime || null
 | 
						|
 | 
						|
    this.filter = typeof opt.filter === 'function' ? opt.filter : _ => true
 | 
						|
 | 
						|
    this[QUEUE] = new Yallist()
 | 
						|
    this[JOBS] = 0
 | 
						|
    this.jobs = +opt.jobs || 4
 | 
						|
    this[PROCESSING] = false
 | 
						|
    this[ENDED] = false
 | 
						|
  }
 | 
						|
 | 
						|
  [WRITE] (chunk) {
 | 
						|
    return super.write(chunk)
 | 
						|
  }
 | 
						|
 | 
						|
  add (path) {
 | 
						|
    this.write(path)
 | 
						|
    return this
 | 
						|
  }
 | 
						|
 | 
						|
  end (path) {
 | 
						|
    if (path) {
 | 
						|
      this.write(path)
 | 
						|
    }
 | 
						|
    this[ENDED] = true
 | 
						|
    this[PROCESS]()
 | 
						|
    return this
 | 
						|
  }
 | 
						|
 | 
						|
  write (path) {
 | 
						|
    if (this[ENDED]) {
 | 
						|
      throw new Error('write after end')
 | 
						|
    }
 | 
						|
 | 
						|
    if (path instanceof ReadEntry) {
 | 
						|
      this[ADDTARENTRY](path)
 | 
						|
    } else {
 | 
						|
      this[ADDFSENTRY](path)
 | 
						|
    }
 | 
						|
    return this.flowing
 | 
						|
  }
 | 
						|
 | 
						|
  [ADDTARENTRY] (p) {
 | 
						|
    const absolute = normPath(path.resolve(this.cwd, p.path))
 | 
						|
    // in this case, we don't have to wait for the stat
 | 
						|
    if (!this.filter(p.path, p)) {
 | 
						|
      p.resume()
 | 
						|
    } else {
 | 
						|
      const job = new PackJob(p.path, absolute, false)
 | 
						|
      job.entry = new WriteEntryTar(p, this[ENTRYOPT](job))
 | 
						|
      job.entry.on('end', _ => this[JOBDONE](job))
 | 
						|
      this[JOBS] += 1
 | 
						|
      this[QUEUE].push(job)
 | 
						|
    }
 | 
						|
 | 
						|
    this[PROCESS]()
 | 
						|
  }
 | 
						|
 | 
						|
  [ADDFSENTRY] (p) {
 | 
						|
    const absolute = normPath(path.resolve(this.cwd, p))
 | 
						|
    this[QUEUE].push(new PackJob(p, absolute))
 | 
						|
    this[PROCESS]()
 | 
						|
  }
 | 
						|
 | 
						|
  [STAT] (job) {
 | 
						|
    job.pending = true
 | 
						|
    this[JOBS] += 1
 | 
						|
    const stat = this.follow ? 'stat' : 'lstat'
 | 
						|
    fs[stat](job.absolute, (er, stat) => {
 | 
						|
      job.pending = false
 | 
						|
      this[JOBS] -= 1
 | 
						|
      if (er) {
 | 
						|
        this.emit('error', er)
 | 
						|
      } else {
 | 
						|
        this[ONSTAT](job, stat)
 | 
						|
      }
 | 
						|
    })
 | 
						|
  }
 | 
						|
 | 
						|
  [ONSTAT] (job, stat) {
 | 
						|
    this.statCache.set(job.absolute, stat)
 | 
						|
    job.stat = stat
 | 
						|
 | 
						|
    // now we have the stat, we can filter it.
 | 
						|
    if (!this.filter(job.path, stat)) {
 | 
						|
      job.ignore = true
 | 
						|
    }
 | 
						|
 | 
						|
    this[PROCESS]()
 | 
						|
  }
 | 
						|
 | 
						|
  [READDIR] (job) {
 | 
						|
    job.pending = true
 | 
						|
    this[JOBS] += 1
 | 
						|
    fs.readdir(job.absolute, (er, entries) => {
 | 
						|
      job.pending = false
 | 
						|
      this[JOBS] -= 1
 | 
						|
      if (er) {
 | 
						|
        return this.emit('error', er)
 | 
						|
      }
 | 
						|
      this[ONREADDIR](job, entries)
 | 
						|
    })
 | 
						|
  }
 | 
						|
 | 
						|
  [ONREADDIR] (job, entries) {
 | 
						|
    this.readdirCache.set(job.absolute, entries)
 | 
						|
    job.readdir = entries
 | 
						|
    this[PROCESS]()
 | 
						|
  }
 | 
						|
 | 
						|
  [PROCESS] () {
 | 
						|
    if (this[PROCESSING]) {
 | 
						|
      return
 | 
						|
    }
 | 
						|
 | 
						|
    this[PROCESSING] = true
 | 
						|
    for (let w = this[QUEUE].head;
 | 
						|
      w !== null && this[JOBS] < this.jobs;
 | 
						|
      w = w.next) {
 | 
						|
      this[PROCESSJOB](w.value)
 | 
						|
      if (w.value.ignore) {
 | 
						|
        const p = w.next
 | 
						|
        this[QUEUE].removeNode(w)
 | 
						|
        w.next = p
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    this[PROCESSING] = false
 | 
						|
 | 
						|
    if (this[ENDED] && !this[QUEUE].length && this[JOBS] === 0) {
 | 
						|
      if (this.zip) {
 | 
						|
        this.zip.end(EOF)
 | 
						|
      } else {
 | 
						|
        super.write(EOF)
 | 
						|
        super.end()
 | 
						|
      }
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  get [CURRENT] () {
 | 
						|
    return this[QUEUE] && this[QUEUE].head && this[QUEUE].head.value
 | 
						|
  }
 | 
						|
 | 
						|
  [JOBDONE] (job) {
 | 
						|
    this[QUEUE].shift()
 | 
						|
    this[JOBS] -= 1
 | 
						|
    this[PROCESS]()
 | 
						|
  }
 | 
						|
 | 
						|
  [PROCESSJOB] (job) {
 | 
						|
    if (job.pending) {
 | 
						|
      return
 | 
						|
    }
 | 
						|
 | 
						|
    if (job.entry) {
 | 
						|
      if (job === this[CURRENT] && !job.piped) {
 | 
						|
        this[PIPE](job)
 | 
						|
      }
 | 
						|
      return
 | 
						|
    }
 | 
						|
 | 
						|
    if (!job.stat) {
 | 
						|
      if (this.statCache.has(job.absolute)) {
 | 
						|
        this[ONSTAT](job, this.statCache.get(job.absolute))
 | 
						|
      } else {
 | 
						|
        this[STAT](job)
 | 
						|
      }
 | 
						|
    }
 | 
						|
    if (!job.stat) {
 | 
						|
      return
 | 
						|
    }
 | 
						|
 | 
						|
    // filtered out!
 | 
						|
    if (job.ignore) {
 | 
						|
      return
 | 
						|
    }
 | 
						|
 | 
						|
    if (!this.noDirRecurse && job.stat.isDirectory() && !job.readdir) {
 | 
						|
      if (this.readdirCache.has(job.absolute)) {
 | 
						|
        this[ONREADDIR](job, this.readdirCache.get(job.absolute))
 | 
						|
      } else {
 | 
						|
        this[READDIR](job)
 | 
						|
      }
 | 
						|
      if (!job.readdir) {
 | 
						|
        return
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    // we know it doesn't have an entry, because that got checked above
 | 
						|
    job.entry = this[ENTRY](job)
 | 
						|
    if (!job.entry) {
 | 
						|
      job.ignore = true
 | 
						|
      return
 | 
						|
    }
 | 
						|
 | 
						|
    if (job === this[CURRENT] && !job.piped) {
 | 
						|
      this[PIPE](job)
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  [ENTRYOPT] (job) {
 | 
						|
    return {
 | 
						|
      onwarn: (code, msg, data) => this.warn(code, msg, data),
 | 
						|
      noPax: this.noPax,
 | 
						|
      cwd: this.cwd,
 | 
						|
      absolute: job.absolute,
 | 
						|
      preservePaths: this.preservePaths,
 | 
						|
      maxReadSize: this.maxReadSize,
 | 
						|
      strict: this.strict,
 | 
						|
      portable: this.portable,
 | 
						|
      linkCache: this.linkCache,
 | 
						|
      statCache: this.statCache,
 | 
						|
      noMtime: this.noMtime,
 | 
						|
      mtime: this.mtime,
 | 
						|
      prefix: this.prefix,
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  [ENTRY] (job) {
 | 
						|
    this[JOBS] += 1
 | 
						|
    try {
 | 
						|
      return new this[WRITEENTRYCLASS](job.path, this[ENTRYOPT](job))
 | 
						|
        .on('end', () => this[JOBDONE](job))
 | 
						|
        .on('error', er => this.emit('error', er))
 | 
						|
    } catch (er) {
 | 
						|
      this.emit('error', er)
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  [ONDRAIN] () {
 | 
						|
    if (this[CURRENT] && this[CURRENT].entry) {
 | 
						|
      this[CURRENT].entry.resume()
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  // like .pipe() but using super, because our write() is special
 | 
						|
  [PIPE] (job) {
 | 
						|
    job.piped = true
 | 
						|
 | 
						|
    if (job.readdir) {
 | 
						|
      job.readdir.forEach(entry => {
 | 
						|
        const p = job.path
 | 
						|
        const base = p === './' ? '' : p.replace(/\/*$/, '/')
 | 
						|
        this[ADDFSENTRY](base + entry)
 | 
						|
      })
 | 
						|
    }
 | 
						|
 | 
						|
    const source = job.entry
 | 
						|
    const zip = this.zip
 | 
						|
 | 
						|
    if (zip) {
 | 
						|
      source.on('data', chunk => {
 | 
						|
        if (!zip.write(chunk)) {
 | 
						|
          source.pause()
 | 
						|
        }
 | 
						|
      })
 | 
						|
    } else {
 | 
						|
      source.on('data', chunk => {
 | 
						|
        if (!super.write(chunk)) {
 | 
						|
          source.pause()
 | 
						|
        }
 | 
						|
      })
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  pause () {
 | 
						|
    if (this.zip) {
 | 
						|
      this.zip.pause()
 | 
						|
    }
 | 
						|
    return super.pause()
 | 
						|
  }
 | 
						|
})
 | 
						|
 | 
						|
class PackSync extends Pack {
 | 
						|
  constructor (opt) {
 | 
						|
    super(opt)
 | 
						|
    this[WRITEENTRYCLASS] = WriteEntrySync
 | 
						|
  }
 | 
						|
 | 
						|
  // pause/resume are no-ops in sync streams.
 | 
						|
  pause () {}
 | 
						|
  resume () {}
 | 
						|
 | 
						|
  [STAT] (job) {
 | 
						|
    const stat = this.follow ? 'statSync' : 'lstatSync'
 | 
						|
    this[ONSTAT](job, fs[stat](job.absolute))
 | 
						|
  }
 | 
						|
 | 
						|
  [READDIR] (job, stat) {
 | 
						|
    this[ONREADDIR](job, fs.readdirSync(job.absolute))
 | 
						|
  }
 | 
						|
 | 
						|
  // gotta get it all in this tick
 | 
						|
  [PIPE] (job) {
 | 
						|
    const source = job.entry
 | 
						|
    const zip = this.zip
 | 
						|
 | 
						|
    if (job.readdir) {
 | 
						|
      job.readdir.forEach(entry => {
 | 
						|
        const p = job.path
 | 
						|
        const base = p === './' ? '' : p.replace(/\/*$/, '/')
 | 
						|
        this[ADDFSENTRY](base + entry)
 | 
						|
      })
 | 
						|
    }
 | 
						|
 | 
						|
    if (zip) {
 | 
						|
      source.on('data', chunk => {
 | 
						|
        zip.write(chunk)
 | 
						|
      })
 | 
						|
    } else {
 | 
						|
      source.on('data', chunk => {
 | 
						|
        super[WRITE](chunk)
 | 
						|
      })
 | 
						|
    }
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
Pack.Sync = PackSync
 | 
						|
 | 
						|
module.exports = Pack
 |