mirror of
https://github.com/angular/angular-cli.git
synced 2025-05-21 14:02:43 +08:00
607 lines
16 KiB
JavaScript
607 lines
16 KiB
JavaScript
'use strict';
|
|
|
|
/**
|
|
@module ember-cli
|
|
*/
|
|
const denodeify = require('denodeify');
|
|
const path = require('path');
|
|
const findUp = require('../../../utilities/find-up').findUp;
|
|
let resolve = denodeify(require('resolve'));
|
|
const fs = require('fs-extra');
|
|
const _ = require('lodash');
|
|
let logger = require('heimdalljs-logger')('ember-cli:project');
|
|
const nodeModulesPath = require('node-modules-path');
|
|
const heimdall = require('heimdalljs');
|
|
|
|
let processCwd = process.cwd();
|
|
// ensure NULL_PROJECT is a singleton
|
|
let NULL_PROJECT;
|
|
|
|
class Project {
|
|
/**
|
|
The Project model is tied to your package.json. It is instantiated
|
|
by giving {{#crossLink "Project/closestSync:method"}}{{/crossLink}}
|
|
the path to your project.
|
|
@class Project
|
|
@constructor
|
|
@param {String} root Root directory for the project
|
|
@param {Object} pkg Contents of package.json
|
|
@param {UI} ui
|
|
@param {CLI} cli
|
|
*/
|
|
constructor(root, pkg, ui, cli) {
|
|
logger.info('init root: %s', root);
|
|
|
|
this.root = root;
|
|
this.pkg = pkg;
|
|
this.ui = ui;
|
|
this.cli = cli;
|
|
this.addonPackages = {};
|
|
this.addons = [];
|
|
this.liveReloadFilterPatterns = [];
|
|
this.setupNodeModulesPath();
|
|
|
|
/**
|
|
Set when the `Watcher.detectWatchman` helper method finishes running,
|
|
so that other areas of the system can be aware that watchman is being used.
|
|
For example, this information is used in the broccoli build pipeline to know
|
|
if we can watch additional directories (like bower_components) "cheaply".
|
|
Contains `enabled` and `version`.
|
|
@private
|
|
@property _watchmanInfo
|
|
@return {Object}
|
|
@default false
|
|
*/
|
|
this._watchmanInfo = {
|
|
enabled: false,
|
|
version: null,
|
|
canNestRoots: false,
|
|
};
|
|
|
|
let instrumentation = this._instrumentation = ensureInstrumentation(cli, ui);
|
|
if (instrumentation) {
|
|
instrumentation.project = this;
|
|
}
|
|
}
|
|
|
|
hasDependencies() {
|
|
return !!this.nodeModulesPath;
|
|
}
|
|
|
|
/**
|
|
Sets the path to the node_modules directory for this
|
|
project.
|
|
@private
|
|
@method setupNodeModulesPath
|
|
*/
|
|
setupNodeModulesPath() {
|
|
this.nodeModulesPath = nodeModulesPath(this.root);
|
|
|
|
logger.info('nodeModulesPath: %s', this.nodeModulesPath);
|
|
}
|
|
|
|
static nullProject(ui, cli) {
|
|
if (NULL_PROJECT) { return NULL_PROJECT; }
|
|
|
|
NULL_PROJECT = new Project(processCwd, {}, ui, cli);
|
|
|
|
NULL_PROJECT.isEmberCLIProject = function() {
|
|
return false;
|
|
};
|
|
|
|
NULL_PROJECT.isEmberCLIAddon = function() {
|
|
return false;
|
|
};
|
|
|
|
NULL_PROJECT.name = function() {
|
|
return path.basename(process.cwd());
|
|
};
|
|
|
|
NULL_PROJECT.initializeAddons();
|
|
|
|
return NULL_PROJECT;
|
|
}
|
|
|
|
/**
|
|
Returns the name from package.json.
|
|
@private
|
|
@method name
|
|
@return {String} Package name
|
|
*/
|
|
name() {
|
|
const getPackageBaseName = require('../utilities/get-package-base-name');
|
|
|
|
return getPackageBaseName(this.pkg.name);
|
|
}
|
|
|
|
/**
|
|
Returns whether or not this is an Ember CLI project.
|
|
This checks whether ember-cli is listed in devDependencies.
|
|
@private
|
|
@method isEmberCLIProject
|
|
@return {Boolean} Whether this is an Ember CLI project
|
|
*/
|
|
isEmberCLIProject() {
|
|
return 'angular-cli' in this.dependencies()
|
|
|| '@angular/cli' in this.dependencies();
|
|
}
|
|
|
|
/**
|
|
Returns whether or not this is an Ember CLI addon.
|
|
@method isEmberCLIAddon
|
|
@return {Boolean} Whether or not this is an Ember CLI Addon.
|
|
*/
|
|
isEmberCLIAddon() {
|
|
return !!this.pkg.keywords && this.pkg.keywords.indexOf('ember-addon') > -1;
|
|
}
|
|
|
|
/**
|
|
Loads the configuration for this project and its addons.
|
|
@public
|
|
@method config
|
|
@param {String} env Environment name
|
|
@return {Object} Merged confiration object
|
|
*/
|
|
config(env) {
|
|
this.initializeAddons();
|
|
|
|
let initialConfig = {};
|
|
|
|
return this.addons.reduce((config, addon) => {
|
|
if (addon.config) {
|
|
_.merge(config, addon.config(env, config));
|
|
}
|
|
|
|
return config;
|
|
}, initialConfig);
|
|
}
|
|
|
|
/**
|
|
Returns whether or not the given file name is present in this project.
|
|
@private
|
|
@method has
|
|
@param {String} file File name
|
|
@return {Boolean} Whether or not the file is present
|
|
*/
|
|
has(file) {
|
|
return fs.existsSync(path.join(this.root, file)) || fs.existsSync(path.join(this.root, `${file}.js`));
|
|
}
|
|
|
|
/**
|
|
Resolves the absolute path to a file.
|
|
@private
|
|
@method resolve
|
|
@param {String} file File to resolve
|
|
@return {String} Absolute path to file
|
|
*/
|
|
resolve(file) {
|
|
return resolve(file, {
|
|
basedir: this.root,
|
|
});
|
|
}
|
|
|
|
/**
|
|
Resolves the absolute path to a file synchronously
|
|
@private
|
|
@method resolveSync
|
|
@param {String} file File to resolve
|
|
@return {String} Absolute path to file
|
|
*/
|
|
resolveSync(file) {
|
|
return resolve.sync(file, {
|
|
basedir: this.root,
|
|
});
|
|
}
|
|
|
|
/**
|
|
Calls `require` on a given module from the context of the project. For
|
|
instance, an addon may want to require a class from the root project's
|
|
version of ember-cli.
|
|
@public
|
|
@method require
|
|
@param {String} file File path or module name
|
|
@return {Object} Imported module
|
|
*/
|
|
require(file) {
|
|
if (/^\.\//.test(file)) { // Starts with ./
|
|
return require(path.join(this.root, file));
|
|
} else if (file.slice(0, this.root.length) === this.root) { // Starts with this.root
|
|
return require(file);
|
|
} else {
|
|
return require(path.join(this.nodeModulesPath, file));
|
|
}
|
|
}
|
|
|
|
/**
|
|
Returns the dependencies from a package.json
|
|
@private
|
|
@method dependencies
|
|
@param {Object} [pkg=this.pkg] Package object
|
|
@param {Boolean} [excludeDevDeps=false] Whether or not development dependencies should be excluded
|
|
@return {Object} Dependencies
|
|
*/
|
|
dependencies(pkg, excludeDevDeps) {
|
|
pkg = pkg || this.pkg || {};
|
|
|
|
let devDependencies = pkg['devDependencies'];
|
|
if (excludeDevDeps) {
|
|
devDependencies = {};
|
|
}
|
|
|
|
return _.assign({}, devDependencies, pkg['dependencies']);
|
|
}
|
|
|
|
/**
|
|
Provides the list of paths to consult for addons that may be provided
|
|
internally to this project. Used for middleware addons with built-in support.
|
|
@private
|
|
@method supportedInternalAddonPaths
|
|
*/
|
|
supportedInternalAddonPaths() {
|
|
if (!this.root) { return []; }
|
|
|
|
let internalMiddlewarePath = path.join(__dirname, '../tasks/server/middleware');
|
|
let legacyBlueprintsPath = require.resolve('ember-cli-legacy-blueprints');
|
|
let emberTryPath = require.resolve('ember-try');
|
|
return [
|
|
path.join(internalMiddlewarePath, 'testem-url-rewriter'),
|
|
path.join(internalMiddlewarePath, 'tests-server'),
|
|
path.join(internalMiddlewarePath, 'history-support'),
|
|
path.join(internalMiddlewarePath, 'broccoli-watcher'),
|
|
path.join(internalMiddlewarePath, 'broccoli-serve-files'),
|
|
path.join(internalMiddlewarePath, 'proxy-server'),
|
|
path.dirname(legacyBlueprintsPath),
|
|
path.dirname(emberTryPath),
|
|
];
|
|
}
|
|
|
|
/**
|
|
Loads and initializes all addons for this project.
|
|
@private
|
|
@method initializeAddons
|
|
*/
|
|
initializeAddons() {
|
|
if (this._addonsInitialized) {
|
|
return;
|
|
}
|
|
this._addonsInitialized = true;
|
|
|
|
logger.info('initializeAddons for: %s', this.name());
|
|
}
|
|
|
|
/**
|
|
Returns what commands are made available by addons by inspecting
|
|
`includedCommands` for every addon.
|
|
@private
|
|
@method addonCommands
|
|
@return {Object} Addon names and command maps as key-value pairs
|
|
*/
|
|
addonCommands() {
|
|
const Command = require('../models/command');
|
|
let commands = {};
|
|
this.addons.forEach(addon => {
|
|
if (!addon.includedCommands) { return; }
|
|
|
|
let token = heimdall.start({
|
|
name: `lookup-commands: ${addon.name}`,
|
|
addonName: addon.name,
|
|
addonCommandInitialization: true,
|
|
});
|
|
|
|
let includedCommands = addon.includedCommands();
|
|
let addonCommands = {};
|
|
|
|
for (let key in includedCommands) {
|
|
if (typeof includedCommands[key] === 'function') {
|
|
addonCommands[key] = includedCommands[key];
|
|
} else {
|
|
addonCommands[key] = Command.extend(includedCommands[key]);
|
|
}
|
|
}
|
|
if (Object.keys(addonCommands).length) {
|
|
commands[addon.name] = addonCommands;
|
|
}
|
|
|
|
token.stop();
|
|
});
|
|
return commands;
|
|
}
|
|
|
|
/**
|
|
Execute a given callback for every addon command.
|
|
Example:
|
|
```
|
|
project.eachAddonCommand(function(addonName, commands) {
|
|
console.log('Addon ' + addonName + ' exported the following commands:' + commands.keys().join(', '));
|
|
});
|
|
```
|
|
@private
|
|
@method eachAddonCommand
|
|
@param {Function} callback [description]
|
|
*/
|
|
eachAddonCommand(callback) {
|
|
if (this.initializeAddons && this.addonCommands) {
|
|
this.initializeAddons();
|
|
let addonCommands = this.addonCommands();
|
|
|
|
_.forOwn(addonCommands, (commands, addonName) => callback(addonName, commands));
|
|
}
|
|
}
|
|
|
|
/**
|
|
Path to the blueprints for this project.
|
|
@private
|
|
@method localBlueprintLookupPath
|
|
@return {String} Path to blueprints
|
|
*/
|
|
localBlueprintLookupPath() {
|
|
return path.join(this.cli.root, 'blueprints');
|
|
}
|
|
|
|
/**
|
|
Returns a list of paths (including addon paths) where blueprints will be looked up.
|
|
@private
|
|
@method blueprintLookupPaths
|
|
@return {Array} List of paths
|
|
*/
|
|
blueprintLookupPaths() {
|
|
return [this.localBlueprintLookupPath()];
|
|
}
|
|
|
|
/**
|
|
Returns a list of addon paths where blueprints will be looked up.
|
|
@private
|
|
@method addonBlueprintLookupPaths
|
|
@return {Array} List of paths
|
|
*/
|
|
addonBlueprintLookupPaths() {
|
|
let addonPaths = this.addons.map(addon => {
|
|
if (addon.blueprintsPath) {
|
|
return addon.blueprintsPath();
|
|
}
|
|
}, this);
|
|
|
|
return addonPaths.filter(Boolean).reverse();
|
|
}
|
|
|
|
/**
|
|
Reloads package.json
|
|
@private
|
|
@method reloadPkg
|
|
@return {Object} Package content
|
|
*/
|
|
reloadPkg() {
|
|
let pkgPath = path.join(this.root, 'package.json');
|
|
|
|
// We use readFileSync instead of require to avoid the require cache.
|
|
this.pkg = fs.readJsonSync(pkgPath);
|
|
|
|
return this.pkg;
|
|
}
|
|
|
|
/**
|
|
Re-initializes addons.
|
|
@private
|
|
@method reloadAddons
|
|
*/
|
|
reloadAddons() {
|
|
this.reloadPkg();
|
|
this._addonsInitialized = false;
|
|
return this.initializeAddons();
|
|
}
|
|
|
|
/**
|
|
Find an addon by its name
|
|
@private
|
|
@method findAddonByName
|
|
@param {String} name Addon name as specified in package.json
|
|
@return {Addon} Addon instance
|
|
*/
|
|
findAddonByName(name) {
|
|
this.initializeAddons();
|
|
|
|
var exactMatch = _.find(this.addons, function(addon) {
|
|
return name === addon.name || (addon.pkg && name === addon.pkg.name);
|
|
});
|
|
|
|
if (exactMatch) {
|
|
return exactMatch;
|
|
}
|
|
|
|
return _.find(this.addons, function(addon) {
|
|
return name.indexOf(addon.name) > -1 || (addon.pkg && name.indexOf(addon.pkg.name) > -1);
|
|
});
|
|
}
|
|
|
|
/**
|
|
Generate test file contents.
|
|
This method is supposed to be overwritten by test framework addons
|
|
like `ember-cli-qunit` and `ember-cli-mocha`.
|
|
@public
|
|
@method generateTestFile
|
|
@param {String} moduleName Name of the test module (e.g. `JSHint`)
|
|
@param {Object[]} tests Array of tests with `name`, `passed` and `errorMessage` properties
|
|
@return {String} The test file content
|
|
*/
|
|
generateTestFile() {
|
|
let message = 'Please install an Ember.js test framework addon or update your dependencies.';
|
|
|
|
if (this.ui) {
|
|
this.ui.writeDeprecateLine(message);
|
|
} else {
|
|
console.warn(message);
|
|
}
|
|
|
|
return '';
|
|
}
|
|
|
|
/**
|
|
Returns a new project based on the first package.json that is found
|
|
in `pathName`.
|
|
@deprecated
|
|
@private
|
|
@static
|
|
@method closest
|
|
@param {String} pathName Path to your project
|
|
@return {Promise} Promise which resolves to a {Project}
|
|
*/
|
|
static closest(pathName, _ui, _cli) {
|
|
let ui = ensureUI(_ui);
|
|
|
|
ui.writeDeprecateLine('`Project.closest` is a private method that will be removed, please use `Project.closestSync` instead.');
|
|
|
|
return closestPackageJSON(pathName).then(result => {
|
|
logger.info('closest %s -> %s', pathName, result);
|
|
if (result.pkg && result.pkg.name === 'ember-cli') {
|
|
return Project.nullProject(_ui, _cli);
|
|
}
|
|
|
|
return new Project(result.directory, result.pkg, ui, _cli);
|
|
});
|
|
}
|
|
|
|
/**
|
|
Returns a new project based on the first package.json that is found
|
|
in `pathName`.
|
|
@private
|
|
@static
|
|
@method closestSync
|
|
@param {String} pathName Path to your project
|
|
@param {UI} _ui The UI instance to provide to the created Project.
|
|
@return {Project} Project instance
|
|
*/
|
|
static closestSync(pathName, _ui, _cli) {
|
|
logger.info('looking for package.json starting at %s', pathName);
|
|
|
|
let ui = ensureUI(_ui);
|
|
|
|
let directory = findupPath(pathName);
|
|
logger.info('found package.json at %s', directory);
|
|
|
|
let relative = path.relative(directory, pathName);
|
|
if (relative.indexOf('tmp') === 0) {
|
|
logger.info('ignoring parent project since we are in the tmp folder of the project');
|
|
return Project.nullProject(_ui, _cli);
|
|
}
|
|
|
|
let pkg = fs.readJsonSync(path.join(directory, 'package.json'));
|
|
logger.info('project name: %s', pkg && pkg.name);
|
|
|
|
if (!isEmberCliProject(pkg)) {
|
|
logger.info('ignoring parent project since it is not an Angular CLI project');
|
|
return Project.nullProject(_ui, _cli);
|
|
}
|
|
|
|
return new Project(directory, pkg, ui, _cli);
|
|
}
|
|
|
|
/**
|
|
Returns a new project based on the first package.json that is found
|
|
in `pathName`, or the nullProject.
|
|
The nullProject signifies no-project, but abides by the null object pattern
|
|
@private
|
|
@static
|
|
@method projectOrnullProject
|
|
@param {UI} _ui The UI instance to provide to the created Project.
|
|
@return {Project} Project instance
|
|
*/
|
|
static projectOrnullProject(_ui, _cli) {
|
|
try {
|
|
return Project.closestSync(process.cwd(), _ui, _cli);
|
|
} catch (reason) {
|
|
if (reason instanceof Project.NotFoundError) {
|
|
return Project.nullProject(_ui, _cli);
|
|
} else {
|
|
throw reason;
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
Returns the project root based on the first package.json that is found
|
|
@static
|
|
@method getProjectRoot
|
|
@return {String} The project root directory
|
|
*/
|
|
static getProjectRoot() {
|
|
let packagePath = findUp(process.cwd(), 'package.json');
|
|
if (!packagePath) {
|
|
logger.info('getProjectRoot: not found. Will use cwd: %s', process.cwd());
|
|
return process.cwd();
|
|
}
|
|
|
|
let directory = path.dirname(packagePath);
|
|
const pkg = require(packagePath);
|
|
|
|
if (pkg && pkg.name === 'ember-cli') {
|
|
logger.info('getProjectRoot: named \'ember-cli\'. Will use cwd: %s', process.cwd());
|
|
return process.cwd();
|
|
}
|
|
|
|
logger.info('getProjectRoot %s -> %s', process.cwd(), directory);
|
|
return directory;
|
|
}
|
|
}
|
|
|
|
class NotFoundError extends Error {
|
|
constructor(message) {
|
|
super(message);
|
|
this.name = 'NotFoundError';
|
|
this.stack = (new Error()).stack;
|
|
}
|
|
}
|
|
|
|
Project.NotFoundError = NotFoundError;
|
|
|
|
function ensureInstrumentation(cli, ui) {
|
|
if (cli && cli.instrumentation) {
|
|
return cli.instrumentation;
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
function ensureUI(_ui) {
|
|
let ui = _ui;
|
|
|
|
if (!ui) {
|
|
// TODO: one UI (lib/cli/index.js also has one for now...)
|
|
const UI = require('../ui');
|
|
ui = new UI({
|
|
inputStream: process.stdin,
|
|
outputStream: process.stdout,
|
|
ci: process.env.CI || (/^(dumb|emacs)$/).test(process.env.TERM),
|
|
writeLevel: (process.argv.indexOf('--silent') !== -1) ? 'ERROR' : undefined,
|
|
});
|
|
}
|
|
|
|
return ui;
|
|
}
|
|
|
|
function closestPackageJSON(pathName) {
|
|
return Promise.resolve()
|
|
.then(() => findUp('package.json', pathName))
|
|
.then(filePath => ({
|
|
directory: path.dirname(filePath),
|
|
pkg: require(filePath)
|
|
}));
|
|
}
|
|
|
|
function findupPath(pathName) {
|
|
try {
|
|
return path.dirname(findUp('package.json', pathName));
|
|
} catch (reason) {
|
|
throw new NotFoundError(`No project found at or up from: \`${pathName}\``);
|
|
}
|
|
}
|
|
|
|
function isEmberCliProject(pkg) {
|
|
return pkg && (
|
|
(pkg.dependencies && Object.keys(pkg.dependencies).indexOf('@angular/cli') !== -1) ||
|
|
(pkg.devDependencies && Object.keys(pkg.devDependencies).indexOf('@angular/cli') !== -1)
|
|
);
|
|
}
|
|
|
|
// Export
|
|
module.exports = Project;
|