замок PerRequestLifestyle не узнают - PullRequest
5 голосов
/ 20 апреля 2010

Впервые в Касл / Виндзор, пожалуйста, потерпите меня.

В настоящее время я использую фреймворк System.Web.Mvc.Extensibility и в своем коде запуска он зарегистрировал HttpContextBase следующим образом:

container.Register(Component.For<HttpContextBase>().LifeStyle.Transient.UsingFactoryMethod(() => new HttpContextWrapper(HttpContext.Current)));

Я хотел изменить поведение и изменить образ жизни httpContextBase на PerWebRequest.

поэтому я изменил код на следующий:

container.Register(Component.For<HttpContextBase>().LifeStyle.PerWebRequest.UsingFactoryMethod(() => new HttpContextWrapper(HttpContext.Current)));

Однако, когда я делаю это, я получаю следующую ошибку:

 System.Configuration.ConfigurationErrorsException: Looks like you forgot to 
 register the http module Castle.MicroKernel.Lifestyle.PerWebRequestLifestyleModule
 Add '<add name="PerRequestLifestyle" 
 type="Castle.MicroKernel.Lifestyle.PerWebRequestLifestyleModule, Castle.MicroKernel" 
 />' to the <httpModules> section on your web.config

, что я сделал в <system.web> и <system.webServer>, однако, я все еще получаю ту же ошибку. Есть намеки?

Заранее спасибо.

Обновление

добавлен код блока для запроса

В рамках system.web.mvc.extensibility существует класс extendedMvcApplication, который наследуется от HttpApplication, а в методе Application_start он вызывает BootStrapper.Execute (). Эта реализация этого метода следующая:

public void Execute()
    {
        bool shouldSkip = false;

        foreach (IBootstrapperTask task in ServiceLocator.GetAllInstances<IBootstrapperTask>().OrderBy(task => task.Order))
        {
            if (shouldSkip)
            {
                shouldSkip = false;
                continue;
            }

            TaskContinuation continuation = task.Execute(ServiceLocator);

            if (continuation == TaskContinuation.Break)
            {
                break;
            }

            shouldSkip = continuation == TaskContinuation.Skip;
        }
    }

Как видите, он просматривает список IBootStrapperTask и пытается выполнить их. Случилось так, что у меня есть одна задача, которая регистрирует маршруты в моем приложении mvc:

public class RegisterRoutes : RegisterRoutesBase
{
    private HttpContextBase contextBase;

    protected override TaskContinuation ExecuteCore(IServiceLocator serviceLocator)
    {
        contextBase = serviceLocator.GetInstance<HttpContextBase>();
        return base.ExecuteCore(serviceLocator);
    }

    protected override void Register(RouteCollection routes)
    {
        routes.IgnoreRoute("{resource}.axd/{*pathInfo}");
        routes.IgnoreRoute("{*favicon}", new { favicon = @"(.*/)?favicon.ico(/.*)?" });
        routes.IgnoreRoute("{*robotstxt}", new { robotstxt = @"(.*/)?robots.txt(/.*)?" });

        XmlRouting.SetAppRoutes(routes, contextBase.Server.MapPath("~/Configuration/Routes.xml"));
    }
}

вы можете видеть, что мне нужно получить Instance (разрешить) объект httpcontextbase, чтобы я мог получить путь к серверу XML-файла.

Ответы [ 2 ]

7 голосов
/ 20 апреля 2010

На момент написания статьи стиль жизни PerWebRequest не поддерживает разрешение в Application_Start ().

См. Описание проблемы и обсуждение:

Обходные пути для этого конкретного случая:

  1. Зарегистрировать RegisterRoutes в качестве экземпляра, явно передав ему текущий контекст в качестве параметра конструктора, например ::

    container.Register(Component.For<IBootstrapperTask>()
                                .Instance(new RegisterRoutes(Context)));
    
  2. Используйте HostingEnvironment.MapPath вместо contextBase.Server.MapPath. Хотите сделать это издевательским? Используйте его через простой интерфейс, например ::1010

    interface IServerMapPath {
        string MapPath(string virtualPath);
    }
    
    class ServerMapPath: IServerMapPath {
        public string MapPath(string virtualPath) {
            return HostingEnvironment.MapPath(virtualPath);
        }
    }
    
    container.AddComponent<IServerMapPath, ServerMapPath>();
    

Затем введите IServerMapPath в ваш RegisterRoutes.

0 голосов
/ 20 апреля 2010
...