Description
π Search Terms
javascript ecmascript standard decorator this static field accessor
π Version & Regression Information
- This is an error in every version I tried (from 5.0 upward)
β― Playground Link
π» Code
The following code incorrectly crashes when it's transformed by TypeScript and then run:
const dec: Function = () => { }
const ctx = { dec }
function wrapper(this: typeof ctx) {
@(this.dec) class Foo {
@(this.dec) method() { }
@(this.dec) static method() { }
@(this.dec) field: undefined
@(this.dec) static field: undefined
@(this.dec) get getter(): undefined { return }
@(this.dec) static get getter(): undefined { return }
@(this.dec) set setter(x: undefined) { }
@(this.dec) static set setter(x: undefined) { }
@(this.dec) accessor accessor: undefined
@(this.dec) static accessor accessor: undefined
}
}
wrapper.call(ctx)
console.log('success')
π Actual behavior
The value of this
in decorators of static fields and static accessors is inconsistent with the value of this
in all other kinds of decorators (run this code vs. this code).
π Expected behavior
I expected the value of this
to be the same for all kinds of decorators. The value of this
is consistent when this code is compiled with Babel.
Additional information about the issue
For context, I'm starting to look at implementing JavaScript decorators in esbuild. I know this feature hasn't been standardized yet and is still being iterated on, but many of my users are asking for esbuild to implement it. My first step was to write a lot of tests: https://github.com/evanw/decorator-tests. That led to me discovering this bug in TypeScript.
Other bugs I noticed:
-
TypeScript thinks the
@
in() => @dec class {}
is a syntax error (demo here). My reading of the specification is that this should be valid asPrimaryExpression
containsClassExpression
which starts withDecoratorList opt
. Babel accepts this code. -
TypeScript generates code containing syntax errors if
await
is used in a decorator. This is allowed by the specification as far as I can tell. Here is an example: (link). Babel printstrue
for this code while TypeScript emits invalid code. One simple hack to fix this could be for TypeScript to useawait (async () => { await ... })()
instead of(() => { await ... })()
in this case. -
Referencing the class name within a decorator seems to return
undefined
instead of throwing aReferenceError
. Both TypeScript and Babel do this so I'm not sure if this is a bug with both, or if it's not a bug and the specification is outdated, or if I'm misreading the specification. Assuming this is a bug with both, it may be somewhat important to fix as people will surely try to do that (e.g. like this) and aReferenceError
seems more clear than a value ofundefined
.