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.
		
		
		
		
		
			
		
			
				
					
					
						
							924 lines
						
					
					
						
							25 KiB
						
					
					
				
			
		
		
	
	
							924 lines
						
					
					
						
							25 KiB
						
					
					
				'use strict'
 | 
						|
 | 
						|
// the PEND/UNPEND stuff tracks whether we're ready to emit end/close yet.
 | 
						|
// but the path reservations are required to avoid race conditions where
 | 
						|
// parallelized unpack ops may mess with one another, due to dependencies
 | 
						|
// (like a Link depending on its target) or destructive operations (like
 | 
						|
// clobbering an fs object to create one of a different type.)
 | 
						|
 | 
						|
const assert = require('assert')
 | 
						|
const Parser = require('./parse.js')
 | 
						|
const fs = require('fs')
 | 
						|
const fsm = require('fs-minipass')
 | 
						|
const path = require('path')
 | 
						|
const mkdir = require('./mkdir.js')
 | 
						|
const wc = require('./winchars.js')
 | 
						|
const pathReservations = require('./path-reservations.js')
 | 
						|
const stripAbsolutePath = require('./strip-absolute-path.js')
 | 
						|
const normPath = require('./normalize-windows-path.js')
 | 
						|
const stripSlash = require('./strip-trailing-slashes.js')
 | 
						|
const normalize = require('./normalize-unicode.js')
 | 
						|
 | 
						|
const ONENTRY = Symbol('onEntry')
 | 
						|
const CHECKFS = Symbol('checkFs')
 | 
						|
const CHECKFS2 = Symbol('checkFs2')
 | 
						|
const PRUNECACHE = Symbol('pruneCache')
 | 
						|
const ISREUSABLE = Symbol('isReusable')
 | 
						|
const MAKEFS = Symbol('makeFs')
 | 
						|
const FILE = Symbol('file')
 | 
						|
const DIRECTORY = Symbol('directory')
 | 
						|
const LINK = Symbol('link')
 | 
						|
const SYMLINK = Symbol('symlink')
 | 
						|
const HARDLINK = Symbol('hardlink')
 | 
						|
const UNSUPPORTED = Symbol('unsupported')
 | 
						|
const CHECKPATH = Symbol('checkPath')
 | 
						|
const MKDIR = Symbol('mkdir')
 | 
						|
const ONERROR = Symbol('onError')
 | 
						|
const PENDING = Symbol('pending')
 | 
						|
const PEND = Symbol('pend')
 | 
						|
const UNPEND = Symbol('unpend')
 | 
						|
const ENDED = Symbol('ended')
 | 
						|
const MAYBECLOSE = Symbol('maybeClose')
 | 
						|
const SKIP = Symbol('skip')
 | 
						|
const DOCHOWN = Symbol('doChown')
 | 
						|
const UID = Symbol('uid')
 | 
						|
const GID = Symbol('gid')
 | 
						|
const CHECKED_CWD = Symbol('checkedCwd')
 | 
						|
const crypto = require('crypto')
 | 
						|
const getFlag = require('./get-write-flag.js')
 | 
						|
const platform = process.env.TESTING_TAR_FAKE_PLATFORM || process.platform
 | 
						|
const isWindows = platform === 'win32'
 | 
						|
const DEFAULT_MAX_DEPTH = 1024
 | 
						|
 | 
						|
// Unlinks on Windows are not atomic.
 | 
						|
//
 | 
						|
// This means that if you have a file entry, followed by another
 | 
						|
// file entry with an identical name, and you cannot re-use the file
 | 
						|
// (because it's a hardlink, or because unlink:true is set, or it's
 | 
						|
// Windows, which does not have useful nlink values), then the unlink
 | 
						|
// will be committed to the disk AFTER the new file has been written
 | 
						|
// over the old one, deleting the new file.
 | 
						|
//
 | 
						|
// To work around this, on Windows systems, we rename the file and then
 | 
						|
// delete the renamed file.  It's a sloppy kludge, but frankly, I do not
 | 
						|
// know of a better way to do this, given windows' non-atomic unlink
 | 
						|
// semantics.
 | 
						|
//
 | 
						|
// See: https://github.com/npm/node-tar/issues/183
 | 
						|
/* istanbul ignore next */
 | 
						|
const unlinkFile = (path, cb) => {
 | 
						|
  if (!isWindows) {
 | 
						|
    return fs.unlink(path, cb)
 | 
						|
  }
 | 
						|
 | 
						|
  const name = path + '.DELETE.' + crypto.randomBytes(16).toString('hex')
 | 
						|
  fs.rename(path, name, er => {
 | 
						|
    if (er) {
 | 
						|
      return cb(er)
 | 
						|
    }
 | 
						|
    fs.unlink(name, cb)
 | 
						|
  })
 | 
						|
}
 | 
						|
 | 
						|
/* istanbul ignore next */
 | 
						|
const unlinkFileSync = path => {
 | 
						|
  if (!isWindows) {
 | 
						|
    return fs.unlinkSync(path)
 | 
						|
  }
 | 
						|
 | 
						|
  const name = path + '.DELETE.' + crypto.randomBytes(16).toString('hex')
 | 
						|
  fs.renameSync(path, name)
 | 
						|
  fs.unlinkSync(name)
 | 
						|
}
 | 
						|
 | 
						|
