-
Notifications
You must be signed in to change notification settings - Fork 12.6k
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
'identity' modifier to indicate a function's parameter-less returns should be narrowed like a value #60948
Comments
Keeping the narrowing inside the declare const value: () => string | undefined;
if (value() !== undefined) {
whatever();
console.log(value().toUpperCase());
} Also this working only in getter/setter-style signals is potentially limiting, for example I use unified signals, with a single function that is both a getter and a setter, because there are many benefits to doing that in a typed codebase (which essentially prevents the common issues of this approach from happening), and if this worked only for getters it would be of no use to me, basically. In general the deeper problem seems about somehow detecting when various types of narrowing should be invalidated, which seems very hard to express and very difficult for the type checker to check for. Worth exploring this area though, as signals are getting more popular. |
The same already happens with property narrowing so this wouldn't be a new problem at all. This would just use the same tradeoffs as the ones mentioned in #9998 |
Isn't the core problem in this example that declare const value: (() => string) | (() => undefined);
if (value() !== undefined) {
console.log(value().toUpperCase());
} Then there's a very straightforward path to adding a narrowing rule that allows a normal interpretation of narrowing |
Also an ELI5 explanation for why it's not correct to write |
Such a type might work for the simple case, but isn't really generalizable. If you consider the most basic implementation of a signal-like type: class Signal<T> {
constructor(private value: T) {}
get(): T { return this.value; }
set(value: T): void { this.value = value }
} it would be very difficult / infeasible to type
It's overhead compared to the experience with plain properties. Sometimes you have multiple levels of operations or multiple reads, which would result in a proliferation of temporary variables. More critically, while if (x()) {
createNewContext();
x().value;
} it might be important to record that |
A distributive conditional type would be correct, though you would only be able to narrow a union ( type PossibleFuncs<T> = T extends unknown ? () => T : never;
declare class Signal<T> {
constructor(value: T) {}
get: PossibleFuncs<T>
set(value: T): void;
} |
Since where the read(function execution) happens matters in Signals libraries hoisting is incredibly clunky for a lot of cases. Especially in templating. Like think of JSX where everything is an expression not a statement. Most templating languages are effectively similar. Signal libraries tend to be granular in their rendering so components/templates don't re-run on a whole. Only parts that change re-execute. So the Signal function needs to be accessed in a very specific scope to trigger the right execution. Not sure of people's familiarity but this is why often these sort of libraries can't destructure props. Because you can't access the getters at the top of the component but instead in the expression closest to where they are used. Different problem but part of the same mechanisms that are present here. You don't always have a place to define variables and access the signal that might need to be used way nested down in your template. Places that are conditionally rendered, or parts of loops. A map function atleast can be made a block statement to be fair but even inside it there will be nested expressions so it can become onerous. This is a fundamental aspect of Signals and the more granular people leverage them the more inevitable it will come that it will be painful to try to hoist stuff. |
How do you actually write this in non-declaration code? This fails to type, and doesn't actually put type PossibleFuncs<T> = T extends unknown ? () => T : never;
class Signal<T> {
private value: T;
constructor(value: T) {
this.value = value;
}
get: PossibleFuncs<T> = () => {
return this.value;
}
set(value: T): void {
this.value = value;
}
} You can do this instead, but type PossibleFuncs<T> = T extends unknown ? () => T : never;
interface Signal<T> {
get: PossibleFuncs<T>;
}
class Signal<T> {
private value: T;
constructor(value: T) {
this.value = value;
}
set(value: T): void {
this.value = value;
}
}
Signal.prototype.get = function () {
return this.value;
} As a side-note, Solid calls these kinds of functions |
This also only works for types in the top-level union. If a property of |
Wait, if the idea is that |
I mean, this violates the spirit of your argument as well: class Foo {
value: number | null = null
reset() {
this.value = null
}
}
const foo = new Foo()
foo.value = 123
foo.reset()
// not a type error
foo.value.toFixed() |
I suppose if you're building with explicitly TypeScript in mind the ideal implementation is something like this: class Signal<T> {
private _value: T;
constructor(value: T) {
this._value = value;
}
get value(): T {
// ... signal tracking logic
return this._value;
}
set value(value: T) {
// ... signal tracking logic
this._value = value;
}
} But this wouldn't work for e.g. Solid, which just has accessors like EDIT: Also, the explicit function call syntax (so |
Okay, but it seems extremely nearsighted to ship a new CFA feature that is going to immediately run into another feature request before it's considered useful. Bringing up the current CFA trade-offs as relates to properties doesn't seem like a useful way to address this concern. |
You're not wrong, but I think the feature would still be helpful regardless. The trade-offs are a "known quantity" to me, and this doesn't introduce anymore of them, right? It would be basically identical to how getters behave (except with no setter, I guess) EDIT: I think with this feature it would be important for the naming and docs to get across it's intended for accessors, so as long as you don't yield to anything, you will know what the value is and it can't change on you (...usually), like a property. |
As an interesting anecdote, Svelte's old style of stores basically handled this problem via the magic <script>
import { writable } from 'svelte/store';
const value = writable<number | null>(null)
// magic reactive store access
if ($value !== null) {
// $value is now typed as non-null
console.log($value.toFixed())
}
</script> Kind of like Signals, Svelte's compiler would track where you used these magic accesses. The Svelte LSP would basically trick the TypeScript LSP into thinking EDIT: There is also an old references proposal for JS which also touches on this problem, specifically this issue about making the concept of references extensible. Food for thought. |
The idea is not that it always returns the same value, but that it has the same expectations as a property accessor - it can be assumed to be stable within the narrowing context of a conditional statement or expression. For Angular, we see significant value in narrowing the getter type and don't view invalidation of this narrowing on Even with property getters, narrowing on the setter has never really been sufficient. Many constructs have other methods which invalidate their getters, and it tends not to be an issue in practice. Basically, we'd strongly prefer the convenience of narrowing the getter function regardless of whether that narrowing was invalidated by the setter. |
I think that is a good summary for why this feature might be useful. I don't know if it would actually have the "same" expectations though, because all property accesses are type-checked like that, but not all function calls would be type-checked like that, which seems inconsistent/confusing in a way type-checking property accesses isn't. In general I think a significant problem regarding the utility of this feature is that a signal is not actually the smallest "possibly-reactive" unit, a function is the fundamental "possibly-reactive" unit. Like let's say we have a component like this: function Paragraph({value}: Props) {
return <p>{value}</p>;
} Or a primitive/hook like this: function useDoubled(value: Value) {
return () => unwrap(value) * 2;
} Or a derivation somewhere that looks like this: const doubled = () => value() * 2; For the component and the hook in general you don't want to say that you accept only signal values, that would be ridiculous, and you don't want to say that you accept only signal values or primitives values either, that's still unnecessarily limiting and weird, what you really should say is that you accept a primitive value or a function to a primitive value, basically a non-reactive thing or a possibly-reactive wrapper to the thing, i.e. "if you give me a reactive version of this thing I support reacting to it". For that derivation you don't want to create a signal because that's unnecessary overhead, every time "value" will change "doubled" will change too (assuming -0 and +0 don't matter here), and you are listening to a single signal. Wrapping that function in a memo/computed today would give you absolutely nothing other than verbosity and overhead. Basically the problem is that the second you are dealing with plain functions this special narrowing wouldn't apply anymore, and you want to say you accept plain functions as inputs, because they are the fundamental "possibly-reactive" unit, so the usefulness of narrowing signal getters seems pretty limited. What we actually want, ideally, is for TS to understand when the same function called again will return the same type as before because its return value depends only on the values of the signals it read the last time, and those values couldn't have possibly changed since the last call of the function. What this feature would give us is instead special-casing type checking for signal getters like property accesses, which is a very different beast. Maybe it's still a useful one though? Personally I'm not convinced it would solve a big-enough slice of the problem to be worth supporting, but the problem it is trying to solve is a real problem. |
Worth mentioning also that even if you say that you accept only primitives or signals to primitives, which again is overly limiting but let's pretend it's fine, you just can't reasonably take advantage of this special narrowing either, normally, because are you going to check if every value is a signal before doing something with it? No, you are going to want to have a function that unwraps possibly-reactive values, to delete this annoying branching, so the type narrowing of the signal would not be taken advantage of in many cases. This would only really largely address the problem when one accepts only signals (not unreactive values, nor plain functions to unreactive values), and one makes only signals (not plain functions), which presumably everybody should agree nobody should be doing? That means, just to look at it syntactically, instead of writing |
For what it's worth, if the below would be working for primitive types (including unions, intersections, null, undefined--but not object types), wouldn't it already be a big step forward for signals as implemented in SolidJS?
By casting the signal getter (as returned from createSignal) to PossibleFuncs SolidJS code could nicely expresses to TS that the getter returns a "stable" result. It might not be required to solve use cases with object types, because they are covered by SolidJS's store (which is using proxies and getters) and, I think, it already gets desired type narrowing. Would adding this limitation to primitive types allow to get combinatorial complexity problem under control? Edit: Seems related Method return type cannot be used as discriminant |
We discussed offline for a bit and wanted to find a solution that would be a bit more general-purpose and solve this problem more completely. The basic sketch was something like // Declaration
interface SomeSignal {
// TODO: bikeshed syntax
// When a function is 'stable', repeated calls
// are assumed yield the same value. It's probably
// an error to specify it on not-zero-arg functions?
stable getValue(): string | number;
// This function isn't assumed-stable
rand(): number;
// TODO: *really* bikeshed this syntax
// A mutating method resets all narrowings on the object it's called on
mutating setValue(value: string | number): void;
}
// Usage
function myFunc(s: SomeSignal) {
// Existing discriminant/parent logic makes handling
// the destructuring here a fairly straightforward thing
const { getValue, setValue, rand } = s;
if (typeof getValue() === "string") {
// OK
const ok: string = getValue();
setValue(42);
// This is an error; 'setValue' reset all narrowings
// on s and we know that getValue is from s
const err: string = getValue();
}
// It's useful everywhere
// New declarations in lib.d.ts would include e.g.
interface Array<T> {
mutating sort(): this;
}
const p = ["world", "hello"] as [string, string];
if (p[0] === "world") {
// *All* narrowings on 'p' reset by this call
p.sort();
// Today: no error
// With 'mutating': would error
const m: "world" = p[0];
} However, we're not experts on the whole signal landscape as it stands right now, and aren't sure if this would handle all the different variants currently in the ecosystem. It'd be helpful for people who are familiar with the various libraries to weigh in on whether this would be a sufficient solution so we can better prioritize further investigation on it. |
Seems very promising! I guess with solid's tuple syntax it could look something like this? TS Playground type SignalPair<T> = [getValue: stable () => T, setValue: mutating (value: T) => void];
declare function createSignal<T>(value: T): SignalPair<T>
const [count, setCount] = createSignal<number | undefined>(0);
const [count2, setCount2] = createSignal<number | undefined>(0);
if (typeof count() === 'number') {
// ^? () => number | undefined
count2(); // Unaffected by narrowing
// ^? () => number | undefined
count(); // Narrowed to non-nullable
// ^? () => number
setCount2(2); // Calling setCount2 doesn't affect count, right?
count();
// ^? () => number
setCount(2); // Calling setCount resets count
count();
// ^? () => number | undefined
} |
Probably a subtlety here is that the modifier is attached to the containing type's member list (not the functions themselves) the same way // Calling this does *what*, exactly?
declare const f: mutating () => void; So it implies you'd have // Note modifier movement here
type SignalPair<T> = [stable getValue: () => T, mutating setValue: (value: T) => void]; which in turn means that tuple members need to support modifiers - not a huge problem IMO, it's actually kind of odd that you can't write const p: [readonly s: string, n: number] = ["", 32]; since it's all sort of just sugar for |
declare const f: stable () => string | number; This one could make sense though. Im not familiar with signals myself but lets say a component receives its getter alone. How do u type that property of the props? Ofc u can add the modifier to ur propsβ property but uβd have to remember about doing this. It ends up coloring all consumers and it seems u couldnt just declare a reusable type for this getter function alone. One could use Unless thatβs just not a pattern people are using. Maybe the expectation is that a full signal gets always passed down - in its readonly variant if necessary or smth |
Would wrapping functions inherit the stability of signals it references? Or would we have to declare them stable manually as well? This is a core part of how signals are used in SolidJS. type SignalPair<T> = [stable getValue: () => T, mutating setValue: (value: T) => void]
declare function createSignal<T>(value: T): SignalPair<T>
const [value, setValue] = createSignal<number |Β undefined>()
// Will this derived signal be stable?
const doubleValue = () => value && value * 2 |
@RyanCavanaugh that looks fantastic! Definitely interface Signal<T> {
// slight nuance - Signal<T> is a function with additional members. TS would need to understand that
// `mutating` on the members relates to `stable` on the function itself.
stable (): T;
mutating set(value: T): void;
mutating update(fn: (value: T) => T): void;
} If so, I can confidently say that this design would be ideal for Angular Signals. |
Good points above about how interface Signal<T> {
stable (): T; // NOTE: how to differentiate this from a _method_ named "stable"?
} (which I think is necessary) would suggest that Other interesting use cases to consider:
|
I'm really not sure about this proposal, there's a Example of a confusing case that could be quite common: interface MutableSignal<T> {
stable get(): T;
mutating set(v: T): undefined;
}
interface Signal<T> {
stable get(): T;
}
const a: MutableSignal<string> = state("hello");
const b: Signal<string> = derived(() => a.get().toUpperCase());
if (b.get() === "HELLO") {
a.set("world");
b.get(); // BUG: type is still narrowed to "HELLO" despite the value changing to "WORLD"
} Signals are built for the purpose of composing state derivation. Getters have the same issue, but I don't think it would be as commonly encountered. Further narrowing after a dependency has mutated would also become very problematic. if (b.get() === "HELLO") {
a.set("world");
if (b.get() === "WORLD") {
// ^ This comparison appears to be unintentional because the types '"HELLO"' and '"WORLD"' have no overlap.(2367)
}
} Even with casting to allow the condition the narrowed type never changes. if (b.get() === "HELLO") {
a.set("world");
if ((b as Signal<"WORLD">).get() === "WORLD") {
b.get(); // WRONG type "HELLO"
}
}
function assertType <T>(v: any): asserts v is T {}
if (b.get() === "HELLO") {
a.set("world");
assertType<Signal<"WORLD">>(b);
if (b.get() === "WORLD") {
b.get(); // WRONG type never
}
} These aren't new problems, they exist with getters (TS playground). However I do think it's far more likely to run into the issue with signals. I'd much rather encourage users to assign the value to a local variable: let bVal = b.get();
if (bVal === "HELLO") {
a.set("world");
bVal = b.get();
if (bVal === "WORLD") {
bVal; // CORRECT type "WORLD"
}
} Kinda wish there was a strict option to disable property access narrowing, and if this lands I'd defiantly want a way to disable it for my own projects. |
I understand being tooling authors/maintainers we are accustomed to thinking about edge cases. Because let's face it, if we make something possible people will abuse it.
The reason is with Signals mutation for the most part(if not always) should be limited to specific scopes. Pure computation(derivations) shouldn't mutate. This limits it to event handlers, and denoted effectful portions of the code. Code that is effectively imperative. Code where block statements exist and reactive scope does not matter. Not only is this a much smaller fraction of the code(like 10% maybe) but it is place where you can set a local variable. But the other 90% of the code where there is no mutation, where templating logic lives, where primitives compose, where you build the graph, and where assigning local variables is unreasonable benefit immensely from I'm also fine with the current proposal because we are talking about such a small slice of a slice. It is consistent with getter/setters today and atleast this way it is explicit and can apply to other functions. |
@ryansolid Can you give an example outside of templates where assigning to a local variable is unreasonable? I do not think we should be creating a feature which has it's primary use case tailored to non standard template syntax, where something like a TS language service plugin could do the work. My concern is this changes the failure mode of signals in block scope reactive areas from a non narrowed type safe annoyance, to one which reduces type safety and can introduces runtime errors. Users are unlikely to know TS is producing incorrect types from overly optimistic narrowing, so they wouldn't do the variable assignment to workaround. Not all signals libs have restriction about reactive scopes, in some it's totally fine to mutate signals anywhere even inside derivations, and reactivity can remain stable and active. Even still just considering event handler contexts I think this would be enough of a footgun there. This change has the potential to steer signal libs into queuing reactive propagation till after event handlers, since that would remove the type safety issue of mutations affecting derivations. #57725 (pure/impure fn/accessor distinction) has the potential to allow a safe way for narrowing signals, and improve the safety of getter/prop narrowing. |
Wouldn't Ryan's suggestion (using function unions) address most of the declare const value: (() => string) | (() => undefined);
if (value() !== undefined) {
console.log(value().toUpperCase());
} The type Signal<T> = {
value: () => T | undefined,
setValue: (value: T) => this.value is () => T, // Calling this function removes undefined from 'value'
} |
Nuclear option for |
This is generally how we see this feature. It's always possible to construct examples where TypeScript's soundness falters, but our experience with signals so far mirrors @ryansolid's - this tends not to be an issue in everyday code. Re: @robbiespeed
For what it's worth, we explored the idea of implementing narrowing as a feature of our language service pretty extensively, and concluded that it was infeasible. The only workable way to achieve the narrowing semantics is to transform incoming expressions and extract the result of stable function calls into temporary variables. This assumes that you can know which of those function calls are stable (are signal reads), which is already not possible without running through type-checking already. We were even willing to treat all zero-arg calls as stable, but it turns out this variable extraction is difficult to impossible to do while preserving all the other type flow operations and semantics. Additionally, we generally want TypeScript to behave the same inside and outside of templates - narrowing within |
I argee that function foo(getValue: stable () => null | Foo) {
return derived(() => {
if (getValue()) {
getValue(); // should be of type Foo
}
});
} |
π Search Terms
indicate function returns identical value cached memoized signals
β Viability Checklist
β Suggestion
Many apps today are built on the concept of using small "getter" functions as wrappers around values. For example, Signals as implemented in Angular, Solid, and the stage 1 TC39 Signals proposal often look something like:
Signals users have struggled with using them in TypeScript because, at present, there isn't a way to get that code block to type check without type errors. Signals users know that the result of
value()
must bestring
inside theif
, but TypeScript doesn't have a way to note that the result should be type narrowed. Common workarounds today include!
,?.
, and refactoring to store intermediate values. All of which are at best unnecessary verbosity, and at worst conflict with frameworks.Request: can we have a keyword -or, failing that, built-in / intrinsic type- to indicate that calls to a function produce a referentially equal, structurally unchanging value? In other words, that the function call (
value()
) should be treated by type narrowing as if it was just a variable reference (value
)?Proposal: how about an
identity
modifier keyword for function types that goes before the()
? It would be treated in syntax space similarly to other modifier keywords such asabstract
andreadonly
.π Motivating Example
When an
identity
function is called, it is given the same type narrowing as variables. Code like this would now type check without type errors, as ifvalue
was declared asconst value: string | undefined
:Narrowing would be cleared the same as variables when, say, a new closure/scope can't be guaranteed to preserve narrowing:
π» Use Cases
One difficult-to-answer design question is: how could
identity
handle functions with parameters? I propose the modifier not be allowed on function signaturess with parameters to start. It should produce a type error for now. The vast majority of Signals users wouldn't need signatures with parameters, so I don't think solidifying that needs to block this proposal. IMO that can always be worked on later.Furthermore, it's common for frameworks to set up functions with a parameter-less "getter" signature and a single-parameter "setter" signature. I propose for an initial version of the feature, calling any other methods or setting to any properties on the type should clear type narrowing:
More details on the difficulties of signals with TypeScript:
If a new modifier keyword isn't palatable, a fallback proposal could be a built-in type like
Identity<T>
. This wouldn't be a new utility type (FAQ: no new utility types); it'd be closer to the built-in template string manipulation types.The text was updated successfully, but these errors were encountered: