MyRepo-Ums/node_modules/@schematics/angular/utility/standalone/rules.js
2024-01-19 11:09:11 +01:00

190 lines
8.3 KiB
JavaScript
Executable File

"use strict";
/**
* @license
* Copyright Google LLC 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
*/
Object.defineProperty(exports, "__esModule", { value: true });
exports.addRootProvider = exports.addRootImport = void 0;
const core_1 = require("@angular-devkit/core");
const schematics_1 = require("@angular-devkit/schematics");
const ast_utils_1 = require("../ast-utils");
const change_1 = require("../change");
const ng_ast_utils_1 = require("../ng-ast-utils");
const app_config_1 = require("./app_config");
const code_block_1 = require("./code_block");
const util_1 = require("./util");
/**
* Adds an import to the root of the project.
* @param project Name of the project to which to add the import.
* @param callback Function that generates the code block which should be inserted.
* @example
*
* ```ts
* import { Rule } from '@angular-devkit/schematics';
* import { addRootImport } from '@schematics/angular/utility';
*
* export default function(): Rule {
* return addRootImport('default', ({code, external}) => {
* return code`${external('MyModule', '@my/module')}.forRoot({})`;
* });
* }
* ```
*/
function addRootImport(project, callback) {
return getRootInsertionRule(project, callback, 'imports', {
name: 'importProvidersFrom',
module: '@angular/core',
});
}
exports.addRootImport = addRootImport;
/**
* Adds a provider to the root of the project.
* @param project Name of the project to which to add the import.
* @param callback Function that generates the code block which should be inserted.
* @example
*
* ```ts
* import { Rule } from '@angular-devkit/schematics';
* import { addRootProvider } from '@schematics/angular/utility';
*
* export default function(): Rule {
* return addRootProvider('default', ({code, external}) => {
* return code`${external('provideLibrary', '@my/library')}({})`;
* });
* }
* ```
*/
function addRootProvider(project, callback) {
return getRootInsertionRule(project, callback, 'providers');
}
exports.addRootProvider = addRootProvider;
/**
* Creates a rule that inserts code at the root of either a standalone or NgModule-based project.
* @param project Name of the project into which to inser tthe code.
* @param callback Function that generates the code block which should be inserted.
* @param ngModuleField Field of the root NgModule into which the code should be inserted, if the
* app is based on NgModule
* @param standaloneWrapperFunction Function with which to wrap the code if the app is standalone.
*/
function getRootInsertionRule(project, callback, ngModuleField, standaloneWrapperFunction) {
return async (host) => {
const mainFilePath = await (0, util_1.getMainFilePath)(host, project);
const codeBlock = new code_block_1.CodeBlock();
if ((0, ng_ast_utils_1.isStandaloneApp)(host, mainFilePath)) {
return (tree) => addProviderToStandaloneBootstrap(tree, callback(codeBlock), mainFilePath, standaloneWrapperFunction);
}
const modulePath = (0, ng_ast_utils_1.getAppModulePath)(host, mainFilePath);
const pendingCode = code_block_1.CodeBlock.transformPendingCode(callback(codeBlock), modulePath);
return (0, schematics_1.chain)([
...pendingCode.rules,
(tree) => {
const changes = (0, ast_utils_1.addSymbolToNgModuleMetadata)((0, util_1.getSourceFile)(tree, modulePath), modulePath, ngModuleField, pendingCode.code.expression,
// Explicitly set the import path to null since we deal with imports here separately.
null);
(0, util_1.applyChangesToFile)(tree, modulePath, changes);
},
]);
};
}
/**
* Adds a provider to the root of a standalone project.
* @param host Tree of the root rule.
* @param pendingCode Code that should be inserted.
* @param mainFilePath Path to the project's main file.
* @param wrapperFunction Optional function with which to wrap the provider.
*/
function addProviderToStandaloneBootstrap(host, pendingCode, mainFilePath, wrapperFunction) {
const bootstrapCall = (0, util_1.findBootstrapApplicationCall)(host, mainFilePath);
const fileToEdit = (0, app_config_1.findAppConfig)(bootstrapCall, host, mainFilePath)?.filePath || mainFilePath;
const { code, rules } = code_block_1.CodeBlock.transformPendingCode(pendingCode, fileToEdit);
return (0, schematics_1.chain)([
...rules,
() => {
let wrapped;
let additionalRules;
if (wrapperFunction) {
const block = new code_block_1.CodeBlock();
const result = code_block_1.CodeBlock.transformPendingCode(block.code `${block.external(wrapperFunction.name, wrapperFunction.module)}(${code.expression})`, fileToEdit);
wrapped = result.code;
additionalRules = result.rules;
}
else {
wrapped = code;
additionalRules = [];
}
return (0, schematics_1.chain)([
...additionalRules,
(tree) => insertStandaloneRootProvider(tree, mainFilePath, wrapped.expression),
]);
},
]);
}
/**
* Inserts a string expression into the root of a standalone project.
* @param tree File tree used to modify the project.
* @param mainFilePath Path to the main file of the project.
* @param expression Code expression to be inserted.
*/
function insertStandaloneRootProvider(tree, mainFilePath, expression) {
const bootstrapCall = (0, util_1.findBootstrapApplicationCall)(tree, mainFilePath);
const appConfig = (0, app_config_1.findAppConfig)(bootstrapCall, tree, mainFilePath);
if (bootstrapCall.arguments.length === 0) {
throw new schematics_1.SchematicsException(`Cannot add provider to invalid bootstrapApplication call in ${bootstrapCall.getSourceFile().fileName}`);
}
if (appConfig) {
addProvidersExpressionToAppConfig(tree, appConfig, expression);
return;
}
const newAppConfig = `, {\n${core_1.tags.indentBy(2) `providers: [${expression}]`}\n}`;
let targetCall;
if (bootstrapCall.arguments.length === 1) {
targetCall = bootstrapCall;
}
else if ((0, util_1.isMergeAppConfigCall)(bootstrapCall.arguments[1])) {
targetCall = bootstrapCall.arguments[1];
}
else {
throw new schematics_1.SchematicsException(`Cannot statically analyze bootstrapApplication call in ${bootstrapCall.getSourceFile().fileName}`);
}
(0, util_1.applyChangesToFile)(tree, mainFilePath, [
(0, ast_utils_1.insertAfterLastOccurrence)(targetCall.arguments, newAppConfig, mainFilePath, targetCall.getEnd() - 1),
]);
}
/**
* Adds a string expression to an app config object.
* @param tree File tree used to modify the project.
* @param appConfig Resolved configuration object of the project.
* @param expression Code expression to be inserted.
*/
function addProvidersExpressionToAppConfig(tree, appConfig, expression) {
const { node, filePath } = appConfig;
const configProps = node.properties;
const providersLiteral = (0, util_1.findProvidersLiteral)(node);
// If there's a `providers` property, we can add the provider
// to it, otherwise we need to declare it ourselves.
if (providersLiteral) {
const hasTrailingComma = providersLiteral.elements.hasTrailingComma;
(0, util_1.applyChangesToFile)(tree, filePath, [
(0, ast_utils_1.insertAfterLastOccurrence)(providersLiteral.elements, (hasTrailingComma || providersLiteral.elements.length === 0 ? '' : ', ') + expression, filePath, providersLiteral.getStart() + 1),
]);
}
else {
const prop = core_1.tags.indentBy(2) `providers: [${expression}]`;
let toInsert;
let insertPosition;
if (configProps.length === 0) {
toInsert = '\n' + prop + '\n';
insertPosition = node.getEnd() - 1;
}
else {
const hasTrailingComma = configProps.hasTrailingComma;
toInsert = (hasTrailingComma ? '' : ',') + '\n' + prop;
insertPosition = configProps[configProps.length - 1].getEnd() + (hasTrailingComma ? 1 : 0);
}
(0, util_1.applyChangesToFile)(tree, filePath, [new change_1.InsertChange(filePath, insertPosition, toInsert)]);
}
}