Register / Resolve T or IEnumerable<T>
dansiegel opened this issue · 8 comments
Description
As initially discussed in issue #115 we have an issue where IEnumerable may need to be resolved. This is a particular issue with Shiny integration where multiple delegates may be registered and will need to be resolved.
Problem
Prism registers services with an expectation that we will internally only want to resolve the last registration for any particular service. However it is possible for developers, or 3rd party plugins such as Shiny to require registering multiple services as is the case for Delegates. In this scenario it would be normal to see:
c.Register<ISomeDelegate, DelegateA>();
c.Register<ISomeDelegate, DelegateB>();
Expectations
What we need is a proper ruleset for the Container so that:
- When we
Resolve<ISomeDelegate>()
we should getDelegateB
- When we
Resolve<IEnumerable<ISomeDelegate>>()
we should get back a collection with bothDelegateA
andDelegateB
.
NOTE: The registrations cannot directly reference the underlying container since this should transparently work with the Prism Container Abstractions and the integration where we convert from IServiceCollection to register each Service Descriptor with the Prism Container.
Observations
Currently we use the default IfAlreadyRegistered.Replace
, this satisfies the first requirement, but fails on the second as we only get a collection with DelegateB.
If we do not set the IfAlreadyRegistered behavior or set it explicitly to Append, we succeed on the second requirement but fail on the first.
cc: @dadhi... any thoughts?
@dansiegel Likely you'll need
rules.WithFactorySelector(Rules.SelectLastRegisteredFactory())
instead of IfAlreadyRegistered.Replace
, which is rather destructive policy anyway (because you overriding the regisrations).
Here is the docs: https://github.com/dadhi/DryIoc/blob/master/docs/DryIoc.Docs/RulesAndDefaultConventions.md#resolving-from-multiple-default-services
@dadhi sorry to keep bugging...
As was being discussed in @dahlbyk's PR... there is a related issue here that in another place we have a check for IsRegistered...
Since we assume IsRegistered should return true whether we are checking for T or IEnumerable. The following change was proposed, but I'm not sure if this is ideal or if it can be simplified.
return Instance.IsRegistered(type) ||
Instance.IsRegistered(type, factoryType: FactoryType.Wrapper);
Also in further testing I realized that we do have an issue with keyed services not appending... is there a way to get keyed services to append instead of throwing?
The only work around I've found there is to force a Replace for keyed services
The only work around I've found there is to force a Replace for keyed services
Yes, for keyed this is the only way I imagine at the moment, because the keyed services are supposed to be unique.
IsRegistered should return true whether we are checking for T or IEnumerable.
IEnumerable wrapper is always registered whether or not you are registered a service type, so that the enpty collection will be returned for the non registered type.
So you may just check for the registered type or check for the type.IsGeneric && type.GetGenericTypeDefinition() == typeof(IEnumerable<>)
Thanks @dadhi... we could certainly check if it's an IEnumerable<T>
in the resolve. I feel that it would potentially create some bugs though if we did that as part of our IsRegistered check since you really don't have anything registered for it.
we could certainly check if it's an
IEnumerable<T>
in the resolve. I feel that it would potentially create some bugs though if we did that as part of our IsRegistered check since you really don't have anything registered for it.
Since DryIoc supports resolving a few different collection types (everything implemented by object[]
, if I understand the code correctly), it seems preferable to avoid repeating that logic.
Is there a downside to checking if a wrapper is registered for the specified type? Expected behavior for resolving a collection for an unregistered type is an empty collection, which seems consistent with considering IEnumerable<>
or whatever to indeed be registered.
DryIoc supports resolving a few different collection types (everything implemented by object[]
It is correct. Plus its own LazyEnumerable
.
Is there a downside to checking if a wrapper is registered for the specified type?
No downside I am aware of. That way you will check for whatever wrapper suported by DryIoc.