Originally published at
www.ikriv.com. Please leave any
comments there.
After upgrading to a newer version of Unity, we found that our process began to fail with StackOverflowException. Investigation showed that it happened because the following registration behaves differently depending on the version of Unity:
container.RegisterType(new InjectionFactory(...));
Github project ikriv/UnityDecoratorBug explores the behavior for different versions. Strictly speaking, the registration is self-contradictory: it may mean either
// (1) Resolve TInterface to whatever is returned by the InjectionFactory
container.RegisterType(new InjectionFactory(...));
or
// (2) Resolve TInterface to container-built instance of TType
container.RegisterType();
Early versions of Unity ignored the TType part and used interpretation #1. In the
release notes for version 5.2.1 it was declared that such registrations are no longer allowed, but in reality they were silently interpreted as (2), ignoring the InjectionFactory part.
This led to a catastrophic problem in our code. We had a decorator type similar to this:
public class Decorator : IService
{
public Decorator(IService service) {...}
}
and the registration declaration was
container.RegisterType(
new InjectionFactory(c => new Decorator(c.Resolve())));
If the injection factory is ignored, this leads to an infinite recursion when resolving IService: IService is resolved to Decorator, that accepts IService as a parameter, that is resolved to Decorator, that needs IService as a paramter, that is resolved to Decorator…
In Unity version 5.7.0 this was fixed and such registrations throw an InvalidOperationException.
To recap:
Unity version
Effect of the registration
<=5.2.0
Injection factory used
5.2.1 to 5.6.1
Injection factory ignored, StackOverflowException for decorators
>=5.7.0
InvalidOperationException
Lessons learnt:
- If something is not allowed, it must throw an exception
- Silently changing interpretation of existing code is evil.
PS. This is probably the first time when
stackoverflow.com actually lived to its name: usually I search for other kinds of exceptions there. 🙂