// this.gid, entry.gid, this.processUid
 | 
						|
const uint32 = (a, b, c) =>
 | 
						|
  a === a >>> 0 ? a
 | 
						|
  : b === b >>> 0 ? b
 | 
						|
  : c
 | 
						|
 | 
						|
// clear the cache if it's a case-insensitive unicode-squashing match.
 | 
						|
// we can't know if the current file system is case-sensitive or supports
 | 
						|
// unicode fully, so we check for similarity on the maximally compatible
 | 
						|
// representation.  Err on the side of pruning, since all it's doing is
 | 
						|
// preventing lstats, and it's not the end of the world if we get a false
 | 
						|
// positive.
 | 
						|
// Note that on windows, we always drop the entire cache whenever a
 | 
						|
// symbolic link is encountered, because 8.3 filenames are impossible
 | 
						|
// to reason about, and collisions are hazards rather than just failures.
 | 
						|
const cacheKeyNormalize = path => stripSlash(normPath(normalize(path)))
 | 
						|
  .toLowerCase()
 | 
						|
 | 
						|
const pruneCache = (cache, abs) => {
 | 
						|
  abs = cacheKeyNormalize(abs)
 | 
						|
  for (const path of cache.keys()) {
 | 
						|
    const pnorm = cacheKeyNormalize(path)
 | 
						|
    if (pnorm === abs || pnorm.indexOf(abs + '/') === 0) {
 | 
						|
      cache.delete(path)
 | 
						|
    }
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
const dropCache = cache => {
 | 
						|
  for (const key of cache.keys()) {
 | 
						|
    cache.delete(key)
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
class Unpack extends Parser {
 | 
						|
  constructor (opt) {
 | 
						|
    if (!opt) {
 | 
						|
      opt = {}
 | 
						|
    }
 | 
						|
 | 
						|
    opt.ondone = _ => {
 | 
						|
      this[ENDED] = true
 | 
						|
      this[MAYBECLOSE]()
 | 
						|
    }
 | 
						|
 | 
						|
    super(opt)
 | 
						|
 | 
						|
    this[CHECKED_CWD] = false
 | 
						|
 | 
						|
    this.reservations = pathReservations()
 | 
						|
 | 
						|
    this.transform = typeof opt.transform === 'function' ? opt.transform : null
 | 
						|
 | 
						|
    this.writable = true
 | 
						|
    this.readable = false
 | 
						|
 | 
						|
    this[PENDING] = 0
 | 
						|
    this[ENDED] = false
 | 
						|
 | 
						|
    this.dirCache = opt.dirCache || new Map()
 | 
						|
 | 
						|
    if (typeof opt.uid === 'number' || typeof opt.gid === 'number') {
 | 
						|
      // need both or neither
 | 
						|
      if (typeof opt.uid !== 'number' || typeof opt.gid !== 'number') {
 | 
						|
        throw new TypeError('cannot set owner without number uid and gid')
 | 
						|
      }
 | 
						|
      if (opt.preserveOwner) {
 | 
						|
        throw new TypeError(
 | 
						|
          'cannot preserve owner in archive and also set owner explicitly')
 | 
						|
      }
 | 
						|
      this.uid = opt.uid
 | 
						|
      this.gid = opt.gid
 | 
						|
      this.setOwner = true
 | 
						|
    } else {
 | 
						|
      this.uid = null
 | 
						|
      this.gid = null
 | 
						|
      this.setOwner = false
 | 
						|
    }
 | 
						|
 | 
						|
    // default true for root
 | 
						|
    if (opt.preserveOwner === undefined && typeof opt.uid !== 'number') {
 | 
						|
      this.preserveOwner = process.getuid && process.getuid() === 0
 | 
						|
    } else {
 | 
						|
      this.preserveOwner = !!opt.preserveOwner
 | 
						|
    }
 | 
						|
 | 
						|
    this.processUid = (this.preserveOwner || this.setOwner) && process.getuid ?
 | 
						|
      process.getuid() : null
 | 
						|
    this.processGid = (this.preserveOwner || this.setOwner) && process.getgid ?
 | 
						|
      process.getgid() : null
 | 
						|
 | 
						|
    // prevent excessively deep nesting of subfolders
 | 
						|
    // set to `Infinity` to remove this restriction
 | 
						|
    this.maxDepth = typeof opt.maxDepth === 'number'
 | 
						|
      ? opt.maxDepth
 | 
						|
      : DEFAULT_MAX_DEPTH
 | 
						|
 | 
						|
    // mostly just for testing, but useful in some cases.
 | 
						|
    // Forcibly trigger a chown on every entry, no matter what
 | 
						|
    this.forceChown = opt.forceChown === true
 | 
						|
 | 
						|
    // turn ><?| in filenames into 0xf000-higher encoded forms
 | 
						|
    this.win32 = !!opt.win32 || isWindows
 | 
						|
 | 
						|
    // do not unpack over files that are newer than what's in the archive
 | 
						|
    this.newer = !!opt.newer
 | 
						|
 | 
						|
    // do not unpack over ANY files
 | 
						|
    this.keep = !!opt.keep
 | 
						|
 | 
						|
    // do not set mtime/atime of extracted entries
 | 
						|
    this.noMtime = !!opt.noMtime
 | 
						|
 | 
						|
    // allow .., absolute path entries, and unpacking through symlinks
 | 
						|
    // without this, warn and skip .., relativize absolutes, and error
 | 
						|
    // on symlinks in extraction path
 | 
						|
    this.preservePaths = !!opt.preservePaths
 | 
						|
 | 
						|
    // unlink files and links before writing. This breaks existing hard
 | 
						|
    // links, and removes symlink directories rather than erroring
 | 
						|
    this.unlink = !!opt.unlink
 | 
						|
 | 
						|
    this.cwd = normPath(path.resolve(opt.cwd || process.cwd()))
 | 
						|
    this.strip = +opt.strip || 0
 | 
						|
    // if we're not chmodding, then we don't need the process umask
 | 
						|
    this.processUmask = opt.noChmod ? 0 : process.umask()
 | 
						|
    this.umask = typeof opt.umask === 'number' ? opt.umask : this.processUmask
 | 
						|
 | 
						|
    // default mode for dirs created as parents
 | 
						|
    this.dmode = opt.dmode || (0o0777 & (~this.umask))
 | 
						|
    this.fmode = opt.fmode || (0o0666 & (~this.umask))
 | 
						|
 | 
						|
    this.on('entry', entry => this[ONENTRY](entry))
 | 
						|
  }
 | 
						|
 | 
						|
  // a bad or damaged archive is a warning for Parser, but an error
 | 
						|
  // when extracting.  Mark those errors as unrecoverable, because
 | 
						|
  // the Unpack contract cannot be met.
 | 
						|
  warn (code, msg, data = {}) {
 | 
						|
    if (code === 'TAR_BAD_ARCHIVE' || code === 'TAR_ABORT') {
 | 
						|
      data.recoverable = false
 | 
						|
    }
 | 
						|
    return super.warn(code, msg, data)
 | 
						|
  }
 | 
						|
 | 
						|
  [MAYBECLOSE] () {
 | 
						|
    if (this[ENDED] && this[PENDING] === 0) {
 | 
						|
      this.emit('prefinish')
 | 
						|
      this.emit('finish')
 | 
						|
      this.emit('end')
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  [CHECKPATH] (entry) {
 | 
						|
    const p = normPath(entry.path)
 | 
						|
    const parts = p.split('/')
 | 
						|
 | 
						|
    if (this.strip) {
 | 
						|
      if (parts.length < this.strip) {
 | 
						|
        return false
 | 
						|
      }
 | 
						|
      if (entry.type === 'Link') {
 | 
						|
        const linkparts = normPath(entry.linkpath).split('/')
 | 
						|
        if (linkparts.length >= this.strip) {
 | 
						|
          entry.linkpath = linkparts.slice(this.strip).join('/')
 | 
						|
        } else {
 | 
						|
          return false
 | 
						|
        }
 | 
						|
      }
 | 
						|
      parts.splice(0, this.strip)
 | 
						|
      entry.path = parts.join('/')
 | 
						|
    }
 | 
						|
 | 
						|
    if (isFinite(this.maxDepth) && parts.length > this.maxDepth) {
 | 
						|
      this.warn('TAR_ENTRY_ERROR', 'path excessively deep', {
 | 
						|
        entry,
 | 
						|
        path: p,
 | 
						|
        depth: parts.length,
 | 
						|
        maxDepth: this.maxDepth,
 | 
						|
      })
 | 
						|
      return false
 | 
						|
    }
 | 
						|
 | 
						|
    if (!this.preservePaths) {
 | 
						|
      if (parts.includes('..') || isWindows && /^[a-z]:\.\.$/i.test(parts[0])) {
 | 
						|
        this.warn('TAR_ENTRY_ERROR', `path contains '..'`, {
 | 
						|
          entry,
 | 
						|
          path: p,
 | 
						|
        })
 | 
						|
        return false
 | 
						|
      }
 | 
						|
 | 
						|
      // strip off the root
 | 
						|
      const [root, stripped] = stripAbsolutePath(p)
 | 
						|
      if (root) {
 | 
						|
        entry.path = stripped
 | 
						|
        this.warn('TAR_ENTRY_INFO', `stripping ${root} from absolute path`, {
 | 
						|
          entry,
 | 
						|
          path: p,
 | 
						|
        })
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    if (path.isAbsolute(entry.path)) {
 | 
						|
      entry.absolute = normPath(path.resolve(entry.path))
 | 
						|
    } else {
 | 
						|
      entry.absolute = normPath(path.resolve(this.cwd, entry.path))
 | 
						|
    }
 | 
						|
 | 
						|
    // if we somehow ended up with a path that escapes the cwd, and we are
 | 
						|
    // not in preservePaths mode, then something is fishy!  This should have
 | 
						|
    // been prevented above, so ignore this for coverage.
 | 
						|
    /* istanbul ignore if - defense in depth */
 | 
						|
    if (!this.preservePaths &&
 | 
						|
        entry.absolute.indexOf(this.cwd + '/') !== 0 &&
 | 
						|
        entry.absolute !== this.cwd) {
 | 
						|
      this.warn('TAR_ENTRY_ERROR', 'path escaped extraction target', {
 | 
						|
        entry,
 | 
						|
        path: normPath(entry.path),
 | 
						|
        resolvedPath: entry.absolute,
 | 
						|
        cwd: this.cwd,
 | 
						|
      })
 | 
						|
      return false
 | 
						|
    }
 | 
						|
 | 
						|
    // an archive can set properties on the extraction directory, but it
 | 
						|
    // may not replace the cwd with a different kind of thing entirely.
 | 
						|
    if (entry.absolute === this.cwd &&
 | 
						|
        entry.type !== 'Directory' &&
 | 
						|
        entry.type !== 'GNUDumpDir') {
 | 
						|
      return false
 | 
						|
    }
 | 
						|
 | 
						|
    // only encode : chars that aren't drive letter indicators
 | 
						|
    if (this.win32) {
 | 
						|
      const { root: aRoot } = path.win32.parse(entry.absolute)
 | 
						|
      entry.absolute = aRoot + wc.encode(entry.absolute.slice(aRoot.length))
 | 
						|
      const { root: pRoot } = path.win32.parse(entry.path)
 | 
						|
      entry.path = pRoot + wc.encode(entry.path.slice(pRoot.length))
 | 
						|
    }
 | 
						|
 | 
						|
    return true
 | 
						|
  }
 | 
						|
 | 
						|
  [ONENTRY] (entry) {
 | 
						|
    if (!this[CHECKPATH](entry)) {
 | 
						|
      return entry.resume()
 | 
						|
    }
 | 
						|
 | 
						|
    assert.equal(typeof entry.absolute, 'string')
 | 
						|
 | 
						|
    switch (entry.type) {
 | 
						|
      case 'Directory':
 | 
						|
      case 'GNUDumpDir':
 | 
						|
        if (entry.mode) {
 | 
						|
          entry.mode = entry.mode | 0o700
 | 
						|
        }
 | 
						|
 | 
						|
      // eslint-disable-next-line no-fallthrough
 | 
						|
      case 'File':
 | 
						|
      case 'OldFile':
 | 
						|
      case 'ContiguousFile':
 | 
						|
      case 'Link':
 | 
						|
      case 'SymbolicLink':
 | 
						|
        return this[CHECKFS](entry)
 | 
						|
 | 
						|
      case 'CharacterDevice':
 | 
						|
      case 'BlockDevice':
 | 
						|
      case 'FIFO':
 | 
						|
      default:
 | 
						|
        return this[UNSUPPORTED](entry)
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  [ONERROR] (er, entry) {
 | 
						|
    // Cwd has to exist, or else nothing works. That's serious.
 | 
						|
    // Other errors are warnings, which raise the error in strict
 | 
						|
    // mode, but otherwise continue on.
 | 
						|
    if (er.name === 'CwdError') {
 | 
						|
      this.emit('error', er)
 | 
						|
    } else {
 | 
						|
      this.warn('TAR_ENTRY_ERROR', er, { entry })
 | 
						|
      this[UNPEND]()
 | 
						|
      entry.resume()
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  [MKDIR] (dir, mode, cb) {
 | 
						|
    mkdir(normPath(dir), {
 | 
						|
      uid: this.uid,
 | 
						|
      gid: this.gid,
 | 
						|
      processUid: this.processUid,
 | 
						|
      processGid: this.processGid,
 | 
						|
      umask: this.processUmask,
 | 
						|
      preserve: this.preservePaths,
 | 
						|
      unlink: this.unlink,
 | 
						|
      cache: this.dirCache,
 | 
						|
      cwd: this.cwd,
 | 
						|
      mode: mode,
 | 
						|
      noChmod: this.noChmod,
 | 
						|
    }, cb)
 | 
						|
  }
 | 
						|
 | 
						|
  [DOCHOWN] (entry) {
 | 
						|
    // in preserve owner mode, chown if the entry doesn't match process
 | 
						|
    // in set owner mode, chown if setting doesn't match process
 | 
						|
    return this.forceChown ||
 | 
						|
      this.preserveOwner &&
 | 
						|
      (typeof entry.uid === 'number' && entry.uid !== this.processUid ||
 | 
						|
        typeof entry.gid === 'number' && entry.gid !== this.processGid)
 | 
						|
      ||
 | 
						|
      (typeof this.uid === 'number' && this.uid !== this.processUid ||
 | 
						|
        typeof this.gid === 'number' && this.gid !== this.processGid)
 | 
						|
  }
 | 
						|
 | 
						|
  [UID] (entry) {
 | 
						|
    return uint32(this.uid, entry.uid, this.processUid)
 | 
						|
  }
 | 
						|
 | 
						|
  [GID] (entry) {
 | 
						|
    return uint32(this.gid, entry.gid, this.processGid)
 | 
						|
  }
 | 
						|
 | 
						|
  [FILE] (entry, fullyDone) {
 | 
						|
    const mode = entry.mode & 0o7777 || this.fmode
 | 
						|
    const stream = new fsm.WriteStream(entry.absolute, {
 | 
						|
      flags: getFlag(entry.size),
 | 
						|
      mode: mode,
 | 
						|
      autoClose: false,
 | 
						|
    })
 | 
						|
    stream.on('error', er => {
 | 
						|
      if (stream.fd) {
 | 
						|
        fs.close(stream.fd, () => {})
 | 
						|
      }
 | 
						|
 | 
						|
      // flush all the data out so that we aren't left hanging
 | 
						|
      // if the error wasn't actually fatal.  otherwise the parse
 | 
						|
      // is blocked, and we never proceed.
 | 
						|
      stream.write = () => true
 | 
						|
      this[ONERROR](er, entry)
 | 
						|
      fullyDone()
 | 
						|
    })
 | 
						|
 | 
						|
    let actions = 1
 | 
						|
    const done = er => {
 | 
						|
      if (er) {
 | 
						|
        /* istanbul ignore else - we should always have a fd by now */
 | 
						|
        if (stream.fd) {
 | 
						|
          fs.close(stream.fd, () => {})
 | 
						|
        }
 | 
						|
 | 
						|
        this[ONERROR](er, entry)
 | 
						|
        fullyDone()
 | 
						|
        return
 | 
						|
      }
 | 
						|
 | 
						|
      if (--actions === 0) {
 | 
						|
        fs.close(stream.fd, er => {
 | 
						|
          if (er) {
 | 
						|
            this[ONERROR](er, entry)
 | 
						|
          } else {
 | 
						|
            this[UNPEND]()
 | 
						|
          }
 | 
						|
          fullyDone()
 | 
						|
        })
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    stream.on('finish', _ => {
 | 
						|
      // if futimes fails, try utimes
 | 
						|
      // if utimes fails, fail with the original error
 | 
						|
      // same for fchown/chown
 | 
						|
      const abs = entry.absolute
 | 
						|
      const fd = stream.fd
 | 
						|
 | 
						|
      if (entry.mtime && !this.noMtime) {
 | 
						|
        actions++
 | 
						|
        const atime = entry.atime || new Date()
 | 
						|
        const mtime = entry.mtime
 | 
						|
        fs.futimes(fd, atime, mtime, er =>
 | 
						|
          er ? fs.utimes(abs, atime, mtime, er2 => done(er2 && er))
 | 
						|
          : done())
 | 
						|
      }
 | 
						|
 | 
						|
      if (this[DOCHOWN](entry)) {
 | 
						|
        actions++
 | 
						|
        const uid = this[UID](entry)
 | 
						|
        const gid = this[GID](entry)
 | 
						|
        fs.fchown(fd, uid, gid, er =>
 | 
						|
          er ? fs.chown(abs, uid, gid, er2 => done(er2 && er))
 | 
						|
          : done())
 | 
						|
      }
 | 
						|
 | 
						|
      done()
 | 
						|
    })
 | 
						|
 | 
						|
    const tx = this.transform ? this.transform(entry) || entry : entry
 | 
						|
    if (tx !== entry) {
 | 
						|
      tx.on('error', er => {
 | 
						|
        this[ONERROR](er, entry)
 | 
						|
        fullyDone()
 | 
						|
      })
 | 
						|
      entry.pipe(tx)
 | 
						|
    }
 | 
						|
    tx.pipe(stream)
 | 
						|
  }
 | 
						|
 | 
						|
  [DIRECTORY] (entry, fullyDone) {
 | 
						|
    const mode = entry.mode & 0o7777 || this.dmode
 | 
						|
    this[MKDIR](entry.absolute, mode, er => {
 | 
						|
      if (er) {
 | 
						|
        this[ONERROR](er, entry)
 | 
						|
        fullyDone()
 | 
						|
        return
 | 
						|
      }
 | 
						|
 | 
						|
      let actions = 1
 | 
						|
      const done = _ => {
 | 
						|
        if (--actions === 0) {
 | 
						|
          fullyDone()
 | 
						|
          this[UNPEND]()
 | 
						|
          entry.resume()
 | 
						|
        }
 | 
						|
      }
 | 
						|
 | 
						|
      if (entry.mtime && !this.noMtime) {
 | 
						|
        actions++
 | 
						|
        fs.utimes(entry.absolute, entry.atime || new Date(), entry.mtime, done)
 | 
						|
      }
 | 
						|
 | 
						|
      if (this[DOCHOWN](entry)) {
 | 
						|
        actions++
 | 
						|
        fs.chown(entry.absolute, this[UID](entry), this[GID](entry), done)
 | 
						|
      }
 | 
						|
 | 
						|
      done()
 | 
						|
    })
 | 
						|
  }
 | 
						|
 | 
						|
  [UNSUPPORTED] (entry) {
 | 
						|
    entry.unsupported = true
 | 
						|
    this.warn('TAR_ENTRY_UNSUPPORTED',
 | 
						|
      `unsupported entry type: ${entry.type}`, { entry })
 | 
						|
    entry.resume()
 | 
						|
  }
 | 
						|
 | 
						|
  [SYMLINK] (entry, done) {
 | 
						|
    this[LINK](entry, entry.linkpath, 'symlink', done)
 | 
						|
  }
 | 
						|
 | 
						|
  [HARDLINK] (entry, done) {
 | 
						|
    const linkpath = normPath(path.resolve(this.cwd, entry.linkpath))
 | 
						|
    this[LINK](entry, linkpath, 'link', done)
 | 
						|
  }
 | 
						|
 | 
						|
  [PEND] () {
 | 
						|
    this[PENDING]++
 | 
						|
  }
 | 
						|
 | 
						|
  [UNPEND] () {
 | 
						|
    this[PENDING]--
 | 
						|
    this[MAYBECLOSE]()
 | 
						|
  }
 | 
						|
 | 
						|
  [SKIP] (entry) {
 | 
						|
    this[UNPEND]()
 | 
						|
    entry.resume()
 | 
						|
  }
 | 
						|
 | 
						|
  // Check if we can reuse an existing filesystem entry safely and
 | 
						|
  // overwrite it, rather than unlinking and recreating
 | 
						|
  // Windows doesn't report a useful nlink, so we just never reuse entries
 | 
						|
  [ISREUSABLE] (entry, st) {
 | 
						|
    return entry.type === 'File' &&
 | 
						|
      !this.unlink &&
 | 
						|
      st.isFile() &&
 | 
						|
      st.nlink <= 1 &&
 | 
						|
      !isWindows
 | 
						|
  }
 | 
						|
 | 
						|
  // check if a thing is there, and if so, try to clobber it
 | 
						|
  [CHECKFS] (entry) {
 | 
						|
    this[PEND]()
 | 
						|
    const paths = [entry.path]
 | 
						|
    if (entry.linkpath) {
 | 
						|
      paths.push(entry.linkpath)
 | 
						|
    }
 | 
						|
    this.reservations.reserve(paths, done => this[CHECKFS2](entry, done))
 | 
						|
  }
 | 
						|
 | 
						|
  [PRUNECACHE] (entry) {
 | 
						|
    // if we are not creating a directory, and the path is in the dirCache,
 | 
						|
    // then that means we are about to delete the directory we created
 | 
						|
    // previously, and it is no longer going to be a directory, and neither
 | 
						|
    // is any of its children.
 | 
						|
    // If a symbolic link is encountered, all bets are off.  There is no
 | 
						|
    // reasonable way to sanitize the cache in such a way we will be able to
 | 
						|
    // avoid having filesystem collisions.  If this happens with a non-symlink
 | 
						|
    // entry, it'll just fail to unpack, but a symlink to a directory, using an
 | 
						|
    // 8.3 shortname or certain unicode attacks, can evade detection and lead
 | 
						|
    // to arbitrary writes to anywhere on the system.
 | 
						|
    if (entry.type === 'SymbolicLink') {
 | 
						|
      dropCache(this.dirCache)
 | 
						|
    } else if (entry.type !== 'Directory') {
 | 
						|
      pruneCache(this.dirCache, entry.absolute)
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  [CHECKFS2] (entry, fullyDone) {
 | 
						|
    this[PRUNECACHE](entry)
 | 
						|
 | 
						|
    const done = er => {
 | 
						|
      this[PRUNECACHE](entry)
 | 
						|
      fullyDone(er)
 | 
						|
    }
 | 
						|
 | 
						|
    const checkCwd = () => {
 | 
						|
      this[MKDIR](this.cwd, this.dmode, er => {
 | 
						|
        if (er) {
 | 
						|
          this[ONERROR](er, entry)
 | 
						|
          done()
 | 
						|
          return
 | 
						|
        }
 | 
						|
        this[CHECKED_CWD] = true
 | 
						|
        start()
 | 
						|
      })
 | 
						|
    }
 | 
						|
 | 
						|
    const start = () => {
 | 
						|
      if (entry.absolute !== this.cwd) {
 | 
						|
        const parent = normPath(path.dirname(entry.absolute))
 | 
						|
        if (parent !== this.cwd) {
 | 
						|
          return this[MKDIR](parent, this.dmode, er => {
 | 
						|
            if (er) {
 | 
						|
              this[ONERROR](er, entry)
 | 
						|
              done()
 | 
						|
              return
 | 
						|
            }
 | 
						|
            afterMakeParent()
 | 
						|
          })
 | 
						|
        }
 | 
						|
      }
 | 
						|
      afterMakeParent()
 | 
						|
    }
 | 
						|
 | 
						|
    const afterMakeParent = () => {
 | 
						|
      fs.lstat(entry.absolute, (lstatEr, st) => {
 | 
						|
        if (st && (this.keep || this.newer && st.mtime > entry.mtime)) {
 | 
						|
          this[SKIP](entry)
 | 
						|
          done()
 | 
						|
          return
 | 
						|
        }
 | 
						|
        if (lstatEr || this[ISREUSABLE](entry, st)) {
 | 
						|
          return this[MAKEFS](null, entry, done)
 | 
						|
        }
 | 
						|
 | 
						|
        if (st.isDirectory()) {
 | 
						|
          if (entry.type === 'Directory') {
 | 
						|
            const needChmod = !this.noChmod &&
 | 
						|
              entry.mode &&
 | 
						|
              (st.mode & 0o7777) !== entry.mode
 | 
						|
            const afterChmod = er => this[MAKEFS](er, entry, done)
 | 
						|
            if (!needChmod) {
 | 
						|
              return afterChmod()
 | 
						|
            }
 | 
						|
            return fs.chmod(entry.absolute, entry.mode, afterChmod)
 | 
						|
          }
 | 
						|
          // Not a dir entry, have to remove it.
 | 
						|
          // NB: the only way to end up with an entry that is the cwd
 | 
						|
          // itself, in such a way that == does not detect, is a
 | 
						|
          // tricky windows absolute path with UNC or 8.3 parts (and
 | 
						|
          // preservePaths:true, or else it will have been stripped).
 | 
						|
          // In that case, the user has opted out of path protections
 | 
						|
          // explicitly, so if they blow away the cwd, c'est la vie.
 | 
						|
          if (entry.absolute !== this.cwd) {
 | 
						|
            return fs.rmdir(entry.absolute, er =>
 | 
						|
              this[MAKEFS](er, entry, done))
 | 
						|
          }
 | 
						|
        }
 | 
						|
 | 
						|
        // not a dir, and not reusable
 | 
						|
        // don't remove if the cwd, we want that error
 | 
						|
        if (entry.absolute === this.cwd) {
 | 
						|
          return this[MAKEFS](null, entry, done)
 | 
						|
        }
 | 
						|
 | 
						|
        unlinkFile(entry.absolute, er =>
 | 
						|
          this[MAKEFS](er, entry, done))
 | 
						|
      })
 | 
						|
    }
 | 
						|
 | 
						|
    if (this[CHECKED_CWD]) {
 | 
						|
      start()
 | 
						|
    } else {
 | 
						|
      checkCwd()
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  [MAKEFS] (er, entry, done) {
 | 
						|
    if (er) {
 | 
						|
      this[ONERROR](er, entry)
 | 
						|
      done()
 | 
						|
      return
 | 
						|
    }
 | 
						|
 | 
						|
    switch (entry.type) {
 | 
						|
      case 'File':
 | 
						|
      case 'OldFile':
 | 
						|
      case 'ContiguousFile':
 | 
						|
        return this[FILE](entry, done)
 | 
						|
 | 
						|
      case 'Link':
 | 
						|
        return this[HARDLINK](entry, done)
 | 
						|
 | 
						|
      case 'SymbolicLink':
 | 
						|
        return this[SYMLINK](entry, done)
 | 
						|
 | 
						|
      case 'Directory':
 | 
						|
      case 'GNUDumpDir':
 | 
						|
        return this[DIRECTORY](entry, done)
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  [LINK] (entry, linkpath, link, done) {
 | 
						|
    // XXX: get the type ('symlink' or 'junction') for windows
 | 
						|
    fs[link](linkpath, entry.absolute, er => {
 | 
						|
      if (er) {
 | 
						|
        this[ONERROR](er, entry)
 | 
						|
      } else {
 | 
						|
        this[UNPEND]()
 | 
						|
        entry.resume()
 | 
						|
      }
 | 
						|
      done()
 | 
						|
    })
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
const callSync = fn => {
 | 
						|
  try {
 | 
						|
    return [null, fn()]
 | 
						|
  } catch (er) {
 | 
						|
    return [er, null]
 | 
						|
  }
 | 
						|
}
 | 
						|
class UnpackSync extends Unpack {
 | 
						|
  [MAKEFS] (er, entry) {
 | 
						|
    return super[MAKEFS](er, entry, () => {})
 | 
						|
  }
 | 
						|
 | 
						|
  [CHECKFS] (entry) {
 | 
						|
    this[PRUNECACHE](entry)
 | 
						|
 | 
						|
    if (!this[CHECKED_CWD]) {
 | 
						|
      const er = this[MKDIR](this.cwd, this.dmode)
 | 
						|
      if (er) {
 | 
						|
        return this[ONERROR](er, entry)
 | 
						|
      }
 | 
						|
      this[CHECKED_CWD] = true
 | 
						|
    }
 | 
						|
 | 
						|
    // don't bother to make the parent if the current entry is the cwd,
 | 
						|
    // we've already checked it.
 | 
						|
    if (entry.absolute !== this.cwd) {
 | 
						|
      const parent = normPath(path.dirname(entry.absolute))
 | 
						|
      if (parent !== this.cwd) {
 | 
						|
        const mkParent = this[MKDIR](parent, this.dmode)
 | 
						|
        if (mkParent) {
 | 
						|
          return this[ONERROR](mkParent, entry)
 | 
						|
        }
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    const [lstatEr, st] = callSync(() => fs.lstatSync(entry.absolute))
 | 
						|
    if (st && (this.keep || this.newer && st.mtime > entry.mtime)) {
 | 
						|
      return this[SKIP](entry)
 | 
						|
    }
 | 
						|
 | 
						|
    if (lstatEr || this[ISREUSABLE](entry, st)) {
 | 
						|
      return this[MAKEFS](null, entry)
 | 
						|
    }
 | 
						|
 | 
						|
    if (st.isDirectory()) {
 | 
						|
      if (entry.type === 'Directory') {
 | 
						|
        const needChmod = !this.noChmod &&
 | 
						|
          entry.mode &&
 | 
						|
          (st.mode & 0o7777) !== entry.mode
 | 
						|
        const [er] = needChmod ? callSync(() => {
 | 
						|
          fs.chmodSync(entry.absolute, entry.mode)
 | 
						|
        }) : []
 | 
						|
        return this[MAKEFS](er, entry)
 | 
						|
      }
 | 
						|
      // not a dir entry, have to remove it
 | 
						|
      const [er] = callSync(() => fs.rmdirSync(entry.absolute))
 | 
						|
      this[MAKEFS](er, entry)
 | 
						|
    }
 | 
						|
 | 
						|
    // not a dir, and not reusable.
 | 
						|
    // don't remove if it's the cwd, since we want that error.
 | 
						|
    const [er] = entry.absolute === this.cwd ? []
 | 
						|
      : callSync(() => unlinkFileSync(entry.absolute))
 | 
						|
    this[MAKEFS](er, entry)
 | 
						|
  }
 | 
						|
 | 
						|
  [FILE] (entry, done) {
 | 
						|
    const mode = entry.mode & 0o7777 || this.fmode
 | 
						|
 | 
						|
    const oner = er => {
 | 
						|
      let closeError
 | 
						|
      try {
 | 
						|
        fs.closeSync(fd)
 | 
						|
      } catch (e) {
 | 
						|
        closeError = e
 | 
						|
      }
 | 
						|
      if (er || closeError) {
 | 
						|
        this[ONERROR](er || closeError, entry)
 | 
						|
      }
 | 
						|
      done()
 | 
						|
    }
 | 
						|
 | 
						|
    let fd
 | 
						|
    try {
 | 
						|
      fd = fs.openSync(entry.absolute, getFlag(entry.size), mode)
 | 
						|
    } catch (er) {
 | 
						|
      return oner(er)
 | 
						|
    }
 | 
						|
    const tx = this.transform ? this.transform(entry) || entry : entry
 | 
						|
    if (tx !== entry) {
 | 
						|
      tx.on('error', er => this[ONERROR](er, entry))
 | 
						|
      entry.pipe(tx)
 | 
						|
    }
 | 
						|
 | 
						|
    tx.on('data', chunk => {
 | 
						|
      try {
 | 
						|
        fs.writeSync(fd, chunk, 0, chunk.length)
 | 
						|
      } catch (er) {
 | 
						|
        oner(er)
 | 
						|
      }
 | 
						|
    })
 | 
						|
 | 
						|
    tx.on('end', _ => {
 | 
						|
      let er = null
 | 
						|
      // try both, falling futimes back to utimes
 | 
						|
      // if either fails, handle the first error
 | 
						|
      if (entry.mtime && !this.noMtime) {
 | 
						|
        const atime = entry.atime || new Date()
 | 
						|
        const mtime = entry.mtime
 | 
						|
        try {
 | 
						|
          fs.futimesSync(fd, atime, mtime)
 | 
						|
        } catch (futimeser) {
 | 
						|
          try {
 | 
						|
            fs.utimesSync(entry.absolute, atime, mtime)
 | 
						|
          } catch (utimeser) {
 | 
						|
            er = futimeser
 | 
						|
          }
 | 
						|
        }
 | 
						|
      }
 | 
						|
 | 
						|
      if (this[DOCHOWN](entry)) {
 | 
						|
        const uid = this[UID](entry)
 | 
						|
        const gid = this[GID](entry)
 | 
						|
 | 
						|
        try {
 | 
						|
          fs.fchownSync(fd, uid, gid)
 | 
						|
        } catch (fchowner) {
 | 
						|
          try {
 | 
						|
            fs.chownSync(entry.absolute, uid, gid)
 | 
						|
          } catch (chowner) {
 | 
						|
            er = er || fchowner
 | 
						|
          }
 | 
						|
        }
 | 
						|
      }
 | 
						|
 | 
						|
      oner(er)
 | 
						|
    })
 | 
						|
  }
 | 
						|
 | 
						|
  [DIRECTORY] (entry, done) {
 | 
						|
    const mode = entry.mode & 0o7777 || this.dmode
 | 
						|
    const er = this[MKDIR](entry.absolute, mode)
 | 
						|
    if (er) {
 | 
						|
      this[ONERROR](er, entry)
 | 
						|
      done()
 | 
						|
      return
 | 
						|
    }
 | 
						|
    if (entry.mtime && !this.noMtime) {
 | 
						|
      try {
 | 
						|
        fs.utimesSync(entry.absolute, entry.atime || new Date(), entry.mtime)
 | 
						|
      } catch (er) {}
 | 
						|
    }
 | 
						|
    if (this[DOCHOWN](entry)) {
 | 
						|
      try {
 | 
						|
        fs.chownSync(entry.absolute, this[UID](entry), this[GID](entry))
 | 
						|
      } catch (er) {}
 | 
						|
    }
 | 
						|
    done()
 | 
						|
    entry.resume()
 | 
						|
  }
 | 
						|
 | 
						|
  [MKDIR] (dir, mode) {
 | 
						|
    try {
 | 
						|
      return mkdir.sync(normPath(dir), {
 | 
						|
        uid: this.uid,
 | 
						|
        gid: this.gid,
 | 
						|
        processUid: this.processUid,
 | 
						|
        processGid: this.processGid,
 | 
						|
        umask: this.processUmask,
 | 
						|
        preserve: this.preservePaths,
 | 
						|
        unlink: this.unlink,
 | 
						|
        cache: this.dirCache,
 | 
						|
        cwd: this.cwd,
 | 
						|
        mode: mode,
 | 
						|
      })
 | 
						|
    } catch (er) {
 | 
						|
      return er
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  [LINK] (entry, linkpath, link, done) {
 | 
						|
    try {
 | 
						|
      fs[link + 'Sync'](linkpath, entry.absolute)
 | 
						|
      done()
 | 
						|
      entry.resume()
 | 
						|
    } catch (er) {
 | 
						|
      return this[ONERROR](er, entry)
 | 
						|
    }
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
Unpack.Sync = UnpackSync
 | 
						|
module.exports = Unpack
 |