在IIS预热步骤中加载ASP.NET MVC视图

8
我最近开始尝试使用IIS的 IProcessHostPreloadClient 接口通过一个预热步骤来优化我的Web应用程序(请参见此处了解如何设置)。这个方案看起来非常棒,或者至少我认为是这样的,因为我实现了一些“聪明”的事情,比如遍历我的控制器并渲染它们来预加载视图。
经过一些尝试和错误,我让它工作了,一切都很好。直到我注意到系统的所有验证都不再有效,无论是客户端还是服务器验证。我想验证通常是在MVC第一次检索视图时钩入的,而我没有这样做。是否有人知道如何在我的解决方案中包含验证,或使用其他方式实现?
代码:
public class Warmup : IProcessHostPreloadClient
{
    public void Preload(string[] parameters)
    {
        //Pre-render all views
        AutoPrimeViewCache("QASW.Web.Mvc.Controllers", @"Views\");
        AutoPrimeViewCache("QASW.Web.Mvc.Areas.Api.Controllers", @"Areas\Api\Views\", "Api");
    }

    private void AutoPrimeViewCache(string controllerNamespace, string relativeViewPath, string area = null)
    {
        var controllerTypes = typeof(Warmup).Assembly.GetTypes().Where(t => t.Namespace == controllerNamespace && (t == typeof(Controller) || t.IsSubclassOf(typeof(Controller))));
        var controllers = controllerTypes.Select(t => new { Instance = (Controller)Activator.CreateInstance(t), Name = t.Name.Remove("Controller") });

        foreach (var controller in controllers)
        {
            var viewPath = Path.Combine(HostingEnvironment.ApplicationPhysicalPath, relativeViewPath + controller.Name);
            var viewDir = new DirectoryInfo(viewPath);
            if (viewDir.Exists)
            {
                var viewNames = viewDir.EnumerateFiles("*.cshtml").Select(f => f.Name.Remove(".cshtml")).ToArray();
                PreloadController(controller.Instance, area, viewNames);
            }
        }
    }

    private void PreloadController(Controller controller, string area, params string[] views)
    {
        var viewEngine = new RazorViewEngine();

        var controllerName = controller.GetType().Name.Remove("Controller");
        var http = new HttpContextWrapper(new HttpContext(new HttpRequest(null, "http://a.b.com", null), new HttpResponse(TextWriter.Null)));
        var routeDescription = area == null ? "{controller}/{action}/{id}" : area + "/{controller}/{action}/{id}";
        var route = new RouteCollection().MapRoute(
            "Default", // Route name
            routeDescription, // URL with parameters
            new { controller = "Home", action = "Index", id = UrlParameter.Optional }
        );

        var routeData = new RouteData(route, route.RouteHandler);
        routeData.Values.Add("controller", controllerName);
        if (area != null)
        {
            routeData.Values.Add("area", area);
            routeData.DataTokens.Add("area", area);
        }
        routeData.DataTokens.Add("controller", controllerName);
        routeData.Values.Add("id", 1);
        routeData.DataTokens.Add("id", 1);
        var controllerContext = new ControllerContext(http, routeData, controller);
        var vDic = new ViewDataDictionary();
        var vTemp = new TempDataDictionary();

        foreach (var view in views)
        {
            var viewResult = viewEngine.FindView(controllerContext, view, null, false);
            if (viewResult.View == null)
                throw new ArgumentException("View not found: {0} (Controller: {1})".Args(view, controllerName));
            var viewContext = new ViewContext(controllerContext, viewResult.View, vDic, vTemp, TextWriter.Null);
            try { viewResult.View.Render(viewContext, TextWriter.Null); }
            catch { }
        }
    }
}
3个回答

3
微软发布了一个新的模块作为IIS 8.0的一部分,该模块替代了以前的预热模块。这个Application Initialization Module for IIS 7.5可以单独下载。下载链接
该模块将创建一个预热阶段,在此阶段中,您可以指定必须完成的请求数量,然后服务器才开始接受请求。这些请求将以比您尝试实现的更强大的方式执行和编译所有视图。
我在如何在IIS 7.5上预热ASP.NET MVC应用程序?中回答了类似的问题,并提供了更多细节。

3

这不是直接回答你的问题,但我认为你应该看一下David Ebbo的Precompiling MVC Razor views using RazorGenerator

这么做的一个原因是避免在站点启动时有任何运行时影响,因为在运行时没有剩余的编译工作。对于视图较多的站点,这可能会产生很大的影响。


我使用了一些全局的Razor助手方法,但无法使Razor生成器与它们配合工作。除此之外,它看起来是一个非常有用的工具。 - Morten Christiansen
1
我不确定预编译视图是否值得这样做。我认为新的IIS应用程序初始化及其支持重叠进程回收更好地解决了这个问题。 - Jacob Hamacher

3
问题不在于问题中的代码,而是执行代码的时间。将代码移动到操作中允许我无问题地执行预热步骤。在我的情况下,我想我只需让安装过程在系统配置完成后调用预热操作即可。

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接