UNPKG

ionic

Version:

A tool for creating and developing Ionic Framework mobile apps.

267 lines (236 loc) • 9.39 kB
/** 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; }