Introduce new flag for minifier optimization hints
See original GitHub issueSearch Terms
Optimization Minifier Bundler IIFE Wrapper Terser
Suggestion
TypeScript 3.9 pulled in PR #32011, which tailors TypeScript’s output for optimization with minifiers such as Terser. This PR definitely goes in the right direction for optimizing TypeScript’s emit for most users, but doesn’t necessarily work for all toolchains out there. At Google we have optimization processes after TypeScript compilation that use static analysis to look for classes, do some rough type-checking, and for example rename class properties and drop methods. The wrapper introduced in TypeScript 3.9 hinders our ability to analyze these classes.
We’d like to propose a new flag to toggle this behavior, which I’ll call --add-optimization-hints
for now. This flag would control if TypeScript decides to emit these hints for optimizers and bundlers, which is not necessarily limited to the /** @class */
form introduced in TypeScript 3.9. For users such as Google, we can disable this flag and instead rely on our own optimizers to handle this, but users piping directly through Webpack and Terser can still receive the benefits from #32011.
Use Cases
Removing IIFE Wrapper for ES2015 Classes
Expanding on our toolchain a bit from above, at Google we use a combination of tsickle, closure-compiler, and other tools to optimize our TypeScript code. Tsickle is used to add type annotations and use the goog.module
module system to TypeScript’s emit such that Closure Compiler can optimize the code. Using the TypeScript 3.9 RC which emits IIFE wrappers around ES2015 classes Closure Compiler is unable to understand that this is an alias for the inner class. By setting --add-optimization-hints=false
we would be able to statically analyze TypeScript’s output without doing additional special processing.
Adding Further Optimization Hints
PR #32011 isn’t the first to request optimization hints from the TypeScript output, for example #13721 requested a /** #pure */
annotation as well, and there could be more in the future that TypeScript would like to hint at. Adding this flag allows TypeScript to produce messier code that is less similar to the input, but more compatible with bundlers and minifiers.
Examples
This would be a new flag for the TypeScript compiler. Assuming TypeScript 3.9 is being used, given the input:
// input.ts
class C {
static prop = 10;
getProp() {
return C.prop;
}
}
Compiling with tsc --add-optimization-hints=true
:
// output.js
"use strict";
let C = /** @class */ (() => {
class C {
getProp() {
return C.prop;
}
}
C.prop = 10;
return C;
})();
Compiling with tsc --add-optimization-hints=false
:
// output.js
"use strict";
class C {
getProp() {
return C.prop;
}
}
C.prop = 10;
Checklist
My suggestion meets these guidelines:
- This wouldn’t be a breaking change in existing TypeScript/JavaScript code
- This wouldn’t change the runtime behavior of existing JavaScript code
- This could be implemented without emitting different JS based on the types of the expressions
- This isn’t a runtime feature (e.g. library functionality, non-ECMAScript syntax with JavaScript output, etc.)
- This feature would agree with the rest of TypeScript’s Design Goals.
Issue Analytics
- State:
- Created 3 years ago
- Reactions:15
- Comments:17 (8 by maintainers)
Top GitHub Comments
Thanks for taking a look into this! We really appreciate the thought going into this issue. We (@IgorMinar, @mprobst, @rrdelaney) met, here’s our perspective:
We think rolling back for the time being is the right thing to do, and we should reconsider the approach taken in the PR.
A potential solution might be annotating TS emit on a semantic level (e.g.
/** @staticproperty */ MyClass.staticProp = 1
), which would allow downstream tools to easily process the output without special casing for a particular optimizer. This also seems to be more in line with TypeScript’s design goals, in particular 3. and 4. This will need more research though.This is already 6 months old, what s the status on this? When can we see this at list with a flag? I stumbled upon as i was trying to figure out why tree shaking was not working in Nativescript. Nativescript is heavily based on Typescript and decorators. Tree shaking makes a big difference in app startup time. Without that flag/fix we will never be able to improve things further.
EDIT: did anyone did that with a transformer. We might be able to detect static/decorators and wrap in that case? I am not that familiar with transformers but maybe someone found a way.