/** * Copyright 2022 Google Inc. All rights reserved. * * Licensed 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 * * https://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. */ import {relative, resolve} from 'path'; import {getSystemPath, normalize, strings} from '@angular-devkit/core'; import { type SchematicContext, type Tree, apply, applyTemplates, chain, mergeWith, move, url, } from '@angular-devkit/schematics'; import {type AngularProject, type TestRunner} from './types.js'; export interface FilesOptions { options: { testRunner: TestRunner; port: number; name?: string; exportConfig?: boolean; ext?: string; route?: string; }; applyPath: string; relativeToWorkspacePath: string; movePath?: string; } export function addFilesToProjects( tree: Tree, context: SchematicContext, projects: Record, options: FilesOptions ): any { return chain( Object.keys(projects).map(name => { return addFilesSingle( tree, context, name, projects[name] as AngularProject, options ); }) )(tree, context); } export function addFilesSingle( _tree: Tree, _context: SchematicContext, name: string, project: AngularProject, {options, applyPath, movePath, relativeToWorkspacePath}: FilesOptions ): any { const projectPath = resolve(getSystemPath(normalize(project.root))); const workspacePath = resolve(getSystemPath(normalize(''))); const relativeToWorkspace = relative( `${projectPath}${relativeToWorkspacePath}`, workspacePath ); const baseUrl = getProjectBaseUrl(project, options.port); const tsConfigPath = getTsConfigPath(project); return mergeWith( apply(url(applyPath), [ move(movePath ? `${project.root}${movePath}` : project.root), applyTemplates({ ...options, ...strings, root: project.root ? `${project.root}/` : project.root, baseUrl, tsConfigPath, project: name, relativeToWorkspace, }), ]) ); } function getProjectBaseUrl(project: any, port: number): string { let options = {protocol: 'http', port, host: 'localhost'}; if (project.architect?.serve?.options) { const projectOptions = project.architect?.serve?.options; const projectPort = port !== 4200 ? port : projectOptions?.port ?? port; options = {...options, ...projectOptions, port: projectPort}; options.protocol = projectOptions.ssl ? 'https' : 'http'; } return `${options.protocol}://${options.host}:${options.port}`; } function getTsConfigPath(project: AngularProject): string { if (!project.root) { return '../tsconfig.json'; } return `../tsconfig.app.json`; } export function addCommonFiles( tree: Tree, context: SchematicContext, projects: Record, filesOptions: Omit ): any { const options: FilesOptions = { ...filesOptions, applyPath: './files/common', relativeToWorkspacePath: `/`, }; return addFilesToProjects(tree, context, projects, options); } export function addFrameworkFiles( tree: Tree, context: SchematicContext, projects: Record, filesOptions: Omit ): any { const testRunner = filesOptions.options.testRunner; const options: FilesOptions = { ...filesOptions, applyPath: `./files/${testRunner}`, relativeToWorkspacePath: `/`, }; return addFilesToProjects(tree, context, projects, options); } export function hasE2ETester( projects: Record ): boolean { return Object.values(projects).some((project: AngularProject) => { return Boolean(project.architect?.e2e); }); } export function getNgCommandName( projects: Record ): string { if (!hasE2ETester(projects)) { return 'e2e'; } return 'puppeteer'; }