Open abhinav opened 1 year ago
I've run across the exact same scenario, with caches but also things like SMS, Email, S3 mocks. Fx seems to be fighting against idiomatic interface usage here and makes it difficult to effectively use interfaces to describe varying implementations.
The workaround I've used for now is to simply construct everything together, rather than provide a redis client as a separate dependency in the graph. This does work fine for now and I can't see it becoming a bigger problem unless this issue arises with deeper dependency trees (though I tend to avoid that with this sort of infrastructure)
func Build() fx.Option {
return fx.Options(
fx.Provide(func(cfg config.Config) (Store, error) {
switch cfg.CacheProvider {
case "":
return local.New(), nil
case "redis":
client, err := rueidis.NewClient(rueidis.ClientOption{
InitAddress: []string{cfg.RedisHost},
})
if err != nil {
return nil, fault.Wrap(err, fmsg.With("failed to connect to redis"))
}
return redis.New(client), nil
}
panic("unknown cache provider: " + cfg.CacheProvider)
}),
)
}
Feature request
Is your feature request related to a problem? Please describe.
This scenario was previously described in #1131. It's reproduced below to save a click.
Suppose a program has a
Store
interface with two implementations: a memory-backed version and a Redis-backed version.The memory store has no dependencies, while the Redis store depends on a Redis client.
The application wants to pick between the two implementations based on a configuration parameter. Imagine:
However, as a result of this,
NewStore
has a hard transitive dependency on the Redis client, even if the Redis store won't be used.Ideally,
NewStore
should be able to specify which dependency set it wants to use after looking at the configuration.Describe the solution you'd like
I'd like to propose adding a new intrinsic operation to the Fx model to accompany the existing intrinsics: Provide, Invoke, and Decorate.
The new operation, tentatively named
fx.Evaluate
, will represent an unevaluated part of the graph.Functions passed to
fx.Evaluate
will always be evaluated. They will feed new information to the graph in the form offx.Option
return values. These new operations may form connections that previously did not exist.With this operation, the scenario described above would be solved like so:
Describe alternatives you've considered
fx.Replace
was considered but is insufficient for this functionality. It cannot conditionally severe the dependency betweenNewStore
and*redis.Client
. See this comment.Is this a breaking change? No, this is not a breaking change.
Additional context This functionality has been brought up before and has been referred to with names such as: monadic graph, dynamic provides, late provides.
There was a prior attempt to implement this in #699 (tracked in #698). This attempt was rejected because of the following reasons per this comment.
fx.Provide
. This is not an issue if we introduce a new intrinsic.fx.Decorate
. This is done now.Fx startup roughly takes the following shape:
With the new intrinsic, this could effectively become: