Skip to content

Commit

Permalink
perf(compiler-cli): reduce duplicate component style resolution (#57502)
Browse files Browse the repository at this point in the history
Previously, the component handler was processing and resolving stylesheets
referenced via `styleUrl`/`styleUrls` multiple times when generating the
compiler metadata for components. The style resource information collection
for such styles has been further consolidated to avoid repeat resource loader
resolve calls which potentially could be expensive. Further optimization is
possible for the inline style case. However, inline styles here only require
AST traversal and no potentially expensive external resolve calls.

PR Close #57502
  • Loading branch information
clydin authored and alxhub committed Aug 26, 2024
1 parent 34ce46f commit 4716c3b
Show file tree
Hide file tree
Showing 2 changed files with 23 additions and 55 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -160,7 +160,7 @@ import {
import {
_extractTemplateStyleUrls,
extractComponentStyleUrls,
extractStyleResources,
extractInlineStyleResources,
extractTemplate,
makeResourceNotFoundError,
ParsedTemplateWithSource,
Expand Down Expand Up @@ -681,7 +681,7 @@ export class ComponentDecoratorHandler
// component.
let styles: string[] = [];

const styleResources = extractStyleResources(this.resourceLoader, component, containingFile);
const styleResources = extractInlineStyleResources(component);
const styleUrls: StyleUrlMeta[] = [
...extractComponentStyleUrls(this.evaluator, component),
..._extractTemplateStyleUrls(template),
Expand All @@ -690,6 +690,16 @@ export class ComponentDecoratorHandler
for (const styleUrl of styleUrls) {
try {
const resourceUrl = this.resourceLoader.resolve(styleUrl.url, containingFile);
if (
styleUrl.source === ResourceTypeForDiagnostics.StylesheetFromDecorator &&
ts.isStringLiteralLike(styleUrl.expression)
) {
// Only string literal values from the decorator are considered style resources
styleResources.add({
path: absoluteFrom(resourceUrl),
expression: styleUrl.expression,
});
}
const resourceStr = this.resourceLoader.load(resourceUrl);
styles.push(resourceStr);
if (this.depTracker !== null) {
Expand All @@ -711,11 +721,7 @@ export class ComponentDecoratorHandler
? ResourceTypeForDiagnostics.StylesheetFromDecorator
: ResourceTypeForDiagnostics.StylesheetFromTemplate;
diagnostics.push(
makeResourceNotFoundError(
styleUrl.url,
styleUrl.nodeForError,
resourceType,
).toDiagnostic(),
makeResourceNotFoundError(styleUrl.url, styleUrl.expression, resourceType).toDiagnostic(),
);
}
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -38,7 +38,7 @@ import {
*/
export interface StyleUrlMeta {
url: string;
nodeForError: ts.Node;
expression: ts.Expression;
source:
| ResourceTypeForDiagnostics.StylesheetFromTemplate
| ResourceTypeForDiagnostics.StylesheetFromDecorator;
Expand Down Expand Up @@ -123,7 +123,9 @@ export interface ExternalTemplateDeclaration extends CommonTemplateDeclaration {
export type TemplateDeclaration = InlineTemplateDeclaration | ExternalTemplateDeclaration;

/** Determines the node to use for debugging purposes for the given TemplateDeclaration. */
export function getTemplateDeclarationNodeForError(declaration: TemplateDeclaration): ts.Node {
export function getTemplateDeclarationNodeForError(
declaration: TemplateDeclaration,
): ts.Expression {
return declaration.isInline ? declaration.expression : declaration.templateUrlExpression;
}

Expand Down Expand Up @@ -629,7 +631,7 @@ export function extractComponentStyleUrls(
{
url: styleUrl,
source: ResourceTypeForDiagnostics.StylesheetFromDecorator,
nodeForError: styleUrlExpr,
expression: styleUrlExpr,
},
];
}
Expand Down Expand Up @@ -657,7 +659,7 @@ function extractStyleUrlsFromExpression(
styleUrls.push({
url: styleUrl,
source: ResourceTypeForDiagnostics.StylesheetFromDecorator,
nodeForError: styleUrlExpr,
expression: styleUrlExpr,
});
}
}
Expand All @@ -675,44 +677,20 @@ function extractStyleUrlsFromExpression(
styleUrls.push({
url: styleUrl,
source: ResourceTypeForDiagnostics.StylesheetFromDecorator,
nodeForError: styleUrlsExpr,
expression: styleUrlsExpr,
});
}
}

return styleUrls;
}

export function extractStyleResources(
resourceLoader: ResourceLoader,
component: Map<string, ts.Expression>,
containingFile: string,
): ReadonlySet<Resource> {
export function extractInlineStyleResources(component: Map<string, ts.Expression>): Set<Resource> {
const styles = new Set<Resource>();
function stringLiteralElements(array: ts.ArrayLiteralExpression): ts.StringLiteralLike[] {
return array.elements.filter((e): e is ts.StringLiteralLike => ts.isStringLiteralLike(e));
}

// If styleUrls is a literal array, process each resource url individually and register ones that
// are string literals. If `styleUrl` is specified, register a single stylesheet. Note that
// `styleUrl` and `styleUrls` are mutually-exclusive. This is validated in
// `extractComponentStyleUrls`.
const styleUrlExpr = component.get('styleUrl');
const styleUrlsExpr = component.get('styleUrls');
if (styleUrlsExpr !== undefined && ts.isArrayLiteralExpression(styleUrlsExpr)) {
for (const expression of stringLiteralElements(styleUrlsExpr)) {
const resource = stringLiteralUrlToResource(resourceLoader, expression, containingFile);
if (resource !== null) {
styles.add(resource);
}
}
} else if (styleUrlExpr !== undefined && ts.isStringLiteralLike(styleUrlExpr)) {
const resource = stringLiteralUrlToResource(resourceLoader, styleUrlExpr, containingFile);
if (resource !== null) {
styles.add(resource);
}
}

const stylesExpr = component.get('styles');
if (stylesExpr !== undefined) {
if (ts.isArrayLiteralExpression(stylesExpr)) {
Expand All @@ -727,31 +705,15 @@ export function extractStyleResources(
return styles;
}

function stringLiteralUrlToResource(
resourceLoader: ResourceLoader,
expression: ts.StringLiteralLike,
containingFile: string,
): Resource | null {
try {
const resourceUrl = resourceLoader.resolve(expression.text, containingFile);
return {path: absoluteFrom(resourceUrl), expression};
} catch {
// Errors in style resource extraction do not need to be handled here. We will produce
// diagnostics for each one that fails in the analysis, after we evaluate the `styleUrls`
// expression to determine _all_ style resources, not just the string literals.
return null;
}
}

export function _extractTemplateStyleUrls(template: ParsedTemplateWithSource): StyleUrlMeta[] {
if (template.styleUrls === null) {
return [];
}

const nodeForError = getTemplateDeclarationNodeForError(template.declaration);
const expression = getTemplateDeclarationNodeForError(template.declaration);
return template.styleUrls.map((url) => ({
url,
source: ResourceTypeForDiagnostics.StylesheetFromTemplate,
nodeForError,
expression,
}));
}

0 comments on commit 4716c3b

Please sign in to comment.