Skip to content

Commit b19cd6e

Browse files
committedDec 3, 2024·
fix: revert commit ad7a297 to fix perf regression
As reported, the perf issue is caused by the introduction of an extra `Program` instance to fetch to AST transformers next to the hidden `Program` behind `ts.transpileModule`. We should only use one instance of `Program` therefore we have to manually copy the codes of `ts.transpileModule` to use for our case. Fixes #2886
1 parent 99bc772 commit b19cd6e

File tree

17 files changed

+277
-55
lines changed

17 files changed

+277
-55
lines changed
 

‎.github/workflows/source_code_ci.yml

+7
Original file line numberDiff line numberDiff line change
@@ -57,3 +57,10 @@ jobs:
5757
with:
5858
os: ubuntu-latest
5959
node: ${{ matrix.node }}
60+
61+
test_performance:
62+
needs: prepare_cache_ubuntu
63+
uses: ./.github/workflows/test_performance.yml
64+
with:
65+
os: ubuntu-latest
66+
node: 20.x
+55
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,55 @@
1+
name: Test performance
2+
3+
on:
4+
workflow_call:
5+
inputs:
6+
os:
7+
required: true
8+
type: string
9+
node:
10+
required: true
11+
type: string
12+
13+
permissions:
14+
contents: read
15+
16+
jobs:
17+
test:
18+
runs-on: ${{ inputs.os }}
19+
20+
steps:
21+
- name: Checkout 🛎️
22+
uses: actions/checkout@v4
23+
with:
24+
fetch-depth: 20
25+
fetch-tags: false
26+
27+
- name: Restore cached node modules ♻️
28+
id: cache-yarn
29+
uses: actions/cache@v4
30+
with:
31+
path: |
32+
.yarn/cache
33+
node_modules
34+
examples/example-app-monorepo/node_modules
35+
examples/example-app-v17/node_modules
36+
examples/example-app-v18/node_modules
37+
examples/example-app-v19/node_modules
38+
examples/example-app-yarn-workspace/node_modules
39+
key: ${{ inputs.os }}-${{ inputs.node }}-build-${{ hashFiles('**/yarn.lock') }}
40+
restore-keys: |
41+
${{ inputs.os }}-${{ inputs.node }}-build
42+
43+
- name: Setup Node version ⚙️
44+
uses: actions/setup-node@v4
45+
with:
46+
node-version: ${{ inputs.node }}
47+
48+
- name: Build 🏗️
49+
run: yarn build
50+
51+
- name: Install dependencies 📦
52+
run: yarn --cwd performance
53+
54+
- name: Run Performance Tests 🧪
55+
run: yarn test-perf

‎.gitignore

