forked from joliss/node-multidep
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathindex.js
160 lines (143 loc) · 5.6 KB
/
index.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
'use strict'
var fs = require('fs')
var path = require('path')
var spawn = require('spawn-cmd').spawn
var RSVP = require('rsvp')
var rimraf = require('rimraf')
// Check that all referenced versions referenced by the spec exist, and
// additionally pick up master versions. Return a hash of packages.
module.exports = function multidep(specPath) {
var spec = getSpec(specPath)
var packages = {}
Object.keys(spec.versions).sort().forEach(function(packageName) {
packages[packageName] = new PackageCollection
spec.versions[packageName].forEach(function(version) {
var packagePath = path.join(spec.path, packageName + '-' + version)
if (!fs.existsSync(packagePath)) {
throw new Error(packagePath + ': No such file or directory. Run `multidep` to install.')
}
var absPath = fs.realpathSync(path.join(packagePath, 'node_modules', packageName))
packages[packageName][version] = require.bind(global, absPath)
packages[packageName].versions.push(version)
})
var masterPath = path.join(spec.path, packageName + '-master')
if (fs.existsSync(masterPath)) {
var absPath = fs.realpathSync(masterPath)
packages[packageName]['master'] = require.bind(global, absPath)
packages[packageName].versions.push('master')
} else {
packages[packageName]['master'] = function() { return null }
}
})
function multidepRequire(packageName, version) {
if (packages[packageName] == null) {
throw new Error("Package '" + packageName + "' not found in " + specPath)
}
var versions = packages[packageName]
if (versions[version] == null) {
if (version === 'master') {
return null
} else {
throw new Error("Version " + version + " of package '" + packageName + "' not found in " + specPath)
}
}
return versions[version]()
}
multidepRequire.forEachVersion = function forEachVersion(packageName, cb) {
if (packages[packageName] == null) {
throw new Error("Package '" + packageName + "' not found in " + specPath)
}
packages[packageName].forEachVersion(cb)
}
multidepRequire.packages = packages
return multidepRequire
}
module.exports.install = function(specPath) {
var spec = getSpec(specPath)
if (!fs.existsSync(spec.path)) {
fs.mkdirSync(spec.path)
}
var promise = RSVP.resolve()
Object.keys(spec.versions).sort().forEach(function(packageName) {
spec.versions[packageName].forEach(function(version) {
promise = promise.then(function() {
var packagePath
// If version is an array, first arg is NPM path and second is alias.
if (version.constructor === Array) {
packagePath = path.join(spec.path, packageName + '-' + version[1])
} else {
packagePath = path.join(spec.path, packageName + '-' + version)
}
return RSVP.resolve()
.then(function() {
if (!fs.existsSync(packagePath)) {
console.log(packageName + ' ' + version + ': Installing')
fs.mkdirSync(packagePath)
fs.mkdirSync(path.join(packagePath, 'node_modules'))
var installName;
// If version is an array, first arg is NPM path and second is alias.
if (version.constructor === Array) {
installName = version[0]
} else {
installName = packageName + '@' + version
}
var npmCommand = 'npm';
var npmArgs = ['install', installName];
if (process.env.MULTIDEP_NPM_PATH) {
npmCommand = 'node';
npmArgs.unshift(process.env.MULTIDEP_NPM_PATH);
}
if (process.env.MULTIDEP_MEMORY_LIMIT) {
npmArgs.push('--max-old-space-size=' + process.env.MULTIDEP_MEMORY_LIMIT);
}
console.log(npmCommand, npmArgs);
var cp = spawn(npmCommand, npmArgs, {
cwd: packagePath,
stdio: 'inherit',
timeout: 300
})
return new RSVP.Promise(function(resolve, reject) {
cp.on('exit', function(code, signal) {
if (code !== 0 || signal != null) {
reject(new Error('npm exited with exit code ' + code + ', signal ' + signal))
} else {
resolve()
}
})
})
} else {
console.log(packageName + ' ' + version + ': Installed')
}
})
.catch(function(err) {
// We created a nested promise with `RSVP.resolve()` above so this
// .catch clause only applies to the previous .then and doesn't
// catch earlier failures in the chain
rimraf.sync(packagePath)
throw err
})
})
})
})
return promise
}
function getSpec(specPath) {
// specPath is relative to cwd, so we need to call realpathSync
var spec = require(fs.realpathSync(specPath))
if (!spec || !spec.hasOwnProperty('path') || Array.isArray(spec.versions)) {
throw new Error('Invalid version spec; expected { path: "test/multidep_modules", versions: { ... } }, got ' +
require('util').inspect(spec))
}
return spec
}
function PackageCollection() {
// We cannot use Object.keys(this) to get the versions because we want to
// preserve the order in which they are listed in the spec
this.versions = []
}
PackageCollection.prototype.forEachVersion = function(cb) {
for (var i = 0; i < this.versions.length; i++) {
var module = this[this.versions[i]]() // require
cb(this.versions[i], module)
}
}