433 lines
9.8 KiB
JavaScript
433 lines
9.8 KiB
JavaScript
|
'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
|