+1-3
Original file line numberDiff line numberDiff line change
@@ -4,9 +4,7 @@ node_modules
44
.idea
55
*.tgz
66
.yarn/*
7-
e2e/**/.yarn/*
8-
examples/**/.yarn/*
9-
website/.yarn/*
7+
**/.yarn/*
108
tmp
119
.idx
1210
.vscode

‎.npmignore

+1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
# Tests
22
e2e
33
examples
4+
performance
45

56
# sources are inlined
67
src

‎examples/example-app-monorepo/apps/app1/src/app/app.component.ts

+7-2
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
1-
import { Component } from '@angular/core';
1+
import { DOCUMENT } from '@angular/common';
2+
import { Component, Inject } from '@angular/core';
23
import { RouterLink, RouterOutlet } from '@angular/router';
34

45
import { BannerComponent } from './banner/banner.component';
@@ -10,4 +11,8 @@ import { WelcomeComponent } from './welcome/welcome.component';
1011
templateUrl: './app.component.html',
1112
imports: [BannerComponent, WelcomeComponent, RouterOutlet, RouterLink],
1213
})
13-
export class AppComponent {}
14+
export class AppComponent {
15+
constructor(@Inject(DOCUMENT) injectedDoc: Document) {
16+
injectedDoc.title = 'Example App';
17+
}
18+
}

‎examples/example-app-v17/src/app/app.component.ts

+7-2
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
1-
import { Component } from '@angular/core';
1+
import { DOCUMENT } from '@angular/common';
2+
import { Component, Inject } from '@angular/core';
23
import { RouterLink, RouterOutlet } from '@angular/router';
34

45
import { BannerComponent } from './banner/banner.component';
@@ -10,4 +11,8 @@ import { WelcomeComponent } from './welcome/welcome.component';
1011
templateUrl: './app.component.html',
1112
imports: [BannerComponent, WelcomeComponent, RouterOutlet, RouterLink],
1213
})
13-
export class AppComponent {}
14+
export class AppComponent {
15+
constructor(@Inject(DOCUMENT) injectedDoc: Document) {
16+
injectedDoc.title = 'Example App';
17+
}
18+
}

‎examples/example-app-v18/src/app/app.component.ts

+7-2
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
1-
import { Component } from '@angular/core';
1+
import { DOCUMENT } from '@angular/common';
2+
import { Component, Inject } from '@angular/core';
23
import { RouterLink, RouterOutlet } from '@angular/router';
34

45
import { BannerComponent } from './banner/banner.component';
@@ -10,4 +11,8 @@ import { WelcomeComponent } from './welcome/welcome.component';
1011
templateUrl: './app.component.html',
1112
imports: [BannerComponent, WelcomeComponent, RouterOutlet, RouterLink],
1213
})
13-
export class AppComponent {}
14+
export class AppComponent {
15+
constructor(@Inject(DOCUMENT) injectedDoc: Document) {
16+
injectedDoc.title = 'Example App';
17+
}
18+
}

‎examples/example-app-v19/src/app/app.component.ts

+7-2
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
1-
import { Component } from '@angular/core';
1+
import { DOCUMENT } from '@angular/common';
2+
import { Component, Inject } from '@angular/core';
23
import { RouterLink, RouterOutlet } from '@angular/router';
34

45
import { BannerComponent } from './banner/banner.component';
@@ -9,4 +10,8 @@ import { WelcomeComponent } from './welcome/welcome.component';
910
templateUrl: './app.component.html',
1011
imports: [BannerComponent, WelcomeComponent, RouterOutlet, RouterLink],
1112
})
12-
export class AppComponent {}
13+
export class AppComponent {
14+
constructor(@Inject(DOCUMENT) injectedDoc: Document) {
15+
injectedDoc.title = 'Example App';
16+
}
17+
}

‎examples/example-app-yarn-workspace/packages/angular-app/src/app/app.component.ts

+7-2
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
1-
import { Component } from '@angular/core';
1+
import { DOCUMENT } from '@angular/common';
2+
import { Component, Inject } from '@angular/core';
23
import { RouterLink, RouterOutlet } from '@angular/router';
34

45
import { BannerComponent } from './banner/banner.component';
@@ -10,4 +11,8 @@ import { WelcomeComponent } from './welcome/welcome.component';
1011
templateUrl: './app.component.html',
1112
imports: [BannerComponent, WelcomeComponent, RouterOutlet, RouterLink],
1213
})
13-
export class AppComponent {}
14+
export class AppComponent {
15+
constructor(@Inject(DOCUMENT) injectedDoc: Document) {
16+
injectedDoc.title = 'Example App';
17+
}
18+
}

‎package.json

+1
Original file line numberDiff line numberDiff line change
@@ -38,6 +38,7 @@
3838
"test-cjs": "jest -c=jest-cjs.config.ts --no-cache && jest -c=jest-transpile-cjs.config.ts --no-cache",
3939
"test-esm": "node --experimental-vm-modules --no-warnings node_modules/jest/bin/jest.js -c=jest-esm.config.ts --no-cache && node --experimental-vm-modules --no-warnings node_modules/jest/bin/jest.js -c=jest-transpile-esm.config.ts --no-cache",
4040
"test-examples": "yarn build && node scripts/test-examples.js",
41+
"test-perf": "jest -c=performance/jest.config.ts --no-cache",
4142
"doc": "cd website && yarn start",
4243
"doc:build": "cd website && yarn build",
4344
"changelog": "conventional-changelog -p angular -i CHANGELOG.md -s -r 1",

‎performance/__tests__/perf.spec.ts

+40
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,40 @@
1+
const testFunction = async () => {
2+
const { camelCase } = await import('lodash-es');
3+
camelCase('FooBar');
4+
};
5+
6+
describe('Performance Measurement', () => {
7+
it('should complete expensive operation within acceptable performance', async () => {
8+
const runs = 5;
9+
const times: number[] = [];
10+
11+
for (let i = 0; i < runs; i++) {
12+
const start = performance.now();
13+
await testFunction();
14+
const end = performance.now();
15+
times.push(end - start);
16+
}
17+
18+
const average = times.reduce((a, b) => a + b, 0) / runs;
19+
const min = Math.min(...times);
20+
const max = Math.max(...times);
21+
const standardDeviation = Math.sqrt(times.reduce((sq, n) => sq + Math.pow(n - average, 2), 0) / runs);
22+
23+
console.log('Performance Metrics:', {
24+
average: `${average.toFixed(2)}ms`,
25+
min: `${min.toFixed(2)}ms`,
26+
max: `${max.toFixed(2)}ms`,
27+
standardDeviation: `${standardDeviation.toFixed(2)}ms`,
28+
});
29+
30+
const acceptableThresholds = {
31+
averageMax: 700,
32+
standardDeviationMax: 1500,
33+
absoluteMax: 3300,
34+
};
35+
36+
expect(average).toBeLessThan(acceptableThresholds.averageMax);
37+
expect(standardDeviation).toBeLessThan(acceptableThresholds.standardDeviationMax);
38+
expect(max).toBeLessThan(acceptableThresholds.absoluteMax);
39+
});
40+
});

‎performance/jest.config.ts

+16
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
import type { JestConfigWithTsJest } from 'ts-jest';
2+
3+
const config: JestConfigWithTsJest = {
4+
testEnvironment: 'jsdom',
5+
transform: {
6+
'^.+\\.(ts|mjs|js|html)$': [
7+
'<rootDir>/../build/index.js',
8+
{
9+
isolatedModules: true,
10+
},
11+
],
12+
},
13+
transformIgnorePatterns: ['node_modules/(?!lodash-es)'],
14+
};
15+
16+
export default config;

‎performance/package.json

+8
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
{
2+
"name": "performance",
3+
"private": true,
4+
"devDependencies": {
5+
"@types/lodash-es": "^4.17.12",
6+
"lodash-es": "^4.17.21"
7+
}
8+
}

‎performance/tsconfig.json

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
{
2+
"extends": "../tsconfig.json",
3+
}

‎performance/yarn.lock

+38
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,38 @@
1+
# This file is generated by running "yarn install" inside your project.
2+
# Manual changes might be lost - proceed with caution!
3+
4+
__metadata:
5+
version: 8
6+
cacheKey: 10
7+
8+
"@types/lodash-es@npm:^4.17.12":
9+
version: 4.17.12
10+
resolution: "@types/lodash-es@npm:4.17.12"
11+
dependencies:
12+
"@types/lodash": "npm:*"
13+
checksum: 10/56b9a433348b11c31051c6fa9028540a033a08fb80b400c589d740446c19444d73b217cf1471d4036448ef686a83e8cf2a35d1fadcb3f2105f26701f94aebb07
14+
languageName: node
15+
linkType: hard
16+
17+
"@types/lodash@npm:*":
18+
version: 4.14.182
19+
resolution: "@types/lodash@npm:4.14.182"
20+
checksum: 10/6c0d3fa682331d7631676817acf4b8b74842a9df0fb63dacbbc6a31b94e266edca550ac096cec8ce95df4fc72cf550a6321322e27872d4dfa15c1003197f6c85
21+
languageName: node
22+
linkType: hard
23+
24+
"lodash-es@npm:^4.17.21":
25+
version: 4.17.21
26+
resolution: "lodash-es@npm:4.17.21"
27+
checksum: 10/03f39878ea1e42b3199bd3f478150ab723f93cc8730ad86fec1f2804f4a07c6e30deaac73cad53a88e9c3db33348bb8ceeb274552390e7a75d7849021c02df43
28+
languageName: node
29+
linkType: hard
30+
31+
"performance@workspace:.":
32+
version: 0.0.0-use.local
33+
resolution: "performance@workspace:."
34+
dependencies:
35+
"@types/lodash-es": "npm:^4.17.12"
36+
lodash-es: "npm:^4.17.21"
37+
languageName: unknown
38+
linkType: soft

‎src/compiler/ng-jest-compiler.spec.ts

+2-1
Original file line numberDiff line numberDiff line change
@@ -323,9 +323,10 @@ describe('NgJestCompiler', () => {
323323
`);
324324

325325
expect(diagnostics.length).toBe(0);
326+
// The `type: undefined` is ok here which is covered by example apps' tests.
326327
expect(code).toContain(dedent`
327328
MyDir.ctorParameters = () => [
328-
{ type: Document, decorators: [{ type: core_1.Inject, args: [core_1.DOCUMENT,] }] }
329+
{ type: undefined, decorators: [{ type: core_1.Inject, args: [core_1.DOCUMENT,] }] }
329330
];
330331
exports.MyDir = MyDir = tslib_1.__decorate([
331332
(0, core_1.Directive)()

‎src/compiler/ng-jest-compiler.ts

+70-41
Original file line numberDiff line numberDiff line change
@@ -8,77 +8,106 @@ import { angularJitApplicationTransform } from '../transformers/jit_transform';
88
import { replaceResources } from '../transformers/replace-resources';
99

1010
export class NgJestCompiler extends TsCompiler {
11-
private readonly _defaultLibDirPath: string;
12-
private readonly _libSourceFileCache = new Map<string, ts.SourceFile>();
13-
1411
constructor(readonly configSet: ConfigSet, readonly jestCacheFS: Map<string, string>) {
1512
super(configSet, jestCacheFS);
1613

1714
this._logger.debug('created NgJestCompiler');
18-
this._defaultLibDirPath = path.dirname(this._ts.getDefaultLibFilePath(this._compilerOptions));
1915
}
2016

2117
/**
2218
* Copy from https://github.com/microsoft/TypeScript/blob/master/src/services/transpile.ts
2319
* This is required because the exposed function `transpileModule` from TypeScript doesn't allow to access `Program`
24-
* and we need `Program` to be able to use Angular AST transformers.
20+
* and we need `Program` to be able to use Angular `replace-resources` transformer.
2521
*/
2622
protected _transpileOutput(fileContent: string, filePath: string): ts.TranspileOutput {
27-
const scriptTarget = this._compilerOptions.target ?? this._ts.ScriptTarget.Latest;
28-
const sourceFile = this._ts.createSourceFile(filePath, fileContent, scriptTarget);
23+
const diagnostics: ts.Diagnostic[] = [];
24+
const compilerOptions = { ...this._compilerOptions };
25+
const options: ts.CompilerOptions = compilerOptions
26+
? // @ts-expect-error internal TypeScript API
27+
this._ts.fixupCompilerOptions(compilerOptions, diagnostics)
28+
: {};
29+
30+
// mix in default options
31+
const defaultOptions = this._ts.getDefaultCompilerOptions();
32+
for (const key in defaultOptions) {
33+
if (Object.prototype.hasOwnProperty.call(defaultOptions, key) && options[key] === undefined) {
34+
options[key] = defaultOptions[key];
35+
}
36+
}
37+
38+
// @ts-expect-error internal TypeScript API
39+
for (const option of this._ts.transpileOptionValueCompilerOptions) {
40+
options[option.name] = option.transpileOptionValue;
41+
}
42+
43+
/**
44+
* transpileModule does not write anything to disk so there is no need to verify that there are no conflicts between
45+
* input and output paths.
46+
*/
47+
options.suppressOutputPathCheck = true;
48+
49+
// Filename can be non-ts file.
50+
options.allowNonTsExtensions = true;
51+
52+
const sourceFile = this._ts.createSourceFile(
53+
filePath,
54+
fileContent,
55+
options.target ?? this._ts.ScriptTarget.Latest,
56+
);
57+
58+
let outputText: string | undefined;
59+
let sourceMapText: string | undefined;
60+
2961
const compilerHost: ts.CompilerHost = {
3062
getSourceFile: (fileName) => {
31-
if (fileName === path.normalize(filePath)) {
32-
return sourceFile;
33-
}
34-
35-
let libSourceFile = this._libSourceFileCache.get(fileName);
36-
if (!libSourceFile) {
37-
const libFilePath = path.join(this._defaultLibDirPath, fileName);
38-
const libFileContent = this._ts.sys.readFile(libFilePath) ?? '';
39-
if (libFileContent) {
40-
libSourceFile = this._ts.createSourceFile(fileName, libFileContent, scriptTarget);
41-
this._libSourceFileCache.set(fileName, libSourceFile);
42-
}
63+
return path.normalize(fileName) === path.normalize(filePath) ? sourceFile : undefined;
64+
},
65+
writeFile: (name, text) => {
66+
if (path.extname(name) === '.map') {
67+
sourceMapText = text;
68+
} else {
69+
outputText = text;
4370
}
44-
45-
return libSourceFile;
4671
},
47-
// eslint-disable-next-line @typescript-eslint/no-empty-function
48-
writeFile: () => {},
4972
getDefaultLibFileName: () => 'lib.d.ts',
5073
useCaseSensitiveFileNames: () => false,
5174
getCanonicalFileName: (fileName) => fileName,
5275
getCurrentDirectory: () => '',
5376
getNewLine: () => os.EOL,
54-
fileExists: (fileName) => fileName === filePath,
77+
fileExists: (fileName) => {
78+
return path.normalize(fileName) === path.normalize(filePath);
79+
},
5580
readFile: () => '',
5681
directoryExists: () => true,
5782
getDirectories: () => [],
5883
};
59-
this.program = this._ts.createProgram([filePath], this._compilerOptions, compilerHost);
60-
61-
return this._ts.transpileModule(fileContent, {
62-
fileName: filePath,
63-
transformers: this._makeTransformers(this.configSet.resolvedTransformers),
64-
compilerOptions: this._compilerOptions,
65-
reportDiagnostics: this.configSet.shouldReportDiagnostics(filePath),
66-
});
84+
85+
this.program = this._ts.createProgram([filePath], options, compilerHost);
86+
this.program.emit(
87+
undefined,
88+
undefined,
89+
undefined,
90+
undefined,
91+
this._makeTransformers(this.configSet.resolvedTransformers),
92+
);
93+
94+
return { outputText: outputText ?? '', diagnostics, sourceMapText };
6795
}
6896

6997
protected _makeTransformers(customTransformers: TsJestAstTransformer): ts.CustomTransformers {
70-
const allTransformers = super._makeTransformers(customTransformers);
98+
// eslint-disable-next-line @typescript-eslint/no-non-null-assertion
99+
const program = this.program!;
71100

72101
return {
73-
...allTransformers.after,
74-
...allTransformers.afterDeclarations,
102+
...super._makeTransformers(customTransformers).after,
103+
...super._makeTransformers(customTransformers).afterDeclarations,
75104
before: [
76-
...(allTransformers.before ?? []),
77-
// eslint-disable-next-line @typescript-eslint/no-non-null-assertion
78-
replaceResources(this.program!),
79-
// eslint-disable-next-line @typescript-eslint/no-non-null-assertion
80-
angularJitApplicationTransform(this.program!),
81-
],
105+
...customTransformers.before.map((beforeTransformer) =>
106+
beforeTransformer.factory(this, beforeTransformer.options),
107+
),
108+
replaceResources(program),
109+
angularJitApplicationTransform(program),
110+
] as Array<ts.TransformerFactory<ts.SourceFile> | ts.CustomTransformerFactory>,
82111
};
83112
}
84113
}

0 commit comments

Comments
 (0)
Please sign in to comment.