@ionic/cli
Version:
A tool for creating and developing Ionic Framework mobile apps.
122 lines (121 loc) • 5.01 kB
JavaScript
Object.defineProperty(exports, "__esModule", { value: true });
exports.locateHook = exports.removeHook = exports.addHook = exports.Hook = void 0;
const tslib_1 = require("tslib");
const utils_array_1 = require("@ionic/utils-array");
const utils_terminal_1 = require("@ionic/utils-terminal");
const debug_1 = require("debug");
const lodash = tslib_1.__importStar(require("lodash"));
const path = tslib_1.__importStar(require("path"));
const color_1 = require("./color");
const errors_1 = require("./errors");
const debug = (0, debug_1.debug)('ionic:lib:hooks');
class Hook {
constructor(e) {
this.e = e;
}
get script() {
return `ionic:${this.name}`;
}
async run(input) {
const { pkgManagerArgs } = await Promise.resolve().then(() => tslib_1.__importStar(require('./utils/npm')));
const type = this.e.project.type;
if (!type || !this.e.project.directory) {
return; // TODO: will we need hooks outside a project?
}
const [pkg] = await this.e.project.getPackageJson(undefined, { logErrors: false });
if (!pkg) {
return;
}
debug(`Looking for ${(0, color_1.ancillary)(this.script)} npm script.`);
const ctxEnvironment = this.generateCTXEnvironment(input);
if (pkg.scripts && pkg.scripts[this.script]) {
debug(`Invoking ${(0, color_1.ancillary)(this.script)} npm script.`);
const [pkgManager, ...pkgArgs] = await pkgManagerArgs(this.e.config.get('npmClient'), { command: 'run', script: this.script });
await this.e.shell.run(pkgManager, pkgArgs, {
env: ctxEnvironment,
});
}
const projectHooks = this.e.project.config.get('hooks');
const hooks = projectHooks ? (0, utils_array_1.conform)(projectHooks[this.name]) : [];
for (const h of hooks) {
const p = path.resolve(this.e.project.directory, h);
try {
if (path.extname(p) !== '.js') {
throw new Error(`Hooks must be .js files with a function for its default export.`);
}
const hook = await this.loadHookFn(p);
if (!hook) {
throw new Error(`Module must have a function for its default export.`);
}
await hook(lodash.assign({}, input, {
project: {
type,
dir: this.e.project.directory,
srcDir: await this.e.project.getSourceDir(),
},
argv: process.argv,
env: {
...process.env,
...ctxEnvironment,
},
}));
}
catch (e) {
throw new errors_1.HookException(`An error occurred while running an Ionic CLI hook defined in ${(0, color_1.strong)((0, utils_terminal_1.prettyPath)(this.e.project.filePath))}.\n` +
`Hook: ${(0, color_1.strong)(this.name)}\n` +
`File: ${(0, color_1.strong)(p)}\n\n` +
`${(0, color_1.failure)(e.stack ? e.stack : e)}`);
}
}
}
async loadHookFn(p) {
const module = require(p);
if (typeof module === 'function') {
return module;
}
else if (typeof module.default === 'function') {
return module.default;
}
debug(`Could not load hook function ${(0, color_1.strong)(p)}: %o not a function`, module);
}
generateCTXEnvironment(input, path = []) {
let environment = {};
for (const [key, value] of Object.entries(input)) {
if (typeof value === 'object') {
environment = {
...environment,
...this.generateCTXEnvironment(value, [...path, key]),
};
}
else {
const name = [...path, key].join('_');
environment[`IONIC_CLI_HOOK_CTX_${lodash.snakeCase(name)}`.toUpperCase()] = value;
}
}
return environment;
}
}
exports.Hook = Hook;
function addHook(baseDir, hooks, hook) {
const hookPaths = (0, utils_array_1.conform)(hooks);
const resolvedHookPaths = hookPaths.map(p => path.resolve(baseDir, p));
if (!resolvedHookPaths.includes(path.resolve(baseDir, hook))) {
hookPaths.push(hook);
}
return hookPaths;
}
exports.addHook = addHook;
function removeHook(baseDir, hooks, hook) {
const hookPaths = (0, utils_array_1.conform)(hooks);
const i = locateHook(baseDir, hookPaths, hook);
if (i >= 0) {
hookPaths.splice(i, 1);
}
return hookPaths;
}
exports.removeHook = removeHook;
function locateHook(baseDir, hooks, hook) {
return (0, utils_array_1.conform)(hooks).map(p => path.resolve(baseDir, p)).indexOf(path.resolve(baseDir, hook));
}
exports.locateHook = locateHook;
;