blob: c5c03ad796423fbf7780ca7799972d4c04841061 [file] [log] [blame]
/**
Licensed to the Apache Software Foundation (ASF) under one
or more contributor license agreements. See the NOTICE file
distributed with this work for additional information
regarding copyright ownership. The ASF licenses this file
to you under the Apache License, Version 2.0 (the
'License'); you may not use this file except in compliance
with the License. You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing,
software distributed under the License is distributed on an
'AS IS' BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
KIND, either express or implied. See the License for the
specific language governing permissions and limitations
under the License.
*/
const pify = require('pify');
const which = pify(require('which'));
var superspawn = require('cordova-common').superspawn;
var events = require('cordova-common').events;
var path = require('path');
var fs = require('fs-extra');
var CordovaError = require('cordova-common').CordovaError;
const resolve = pify(require('resolve'), { multiArgs: true });
const npa = require('npm-package-arg');
const semver = require('semver');
/**
* Installs a module from npm, a git url or the local file system.
*
* @param {String} target A spec for the package to be installed
* (anything supported by `npm install`)
* @param {String} dest Location where to install the package
* @param {Object} [opts={}] Additional options
*
* @return {Promise<string>} Absolute path to the installed package
*/
module.exports = function (target, dest, opts = {}) {
return Promise.resolve()
.then(function () {
if (!dest || !target) {
throw new CordovaError('Need to supply a target and destination');
}
// Create dest if it doesn't exist yet
fs.ensureDirSync(dest);
})
.then(_ => {
return pathToInstalledPackage(target, dest)
.catch(_ => installPackage(target, dest, opts));
})
.catch(function (err) {
throw new CordovaError(err);
});
};
// Installs the package specified by target and returns the installation path
function installPackage (target, dest, opts) {
return isNpmInstalled()
// Ensure that `npm` installs to `dest` and not any of its ancestors
.then(_ => fs.ensureDir(path.join(dest, 'node_modules')))
// Run `npm` to install requested package
.then(_ => npmArgs(target, opts))
.then(args => {
events.emit('verbose', `fetch: Installing ${target} to ${dest}`);
return superspawn.spawn('npm', args, { cwd: dest });
})
// Resolve path to installed package
.then(getTargetPackageSpecFromNpmInstallOutput)
.then(spec => pathToInstalledPackage(spec, dest));
}
function npmArgs (target, userOptions) {
const opts = Object.assign({ production: true }, userOptions);
const args = ['install', target];
if (opts.production) {
args.push('--production');
}
if (opts.save_exact) {
args.push('--save-exact');
} else if (opts.save) {
args.push('--save');
} else {
args.push('--no-save');
}
return args;
}
function getTargetPackageSpecFromNpmInstallOutput (npmInstallOutput) {
const packageInfoLine = npmInstallOutput.split('\n')
.find(line => line.startsWith('+ '));
if (!packageInfoLine) {
throw new CordovaError('Could not determine package name from output:\n' + npmInstallOutput);
}
return packageInfoLine.slice(2);
}
// Resolves to installation path of package defined by spec if the right version
// is installed, rejects otherwise.
function pathToInstalledPackage (spec, dest) {
return Promise.resolve().then(_ => {
const { name, rawSpec } = npa(spec, dest);
if (!name) {
throw new CordovaError(`Cannot determine package name from spec ${spec}`);
}
return resolvePathToPackage(name, dest)
.then(([pkgPath, { version }]) => {
if (!semver.satisfies(version, rawSpec)) {
throw new CordovaError(`Installed package ${name}@${version} does not satisfy ${name}@${rawSpec}`);
}
return pkgPath;
});
});
}
// Resolves to installation path and package.json of package `name` starting
// from `basedir`
function resolvePathToPackage (name, basedir) {
return Promise.resolve().then(_ => {
const paths = (process.env.NODE_PATH || '')
.split(path.delimiter)
.filter(p => p);
// We resolve the path to the module's package.json to avoid getting the
// path to `main` which could be located anywhere in the package
return resolve(path.join(name, 'package.json'), { paths, basedir })
.then(([pkgJsonPath, pkgJson]) => [
path.dirname(pkgJsonPath), pkgJson
]);
});
}
/**
* Checks to see if npm is installed on the users system
*
* @return {Promise<string>} Absolute path to npm.
*/
function isNpmInstalled () {
return which('npm').catch(_ => {
throw new CordovaError('"npm" command line tool is not installed: make sure it is accessible on your PATH.');
});
}
module.exports.isNpmInstalled = isNpmInstalled;
/**
* Uninstalls the package `target` from `dest` using given options.
*
* @param {String} target Name of the package to be uninstalled
* @param {String} dest Location from where to uninstall the package
* @param {Object} [opts={}] Additional options
*
* @return {Promise<string>} Resolves when removal has finished
*/
module.exports.uninstall = function (target, dest, opts) {
var fetchArgs = ['uninstall'];
opts = opts || {};
// check if npm is installed on the system
return isNpmInstalled()
.then(function () {
if (dest && target) {
// add target to fetchArgs Array
fetchArgs.push(target);
} else throw new CordovaError('Need to supply a target and destination');
// set the directory where npm uninstall will be run
opts.cwd = dest;
// if user added --save flag, pass it to npm uninstall command
if (opts.save) {
fetchArgs.push('--save');
} else {
fetchArgs.push('--no-save');
}
// run npm uninstall, this will remove dependency
// from package.json if --save was used.
return superspawn.spawn('npm', fetchArgs, opts);
})
.catch(function (err) {
throw new CordovaError(err);
});
};