UNPKG

nodekit-cli-lib

Version:

Command-line helpers for NodeKit command line interface

130 lines (114 loc) 5.01 kB
/** Licensed to OffGrid Networks (OGN) under one or more contributor license agreements. See the NOTICE file distributed with this work for additional information regarding copyright ownership. OGN 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. */ // Helper methods to help keep npm operations separated. var npm = require('npm'), path = require('path'), Q = require('q'), unpack = require('./unpack'), util = require('../nodekit/util'), cachedSettings = null, cachedSettingsValues = null; /** * @description Calls npm.load, then initializes npm.config with the specified settings. Then executes a chain of * promises that rely on those npm settings, then restores npm settings back to their previous value. Use this rather * than passing settings to npm.load, since that only works the first time you try to load npm. * @param {Object} settings * @param {Function} promiseChain */ function loadWithSettingsThenRestore(settings, promiseChain) { return loadWithSettings(settings).then(promiseChain).finally(restoreSettings); } function loadWithSettings(settings) { if (cachedSettings) { throw new Error('Trying to initialize npm when settings have not been restored from a previous initialization.'); } return Q.nfcall(npm.load, settings).then(function () { for (var prop in settings) { var currentValue = npm.config.get(prop); var newValue = settings[prop]; if (currentValue !== newValue) { cachedSettingsValues = cachedSettingsValues || {}; cachedSettings = cachedSettings || []; cachedSettings.push(prop); if (typeof currentValue !== 'undefined') { cachedSettingsValues[prop] = currentValue; } npm.config.set(prop, newValue); } } }); } function restoreSettings() { if (cachedSettings) { cachedSettings.forEach(function (prop) { if (prop in cachedSettingsValues) { npm.config.set(prop, cachedSettingsValues[prop]); } else { npm.config.del(prop); } }); cachedSettings = null; cachedSettingsValues = null; } } /** * Fetches the latest version of a package from NPM that matches the specified version. Returns a promise that * resolves to the directory the NPM package is located in. * @param packageName - name of an npm package * @param packageVersion - requested version or version range */ function fetchPackage(packageName, packageVersion) { // Get the latest matching version from NPM if a version range is specified return util.getLatestMatchingNpmVersion(packageName, packageVersion).then( function (latestVersion) { return cachePackage(packageName, latestVersion); } ); } /** * Invokes "npm cache add," and then returns a promise that resolves to a directory containing the downloaded, * or cached package. * @param packageName - name of an npm package * @param packageVersion - requested version (not a version range) */ function cachePackage(packageName, packageVersion) { return Q().then(function () { var cacheDir = path.join(util.libDirectory, 'npm_cache'); // If already cached, use that rather than calling 'npm cache add' again. var packageCacheDir = path.resolve(cacheDir, packageName, packageVersion); var packageTGZ = path.resolve(packageCacheDir, 'package.tgz'); if (util.existsSync(packageTGZ)) { return unpack.unpackTgz(packageTGZ, path.resolve(packageCacheDir, 'package')); } // Load with NPM configuration return loadWithSettingsThenRestore({'cache': cacheDir}, function () { // Invoke NPM Cache Add return Q.ninvoke(npm.commands, 'cache', ['add', (packageName + '@' + packageVersion)]).then( function (info) { var packageDir = path.resolve(npm.cache, info.name, info.version, 'package'); var packageTGZ = path.resolve(npm.cache, info.name, info.version, 'package.tgz'); return unpack.unpackTgz(packageTGZ, packageDir); } ); } ); }); } module.exports.loadWithSettingsThenRestore = loadWithSettingsThenRestore; module.exports.fetchPackage = fetchPackage; module.exports.cachePackage = cachePackage;