Skip to content

RFC: Do Families need the abillity to be overridden? #3995

Open
@rrousselGit

Description

@rrousselGit

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) => ...) to provider(<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?

Metadata

Metadata

Assignees

Labels

Projects

No projects

Milestone

No milestone

Relationships

None yet

Development

No branches or pull requests

Issue actions