I have a custom controller factory (very basic implementation that is relevant to my question):
public class MyControllerFactory : DefaultControllerFactory
{
public override IController CreateController(RequestContext requestContext, string controllerName)
{
var controller = base.CreateController(requestContext, controllerName);
return controller;
}
protected override System.Web.SessionState.SessionStateBehavior GetControllerSessionBehavior(RequestContext requestContext, Type controllerType)
{
return base.GetControllerSessionBehavior(requestContext, controllerType);
}
}
Which is registered in global.asax
like this:
protected void Application_Start()
{
ControllerBuilder.Current.SetControllerFactory(typeof(MyControllerFactory));
// other code here
}
Breakpoints set in this controller factory are hit properly when there is no conflict in resolving the controller (e.g. only one controller is found in the resolution process). However, when I have two controllers with the same name (in my case: one in an area, one not in an area) without using the namespaces constraint in the route, my controller isn't hit, and the DefaultControllerFactory
takes over and throws an (expected) exception.
My Questions
Am I registering MyControllerFactory
properly? Is there some other reason that it's not getting used in the case I've outlined above?
What I'm trying to do
I'm trying to write a ControllerFactory
that automatically uses controllers that are defined in a (config-specified) area (i.e. if a conflict occurs, like above, use the area controller instead of the non-area controller). If the area doesn't contain a matching controller, fall back to the one that is not in an area.
You could try using Global Namespace Prioritization. You use the
Add
method ofControllerBuilder.Current.DefaultNamespaces
to add namespaces that should be given higher priority. All the namespaces that are added are given the same priority, it doesn't matter what order you add them in, but they are given higher priority than namespaces that are not added. So you could add your config-specified namespace with this method and that should cause it to search there first. You just add it in yourApplication_Start
, maybe after your line to register your controller factory:You can also add a ".*" after the namespace name to have it search child namespaces as well.
You could also directly implement the
IControllerFactory
interface instead of extending theDefaultControllerFactory
if the default implementation is getting in the way when there is a name collision.