-
Notifications
You must be signed in to change notification settings - Fork 12.8k
[isolatedDeclarations] Add a syntactic form of computed property name which is always emitted as a computed property name #58800
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. Weβll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Comments
I guess what I donβt understand is, if youβre already capable of doing this check for an arbitrary expression in the |
Given // @filename: node_modules/mod/index.ts
export const something = Math.random() ? 1 : 2; then import {something} from "mod";
export const a = {[something]: 1}; has no errors, while import {something} from "mod";
export const a = {[something satisfies keyof]: 1}; issues an error on and import {something} from "mod";
export const a = {[something]: 1}; emits export const a: {[something: number]: number}; in accordance with the type of the resolved import {something} from "mod";
export const a = {[something satisfies keyof]: 1}; always emits, even if import {something} from "mod";
export const a: {[something]: number}; |
It's basically a way to pull the error the declaration file would get from preserving the computed property name forward into the input file. |
Ah, so it's basically a syntactic hint not to widen it to an index signature, we always want it to show up as a computed property in types. |
Since |
@weswigham Do you think this should be a special case of a more general affordance to annotate a wider range of values to satisfy isolated declarations? For example, if we wanted to do export const values = [Color.ORANGE, Color.PURPLE] as const; we have to write out something like: export const values: readonly [Color.ORANGE, Color.PURPLE] = [Color.ORANGE, Color.PURPLE];
or
export const values = [Color.ORANGE, Color.PURPLE] satisfies readonly [Color.ORANGE, Color.PURPLE] as readonly [Color.ORANGE, Color.PURPLE];
or
export const values = [Color.ORANGE satisfies Color.ORANGE as Color.ORANGE, Color.PURPLE satisfies Color.PURPLE as Color.PURPLE] as const; so we might want something like export const values = [Color.ORANGE satisfies const, Color.PURPLE satisfies const] as const;
or more conveniently
export const values = [Color.ORANGE, Color.PURPLE] satisfies const; // typechecking error if either of these aren't actually `const` Then maybe for consistency we'd have |
Hi, is this still open? |
π Search Terms
isolatedDeclarations transpileModule computed property name
β Viability Checklist
β Suggestion
Background
Computed property names under
isolatedDeclarations
are very limited right now. Today, you can write{[Symbol.iterator]: ...}
and that's about it. This restriction is in place because for an arbitrary{[expression]: ...}
we don't know if the type should be{[expression]: something}
,{[expression: string]: something}
or even{}
(or a future{f1: something} | {f2: something}
). Computed property names in types today have to exactly be a single late bindable name - nothing more, nothing less - meanwhile computed property names in object expressions (and class declarations) are much more flexible in what we allow.Thus far, this has worked pretty well for TS users, since we basically pre-solve and cache whatever the expression computed name resolves to into our declaration files. Unfortunately, for
isolatedDeclarations
users, this poses a problem, since theexpression
in the computed property name may be from or rely on type information from another file. In such a case, it's impossible to know how to emit the type for the expression. You could optimistically emit{[expression]: something}
, but ifexpression
ends up evaluating tostring
orany
in a whole-program context, the declaration file will produce an error and incorrect type information.Proposal
What we could use in such a scenario is a syntactic opt-in to guaranteeing the preservation of a computed property name in the calculated type for an expression. A form of computed property name that, when you see it, always ensures a computed property name appears in the output, and issues checker errors if the types when checked cannot produce a valid computed property name in a declaration file.
I propose we reuse some existing syntax with a bit of a new meaning to accomplish this - a
satisfies keyof
postfix assertion, only valid in computed property name positions, and only on dotted entity name expressions. This would mean you could writeand we would always emit
and issue an error on
something satisfies keyof
ifsomething
isn't exactly a single unique symbol, string literal, or number literal type (as is valid in the type position computed property name).Compatibility
Only
isolatedDeclarations
-concerned authors really need to think about this feature - it's erased from declaration files, since they already check this constraint, so library consumers will never see it. People not usingisolatedDeclarations
will never be driven to use it, since they will always be able to produce a declaration type without an assertion. This is pretty easy to integrate into theisolatedDeclarations
quickfixer. This doesn't conflict with existingsatisfies keyof T
assertions, since they require a type argument forkeyof
. There is also the possibility of allowingsatisfies keyof
in other locations and on arbitrary expression kinds in the future to check the same invariant - that the expression is exactly a single literal key type - if we think such a check has use in broader contexts than just computed property names.Addenda: Making error cases better
Once we have
{[expression satsifies keyof]: ...}
in place, we know that that computed property name should always produce exactly one object key, even ifexpression
doesn't produce a valid key type (and thus an error). In such a scenario, it could be beneficial to override the type ofexpression
with a property key unique to theexpression
symbol, and then fallback to using such a symbol whenever laterobj[expression]
lookups fail. In this way, we can preserve as much user intent as possible, without rapidly reverting to an uncheckedany
state. This is neat (I have a working prototype), especially in the context of single-file checking modes like what our language service does when loading the full program in the background, but isn't really necessary for the feature. The open questions I have for this are justkeyof
result of a type containing one of the fallback property keys be adjusted to bestring | number | symbol
? Should the fallback error property just be filtered fromkeyof
entirely?The text was updated successfully, but these errors were encountered: