260 lines
11 KiB
JavaScript
260 lines
11 KiB
JavaScript
|
"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.FileSystemEngineHostBase = exports.SchematicNameCollisionException = exports.SchematicMissingDescriptionException = exports.SchematicMissingFieldsException = exports.CollectionMissingFieldsException = exports.CollectionMissingSchematicsMapException = exports.FactoryCannotBeResolvedException = exports.SchematicMissingFactoryException = exports.InvalidCollectionJsonException = exports.CollectionCannotBeResolvedException = void 0;
|
||
|
const core_1 = require("@angular-devkit/core");
|
||
|
const node_1 = require("@angular-devkit/core/node");
|
||
|
const fs_1 = require("fs");
|
||
|
const path_1 = require("path");
|
||
|
const rxjs_1 = require("rxjs");
|
||
|
const src_1 = require("../src");
|
||
|
const file_system_utility_1 = require("./file-system-utility");
|
||
|
class CollectionCannotBeResolvedException extends core_1.BaseException {
|
||
|
constructor(name) {
|
||
|
super(`Collection ${JSON.stringify(name)} cannot be resolved.`);
|
||
|
}
|
||
|
}
|
||
|
exports.CollectionCannotBeResolvedException = CollectionCannotBeResolvedException;
|
||
|
class InvalidCollectionJsonException extends core_1.BaseException {
|
||
|
constructor(_name, path, jsonException) {
|
||
|
let msg = `Collection JSON at path ${JSON.stringify(path)} is invalid.`;
|
||
|
if (jsonException) {
|
||
|
msg = `${msg} ${jsonException.message}`;
|
||
|
}
|
||
|
super(msg);
|
||
|
}
|
||
|
}
|
||
|
exports.InvalidCollectionJsonException = InvalidCollectionJsonException;
|
||
|
class SchematicMissingFactoryException extends core_1.BaseException {
|
||
|
constructor(name) {
|
||
|
super(`Schematic ${JSON.stringify(name)} is missing a factory.`);
|
||
|
}
|
||
|
}
|
||
|
exports.SchematicMissingFactoryException = SchematicMissingFactoryException;
|
||
|
class FactoryCannotBeResolvedException extends core_1.BaseException {
|
||
|
constructor(name) {
|
||
|
super(`Schematic ${JSON.stringify(name)} cannot resolve the factory.`);
|
||
|
}
|
||
|
}
|
||
|
exports.FactoryCannotBeResolvedException = FactoryCannotBeResolvedException;
|
||
|
class CollectionMissingSchematicsMapException extends core_1.BaseException {
|
||
|
constructor(name) {
|
||
|
super(`Collection "${name}" does not have a schematics map.`);
|
||
|
}
|
||
|
}
|
||
|
exports.CollectionMissingSchematicsMapException = CollectionMissingSchematicsMapException;
|
||
|
class CollectionMissingFieldsException extends core_1.BaseException {
|
||
|
constructor(name) {
|
||
|
super(`Collection "${name}" is missing fields.`);
|
||
|
}
|
||
|
}
|
||
|
exports.CollectionMissingFieldsException = CollectionMissingFieldsException;
|
||
|
class SchematicMissingFieldsException extends core_1.BaseException {
|
||
|
constructor(name) {
|
||
|
super(`Schematic "${name}" is missing fields.`);
|
||
|
}
|
||
|
}
|
||
|
exports.SchematicMissingFieldsException = SchematicMissingFieldsException;
|
||
|
class SchematicMissingDescriptionException extends core_1.BaseException {
|
||
|
constructor(name) {
|
||
|
super(`Schematics "${name}" does not have a description.`);
|
||
|
}
|
||
|
}
|
||
|
exports.SchematicMissingDescriptionException = SchematicMissingDescriptionException;
|
||
|
class SchematicNameCollisionException extends core_1.BaseException {
|
||
|
constructor(name) {
|
||
|
super(`Schematics/alias ${JSON.stringify(name)} collides with another alias or schematic` +
|
||
|
' name.');
|
||
|
}
|
||
|
}
|
||
|
exports.SchematicNameCollisionException = SchematicNameCollisionException;
|
||
|
/**
|
||
|
* A EngineHost base class that uses the file system to resolve collections. This is the base of
|
||
|
* all other EngineHost provided by the tooling part of the Schematics library.
|
||
|
*/
|
||
|
class FileSystemEngineHostBase {
|
||
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||
|
_transforms = [];
|
||
|
_contextTransforms = [];
|
||
|
_taskFactories = new Map();
|
||
|
listSchematicNames(collection, includeHidden) {
|
||
|
const schematics = [];
|
||
|
for (const key of Object.keys(collection.schematics)) {
|
||
|
const schematic = collection.schematics[key];
|
||
|
if ((schematic.hidden && !includeHidden) || schematic.private) {
|
||
|
continue;
|
||
|
}
|
||
|
// If extends is present without a factory it is an alias, do not return it
|
||
|
// unless it is from another collection.
|
||
|
if (!schematic.extends || schematic.factory) {
|
||
|
schematics.push(key);
|
||
|
}
|
||
|
else if (schematic.extends && schematic.extends.indexOf(':') !== -1) {
|
||
|
schematics.push(key);
|
||
|
}
|
||
|
}
|
||
|
return schematics;
|
||
|
}
|
||
|
registerOptionsTransform(t) {
|
||
|
this._transforms.push(t);
|
||
|
}
|
||
|
registerContextTransform(t) {
|
||
|
this._contextTransforms.push(t);
|
||
|
}
|
||
|
/**
|
||
|
*
|
||
|
* @param name
|
||
|
* @return {{path: string}}
|
||
|
*/
|
||
|
createCollectionDescription(name, requester) {
|
||
|
const path = this._resolveCollectionPath(name, requester?.path);
|
||
|
const jsonValue = (0, file_system_utility_1.readJsonFile)(path);
|
||
|
if (!jsonValue || typeof jsonValue != 'object' || Array.isArray(jsonValue)) {
|
||
|
throw new InvalidCollectionJsonException(name, path);
|
||
|
}
|
||
|
// normalize extends property to an array
|
||
|
if (typeof jsonValue['extends'] === 'string') {
|
||
|
jsonValue['extends'] = [jsonValue['extends']];
|
||
|
}
|
||
|
const description = this._transformCollectionDescription(name, {
|
||
|
...jsonValue,
|
||
|
path,
|
||
|
});
|
||
|
if (!description || !description.name) {
|
||
|
throw new InvalidCollectionJsonException(name, path);
|
||
|
}
|
||
|
// Validate aliases.
|
||
|
const allNames = Object.keys(description.schematics);
|
||
|
for (const schematicName of Object.keys(description.schematics)) {
|
||
|
const aliases = description.schematics[schematicName].aliases || [];
|
||
|
for (const alias of aliases) {
|
||
|
if (allNames.indexOf(alias) != -1) {
|
||
|
throw new SchematicNameCollisionException(alias);
|
||
|
}
|
||
|
}
|
||
|
allNames.push(...aliases);
|
||
|
}
|
||
|
return description;
|
||
|
}
|
||
|
createSchematicDescription(name, collection) {
|
||
|
// Resolve aliases first.
|
||
|
for (const schematicName of Object.keys(collection.schematics)) {
|
||
|
const schematicDescription = collection.schematics[schematicName];
|
||
|
if (schematicDescription.aliases && schematicDescription.aliases.indexOf(name) != -1) {
|
||
|
name = schematicName;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
if (!(name in collection.schematics)) {
|
||
|
return null;
|
||
|
}
|
||
|
const collectionPath = (0, path_1.dirname)(collection.path);
|
||
|
const partialDesc = collection.schematics[name];
|
||
|
if (!partialDesc) {
|
||
|
return null;
|
||
|
}
|
||
|
if (partialDesc.extends) {
|
||
|
const index = partialDesc.extends.indexOf(':');
|
||
|
const collectionName = index !== -1 ? partialDesc.extends.slice(0, index) : null;
|
||
|
const schematicName = index === -1 ? partialDesc.extends : partialDesc.extends.slice(index + 1);
|
||
|
if (collectionName !== null) {
|
||
|
const extendCollection = this.createCollectionDescription(collectionName);
|
||
|
return this.createSchematicDescription(schematicName, extendCollection);
|
||
|
}
|
||
|
else {
|
||
|
return this.createSchematicDescription(schematicName, collection);
|
||
|
}
|
||
|
}
|
||
|
// Use any on this ref as we don't have the OptionT here, but we don't need it (we only need
|
||
|
// the path).
|
||
|
if (!partialDesc.factory) {
|
||
|
throw new SchematicMissingFactoryException(name);
|
||
|
}
|
||
|
const resolvedRef = this._resolveReferenceString(partialDesc.factory, collectionPath, collection);
|
||
|
if (!resolvedRef) {
|
||
|
throw new FactoryCannotBeResolvedException(name);
|
||
|
}
|
||
|
let schema = partialDesc.schema;
|
||
|
let schemaJson = undefined;
|
||
|
if (schema) {
|
||
|
if (!(0, path_1.isAbsolute)(schema)) {
|
||
|
schema = (0, path_1.join)(collectionPath, schema);
|
||
|
}
|
||
|
schemaJson = (0, file_system_utility_1.readJsonFile)(schema);
|
||
|
}
|
||
|
// The schematic path is used to resolve URLs.
|
||
|
// We should be able to just do `dirname(resolvedRef.path)` but for compatibility with
|
||
|
// Bazel under Windows this directory needs to be resolved from the collection instead.
|
||
|
// This is needed because on Bazel under Windows the data files (such as the collection or
|
||
|
// url files) are not in the same place as the compiled JS.
|
||
|
const maybePath = (0, path_1.join)(collectionPath, partialDesc.factory);
|
||
|
const path = (0, fs_1.existsSync)(maybePath) && (0, fs_1.statSync)(maybePath).isDirectory() ? maybePath : (0, path_1.dirname)(maybePath);
|
||
|
return this._transformSchematicDescription(name, collection, {
|
||
|
...partialDesc,
|
||
|
schema,
|
||
|
schemaJson,
|
||
|
name,
|
||
|
path,
|
||
|
factoryFn: resolvedRef.ref,
|
||
|
collection,
|
||
|
});
|
||
|
}
|
||
|
createSourceFromUrl(url) {
|
||
|
switch (url.protocol) {
|
||
|
case null:
|
||
|
case 'file:':
|
||
|
return (context) => {
|
||
|
// Check if context has necessary FileSystemSchematicContext path property
|
||
|
const fileDescription = context.schematic.description;
|
||
|
if (fileDescription.path === undefined) {
|
||
|
throw new Error('Unsupported schematic context. Expected a FileSystemSchematicContext.');
|
||
|
}
|
||
|
// Resolve all file:///a/b/c/d from the schematic's own path, and not the current
|
||
|
// path.
|
||
|
const root = (0, core_1.normalize)((0, path_1.resolve)(fileDescription.path, url.path || ''));
|
||
|
return new src_1.HostCreateTree(new core_1.virtualFs.ScopedHost(new node_1.NodeJsSyncHost(), root));
|
||
|
};
|
||
|
}
|
||
|
return null;
|
||
|
}
|
||
|
transformOptions(schematic, options, context) {
|
||
|
const transform = async () => {
|
||
|
let transformedOptions = options;
|
||
|
for (const transformer of this._transforms) {
|
||
|
const transformerResult = transformer(schematic, transformedOptions, context);
|
||
|
transformedOptions = await ((0, rxjs_1.isObservable)(transformerResult)
|
||
|
? (0, rxjs_1.lastValueFrom)(transformerResult)
|
||
|
: transformerResult);
|
||
|
}
|
||
|
return transformedOptions;
|
||
|
};
|
||
|
return (0, rxjs_1.from)(transform());
|
||
|
}
|
||
|
transformContext(context) {
|
||
|
return this._contextTransforms.reduce((acc, curr) => curr(acc), context);
|
||
|
}
|
||
|
getSchematicRuleFactory(schematic, _collection) {
|
||
|
return schematic.factoryFn;
|
||
|
}
|
||
|
registerTaskExecutor(factory, options) {
|
||
|
this._taskFactories.set(factory.name, () => (0, rxjs_1.from)(factory.create(options)));
|
||
|
}
|
||
|
createTaskExecutor(name) {
|
||
|
const factory = this._taskFactories.get(name);
|
||
|
if (factory) {
|
||
|
return factory();
|
||
|
}
|
||
|
return (0, rxjs_1.throwError)(new src_1.UnregisteredTaskException(name));
|
||
|
}
|
||
|
hasTaskExecutor(name) {
|
||
|
return this._taskFactories.has(name);
|
||
|
}
|
||
|
}
|
||
|
exports.FileSystemEngineHostBase = FileSystemEngineHostBase;
|