Feature Request/Question: Lift the explicit type requirement in assertions for participation in CFA
See original GitHub issueSuggestion
đ Search Terms
explicit type, assertion, CFA
â Viability Checklist
- 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, new syntax sugar for JS, etc.)
- This feature would agree with the rest of TypeScriptâs Design Goals.
â Suggestion
As stated in #32695, functions using asserts
require explicitly typed/annotated like so:
declare let x: unknown;
const aFoo = a.literal("foo");
aFoo(x); // error
// Assertions require every name in the call target to be declared with an explicit type annotation.(2775)
// input.ts(2, 7): 'aFoo' needs an explicit type annotation.
const _aBar = a.literal("bar")
const aBar: typeof _aBar = _aBar;
aBar(x); // works
let test1: "bar" = x
declare let y: unknown;
a.string(y) // works
let test2: string = y;
namespace a {
export function literal<T>(t: InferLiteral<T>){
return function(v: unknown): asserts v is T {
if (v !== t) throw new Error()
}
}
export function string(v: unknown): asserts v is string {
if (typeof v !== "string") throw new Error();
}
}
type InferLiteral<T> =
| (T extends string ? T : string)
| (T extends number ? T : number)
| (T extends boolean ? T : boolean)
The reason stated in the PR is âThis particular rule exists so that control flow analysis of potential assertion calls doesnât circularly trigger further analysis.â
My question is, umm, what does this mean in more layman terms? My impression was itâs to not allow recursive usage but looks like thatâs not the case because this compiles⌠So not sure what the above statement means
namespace a {
export function literal<T>(t: InferLiteral<T>){
return function(v: unknown): asserts v is T {
+ let _aLol = a.literal("lol")
+ let aLol: typeof _aLol = _aLol;
+ let x = {} as unknown;
+ aLol(x)
+ let test: "lol" = x;
if (v !== t) throw new Error()
}
}
Also I understand itâs a âdesign limitationâ but, excuse my ignorance, is it really that hard to lift it? Because itâs quite annoying to make make two variables for the same function then annotated the other, makes me think âEh why canât the compiler do this for itselfâ haha. Maybe lift the restriction in some scenarios?
đ Motivating Example
đť Use Cases
I was writing a fail-fast parser that basically composes assertion functions something like this⌠But I have to use the mentioned workaround. Even if this is too much of a feature request, an explanation why the requirement exists would make me feel less annoyed when I redeclare and annotate assertions đ
Playground (Hit on run to see the ParseError
with message "At Person.age: Expected a number"
)
namespace a {
export const string: Asserter<string> =
(v, p) => invariant(typeof v === "string", "Expected a string", v, p)
export const number: Asserter<number> =
(v, p) => invariant(typeof v === "number", "Expected a number", v, p)
export const object =
<O extends { [_ in string]: Asserter }>(tO: O):
Asserter<{ [K in keyof O]: O[K] extends Asserter<infer T> ? T : never }> =>
(v, p) => {
invariant((v): v is object => typeof v === "object", "Expected an object", v, p);
for (let k in tO) {
(tO[k] as any)((v as any)[k], `${p}.${k}`)
}
}
export class ParseError extends Error {
constructor(message: string, public actual: unknown, public path: string) {
super(`At ${path}: ${message}`)
}
}
function invariant(test: boolean, message: string, actual: unknown, path: string): void
function invariant<T, U extends T>(test: (v: T) => v is U, message: string, actual: T, path: string): asserts actual is U
function invariant<T>(test: (v: T) => boolean, message: string, actual: T, path: string): void
function invariant(test: boolean | ((v: unknown) => boolean), message: string, actual: unknown, path: string) {
if (typeof test === "function" ? !test(actual) : !test)
throw new ParseError(message, actual, path);
}
}
type Asserter<T = unknown> = (v: unknown, path: string) => asserts v is T
const _aPerson = a.object({ name: a.string, age: a.number })
const aPerson: typeof _aPerson = _aPerson;
let person = { name: "Devansh", age: true } as unknown;
aPerson(person, "Person");
let test: string = person.name
Issue Analytics
- State:
- Created 2 years ago
- Reactions:4
- Comments:5 (3 by maintainers)
Top GitHub Comments
I definitely want this, but since itâs apparently too hard to do it, could this at least be a request for a âquick fixâ that gives you some idea what you should be doing? https://github.com/microsoft/TypeScript/issues/34596#issuecomment-548084070
crosslinking to #33622 for the error message implementation and to #32695 for the assertion function implementation