mirror of
https://github.com/nodejs/node.git
synced 2025-05-16 02:01:52 +00:00

PR-URL: https://github.com/nodejs/node/pull/17777 Reviewed-By: Michaël Zasso <targos@protonmail.com> Reviewed-By: James M Snell <jasnell@gmail.com> Reviewed-By: Ruben Bridgewater <ruben@bridgewater.de> Reviewed-By: Anna Henningsen <anna@addaleax.net> Reviewed-By: Gibson Fahnestock <gibfahn@gmail.com>
188 lines
5.9 KiB
JavaScript
188 lines
5.9 KiB
JavaScript
'use strict'
|
|
|
|
const path = require('path')
|
|
const fs = require('graceful-fs')
|
|
const linkIfExists = require('gentle-fs').linkIfExists
|
|
const cmdShimIfExists = require('cmd-shim').ifExists
|
|
const asyncMap = require('slide').asyncMap
|
|
const BB = require('bluebird')
|
|
const open = BB.promisify(fs.open)
|
|
const close = BB.promisify(fs.close)
|
|
const stat = BB.promisify(fs.stat)
|
|
const chmod = BB.promisify(fs.chmod)
|
|
const Transform = require('stream').Transform
|
|
const fsWriteStreamAtomic = require('fs-write-stream-atomic')
|
|
|
|
module.exports = BB.promisify(binLinks)
|
|
|
|
function binLinks (pkg, folder, global, opts, cb) {
|
|
// if it's global, and folder is in {prefix}/node_modules,
|
|
// then bins are in {prefix}/bin
|
|
// otherwise, then bins are in folder/../.bin
|
|
var parent = pkg.name && pkg.name[0] === '@' ? path.dirname(path.dirname(folder)) : path.dirname(folder)
|
|
var gnm = global && opts.globalDir
|
|
var gtop = parent === gnm
|
|
|
|
opts.log.info('linkStuff', opts.pkgId)
|
|
opts.log.silly('linkStuff', opts.pkgId, 'has', parent, 'as its parent node_modules')
|
|
if (global) opts.log.silly('linkStuff', opts.pkgId, 'is part of a global install')
|
|
if (gnm) opts.log.silly('linkStuff', opts.pkgId, 'is installed into a global node_modules')
|
|
if (gtop) opts.log.silly('linkStuff', opts.pkgId, 'is installed into the top-level global node_modules')
|
|
|
|
asyncMap(
|
|
[linkBins, linkMans],
|
|
function (fn, cb) {
|
|
if (!fn) return cb()
|
|
opts.log.verbose(fn.name, opts.pkgId)
|
|
fn(pkg, folder, parent, gtop, opts, cb)
|
|
},
|
|
cb
|
|
)
|
|
}
|
|
|
|
function isHashbangFile (file) {
|
|
return open(file, 'r').then((fileHandle) => {
|
|
return new BB((resolve, reject) => {
|
|
fs.read(fileHandle, Buffer.from(new Array(2)), 0, 2, 0, function (err, bytesRead, buffer) {
|
|
close(fileHandle).then(() => {
|
|
resolve(!err && buffer.toString() === '#!')
|
|
}).catch(reject)
|
|
})
|
|
})
|
|
})
|
|
}
|
|
|
|
function dos2Unix (file) {
|
|
return stat(file).then((stats) => {
|
|
let previousChunkEndedInCR = false
|
|
return new BB((resolve, reject) => {
|
|
fs.createReadStream(file)
|
|
.on('error', reject)
|
|
.pipe(new Transform({
|
|
transform: function (chunk, encoding, done) {
|
|
let data = chunk.toString()
|
|
if (previousChunkEndedInCR) {
|
|
data = '\r' + data
|
|
}
|
|
if (data[data.length - 1] === '\r') {
|
|
data = data.slice(0, -1)
|
|
previousChunkEndedInCR = true
|
|
} else {
|
|
previousChunkEndedInCR = false
|
|
}
|
|
done(null, data.replace(/\r\n/g, '\n'))
|
|
},
|
|
flush: function (done) {
|
|
if (previousChunkEndedInCR) {
|
|
this.push('\r')
|
|
}
|
|
done()
|
|
}
|
|
}))
|
|
.on('error', reject)
|
|
.pipe(fsWriteStreamAtomic(file))
|
|
.on('error', reject)
|
|
.on('finish', function () {
|
|
resolve(chmod(file, stats.mode))
|
|
})
|
|
})
|
|
})
|
|
}
|
|
|
|
function getLinkOpts (opts, gently) {
|
|
return Object.assign({}, opts, { gently: gently })
|
|
}
|
|
|
|
function linkBins (pkg, folder, parent, gtop, opts, cb) {
|
|
if (!pkg.bin || (!gtop && path.basename(parent) !== 'node_modules')) {
|
|
return cb()
|
|
}
|
|
var linkOpts = getLinkOpts(opts, gtop && folder)
|
|
var execMode = parseInt('0777', 8) & (~opts.umask)
|
|
var binRoot = gtop ? opts.globalBin
|
|
: path.resolve(parent, '.bin')
|
|
opts.log.verbose('linkBins', [pkg.bin, binRoot, gtop])
|
|
|
|
asyncMap(Object.keys(pkg.bin), function (bin, cb) {
|
|
var dest = path.resolve(binRoot, bin)
|
|
var src = path.resolve(folder, pkg.bin[bin])
|
|
|
|
linkBin(src, dest, linkOpts, function (er) {
|
|
if (er) return cb(er)
|
|
// bins should always be executable.
|
|
// XXX skip chmod on windows?
|
|
fs.chmod(src, execMode, function (er) {
|
|
if (er && er.code === 'ENOENT' && opts.ignoreScripts) {
|
|
return cb()
|
|
}
|
|
if (er) return cb(er)
|
|
isHashbangFile(src).then((isHashbang) => {
|
|
if (isHashbang) {
|
|
opts.log.silly('linkBins', 'Converting line endings of hashbang file:', src)
|
|
return dos2Unix(src)
|
|
}
|
|
}).then(() => {
|
|
if (!gtop) return cb()
|
|
var dest = path.resolve(binRoot, bin)
|
|
var out = opts.parseable
|
|
? dest + '::' + src + ':BINFILE'
|
|
: dest + ' -> ' + src
|
|
|
|
if (!opts.json && !opts.parseable) {
|
|
opts.log.clearProgress()
|
|
console.log(out)
|
|
opts.log.showProgress()
|
|
}
|
|
cb()
|
|
}).catch(cb)
|
|
})
|
|
})
|
|
}, cb)
|
|
}
|
|
|
|
function linkBin (from, to, opts, cb) {
|
|
if (process.platform !== 'win32') {
|
|
return linkIfExists(from, to, opts, cb)
|
|
} else {
|
|
return cmdShimIfExists(from, to, cb)
|
|
}
|
|
}
|
|
|
|
function linkMans (pkg, folder, parent, gtop, opts, cb) {
|
|
if (!pkg.man || !gtop || process.platform === 'win32') return cb()
|
|
|
|
var manRoot = path.resolve(opts.prefix, 'share', 'man')
|
|
opts.log.verbose('linkMans', 'man files are', pkg.man, 'in', manRoot)
|
|
|
|
// make sure that the mans are unique.
|
|
// otherwise, if there are dupes, it'll fail with EEXIST
|
|
var set = pkg.man.reduce(function (acc, man) {
|
|
acc[path.basename(man)] = man
|
|
return acc
|
|
}, {})
|
|
pkg.man = pkg.man.filter(function (man) {
|
|
return set[path.basename(man)] === man
|
|
})
|
|
|
|
asyncMap(pkg.man, function (man, cb) {
|
|
if (typeof man !== 'string') return cb()
|
|
opts.log.silly('linkMans', 'preparing to link', man)
|
|
var parseMan = man.match(/(.*\.([0-9]+)(\.gz)?)$/)
|
|
if (!parseMan) {
|
|
return cb(new Error(
|
|
man + ' is not a valid name for a man file. ' +
|
|
'Man files must end with a number, ' +
|
|
'and optionally a .gz suffix if they are compressed.'
|
|
))
|
|
}
|
|
|
|
var stem = parseMan[1]
|
|
var sxn = parseMan[2]
|
|
var bn = path.basename(stem)
|
|
var manSrc = path.resolve(folder, man)
|
|
var manDest = path.join(manRoot, 'man' + sxn, bn)
|
|
|
|
linkIfExists(manSrc, manDest, getLinkOpts(opts, gtop && folder), cb)
|
|
}, cb)
|
|
}
|