Description
Context
With the death of macros, I've been investigating how to improve Riverpod's syntax without relying on code-generation.
The goals are:
- Reducing the number of providers (ideally just 1)
- Supporting any form of function for
family
.
Specifically, this issue is triggered by the second bit.
One motivation behind code-generation was to support a broad scope of provider parameters, including named parameters, optional ones, and even default values.. Lately, we even got generic support, and can write:
@riverpod
Future<T> generic<T>(Ref ref, {String search = ''}) => ...;
I've been investigating how to achieve the same without code-generation. I have various ideas in mind, but they have one flaw:
We loose the ability to override an entire Family
:
ProviderScope(
overrides: [
genericProvider.overrideWith((ref, arg) => ...)
]
)
At least, we can't do it in a type-safe way.
But we can still override one provider:
ProviderScope(
overrides: [
genericProvider<int>(search: 'John').overrideWith((ref) => ...)
]
)
Now comes the main topic:
Is loosing the ability to override an entire family
in a type-safe way a blocker?
My assumptions are that:
- overriding an entire
family
is primarily used by tests - tests could get away with overriding one provider instead of the whole family.
Possibly workaround: A non-typed test-only way to override entire families.
Given my assumption that family.overrideWith
is mostly test-specific, it's less important that the API is fully type-safe (because a test will fail if there's an issue)
With that in mind, we could offer a function which looks like:
@visibleForTesting
Override overrideWith(
Object? Function(Ref ref, Object? arg) cb,
)
The idea is that such function would not be typed.
- The return value may not match what the provider expects (in which case we'll get a cast error in the test)
- The argument will be typed as
Object?
, and you'll need to cast it if your stub relies on arguments to work
Of course, code-generation would still be fully typed (and wouldn't be test-only).
Alternatively: Just remove family.overrideWith
and keep it a code-gen-only feature
If the unsoundness of the proposed workaround is a no-go, we could just remove the API entirely.
Folks could migrate their tests by overriding one provider at a time instead of the whole block. That might be more tedious sometimes, but there's not much else we could do.
Or just don't change the family
syntax if we can't have overrideWith
This is the worse-case-scenario. We'd keep the current syntax for the sole sake of overrideWith
.
This feels incorrect, as I doubt many people override entire an family
often ; especially in production code.
So, in summary, here are a few questions for you:
- Do you use the ability to override entire families in your apps?
- Do you do so only inside tests?
- If you do, would you mind migrating from
provider.overrideWith((ref, arg) => ...)
toprovider(<value>).overrideWith((ref) => ...)
? - Would you care if
provider.overrideWith((ref, arg) => )
was test-only and wasn't type-safe? - If you use
provider.overrideWIth((ref, arg) =>...)
in production code, what's the use-case?