Recientemente comencé a jugar con la capacidad de IIS para aplicar un paso de calentamiento a mi aplicación web mediante el uso de la interfaz IProcessHostPreloadClient
(busque here para obtener orientación sobre cómo configurar esto arriba). Esto funcionó muy bien, o al menos pensé que sí, porque una de las cosas 'inteligentes' que hice fue intentar precargar mis puntos de vista al iterar sobre mis controladores y representarlos.Precargar vistas ASP.NET MVC en el paso de calentamiento IIS
Después de un poco de prueba y error, lo hice funcionar y todo estaba bien. Es decir, hasta que noté que ya no funcionaba toda la validación para mi sistema, ni la validación del cliente ni la del servidor. Supongo que la validación normalmente está conectada a las vistas cuando MVC recupera una vista por primera vez y no lo hice. ¿Alguien tiene una idea de cómo esto podría incluirse en mi solución o quizás de otra manera?
El código:
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 { }
}
}
}
Utilizo algunos métodos globales Razor helper y no consigo que el generador Razor funcione con ellos. De lo contrario, parece ser una herramienta muy útil. –
No estoy seguro de que valga la pena precompilar las vistas. Creo que la nueva inicialización de la aplicación IIS con su soporte para el reciclaje de procesos superpuestos resuelve mejor el problema. –