Alan Agius c969152de6 refactor: remove analytics API from core and architect
All analytics is now in the @angular/cli package

BREAKING CHANGE: analytics APIs have been removed without replacement from `@angular-devkit/core` and `@angular-devkit/architect`.
2022-10-11 18:52:30 +02:00

424 lines
12 KiB
TypeScript

/**
* @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
*/
import { json, logging } from '@angular-devkit/core';
import { Observable, from, merge, of, onErrorResumeNext } from 'rxjs';
import {
concatMap,
first,
ignoreElements,
last,
map,
shareReplay,
switchMap,
takeUntil,
} from 'rxjs/operators';
import {
BuilderInfo,
BuilderInput,
BuilderOutput,
BuilderRegistry,
BuilderRun,
Target,
targetStringFromTarget,
} from './api';
import { ArchitectHost, BuilderDescription, BuilderJobHandler } from './internal';
import {
FallbackRegistry,
JobHandler,
JobHandlerContext,
JobInboundMessage,
JobInboundMessageKind,
JobName,
JobOutboundMessageKind,
Registry,
Scheduler,
SimpleJobRegistry,
SimpleScheduler,
createJobHandler,
} from './jobs';
import { scheduleByName, scheduleByTarget } from './schedule-by-name';
const inputSchema = require('./input-schema.json');
const outputSchema = require('./output-schema.json');
function _createJobHandlerFromBuilderInfo(
info: BuilderInfo,
target: Target | undefined,
host: ArchitectHost,
registry: json.schema.SchemaRegistry,
baseOptions: json.JsonObject,
): Observable<BuilderJobHandler> {
const jobDescription: BuilderDescription = {
name: target ? `{${targetStringFromTarget(target)}}` : info.builderName,
argument: { type: 'object' },
input: inputSchema,
output: outputSchema,
info,
};
function handler(argument: json.JsonObject, context: JobHandlerContext) {
// Add input validation to the inbound bus.
const inboundBusWithInputValidation = context.inboundBus.pipe(
concatMap((message) => {
if (message.kind === JobInboundMessageKind.Input) {
const v = message.value as BuilderInput;
const options = {
...baseOptions,
...v.options,
};
// Validate v against the options schema.
return registry.compile(info.optionSchema).pipe(
concatMap((validation) => validation(options)),
map((validationResult: json.schema.SchemaValidatorResult) => {
const { data, success, errors } = validationResult;
if (success) {
return { ...v, options: data } as BuilderInput;
}
throw new json.schema.SchemaValidationException(errors);
}),
map((value) => ({ ...message, value })),
);
} else {
return of(message as JobInboundMessage<BuilderInput>);
}
}),
// Using a share replay because the job might be synchronously sending input, but
// asynchronously listening to it.
shareReplay(1),
);
// Make an inboundBus that completes instead of erroring out.
// We'll merge the errors into the output instead.
const inboundBus = onErrorResumeNext(inboundBusWithInputValidation);
const output = from(host.loadBuilder(info)).pipe(
concatMap((builder) => {
if (builder === null) {
throw new Error(`Cannot load builder for builderInfo ${JSON.stringify(info, null, 2)}`);
}
return builder.handler(argument, { ...context, inboundBus }).pipe(
map((output) => {
if (output.kind === JobOutboundMessageKind.Output) {
// Add target to it.
return {
...output,
value: {
...output.value,
...(target ? { target } : 0),
} as unknown as json.JsonObject,
};
} else {
return output;
}
}),
);
}),
// Share subscriptions to the output, otherwise the the handler will be re-run.
shareReplay(),
);
// Separate the errors from the inbound bus into their own observable that completes when the
// builder output does.
const inboundBusErrors = inboundBusWithInputValidation.pipe(
ignoreElements(),
takeUntil(onErrorResumeNext(output.pipe(last()))),
);
// Return the builder output plus any input errors.
return merge(inboundBusErrors, output);
}
return of(Object.assign(handler, { jobDescription }) as BuilderJobHandler);
}
export interface ScheduleOptions {
logger?: logging.Logger;
}
/**
* A JobRegistry that resolves builder targets from the host.
*/
class ArchitectBuilderJobRegistry implements BuilderRegistry {
constructor(
protected _host: ArchitectHost,
protected _registry: json.schema.SchemaRegistry,
protected _jobCache?: Map<string, Observable<BuilderJobHandler | null>>,
protected _infoCache?: Map<string, Observable<BuilderInfo | null>>,
) {}
protected _resolveBuilder(name: string): Observable<BuilderInfo | null> {
const cache = this._infoCache;
if (cache) {
const maybeCache = cache.get(name);
if (maybeCache !== undefined) {
return maybeCache;
}
const info = from(this._host.resolveBuilder(name)).pipe(shareReplay(1));
cache.set(name, info);
return info;
}
return from(this._host.resolveBuilder(name));
}
protected _createBuilder(
info: BuilderInfo,
target?: Target,
options?: json.JsonObject,
): Observable<BuilderJobHandler | null> {
const cache = this._jobCache;
if (target) {
const maybeHit = cache && cache.get(targetStringFromTarget(target));
if (maybeHit) {
return maybeHit;
}
} else {
const maybeHit = cache && cache.get(info.builderName);
if (maybeHit) {
return maybeHit;
}
}
const result = _createJobHandlerFromBuilderInfo(
info,
target,
this._host,
this._registry,
options || {},
);
if (cache) {
if (target) {
cache.set(targetStringFromTarget(target), result.pipe(shareReplay(1)));
} else {
cache.set(info.builderName, result.pipe(shareReplay(1)));
}
}
return result;
}
get<A extends json.JsonObject, I extends BuilderInput, O extends BuilderOutput>(
name: string,
): Observable<JobHandler<A, I, O> | null> {
const m = name.match(/^([^:]+):([^:]+)$/i);
if (!m) {
return of(null);
}
return from(this._resolveBuilder(name)).pipe(
concatMap((builderInfo) => (builderInfo ? this._createBuilder(builderInfo) : of(null))),
first(null, null),
) as Observable<JobHandler<A, I, O> | null>;
}
}
/**
* A JobRegistry that resolves targets from the host.
*/
class ArchitectTargetJobRegistry extends ArchitectBuilderJobRegistry {
override get<A extends json.JsonObject, I extends BuilderInput, O extends BuilderOutput>(
name: string,
): Observable<JobHandler<A, I, O> | null> {
const m = name.match(/^{([^:]+):([^:]+)(?::([^:]*))?}$/i);
if (!m) {
return of(null);
}
const target = {
project: m[1],
target: m[2],
configuration: m[3],
};
return from(
Promise.all([
this._host.getBuilderNameForTarget(target),
this._host.getOptionsForTarget(target),
]),
).pipe(
concatMap(([builderStr, options]) => {
if (builderStr === null || options === null) {
return of(null);
}
return this._resolveBuilder(builderStr).pipe(
concatMap((builderInfo) => {
if (builderInfo === null) {
return of(null);
}
return this._createBuilder(builderInfo, target, options);
}),
);
}),
first(null, null),
) as Observable<JobHandler<A, I, O> | null>;
}
}
function _getTargetOptionsFactory(host: ArchitectHost) {
return createJobHandler<Target, json.JsonValue, json.JsonObject>(
(target) => {
return host.getOptionsForTarget(target).then((options) => {
if (options === null) {
throw new Error(`Invalid target: ${JSON.stringify(target)}.`);
}
return options;
});
},
{
name: '..getTargetOptions',
output: { type: 'object' },
argument: inputSchema.properties.target,
},
);
}
function _getProjectMetadataFactory(host: ArchitectHost) {
return createJobHandler<Target, json.JsonValue, json.JsonObject>(
(target) => {
return host.getProjectMetadata(target).then((options) => {
if (options === null) {
throw new Error(`Invalid target: ${JSON.stringify(target)}.`);
}
return options;
});
},
{
name: '..getProjectMetadata',
output: { type: 'object' },
argument: {
oneOf: [{ type: 'string' }, inputSchema.properties.target],
},
},
);
}
function _getBuilderNameForTargetFactory(host: ArchitectHost) {
return createJobHandler<Target, never, string>(
async (target) => {
const builderName = await host.getBuilderNameForTarget(target);
if (!builderName) {
throw new Error(`No builder were found for target ${targetStringFromTarget(target)}.`);
}
return builderName;
},
{
name: '..getBuilderNameForTarget',
output: { type: 'string' },
argument: inputSchema.properties.target,
},
);
}
function _validateOptionsFactory(host: ArchitectHost, registry: json.schema.SchemaRegistry) {
return createJobHandler<[string, json.JsonObject], never, json.JsonObject>(
async ([builderName, options]) => {
// Get option schema from the host.
const builderInfo = await host.resolveBuilder(builderName);
if (!builderInfo) {
throw new Error(`No builder info were found for builder ${JSON.stringify(builderName)}.`);
}
return registry
.compile(builderInfo.optionSchema)
.pipe(
concatMap((validation) => validation(options)),
switchMap(({ data, success, errors }) => {
if (success) {
return of(data as json.JsonObject);
}
throw new json.schema.SchemaValidationException(errors);
}),
)
.toPromise();
},
{
name: '..validateOptions',
output: { type: 'object' },
argument: {
type: 'array',
items: [{ type: 'string' }, { type: 'object' }],
},
},
);
}
export class Architect {
private readonly _scheduler: Scheduler;
private readonly _jobCache = new Map<string, Observable<BuilderJobHandler>>();
private readonly _infoCache = new Map<string, Observable<BuilderInfo>>();
constructor(
private _host: ArchitectHost,
registry: json.schema.SchemaRegistry = new json.schema.CoreSchemaRegistry(),
additionalJobRegistry?: Registry,
) {
const privateArchitectJobRegistry = new SimpleJobRegistry();
// Create private jobs.
privateArchitectJobRegistry.register(_getTargetOptionsFactory(_host));
privateArchitectJobRegistry.register(_getBuilderNameForTargetFactory(_host));
privateArchitectJobRegistry.register(_validateOptionsFactory(_host, registry));
privateArchitectJobRegistry.register(_getProjectMetadataFactory(_host));
const jobRegistry = new FallbackRegistry([
new ArchitectTargetJobRegistry(_host, registry, this._jobCache, this._infoCache),
new ArchitectBuilderJobRegistry(_host, registry, this._jobCache, this._infoCache),
privateArchitectJobRegistry,
...(additionalJobRegistry ? [additionalJobRegistry] : []),
] as Registry[]);
this._scheduler = new SimpleScheduler(jobRegistry, registry);
}
has(name: JobName) {
return this._scheduler.has(name);
}
scheduleBuilder(
name: string,
options: json.JsonObject,
scheduleOptions: ScheduleOptions = {},
): Promise<BuilderRun> {
// The below will match 'project:target:configuration'
if (!/^[^:]+:[^:]+(:[^:]+)?$/.test(name)) {
throw new Error('Invalid builder name: ' + JSON.stringify(name));
}
return scheduleByName(name, options, {
scheduler: this._scheduler,
logger: scheduleOptions.logger || new logging.NullLogger(),
currentDirectory: this._host.getCurrentDirectory(),
workspaceRoot: this._host.getWorkspaceRoot(),
});
}
scheduleTarget(
target: Target,
overrides: json.JsonObject = {},
scheduleOptions: ScheduleOptions = {},
): Promise<BuilderRun> {
return scheduleByTarget(target, overrides, {
scheduler: this._scheduler,
logger: scheduleOptions.logger || new logging.NullLogger(),
currentDirectory: this._host.getCurrentDirectory(),
workspaceRoot: this._host.getWorkspaceRoot(),
});
}
}