177 lines
5.4 KiB
TypeScript

/**
* @license
* Copyright Google Inc. All Rights Reserved.
*
* Use of this source code is governed by an MIT-style license that can be
* found in the LICENSE file at https://angular.io/license
*/
import { strings, tags } from '@angular-devkit/core';
import { experimental } from '@angular-devkit/core';
import {
Rule,
SchematicContext,
SchematicsException,
Tree,
apply,
chain,
mergeWith,
move,
template,
url,
} from '@angular-devkit/schematics';
import { getWorkspace, getWorkspacePath } from '../utility/config';
import { Schema as E2eOptions } from './schema';
type WorkspaceSchema = experimental.workspace.WorkspaceSchema;
// TODO: use JsonAST
// function appendPropertyInAstObject(
// recorder: UpdateRecorder,
// node: JsonAstObject,
// propertyName: string,
// value: JsonValue,
// indent = 4,
// ) {
// const indentStr = '\n' + new Array(indent + 1).join(' ');
// if (node.properties.length > 0) {
// // Insert comma.
// const last = node.properties[node.properties.length - 1];
// recorder.insertRight(last.start.offset + last.text.replace(/\s+$/, '').length, ',');
// }
// recorder.insertLeft(
// node.end.offset - 1,
// ' '
// + `"${propertyName}": ${JSON.stringify(value, null, 2).replace(/\n/g, indentStr)}`
// + indentStr.slice(0, -2),
// );
// }
function addAppToWorkspaceFile(options: E2eOptions, workspace: WorkspaceSchema): Rule {
return (host: Tree, context: SchematicContext) => {
// TODO: use JsonAST
// const workspacePath = '/angular.json';
// const workspaceBuffer = host.read(workspacePath);
// if (workspaceBuffer === null) {
// throw new SchematicsException(`Configuration file (${workspacePath}) not found.`);
// }
// const workspaceJson = parseJson(workspaceBuffer.toString());
// if (workspaceJson.value === null) {
// throw new SchematicsException(`Unable to parse configuration file (${workspacePath}).`);
// }
let projectRoot = options.projectRoot !== undefined
? options.projectRoot
: `${workspace.newProjectRoot}/${options.name}`;
if (projectRoot !== '' && !projectRoot.endsWith('/')) {
projectRoot += '/';
}
// tslint:disable-next-line:no-any
const project: any = {
root: projectRoot,
projectType: 'application',
targets: {
e2e: {
builder: '@angular-devkit/build-angular:protractor',
options: {
protractorConfig: `${projectRoot}protractor.conf.js`,
devServerTarget: `${options.relatedAppName}:serve`,
},
configurations: {
production: {
devServerTarget: `${options.relatedAppName}:serve:production`,
},
},
},
lint: {
builder: '@angular-devkit/build-angular:tslint',
options: {
tsConfig: `${projectRoot}tsconfig.e2e.json`,
exclude: [
'**/node_modules/**',
],
},
},
},
};
// tslint:disable-next-line:no-any
// const projects: JsonObject = (<any> workspaceAst.value).projects || {};
// tslint:disable-next-line:no-any
// if (!(<any> workspaceAst.value).projects) {
// // tslint:disable-next-line:no-any
// (<any> workspaceAst.value).projects = projects;
// }
// TODO: throw if the project already exist.
workspace.projects[options.name] = project;
host.overwrite(getWorkspacePath(host), JSON.stringify(workspace, null, 2));
};
}
const projectNameRegexp = /^[a-zA-Z][.0-9a-zA-Z]*(-[.0-9a-zA-Z]*)*$/;
const unsupportedProjectNames = ['test', 'ember', 'ember-cli', 'vendor', 'app'];
function getRegExpFailPosition(str: string): number | null {
const parts = str.indexOf('-') >= 0 ? str.split('-') : [str];
const matched: string[] = [];
parts.forEach(part => {
if (part.match(projectNameRegexp)) {
matched.push(part);
}
});
const compare = matched.join('-');
return (str !== compare) ? compare.length : null;
}
function validateProjectName(projectName: string) {
const errorIndex = getRegExpFailPosition(projectName);
if (errorIndex !== null) {
const firstMessage = tags.oneLine`
Project name "${projectName}" is not valid. New project names must
start with a letter, and must contain only alphanumeric characters or dashes.
When adding a dash the segment after the dash must also start with a letter.
`;
const msg = tags.stripIndent`
${firstMessage}
${projectName}
${Array(errorIndex + 1).join(' ') + '^'}
`;
throw new SchematicsException(msg);
} else if (unsupportedProjectNames.indexOf(projectName) !== -1) {
throw new SchematicsException(`Project name "${projectName}" is not a supported name.`);
}
}
export default function (options: E2eOptions): Rule {
return (host: Tree) => {
validateProjectName(options.name);
const workspace = getWorkspace(host);
let newProjectRoot = workspace.newProjectRoot;
let appDir = `${newProjectRoot}/${options.name}`;
if (options.projectRoot !== undefined) {
newProjectRoot = options.projectRoot;
appDir = newProjectRoot;
}
return chain([
addAppToWorkspaceFile(options, workspace),
mergeWith(
apply(url('./files'), [
template({
utils: strings,
...options,
'dot': '.',
appDir,
}),
move(appDir),
])),
]);
};
}