Your IP : 18.224.54.118


Current Path : /proc/thread-self/root/lib/node_modules/npm/node_modules/gentle-fs/lib/
Upload File :
Current File : //proc/thread-self/root/lib/node_modules/npm/node_modules/gentle-fs/lib/link.js

'use strict'

const path = require('path')
const fs = require('graceful-fs')
const chain = require('slide').chain
const mkdir = require('./mkdir.js')
const rm = require('./rm.js')
const inferOwner = require('infer-owner')
const chown = require('./chown.js')

exports = module.exports = {
  link: link,
  linkIfExists: linkIfExists
}

function linkIfExists (from, to, opts, cb) {
  opts.currentIsLink = false
  opts.currentExists = false
  fs.stat(from, function (er) {
    if (er) return cb()
    fs.readlink(to, function (er, fromOnDisk) {
      if (!er || er.code !== 'ENOENT') {
        opts.currentExists = true
      }
      // if the link already exists and matches what we would do,
      // we don't need to do anything
      if (!er) {
        opts.currentIsLink = true
        var toDir = path.dirname(to)
        var absoluteFrom = path.resolve(toDir, from)
        var absoluteFromOnDisk = path.resolve(toDir, fromOnDisk)
        opts.currentTarget = absoluteFromOnDisk
        if (absoluteFrom === absoluteFromOnDisk) return cb()
      }
      link(from, to, opts, cb)
    })
  })
}

function resolveIfSymlink (maybeSymlinkPath, cb) {
  fs.lstat(maybeSymlinkPath, function (err, stat) {
    if (err) return cb.apply(this, arguments)
    if (!stat.isSymbolicLink()) return cb(null, maybeSymlinkPath)
    fs.readlink(maybeSymlinkPath, cb)
  })
}

function ensureFromIsNotSource (from, to, cb) {
  resolveIfSymlink(from, function (err, fromDestination) {
    if (err) return cb.apply(this, arguments)
    if (path.resolve(path.dirname(from), fromDestination) === path.resolve(to)) {
      return cb(new Error('Link target resolves to the same directory as link source: ' + to))
    }
    cb.apply(this, arguments)
  })
}

function link (from, to, opts, cb) {
  to = path.resolve(to)
  opts.base = path.dirname(to)
  var absTarget = path.resolve(opts.base, from)
  var relativeTarget = path.relative(opts.base, absTarget)
  var target = opts.absolute ? absTarget : relativeTarget

  const tasks = [
    [ensureFromIsNotSource, absTarget, to],
    [fs, 'stat', absTarget],
    [clobberLinkGently, from, to, opts],
    [mkdir, path.dirname(to)],
    [fs, 'symlink', target, to, 'junction']
  ]

  if (chown.selfOwner.uid !== 0) {
    chain(tasks, cb)
  } else {
    inferOwner(to).then(owner => {
      tasks.push([chown, to, owner.uid, owner.gid])
      chain(tasks, cb)
    })
  }
}

exports._clobberLinkGently = clobberLinkGently
function clobberLinkGently (from, to, opts, cb) {
  if (opts.currentExists === false) {
    // nothing to clobber!
    opts.log.silly('gently link', 'link does not already exist', {
      link: to,
      target: from
    })
    return cb()
  }

  if (!opts.clobberLinkGently ||
      opts.force === true ||
      !opts.gently ||
      typeof opts.gently !== 'string') {
    opts.log.silly('gently link', 'deleting existing link forcefully', {
      link: to,
      target: from,
      force: opts.force,
      gently: opts.gently,
      clobberLinkGently: opts.clobberLinkGently
    })
    return rm(to, opts, cb)
  }

  if (!opts.currentIsLink) {
    opts.log.verbose('gently link', 'cannot remove, not a link', to)
    // don't delete.  it'll fail with EEXIST when it tries to symlink.
    return cb()
  }

  if (opts.currentTarget.indexOf(opts.gently) === 0) {
    opts.log.silly('gently link', 'delete existing link', to)
    return rm(to, opts, cb)
  } else {
    opts.log.verbose('gently link', 'refusing to delete existing link', {
      link: to,
      currentTarget: opts.currentTarget,
      newTarget: from,
      gently: opts.gently
    })
    return cb()
  }
}

?>