ionic
Version:
A tool for creating and developing Ionic Framework mobile apps.
267 lines (236 loc) • 9.39 kB
JavaScript
/**
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.
*/
var cordovaUtil = require('../cordova/util'),
events = require('cordova-common').events,
Q = require('q'),
scriptsFinder = require('./scriptsFinder'),
Context = require('./Context'),
CordovaError = require('cordova-common').CordovaError,
path = require('path'),
fs = require('fs'),
os = require('os'),
superspawn = require('cordova-common').superspawn;
var isWindows = os.platform().slice(0, 3) === 'win';
/**
* Tries to create a HooksRunner for passed project root.
* @constructor
*/
function HooksRunner(projectRoot) {
var root = cordovaUtil.isCordova(projectRoot);
if (!root) throw new CordovaError('Not a Cordova project ("' + projectRoot + '"), can\'t use hooks.');
else this.projectRoot = root;
}
/**
* Fires all event handlers and scripts for a passed hook type.
* Returns a promise.
*/
HooksRunner.prototype.fire = function fire(hook, opts) {
if (isHookDisabled(opts, hook)) {
return Q('hook '+hook+' is disabled.');
}
// args check
if (!hook) {
throw new Error('hook type is not specified');
}
opts = this.prepareOptions(opts);
var handlers = events.listeners(hook);
var scripts = scriptsFinder.getHookScripts(hook, opts);
// CB-10193 Emit warning if there is any handlers subscribed to 'pre_package'
if (hook === 'pre_package' && (handlers.length > 0 || scripts.length > 0)) {
events.emit('warn', '"pre_package" hook is deprecated and will be removed in next Windows platform versions. ' +
'Please use "after_prepare" if you want to manipulate files in www before app will be packaged.');
}
// execute hook event listeners first
return executeEventHandlersSerially(hook, opts).then(function() {
// then execute hook script files
var context = new Context(hook, opts);
return runScriptsSerially(scripts, context);
});
};
/**
* Refines passed options so that all required parameters are set.
*/
HooksRunner.prototype.prepareOptions = function(opts) {
opts = opts || {};
opts.projectRoot = this.projectRoot;
opts.cordova = opts.cordova || {};
opts.cordova.platforms = opts.cordova.platforms || opts.platforms || cordovaUtil.listPlatforms(opts.projectRoot);
opts.cordova.platforms = opts.cordova.platforms.map(function(platform) { return platform.split('@')[0]; } );
opts.cordova.plugins = opts.cordova.plugins || opts.plugins || cordovaUtil.findPlugins(path.join(opts.projectRoot, 'plugins'));
try {
opts.cordova.version = opts.cordova.version || require('../../package').version;
} catch(ex) {
events.emit('warn', 'HooksRunner could not load package.json: ' + ex.message);
}
return opts;
};
module.exports = HooksRunner;
/**
* Executes hook event handlers serially. Doesn't require a HooksRunner to be constructed.
* Returns a promise.
*/
module.exports.fire = globalFire;
function globalFire(hook, opts) {
if (isHookDisabled(opts, hook)) {
return Q('hook '+hook+' is disabled.');
}
opts = opts || {};
return executeEventHandlersSerially(hook, opts);
}
// Returns a promise.
function executeEventHandlersSerially(hook, opts) {
var handlers = events.listeners(hook);
if (handlers.length) {
// Chain the handlers in series.
return handlers.reduce(function(soFar, f) {
return soFar.then(function() { return f(opts); });
}, Q());
} else {
return Q(); // Nothing to do.
}
}
/**
* Serially fires scripts either via Q(require(pathToScript)(context)) or via child_process.spawn.
* Returns promise.
*/
function runScriptsSerially (scripts, context) {
return scripts.reduce(function(prevScriptPromise, nextScript) {
return prevScriptPromise.then(function() {
return runScript(nextScript, context);
});
}, Q());
}
/**
* Async runs single script file.
*/
function runScript(script, context) {
if (typeof script.useModuleLoader == 'undefined') {
// if it is not explicitly defined whether we should use modeule loader or not
// we assume we should use module loader for .js files
script.useModuleLoader = path.extname(script.path).toLowerCase() == '.js';
}
if(script.useModuleLoader) {
return runScriptViaModuleLoader(script, context);
} else {
return runScriptViaChildProcessSpawn(script, context);
}
}
/**
* Runs script using require.
* Returns a promise. */
function runScriptViaModuleLoader(script, context) {
if(!fs.existsSync(script.fullPath)) {
events.emit('warn', 'Script file does\'t exist and will be skipped: ' + script.fullPath);
return Q();
}
var scriptFn = require(script.fullPath);
context.scriptLocation = script.fullPath;
context.opts.plugin = script.plugin;
// We can't run script if it is a plain Node script - it will run its commands when we require it.
// This is not a desired case as we want to pass context, but added for compatibility.
if (scriptFn instanceof Function) {
// If hook is async it can return promise instance and we will handle it.
return Q(scriptFn(context));
} else {
return Q();
}
}
/**
* Runs script using child_process spawn method.
* Returns a promise. */
function runScriptViaChildProcessSpawn(script, context) {
var opts = context.opts;
var command = script.fullPath;
var args = [opts.projectRoot];
if (fs.statSync(script.fullPath).isDirectory()) {
events.emit('verbose', 'skipped directory "' + script.fullPath + '" within hook directory');
return Q();
}
if (isWindows) {
// TODO: Make shebang sniffing a setting (not everyone will want this).
var interpreter = extractSheBangInterpreter(script.fullPath);
// we have shebang, so try to run this script using correct interpreter
if (interpreter) {
args.unshift(command);
command = interpreter;
}
}
var execOpts = {cwd: opts.projectRoot, printCommand: true, stdio: 'inherit'};
execOpts.env = {};
execOpts.env.CORDOVA_VERSION = require('../../package').version;
execOpts.env.CORDOVA_PLATFORMS = opts.platforms ? opts.platforms.join() : '';
execOpts.env.CORDOVA_PLUGINS = opts.plugins ? opts.plugins.join() : '';
execOpts.env.CORDOVA_HOOK = script.fullPath;
execOpts.env.CORDOVA_CMDLINE = process.argv.join(' ');
return superspawn.spawn(command, args, execOpts)
.catch(function(err) {
// Don't treat non-executable files as errors. They could be READMEs, or Windows-only scripts.
if (!isWindows && err.code == 'EACCES') {
events.emit('verbose', 'skipped non-executable file: ' + script.fullPath);
} else {
throw new Error('Hook failed with error code ' + err.code + ': ' + script.fullPath);
}
});
}
/**
* Extracts shebang interpreter from script' source. */
function extractSheBangInterpreter(fullpath) {
var fileChunk;
var octetsRead;
var fileData;
var hookFd = fs.openSync(fullpath, 'r');
try {
// this is a modern cluster size. no need to read less
fileData = new Buffer(4096);
octetsRead = fs.readSync(hookFd, fileData, 0, 4096, 0);
fileChunk = fileData.toString();
} finally {
fs.closeSync(hookFd);
}
var hookCmd, shMatch;
// Filter out /usr/bin/env so that "/usr/bin/env node" works like "node".
var shebangMatch = fileChunk.match(/^#!(?:\/usr\/bin\/env )?([^\r\n]+)/m);
if (octetsRead == 4096 && !fileChunk.match(/[\r\n]/))
events.emit('warn', 'shebang is too long for "' + fullpath + '"');
if (shebangMatch)
hookCmd = shebangMatch[1];
// Likewise, make /usr/bin/bash work like "bash".
if (hookCmd)
shMatch = hookCmd.match(/bin\/((?:ba)?sh)$/);
if (shMatch)
hookCmd = shMatch[1];
return hookCmd;
}
/**
* Checks if the given hook type is disabled at the command line option.
* @param {Object} opts - the option object that contains command line options
* @param {String} hook - the hook type
* @returns {Boolean} return true if the passed hook is disabled.
*/
function isHookDisabled(opts, hook) {
if (opts === undefined || opts.nohooks === undefined) {
return false;
}
var disabledHooks = opts.nohooks;
var length = disabledHooks.length;
for (var i=0; i<length; i++) {
if (hook.match(disabledHooks[i]) !== null) {
return true;
}
}
return false;
}