2017-07-27 03:29:17 -04:00
|
|
|
const fs = require('fs-extra');
|
|
|
|
const glob = require('glob');
|
|
|
|
const path = require('canonical-path');
|
|
|
|
const shelljs = require('shelljs');
|
|
|
|
const yargs = require('yargs');
|
|
|
|
|
|
|
|
const SHARED_PATH = path.resolve(__dirname, 'shared');
|
|
|
|
const SHARED_NODE_MODULES_PATH = path.resolve(SHARED_PATH, 'node_modules');
|
|
|
|
const BOILERPLATE_BASE_PATH = path.resolve(SHARED_PATH, 'boilerplate');
|
2017-08-22 15:31:15 -04:00
|
|
|
const BOILERPLATE_COMMON_BASE_PATH = path.resolve(BOILERPLATE_BASE_PATH, 'common');
|
2017-07-27 03:29:17 -04:00
|
|
|
const EXAMPLES_BASE_PATH = path.resolve(__dirname, '../../content/examples');
|
|
|
|
|
2017-08-22 15:31:15 -04:00
|
|
|
const BOILERPLATE_PATHS = {
|
|
|
|
cli: [
|
|
|
|
'src/environments/environment.prod.ts',
|
|
|
|
'src/environments/environment.ts',
|
|
|
|
'src/assets/.gitkeep',
|
2018-04-12 22:20:01 -04:00
|
|
|
'src/browserslist',
|
2017-08-22 15:31:15 -04:00
|
|
|
'src/favicon.ico',
|
2018-04-12 22:20:01 -04:00
|
|
|
'src/karma.conf.js',
|
2017-08-22 15:31:15 -04:00
|
|
|
'src/polyfills.ts',
|
|
|
|
'src/test.ts',
|
|
|
|
'src/tsconfig.app.json',
|
|
|
|
'src/tsconfig.spec.json',
|
2018-04-12 22:20:01 -04:00
|
|
|
'src/tslint.json',
|
|
|
|
'e2e/src/app.po.ts',
|
|
|
|
'e2e/protractor.conf.js',
|
2017-08-22 15:31:15 -04:00
|
|
|
'e2e/tsconfig.e2e.json',
|
|
|
|
'.editorconfig',
|
2018-04-12 22:20:01 -04:00
|
|
|
'angular.json',
|
2017-08-22 15:31:15 -04:00
|
|
|
'package.json',
|
|
|
|
'tsconfig.json',
|
|
|
|
'tslint.json'
|
|
|
|
],
|
|
|
|
systemjs: [
|
|
|
|
'src/systemjs-angular-loader.js',
|
|
|
|
'src/systemjs.config.js',
|
|
|
|
'src/tsconfig.json',
|
|
|
|
'bs-config.json',
|
|
|
|
'bs-config.e2e.json',
|
|
|
|
'package.json',
|
|
|
|
'tslint.json'
|
|
|
|
],
|
|
|
|
common: [
|
|
|
|
'src/styles.css'
|
|
|
|
]
|
|
|
|
};
|
2017-07-27 03:29:17 -04:00
|
|
|
|
2017-10-29 07:59:32 -04:00
|
|
|
// All paths in this tool are relative to the current boilerplate folder, i.e boilerplate/i18n
|
|
|
|
// This maps the CLI files that exists in a parent folder
|
|
|
|
const cliRelativePath = BOILERPLATE_PATHS.cli.map(file => `../cli/${file}`);
|
|
|
|
|
2018-07-16 15:52:59 -04:00
|
|
|
BOILERPLATE_PATHS.elements = [
|
2017-10-29 07:59:32 -04:00
|
|
|
...cliRelativePath,
|
2018-07-16 15:52:59 -04:00
|
|
|
'tsconfig.json'
|
2017-10-29 07:59:32 -04:00
|
|
|
];
|
|
|
|
|
2018-07-16 15:52:59 -04:00
|
|
|
BOILERPLATE_PATHS.i18n = [
|
2017-10-30 18:39:58 -04:00
|
|
|
...cliRelativePath,
|
2018-05-02 08:13:32 -04:00
|
|
|
'angular.json',
|
2017-10-30 18:39:58 -04:00
|
|
|
'package.json'
|
|
|
|
];
|
|
|
|
|
2019-01-10 15:08:51 -05:00
|
|
|
BOILERPLATE_PATHS['service-worker'] = [
|
|
|
|
...cliRelativePath,
|
|
|
|
'angular.json',
|
|
|
|
'package.json'
|
|
|
|
];
|
|
|
|
|
2018-10-05 15:30:35 -04:00
|
|
|
BOILERPLATE_PATHS.testing = [
|
|
|
|
...cliRelativePath,
|
|
|
|
'angular.json'
|
|
|
|
];
|
|
|
|
|
2018-07-16 15:52:59 -04:00
|
|
|
BOILERPLATE_PATHS.universal = [
|
|
|
|
...cliRelativePath,
|
|
|
|
'angular.json',
|
|
|
|
'package.json'
|
|
|
|
];
|
|
|
|
|
2019-02-05 16:20:05 -05:00
|
|
|
BOILERPLATE_PATHS.ivy = {
|
|
|
|
systemjs: [
|
|
|
|
'rollup-config.js',
|
|
|
|
'tsconfig-aot.json'
|
|
|
|
],
|
|
|
|
cli: [
|
|
|
|
'src/tsconfig.app.json'
|
|
|
|
]
|
|
|
|
};
|
|
|
|
|
2017-07-27 03:29:17 -04:00
|
|
|
const EXAMPLE_CONFIG_FILENAME = 'example-config.json';
|
|
|
|
|
|
|
|
class ExampleBoilerPlate {
|
|
|
|
/**
|
|
|
|
* Add boilerplate files to all the examples
|
|
|
|
*/
|
2019-02-05 16:20:05 -05:00
|
|
|
add(ivy = false) {
|
2017-07-27 03:29:17 -04:00
|
|
|
// Get all the examples folders, indicated by those that contain a `example-config.json` file
|
|
|
|
const exampleFolders = this.getFoldersContaining(EXAMPLES_BASE_PATH, EXAMPLE_CONFIG_FILENAME, 'node_modules');
|
|
|
|
|
2019-02-05 16:20:05 -05:00
|
|
|
if (!fs.existsSync(SHARED_NODE_MODULES_PATH)) {
|
|
|
|
throw new Error(`The shared node_modules folder for the examples (${SHARED_NODE_MODULES_PATH}) is missing.\n` +
|
2017-10-06 09:48:37 -04:00
|
|
|
`Perhaps you need to run "yarn example-use-npm" or "yarn example-use-local" to install the dependencies?`);
|
2019-02-05 16:20:05 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
if (ivy) {
|
|
|
|
shelljs.exec(`yarn --cwd ${SHARED_PATH} ivy-ngcc`);
|
|
|
|
}
|
|
|
|
|
|
|
|
exampleFolders.forEach(exampleFolder => {
|
|
|
|
const exampleConfig = this.loadJsonFile(path.resolve(exampleFolder, EXAMPLE_CONFIG_FILENAME));
|
2017-10-06 09:48:37 -04:00
|
|
|
|
2017-07-27 03:29:17 -04:00
|
|
|
// Link the node modules - requires admin access (on Windows) because it adds symlinks
|
|
|
|
const destinationNodeModules = path.resolve(exampleFolder, 'node_modules');
|
|
|
|
fs.ensureSymlinkSync(SHARED_NODE_MODULES_PATH, destinationNodeModules);
|
|
|
|
|
2017-08-22 15:31:15 -04:00
|
|
|
const boilerPlateType = exampleConfig.projectType || 'cli';
|
|
|
|
const boilerPlateBasePath = path.resolve(BOILERPLATE_BASE_PATH, boilerPlateType);
|
2017-07-27 03:29:17 -04:00
|
|
|
|
2017-08-22 15:31:15 -04:00
|
|
|
// Copy the boilerplate specific files
|
|
|
|
BOILERPLATE_PATHS[boilerPlateType].forEach(filePath => this.copyFile(boilerPlateBasePath, exampleFolder, filePath));
|
|
|
|
|
|
|
|
// Copy the boilerplate common files
|
|
|
|
BOILERPLATE_PATHS.common.forEach(filePath => this.copyFile(BOILERPLATE_COMMON_BASE_PATH, exampleFolder, filePath));
|
2019-02-05 16:20:05 -05:00
|
|
|
|
|
|
|
// Copy Ivy specific files
|
|
|
|
if (ivy && BOILERPLATE_PATHS.ivy[boilerPlateType]) {
|
|
|
|
const ivyBoilerPlateBasePath = path.resolve(BOILERPLATE_BASE_PATH, 'ivy', boilerPlateType);
|
|
|
|
BOILERPLATE_PATHS.ivy[boilerPlateType].forEach(filePath => this.copyFile(ivyBoilerPlateBasePath, exampleFolder, filePath));
|
|
|
|
}
|
2017-07-27 03:29:17 -04:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Remove all the boilerplate files from all the examples
|
|
|
|
*/
|
|
|
|
remove() {
|
2019-02-05 16:20:05 -05:00
|
|
|
shelljs.exec('git clean -xdfq', { cwd: EXAMPLES_BASE_PATH });
|
2017-07-27 03:29:17 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
main() {
|
|
|
|
yargs
|
|
|
|
.usage('$0 <cmd> [args]')
|
2019-02-05 16:20:05 -05:00
|
|
|
.command('add', 'add the boilerplate to each example', (yrgs) => this.add(yrgs.argv.ivy))
|
2017-07-27 03:29:17 -04:00
|
|
|
.command('remove', 'remove the boilerplate from each example', () => this.remove())
|
|
|
|
.demandCommand(1, 'Please supply a command from the list above')
|
|
|
|
.argv;
|
|
|
|
}
|
|
|
|
|
|
|
|
getFoldersContaining(basePath, filename, ignore) {
|
|
|
|
const pattern = path.resolve(basePath, '**', filename);
|
|
|
|
const ignorePattern = path.resolve(basePath, '**', ignore, '**');
|
|
|
|
return glob.sync(pattern, { ignore: [ignorePattern] }).map(file => path.dirname(file));
|
|
|
|
}
|
|
|
|
|
|
|
|
copyFile(sourceFolder, destinationFolder, filePath) {
|
|
|
|
const sourcePath = path.resolve(sourceFolder, filePath);
|
2017-10-30 18:39:58 -04:00
|
|
|
|
|
|
|
// normalize path if needed
|
|
|
|
filePath = this.normalizePath(filePath);
|
|
|
|
|
2017-07-27 03:29:17 -04:00
|
|
|
const destinationPath = path.resolve(destinationFolder, filePath);
|
|
|
|
fs.copySync(sourcePath, destinationPath, { overwrite: true });
|
|
|
|
fs.chmodSync(destinationPath, 444);
|
|
|
|
}
|
|
|
|
|
|
|
|
loadJsonFile(filePath) {
|
2019-02-05 16:20:05 -05:00
|
|
|
return fs.readJsonSync(filePath, { throws: false }) || {};
|
2017-07-27 03:29:17 -04:00
|
|
|
}
|
2017-10-30 18:39:58 -04:00
|
|
|
|
|
|
|
normalizePath(filePath) {
|
|
|
|
// transform for example ../cli/src/tsconfig.app.json to src/tsconfig.app.json
|
|
|
|
return filePath.replace(/\.{2}\/\w+\//, '');
|
|
|
|
}
|
2017-07-27 03:29:17 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = new ExampleBoilerPlate();
|
|
|
|
|
|
|
|
// If this file was run directly then run the main function,
|
|
|
|
if (require.main === module) {
|
|
|
|
module.exports.main();
|
2017-08-22 15:31:15 -04:00
|
|
|
}
|