jueves, 25 de agosto de 2016

Battlevoid: Harbinger - Tips para completar el juego

Objetivos

Hola amigos, en esta ocasión les daré algunos tips para pasar el juego Battlevoid: Harbinger (antes llamado Battlestation: Harbinger pero como dice en su cuenta de Twitter @BattlestationPC, por problemas legales se le debió cambiar el nombre). Lo compré para Android cuesta $2.900 (pesos Chilenos), unos 4.5 USD. Para PC lo encuentras en Steam a 9,99 €, unos 11 USD. Este juego es un Roguelike espacial con elementos de batalla y estrategia, muy adictivo y entretenido. Me ha tenido una y otra vea intentado pasarlo, aún no puedo pero estoy a punto. Al ser roguelike dejo en claro que es que si mueres, comienzas todo de nuevo, pero de igual forma ganas experiencia para desbloquear otras naves. Son mapas random, por lo que una partida nunca es igual a otra. Permite jugarlo offline.
Tiene una comunidad (en inglés) activa: http://bugbyte.fi/forums/forumdisplay.php?fid=10, así que les animo a que lo jueguen.

Parte con algo fácil

  • Empieza en fácil, con un mapa autogenerado pero limitado (presionando el dado de abajo a la izquierda), y no infinito.
  • Parte con una nave potente de las pocas que hay al comienzo. Ve las características de cada una (presionando en el triangulo de arriba a la derecha) y quédate con la mejor.

Dinero

  • Chatarra espacial o tuercas: reúne esa chatarra que parecen tuercas (o engranajes) para comprar módulos de naves o naves amigas, las tuercas son tu dinero. La obtienes al ganar batallas o completar misiones. Al eliminar una nave grande obtienes más que un pequeña, pero al completar una misión ganas mucho más.
  • Cubos: a veces recogerás cubos grises que son parte de alguna misión, no debes venderlos, sino llevarlos alguna estación que lo necesite, según los objetivos de dicha misión.
  • Materiales verdes: reúne estos ítems para hacerle upgrade a tus armas, drones o naves ayudantes.
  • Vende lo que no necesites: pero ojo que si lo vendes, obtienes una parte muy pequeña de lo que lo compras, así que úsalo como último recurso.
  • Vende equipo alien pero sólo si encuentras uno mejor: equipo alien se vende bien, pero primero úsalo ya que en general tiene mejoras. Véndelo sólo si encuentras uno mejor.
  • Vende tus armas sólo en caso muy necesario: por ejemplo quieres un cañón de largo alcance, y te faltan 150 tuercas, vende una de tus armas chicas ya que vale la pena, lo mismo para comprar un cañón nuclear.

Equipo

  • Al comienzo tienes 3 naves para seleccionar y cada uno con slots diferentes. Hay 3 tipos de slots:
    • Slot rojos: permite incorporar artillería de ataque como un cañón láser o cañón nuclear. Son los más útiles.
    • Slot azules (o calipso): armas de defensa, que son de menor calibre y rango como el bolter, que es el más barato.
    • Slot verdes: el hangar, donde puedes construir bolters, láser, reparar, naves no tripuladas o drones. Los bolters son los más baratos y más efectivos por el precio, o los drones son la mejor opción.
    • Naves extras: Existe una variedad con diferentes precios. Ve bien tu estrategia para decidirte por una nave cargador (carrie) o una de asalto ya que podrás actualizar estas naves luego. Puedes comprar hasta dos naves ayudantes, si te alcanzan las tuercas claro.
  • Las naves que compres vienen vacías: Si tienes suficientes tuercas y compras una nave, esta viene sin armas, así que no compres a menos que tengas unas 300 tuercas extras para comprar la nave más un arma básica.
  • Mejora tu equipo: cada equipo que compres tiene características para ser mejorado. Una buena idea es mejorar la característica de número de naves para el equipo Drones para poder sacar más drones y de forma rápida.
  • Bolter drones, baratos y eficientes: son la mejor opción para ahorrar, hasta que te alcance para un cañón de asalto. Con 3 bolter drones y un bolter de defensa puedes eliminar una nave grande, lo que te dará buen dinero. Si eres bueno, en poco tiempo puedes eliminar varias naves grandes y te alcanza para el cañón nuclear que es una de las mejores armas del juego.

Estación

  • Reparar es gratis: si estás en la misma pantalla de una Estación, basta tocar tu nave cuando esta dañada para que aparezca el ícono de reparar, si aceptas reparará gratis tu nave. En cambio, si estás en una pantalla donde no está la estación y no quieres volver a la zona de la Estación, te sale dinero reparar. Vuelve a la estación cada vez que puedas, dando saltos espaciales (si estás muy lejos de una, mira primero en el mapa que no hayan enemigos en el camino. Mira las flechas rojas como se mueven, indican la dirección donde se moverán de las naves enemigas)
  • La estación te defiende: una gran ventaja es que si naves enemigas te siguen acá, la estación tiene una muy buena defensa que te ayudará a defenderte. Un truco es atraer las naves enemigas a la Estación con el radar SOS que dejan los enemigos a veces, así las destruyes más rápido.
  • La estación vende buen equipo: la estación tiene un muy buen equipo para vender en general, sobretodo armas aliens poderosas. Con solo tocar la Estación sale el ícono de comprar (un ícono parecido a un carro de compra).

Mapa

  • Los símbolos azules son aliados: Las tuercas azules son Estaciones, las flechas azules naves o drones aliados.
  • Cuando saltes, escoge bien el cuadrante donde caerás: Al moverte a una nueva ubicación, puedes seleccionar en los 9 cuadros donde caerás, si arriba, al centro, abajo-derecha, etc. Ve bien ya que si hay muchos enemigos y no tienes poder de ataque, mejor ataca de lejos y eliges una posición lejana a ellos.
  • Los símbolos rojos son enemigos: algunos tienen forma de flecha, donde apunta la flecha es donde se están moviendo o donde se moverán, por lo que puedes ver por ejemplo si uno se te acercará en el siguiente turno.
  • Salta un turno a veces: en el mapa verás un ícono de un reloj de arena verde, si lo presionas pasará un turno. Es útil para cuando veas que algunas naves enemigas se mueven a lo lejos y quieres pasar por un área donde justo ellas acaban de dejar.
  • Usa la antena SOS: ese icono como radar que se ve en el mapa, a la izquierda del reloj, te la entregan los enemigos, y sirve para llamar la atención de los mismos (como que tu eres un enemigo y estás pidiendo apoyo) y si están a un turno de ti, lo más seguro vendrán a ayudar, y ahí es cuando es bueno para tenderles una emboscada. Sólo úsala cuando estés en una pantalla junto a la Estación, así llamas la atención de los enemigos, se acercarán y la Estación te dará un buen apoyo para destruirlas. Esta antena, solo atrae a cierto tipo de razas, no todas. En inglés a este ícono le llaman Distress Beacons.
  • Recoge tecnología alien verde: luego de terminar una batalla, aparecerá a veces un ícono de exclamación abajo, si lo tocas saldrá la tecnología alien que dejó alguna nave, esta la puedes equipar en tus slot azules o rojos. Muchas veces es la misma que tu ya tienes equipada, pero con mejores upgrades.

Naves

Las Naves jugables, son de dos tipos: Fighters (ataque) o Carriers (cargadores) y estas a la vez, son de dos tipos, cargadores livianos o de asalto:
  • Nightingale: nave tipo cargador y de asalto liviano, es la primera del juego que puedes usar.
  • Raven: nave de asalto, tiene un casco más duro.
  • Guardian: nave cargadora y de asalto liviano, es un poco mejor que Nightingale, pero más lenta, se desbloquea al nivel 4.
  • Zephyr: destructor, se desbloquea al nivel 4.
  • Hurricane – nave de asalto, se desbloquea al nivel 5.
  • Resolution – nave cargadora y de asalto grande, la nave más cara, se desbloquea al nivel 6.
  • Memphis – destructor, se desbloquea al nivel 7.
  • Avalon – nave cargadora y destructor liviano, se desbloquea al nivel 8.
  • Valhalla – nave cargador y destructora grande, se desbloquea al nivel 9.
  • Achilles – nave crucero, se desbloquea al nivel 10.
  • Liberator – nave cargador y crucero liviano, la nave con mejor escudo, se desbloquea al nivel 11.
  • Armada: la mejor, se desbloquea en el nivel 12.
Naves extras que te pueden ayudar:
  • Chimera – nave cargadora chica, una de las más rápidas, con un nivel de 24 de velocidad.
  • Genesis – nave exploradora y cargadora chica.
  • Viper – nave exploradora, la más rápida con los mejores propulsores.
  • Falcon – nave de asalto, tiene uno de los mejores cascos y velocidad.
  • Ravager – nave de asalto.
Define tu estrategia con las naves
  • Naves de asalto no tienen slot verdes: esto afecta a que por ejemplo el destructor Zephyr no podría llevar drones. Tiene varias ranuras para armas, y tres rojas para cañones de grueso calibre.
  • Usa dos light carrier (cargador liviano) con drones: una alternativa es tener una nave de asalto y a la vez naves extras cargadoras ya que es una forma económica de tener una pequeña escuadra, ya que los Drones Luchadores parten de las 250 tuercas, luego puedes mejorar y comprar Drones Bolter por 600 tuercas.
  • Usa una nave principal carrier con drones: otra alternativa es tener un cargador que tienen menos slots de armas pero tienen slot para drones.
  • Las naves ayudantes tienen espacio para ítems: si tienes items que no quieres vender, puedes dejarla en la sección azul de items de las naves ayudantes, así vez si te sirven en un futuro.
  • Mi escuadra favorita: Una BSE Zephyr que tiene tres slot rojos, lo equipo con dos Torrente de Partículas (4600 tuercas) y un Cañón Nuclear (2500 tuercas) y dos naves Nightingale con un dron de misiles, un Torrent de Partículas y un Cañón Nuclear.

Armas

  • Armas Celestial o armas aliens: ve bien los estatus de cada arma si tienes que decidir entre una y otra
  • Las Estaciones venden muy buenas armas: no olvidar que las armas de las estaciones están mejoradas.
  • Los bolter son los más baratos y efectivos.
  • Teleport es para cobardes: si estás a punto de morir y quieres huir, cómpralo, pero personalmente si muero, muero con mi nave y no como un cobarde jaj, además si mueres ganas experiencia para desbloquear mejores naves :P

Nunca te distraigas, y arranca a veces

  • Nunca, pero nunca te distraigas: Me pasó más de una vez, que por estar mirando la barra de salud de una de mis naves amigas, me atacaron la nave principal y morí. Esto luego de horas de juego :(
  • Si tu energía está al 25%, ¡corre!: No te las des de confiado, si ves el marco rojo de la pantalla, es que debes arrancar a un sitio seguro, ojalá a una Estación.

Anexo a los tips: Libro de ilustraciones

Fuera de los tips y estrategias, quería comentarles que me gustó mucho que el juego tiene una historia de fondo, de razas y conquistas. En el foro se puede leer de ella: http://bugbyte.fi/forums/forumdisplay.php?fid=7
También los fans hacen ilustraciones del universo del juego. Entre las más interesantes son las de las razas, colocaré aquí algunas traducciones:

Los Wanderers
Hemos viajado a través de las galaxias durante mucho tiempo. Nuestro planeta nos fue arrebatado por lo que hemos estado obligados a vivir en nuestras naves. Todos nos llaman Los Wanderers, nosotros nos hacemos llamar sobrevivientes. El tiempo nos ha dado fama de despiadados, y ya no vamos a esperar a ser atacados, sino que vamos a atacar primero. Una sombra infinita y eternamente creciente ha surgido en la vida, sin embargo, no vamos a ser los que se detienen a orar. Estamos aquí, queremos tus naves.

Los Trolgars
La historia trata de la esclavitud que ha pasado de generación en generación en este pueblo en el pasado. Los Trolgars quieren defender su planeta con más fuerza que nunca lo que les pertenece. Ahora son libres, y ahora también quieren reclamar más espacio para su pueblo. Nunca volverán a dejar que nadie se acerca a sus tesoros. Los Trolgars mezclan y combinan cada pieza de tecnología que encuentran y son buenos carroñeros en esa área. En las ciudades de los Trolgars el pasado está presente sólo en las chimeneas negras que se elevan a los cielos, y las historias que se cuentan sobre Haik-u, el Trolgar cuyos pies fueron atados con grilletes metálicos.

Las Schillae
Alguna vez fue una raza pacífica, pero ahora está en busca de la guerra, y lo cantan. La canción es sobre el amor, la traición y la esperanza, pero en su mayoría, trata de venganza. Estas mujeres, sirenas abominables, han sido traicionadas de una manera horrible. Les han arrebatado su amor y esperanza y las han empujado al infierno. Les quitaron los machos de su propia raza y rezan a los machos de otras razas, especialmente a los machos humanos. Ahora que quieren venganza, necesitan un plan. Ellas necesitan ayuda, y necesitan la tecnología. Por encima de todo, ellas tienen que saber quién les ha hecho esto. Hasta que sepan, ellas dudarán de todos.

Los Celestials
Estos seres de energía una vez que se unificaron en una sola luz madre. Esta luz fue infectada por la oscuridad, y necesitaba una manera de detener la pérdida de color por completo. Los hijos de la luz, unas diminutas esferas radiantes, vagaron por el espacio. Su única esperanza era chocar con algo que podría ayudarles. Cuando descubrieron a los seres humanos, encontraron una respuesta, no fue una cura, sino una manera de sobrevivir.

Los Unknown (Los Desconocidos)
Desde el principio, los Unknown ya vivían. Acababa de empezar una evolución larga y sinuosa, pero desde la primera fracción de segundo se sentían que estaban vivos. Su morada se expandió a la misma velocidad que el universo. Las galaxias se arremolinaron y condensaron, nacieron de las estrellas. En cada estrella, los Unknown estaban viviendo una vida silenciosa y sin pretensiones. Con curiosidad, ellos mantienen un seguimiento de la evolución del universo, pero sin intervenir. Ellos ven todo lo que pasa, y cada estrella o planeta recién nacido, ya les pertenece por herencia.

Link útil

lunes, 30 de mayo de 2016

Tutorial de uso de la librería Winnovative HTML to PDF con C# y MVC

Hola, hace rato que no publicaba nada...Bueno hoy veremos como usar la librería Winnovative HTML to PDF Converter Library usando C# y MVC .Net. El tema nació en que buscamos una forma buena y elegante de imprimir un PDF a partir de un sitio web hecho en MVC .Net, pudiendo personalizar el PDF, tipo de hoja, orientación, que permita imprimir un sitio que haga request Ajax y tenga imágenes.

Personalmente conozco 3 herramientas que permiten generar PDF desde C# y me quedo con Winnovative por lo anterior dicho:


El tutorial fue probado con IE 11, Chrome 50.0.2661.102, FireFox 37.0.1.

Requisitos


  • Tener la librería Winnovative HTML to PDF Converter Library v12.16, descargarla de http://www.winnovative-software.com/download.aspx. En mi caso bajé Winnovative HTML to PDF Converter Library Optimized for 64-bit ya que tanto los PC de los desarrolladores como servidores son de 64 bit.
  • En mi caso usaré una versión comprada HTML to PDF Converter Redistributable License cuyo valor es de 1200 USD ya que permite instalarlo en N Servidores, incluyendo en PC de desarrolladores y en servidores de clientes. No es obligación tener la licencia para el ejemplo, lo que sí, saldrán tus PDF con la frase Winnovative PDF Tools Demo en la parte inferior:
  • Para el ejemplo usaré Visual Studio 2013 con C#, MVC 4 y Framework 4.0
NOTA: la librería no es compatible con el Framework .Net 3.5, si con las versiones 2.0, 4.0 y 4.5.

Paso a paso


Pasos generales

  • Crear un proyecto MVC 4 usando C# con Framework 4.0.
  • Vamos a instalar la librería Winnovative "a la antigua", es decir manualmente, ya que se puede instalar por NuGet. Descomprimir el archivo descargado WnvHtmlToPdf-v12.16-x64.zip, en la raíz del zip estarán estos tres archivos que son compatibles si usas al Framework 4.0: wnvhtmltopdf.dll, wnvinternal.dat y Help.chm.
  • Si usas el Framework 2.0 debes copiar los 3 archivos de la carpeta NET_2.0 (wnvhtmltopdf.dll, wnvhtmltopdf.xml y wnvinternal.dat) al directorio bin de tu proyecto web. El XML contiene sólo la documentación de los fuentes.
  • Si están con Framework 4.0. debes copiarlos a una nueva carpeta LIB. Preocúpate los archivos que no estén chequeados como "solo lectura".
  • El archivo wnvhtmltopdf.dll dentro de LIB debes referenciarlo desde tu proyecto MVC .NET.
  • Agregar en el HomeController el using:
    //HomeController.cs
    using Winnovative;
    
  • Si usas la Key, y la agregas en el código, no se verá el mensaje "Winnovative PDF Tools Demo":
    //HomeController.cs
    htmlToPdfConverter.LicenseKey = "keyComprada";
    

Imprimir a pdf una vista o página externa

  • Crear el vista principal con un botón dentro de un Form que llamará a la impresión, por ejemplo:
    
    @using (Html.BeginForm("ImprimeVistaExterna", "Home", FormMethod.Post))
    {
        
    }
    
  • Creas el Action y la Vista Externa a ser llamada. Esta vista puede tener imágenes, o un llamado Ajax a otra Vista ¡y todo eso se verá impreso!. Esa es la gracia de la librería ya que mucha no hacen eso, esta ejecuta el request con todo lo que conlleva, llamados Ajax, JS, estilos, imágenes y todo el resultado del HTML final, es el que imprime. En mi caso incluso uso la librería Kendo MVC para mostrar un TreView y esta igual se imprime:
    
    @{
        ViewBag.Title = "Impresion ejemplo";
    }
    
    

    Vista demo de impresión

    @(Html.Kendo().TreeView() .Name("LinksTreeView") .Items(items => { items.Add() .Text("Home") .Action("Index", "Home"); items.Add() .Text("About") .Action("About", "Home"); items.Add() .Text("Contact") .Action("Contact", "Home"); }) )
  • Por último creamos la vista externa (la más externa) que es llamada desde al Ajax anterior:
    
    

    Otra Vista externa invocada por ajax

    holaaa

  • Si abrimos la Vista externa en el navegador se verá así:
  • Creas el Action ImprimeVistaExterna dentro del controlador Home:
    //HomeController.cs
            [HttpPost]
            public ActionResult ImprimeVistaExterna()
            {
                // Create a HTML to PDF converter object with default settings
                HtmlToPdfConverter htmlToPdfConverter = new HtmlToPdfConverter();
    
                // Set HTML Viewer width in pixels which is the equivalent in converter of the browser window width
                htmlToPdfConverter.HtmlViewerWidth = 1024;
                
                htmlToPdfConverter.LicenseKey = "tuKey"; //propiedad opcional
                
                htmlToPdfConverter.HtmlViewerHeight = 768;
    
                // Set PDF page size which can be a predefined size like A4 or a custom size in points 
                // Leave it not set to have a default A4 PDF page
                htmlToPdfConverter.PdfDocumentOptions.PdfPageSize = PdfPageSize.Letter;
    
                // Set PDF page orientation to Portrait or Landscape
                // Leave it not set to have a default Portrait orientation for PDF page
                htmlToPdfConverter.PdfDocumentOptions.PdfPageOrientation = PdfPageOrientation.Portrait;
    
                // Set the maximum time in seconds to wait for HTML page to be loaded 
                // Leave it not set for a default 60 seconds maximum wait time
                htmlToPdfConverter.NavigationTimeout = 4;
    
                // Set an adddional delay in seconds to wait for JavaScript or AJAX calls after page load completed
                // Set this property to 0 if you don't need to wait for such asynchcronous operations to finish
                htmlToPdfConverter.ConversionDelay = 1;
    
                string url = "http://localhost:50422/Home/VistaExterna";
    
                // Convert the HTML page given by an URL to a PDF document in a memory buffer
                byte[] outPdfBuffer = htmlToPdfConverter.ConvertUrl(url);
    
                // Send the PDF file to browser
                FileResult fileResult = new FileContentResult(outPdfBuffer, "application/pdf");
                fileResult.FileDownloadName = "ImpresionVistaExterna.pdf";
    
                return fileResult;
            }
    
  • Notar que en url defines la ruta que se imprimirá. Se le hará un request a dicha ruta. Puede ser cualquier URL, incluso una externa como www.google.com
  • Con htmlToPdfConverter.NavigationTimeout se define el tiempo máximo de espera a dicho request. Yo usaré 4 segundos.
  • Con fileResult.FileDownloadName le defines el nombre del archivo PDF de salida.
  • Con htmlToPdfConverter.PdfDocumentOptions.PdfPageSize = PdfPageSize.Letter defines el tipo de papel, en este caso tipo Carta.
  • Con htmlToPdfConverter.PdfDocumentOptions.PdfPageOrientation = PdfPageOrientation.Portrait defines la orientación del documento.
  • Al hacer clic en "Imprimir página Externa", se bajará el PDF resultante y su lo abres es el mismo contenido HTML que tiene la vista externa:


Imprimir a pdf una sección de la página actual

  • La sección debe ser una Vista Parcial, así que creas la vista Seccion.cshtml:
    
    
    @{
        ViewBag.Title = "Seccion ejemplo";
        var nombre = ViewBag.Nombre;    
    }
    
    
    @using (Html.BeginForm("ImpresionSeccionVistaActual", "Home", FormMethod.Post)) {

    Sección: @nombre


    }
    Este texto está rotado 90 grados
  • El ejemplo contiene un texto rotado en 90 grados que se verá impreso de igual forma, por lo tanto debes agregar en la sección de estilos el siguiente código:
    
    
    
  • Creamos en la vista principal un botón Imprimir que llame al Action que haremos luego. Notar que agregamos dos bloques de texto antes y después para que se vea que solo se imprimirá la vista Seccion.cshtml:
    
    

    Texto anterior

    @{ Html.RenderAction("Seccion", "Home", new { nombre = nombre }); } @using (Html.BeginForm("ImpresionSeccionVistaActual", "Home", FormMethod.Post)) { @Html.Hidden("nombre", (object)nombre) }

    Texto posterior

  • En el HomeController crear el Action ImpresionSeccionVistaActual que procesa la sección:
    //HomeController.cs
            [HttpPost]
            public ActionResult ImpresionSeccionVistaActual(FormCollection collection)
            {
                object model = null;
                ViewDataDictionary viewData = new ViewDataDictionary(model);
    
                // transmit the posted data to view
                viewData.Add("nombre", collection["nombre"]);
    
                // The string writer where to render the HTML code of the view
                StringWriter stringWriter = new StringWriter();
    
                // Render the Index view in a HTML string
                ViewEngineResult viewResult = ViewEngines.Engines.FindView(ControllerContext, "Seccion", null);
                ViewContext viewContext = new ViewContext(
                        ControllerContext,
                        viewResult.View,
                        viewData,
                        new TempDataDictionary(),
                        stringWriter
                        );
                viewResult.View.Render(viewContext, stringWriter);
    
                // Get the view HTML string
                string htmlToConvert = stringWriter.ToString();
    
                // Get the base URL
                String currentPageUrl = this.ControllerContext.HttpContext.Request.Url.AbsoluteUri;
                String baseUrl = currentPageUrl.Substring(0, currentPageUrl.Length - "Home/Seccion".Length);
    
                // Create a HTML to PDF converter object with default settings
                HtmlToPdfConverter htmlToPdfConverter = new HtmlToPdfConverter();
    
                // Set license key received after purchase to use the converter in licensed mode
                // Leave it not set to use the converter in demo mode
                htmlToPdfConverter.LicenseKey = key;
    
                // Set an adddional delay in seconds to wait for JavaScript or AJAX calls after page load completed
                // Set this property to 0 if you don't need to wait for such asynchcronous operations to finish
                htmlToPdfConverter.ConversionDelay = 2;
    
                // Convert the HTML string to a PDF document in a memory buffer
                byte[] outPdfBuffer = htmlToPdfConverter.ConvertHtml(htmlToConvert, baseUrl);
    
                // Send the PDF file to browser
                FileResult fileResult = new FileContentResult(outPdfBuffer, "application/pdf");
                fileResult.FileDownloadName = "ImpresionSeccionVistaActual.pdf";
    
                return fileResult;
            }
    
  • Si haces clic en "Imprimir solo sección", se verá el PDF:
  • La imagen de la aplicación final ejecutándose en el navegador con dos botones, Imprimir Sección o Imprimir página externa, se ve así:

Mostrar PDF en una nueva pestaña del navegador al hacer clic en imprimir (y no en la barra de descargas)


Cambiar la clase Controller:
//HomeController.cs
// Convert the HTML page given by an URL to a PDF document in a memory buffer
byte[] outPdfBuffer = htmlToPdfConverter.ConvertUrl(url);            

// Send the PDF file to browser
//FileResult fileResult = new FileContentResult(outPdfBuffer, "application/pdf");            
//fileResult.FileDownloadName = "ImpresionVistaExterna.pdf";
//return fileResult;

MemoryStream stream = new MemoryStream(outPdfBuffer);
Response.AppendHeader("content-disposition", "inline; filename=file.pdf");
return new FileStreamResult(stream, "application/pdf");
Cambiar la vista para que abra en una nueva pestaña con target _black:

@using (Html.BeginForm("ImprimeVistaExterna", "Home", FormMethod.Post, new { target= "_blank" }))
{
    
}


Mostrar PDF en una nueva pestaña del navegador al hacer clic en imprimir y se abra automáticamente el cuadro de diálogos de las impresoras


Esto es útil si el usuario quiere hacer la menor cantidad posible de clic para imprimir un documento PDF. Se le debe asignar el key (si tienes) de nuevo al Document. Recordar igual lo del target _blank en la vista que tiene el botón.
//HomeController.cs
            string url = "http://localhost:50422/Home/VistaExterna";

            // Convert the HTML page given by an URL to a PDF document in a memory buffer
            byte[] outPdfBuffer = htmlToPdfConverter.ConvertUrl(url);                       

            //mergeResultPdfDocument.OpenAction.Action = New PdfActionJavaScript("print()")

            // Send the PDF file to browser
            //FileResult fileResult = new FileContentResult(outPdfBuffer, "application/pdf");            
            //fileResult.FileDownloadName = "ImpresionVistaExterna.pdf";
            //return fileResult;

            MemoryStream stream = new MemoryStream(outPdfBuffer);

            Document document = new Document(stream);
            document.LicenseKey = key;
            document.OpenAction.Action = new PdfActionJavaScript("print()");
            byte[] b = document.Save();
            Stream strm = new MemoryStream(b);
           
            Response.AppendHeader("content-disposition", "inline; filename=file.pdf");
            return new FileStreamResult(strm, "application/pdf");


Links útiles

miércoles, 18 de noviembre de 2015

Entender la Compilación Dinámica de ASP.NET

Entendiendo la compilación dinámica de ASP.NET


Cuando se hace un request a un sitio web, ASP.NET primero parsea y luego compila el código de la aplicación Web para convertirlo en uno o más assemblies. Cuando se compila el código, es traducido a una representación que es independiente del lenguaje y del CPU llamada Microsoft Intermediate Language (MSIL). En tiempo de ejecución, MSIL corre en el contexto del .NET Framework, el cual traduce MSIL en instrucciones específicas para ese CPU y para el procesador que corre la aplicación en dicho computador.



Compilación en el primer request


Por defecto, las páginas web de ASP.NET y el código embebido se compilan dinámicamente cuando los usuarios hacen el primer request a un recurso, como una página web ASP.NET (un .aspx). Luego que las páginas y el código embebido se compila por primera vez, se dejan estos recursos en un caché, así los subsecuentes request a esa misma página son extremadamente eficientes.
ASP.NET soporta Compilación dinámica de las páginas ASP.NET (.aspx), ASP.NET Web services (.asmx), ASP.NET HTTP handlers (.ashx) y ASP.NET application files (Global.asax), pero también otros archivos de código fuente como (*.aspx.vb) y clases (*.aspx.cs). Para más información de los tipos de archivo de ASP.NET, ver Web Site File Types. Para más información del proceso de compilación de ASP.NET, ver la sección "Compilation Life Cycle" de ASP.NET Application Life Cycle Overview.

Recompilación cuando hay cambios


Cualquier cambio de un archivo dinámicamente compilado invalida el assembly que está en caché y gatilla una recompilación de todos los recursos afectados. La próxima vez que se haga un request, ASP.NET reconoce que el código cambió y recompila los recursos afectados de la aplicación web. Este sistema permite desarrollar aplicaciones con un mínimo de sobrecarga en el proceso de compilación. (Notar que dependiendo del cambio que hay sobre un recurso, ASP.NET recompilará una sola página o todo el sitio Web.)

Dependencias de la compilación


Cuando se hace el primer request a la aplicación, ASP.NET compila los archivos en un orden específico. Los primero ítems en ser compilados son los llamados ítems de primer nivel.
Luego del primer request, los ítems de primero nivel se recompilan sólo si cambian las dependencias.
Los ítems de primer nivel incluyen las carpetas App_GlobalResources, App_WebResources, propiedades del perfil, carpeta App_Code y archivo Global.asax. Luego que los ítems de primer nivel se compilan, ASP.NET compila los ítems adicionales. Estos ítems incluyen la carpeta App_LocalResources, páginas ASP.NET (.aspx files), ASP.NET user controls (.ascx), ASP.NET HTTP Handlers (.ashx) y módulos ASP.NET HTTP (.asmx files), themes, master pages y otros archivos fuente. Para más información ver ASP.NET Web Site Layout y ASP.NET Application Life Cycle Overview.

Salida de la compilación


Cuando el código es compilado, los assemblys resultantes son "cacheados" en una carpeta en el servidor. Esta carpeta requiere los permisos apropiados para que tu código compilado se ejecute correctamente. Puedes configurar tanto la ubicación de la carpeta de compilación como los permisos con los que opera tu código compilado.

Ubicación de la carpeta de compilación

Por defecto, cuando compilas una aplicación web, el código compilado queda en la conocida carpeta Temporary ASP.NET Files. Esta carpeta está dentro de la carpeta donde tienes instalado el Framework .Net. La ruta típica es:
%SystemRoot%\Microsoft.NET\Framework\versionNumber\Temporary ASP.NET Files
Por ejemplo, en una máquina de 64 bits, si está instalado el Framework 2.0 tendrías la carpeta:
C:\Windows\Microsoft.NET\Framework64\v2.0.50727\Temporary ASP.NET Files

Permisos requeridos para la carpeta de compilación

El proceso de instalación del Framework .NET crea la carpeta Temporary ASP.NET Files y le asigna permisos de acceso a la cuenta local de ASP.NET, el cual tiene los permisos necesarios para acceder al código compilado. Si modificas la configuración u opciones de la cuenta de usuario, asegúrate que la cuenta tenga suficientes permisos a la carpeta de Temporary ASP.NET Files. Para más detalles, ve How to: Run aspnet_wp.exe Under a User Account.

Configuración de la carpeta de compilación

ASP.NET crea una subcarpeta dentro de la carpeta Temporary ASP.NET File para cada aplicación. Puedes configurar la ubicación raíz usando el atributo tempDirectory de la sección compilación del archivo de configuración. Este atributo opcional permite especificar el directorio para almacenar los archivos temporales durante la compilación. Por defecto es un string vacío (""). En caso de ser un string vacío y si el proceso actual tiene los permisos de acceso necesario, los archivos son guardados en el siguiente directorio:
%FrameworkInstallLocation%\Temporary ASP.NET Files
Para más información, ver el Elemento compilación (ASP.NET Settings Schema) y la propiedad TempDirectory de la sección CompilationSection.

Soporte multilenguaje


ASP.NET 2.0 soporta múltiples lenguajes de programación en una misma aplicación web. En la carpeta App_Code, puedes especificar una subcarpeta para cada lenguaje, como C# y Visual Basic. ASP.NET creará un assembly separado para cada subcarpeta. Para más información ver Shared Code Folders in ASP.NET Web Sites y Walkthrough: Developing Web Sites Using Multiple Programming Languages.

Optimizando la compilación dinámica


A partir de un HotFix del Framework 3.5 se puede usar el atributo optimizeCompilations del web.config. Más info aquí.

Ventajas y desventajas de las compilación dinámica


La compilación dinámica de ASP.NET permite modificar el código fuente sin tener que explícitamente tener que compilar el código ni hacer un deploy en un servidor (Nota del traductor: como ejemplo práctico, permite que tengas un sitio web en un ambiente pre-productivo por ejemplo, donde tienes las dll tanto como los fuentes *.aspx.cs, y poder cambiar estos fuentes en "caliente", le das F5 al navegador y listo, el código se verá reflejado en el sitio. sirve para probar más rápidamente un cambio de un sitio en la capa web. Ahora si el proyecto web, usa un dll de otro proyecto, aunque tengas los fuentes de ese otro proyecto, si los cambias, no tomará el cambio en el proyecto web ya que ahí si requiere compilar dicha dll). Si modificas un código fuente, ASP.NET automáticamente recompila el archivo y actualiza todos los recursos que usan dicho archivo. No es necesario restetear el IIS para que tome el cambio a menos que la sección del web.config haya cambiado. Adicionalmente, puedes extender el sistema de compilación de ASP.NET al crear proveedores para que nuevos tipos de archivos sean llamados durante la compilación. La compilación dinámica beneficia al sistema de compilación de ASP.NET y es compatible con las estructuras y tipos de aplicación antiguas.

Hay ciertos aspectos que la compilación dinámica no ofrece. La Compilación dinámica puede hacer más lento el tiempo de respuesta de la carga inicial de un sitio por parte del usuario, ya que las páginas y archivos de código deben ser compilados la primera vez que son solicitados. Esto podría ser un problema si es un sitio muy grande y es cambiado frecuentemente.
La Compilación dinámica no identifica errores de compilación antes que los usuarios accedan al sitio.
También, la compilación dinámica no provee la funcionalidad de crear una versión compilada de un sitio que ya ha sido deployado en producción sin tener el código fuente.
Si alguno de esos problemas te afecta, lo mejor es precompilar el sitio y deployarlo a producción sólo con los elementos necesarios (sin todos los fuentes), un ejemplo clásico es la opción Publicar de Visual Studio. Para más información, ver ASP.NET Web Site Precompilation Overview.

Fuente


lunes, 16 de noviembre de 2015

Testeando UrlScan 3.1 para IIS 7.5

Instalación


El otro día estaba viendo algo UrlScan, es una herramienta free de Microsoft para analizar requests así protegerte de ataques SQL Injection o Script Injection de forma simple, definiendo parámetros y opciones en un archivo ini.

Bajé la versión de 64 bits de https://www.microsoft.com/en-us/download/details.aspx?id=5728

Leí la documentación de: https://support.microsoft.com/es-es/kb/326444

El .ini queda instalado por defecto en: C:\Windows\System32\inetsrv\urlscan\UrlScan.ini

(Mi .ini que usé lo dejé al final)

La carpeta con los Logs: C:\Windows\System32\inetsrv\urlscan\logs\ le dí permiso de escritura al usuario IIS. En IIS 6 darle permiso de escritura al usuario IIS_WPG, y para IIS 7 u superior dale permisos a IIS_IUSRS.

En mi caso sólo usaré un filtro global, pero si quieres aplicar el filtro sólo para un sitio web, debes registrar la DLL como filtro en Web Sites -> Properties -> ISAPI Filters como dice aquí.

Al final de la instalación, reinicié el IIS con la línea de comandos iisreset.

Pruebas


NOTA: Por cada cambio en el .ini recomiendan Reciclar el Pool del sitio probado para que tome el cambio, si embargo a veces me tomaba sin necesidad de esto.

Al probar con la URL que tiene un "intento" de SQL Injection:

http://misitio.cl/index.asp?select=1

Resultado: Error 404 con detalles



Resultado desde fuera del sitio: Error 404 sin detalles (este error daría si el UrlScan está en producción y queremos acceder desde afuera):



Como nota, el mismo error da cuando escribimos un "intento" de Script Injection:

http://misitio.cl/index.asp?title=<meta%20http-equiv="refresh"%20content="0;">

Logging


El log, en mi caso, urlscan.111615.log, (es 16-nov-2015) va quedando con las entradas bloqueadas y el string o secuencia inválida:

#Software: Microsoft UrlScan 3.1
#Version: 1.0
#Date: 2015-11-16 20:09:41
#Fields: Date Time c-ip s-siteid cs-method cs-uri x-action x-reason x-context cs-data x-control
2015-11-16 20:09:43 172.31.17.26 9 GET /index.asp?item=%3C Rejected disallowed+query+string+sequence query+string - <

Errores comunes

El log solo logea los títulos de las columnas

-Revisa el archivo ini, que no tengas un [Options] ya que en el ejemplo del mismo .ini dice [Options].

Yo tenía:

[Options]
RuleList=SQL Injection

Y debe ser sólo:

RuleList=SQL Injection

-Revisa que le hayas dado permisos de escritura a la carpeta log al usuario de IIS.

Conclusión


UrlScan es la solución para evitar andar programando en el código (IF request ("%, <, >, etc...) == "error") ya que sólo mediante configuración filtras todo lo malo .

Referencias



Anexo: Mi archivo ini


Mi .ini que usé es este. Notar que dejé ScanHeaders=Cookie para incluir que si se crea una Cookie, se validen los strings que tenga como valor:
[options]

UseAllowVerbs=1                ; If 1, use [AllowVerbs] section, else use the
                               ; [DenyVerbs] section.   The default is 1.

UseAllowExtensions=0           ; If 1, use [AllowExtensions] section, else
                               ; use the [DenyExtensions] section. The
                               ; default is 0.

NormalizeUrlBeforeScan=1       ; If 1, canonicalize URL before processing.
                               ; The default is 1.  Note that setting this
                               ; to 0 will make checks based on extensions,
                               ; and the URL unreliable and is therefore not
                               ; recommend other than for testing.

VerifyNormalization=1          ; If 1, canonicalize URL twice and reject
                               ; request if a change occurs.  The default
                               ; is 1.

AllowHighBitCharacters=0       ; If 1, allow high bit (ie. UTF8 or MBCS)
                               ; characters in URL.  The default is 0.

AllowDotInPath=0               ; If 1, allow dots that are not file
                               ; extensions. The default is 0. Note that
                               ; setting this property to 1 will make checks
                               ; based on extensions unreliable and is
                               ; therefore not recommended other than for
                               ; testing.

RemoveServerHeader=0           ; If 1, remove the 'Server' header from
                               ; response.  The default is 0.

EnableLogging=1                ; If 1, log UrlScan activity.  The
                               ; default is 1.  Changes to this property
                               ; will not take effect until UrlScan is
                               ; restarted.

PerProcessLogging=0            ; This property is deprecated for UrlScan
                               ; 3.0 and later.  UrlScan 3.0 and later can
                               ; safely log output from multiple processes
                               ; to the same log file.  Changes to this
                               ; property will not take effect until
                               ; UrlScan is restarted.

AllowLateScanning=0            ; If 1, then UrlScan will load as a low
                               ; priority filter.  The default is 0.  Note
                               ; that this setting should only be used in
                               ; the case where there another installed
                               ; filter is modifying the URL and you wish
                               ; to have UrlScan apply its rules to the
                               ; rewritten URL.  Changes to this property
                               ; will not take effect until UrlScan is
                               ; restarted.

PerDayLogging=1                ; If 1, UrlScan will produce a new log each
                               ; day with activity in the form
                               ; 'UrlScan.010101.log'. If 0, UrlScan will
                               ; log activity to urlscan.log.  The default
                               ; is 1.  Changes to this setting will not
                               ; take effect until UrlScan is restarted.

UseFastPathReject=0            ; If 1, then UrlScan will not use the
                               ; RejectResponseUrl.  On IIS versions less
                               ; than 6.0, this will also prevent IIS
                               ; from writing rejected requests to the
                               ; W3SVC log.  UrlScan will log rejected
                               ; requests regardless of this setting.  The
                               ; default is 0.

LogLongUrls=0                  ; This property is deprecated for UrlScan 3.0
                               ; and later. UrlScan 3.0 and later will
                               ; always include the complete URL in its log
                               ; file.

UnescapeQueryString=1          ; If 1, UrlScan will perform two passes on
                               ; each query string scan, once with the raw
                               ; query string and once after unescaping it.
                               ; If 0, UrlScan will only look at the raw
                               ; query string as sent by the client.  The
                               ; default is 1. Note that if this property is
                               ; set to 0, then checks based on the query
                               ; string will be unreliable.

;
; If UseFastPathReject is 0, then UrlScan will send
; rejected requests to the URL specified by RejectResponseUrl.
; If not specified, '/Rejected-by-UrlScan' will be used.
; Changes to this setting will not take effect until UrlScan
; is restarted.
;
; Note that setting "RejectResponseUrl=/~*" will put UrlScan into Logging
; Only Mode.  In this mode, UrlScan will process all requests per the
; config settings, but it will only log the results and not actually
; reject the requests.  This mode is useful for testing UrlScan settings
; on a production server without actually interrupting requests.
;

RejectResponseUrl=

;
; LoggingDirectory can be used to specify the directory where the
; log file will be created.  This value should be the absolute path
; (ie. c:\some\path).  If not specified, then UrlScan will create
; the log in the same directory where the UrlScan.dll file is located.
; Changes to this setting will not take effect until UrlScan is
; restarted.
;

LoggingDirectory=Logs

;
; If RemoveServerHeader is 0, then AlternateServerName can be
; used to specify a replacement for IIS's built in 'Server' header
;

AlternateServerName=

;
; UrlScan supports custom rules that can be applied in addition to the other
; checks and options specified in this configuration file.  Rules should be
; listed in a comma separated string in the RuleList property.  Each rule in
; the list corresponds to two sections in this configuration file, one
; containing the options for the rule, and one containing deny strings for
; the rule.
;
; Here is an example:
;
;   [Options]
;   RuleList=Rule1
;
;   [Rule1]
;   AppliesTo=.exe,.dll        ; A comma separated list of file extensions to
;                              ; which the rule applies.  If not specified,
;                              ; the rule will be applied to all requests.
;
;   DenyDataSection=Rule1 Data ; The name of the section containing the
;                              ; rule's deny strings
;
;   ScanURL=0                  ; If 1, the URL will be scanned for deny
;                              ; strings. The default is 0.
;
;   ScanAllRaw=0               ; If 1, then the raw request header data will
;                              ; be scanned for deny strings.  The default
;                              ; is 0.
;
;   ScanQueryString=0          ; If 1, the the query string will be scanned
;                              ; for deny strings.  The default is 0.  Note
;                              ; that if UnescapeQueryString=1 is set in the
;                              ; [Options] section, then two scans will be
;                              ; made of the query string, one with the raw
;                              ; query string and one with the query string
;                              ; unescaped.
;
;   ScanHeaders=               ; A comma separated list of request headers to
;                              ; be scanned for deny strings.  The default is
;                              ; no headers.
;
;   DenyUnescapedPercent=0     ; If 1, UrlScan will scan the specified part
;                              ; of the raw request for a % character that is
;                              ; not used as an escape sequence.  If found,
;                              ; the request will be rejected.  This check
;                              ; can be used with ScanQueryString=1,
;                              ; ScanAllRaw=1, or the list of ScanHeaders.
;                              ; The default is 0.  Note that if you want to
;                              ; deny non-escaped % characters in the URL,
;                              ; you can set VerifyNormalization=0 in the
;                              ; [Options] section and then add % as a
;                              ; [DenyUrlSequences] entry.
;
;   [Rule1 data]
;   string1
;   string2
;

RuleList=SQL Injection

[SQL Injection]
AppliesTo=.asp,.aspx
DenyDataSection=SQL Injection Strings
ScanUrl=0
ScanAllRaw=0
ScanQueryString=1
ScanHeaders=Cookie:


[SQL Injection Strings]
--
@ ; also catches @@
alter
cast
convert
create
declare
delete
drop
exec ; also catches execute
fetch
insert
kill
select

[RequestLimits]

;
; The entries in this section impose limits on the length
; of allowed parts of requests reaching the server.
;
; It is possible to impose a limit on the length of the
; value of a specific request header by prepending "Max-" to the
; name of the header.  For example, the following entry would
; impose a limit of 100 bytes to the value of the
; 'Content-Type' header:
;
;   Max-Content-Type=100
;
; Any headers not listed in this section will not be checked for
; length limits.
;
; There are 3 special case limits:
;
;   - MaxAllowedContentLength specifies the maximum allowed
;     numeric value of the Content-Length request header.  For
;     example, setting this to 1000 would cause any request
;     with a content length that exceeds 1000 to be rejected.
;     The default is 30000000.
;
;   - MaxUrl specifies the maximum length of the request URL,
;     not including the query string. The default is 260 (which
;     is equivalent to MAX_PATH).
;
;   - MaxQueryString specifies the maximum length of the query
;     string.  The default is 2048.
;

MaxAllowedContentLength=30000000
MaxUrl=260
MaxQueryString=2048

[AllowVerbs]

;
; The verbs (aka HTTP methods) listed here are those commonly
; processed by a typical IIS server.
;
; Note that these entries are effective if "UseAllowVerbs=1"
; is set in the [Options] section above.
;

GET
HEAD
POST

[DenyVerbs]

;
; The verbs (aka HTTP methods) listed here are used for publishing
; content to an IIS server via WebDAV.
;
; Note that these entries are effective if "UseAllowVerbs=0"
; is set in the [Options] section above.
;

PROPFIND
PROPPATCH
MKCOL
DELETE
PUT
COPY
MOVE
LOCK
UNLOCK
OPTIONS
SEARCH

[DenyHeaders]

;
; The following request headers alter processing of a
; request by causing the server to process the request
; as if it were intended to be a WebDAV request, instead
; of a request to retrieve a resource.
;

Translate:
If:
Lock-Token:
Transfer-Encoding:

[AllowExtensions]

;
; Extensions listed here are commonly used on a typical IIS server.
;
; Note that these entries are effective if "UseAllowExtensions=1"
; is set in the [Options] section above.
;

.htm
.html
.txt
.jpg
.jpeg
.gif

[DenyExtensions]

;
; Extensions listed here either run code directly on the server,
; are processed as scripts, or are static files that are
; generally not intended to be served out.
;
; Note that these entries are effective if "UseAllowExtensions=0"
; is set in the [Options] section above.
;
; Also note that ASP scripts are denied with the below
; settings.  If you wish to enable ASP, remove the
; following extensions from this list:
;    .asp
;    .cer
;    .cdx
;    .asa
;

; Deny executables that could run on the server
.exe
.bat
.cmd
.com

; Deny infrequently used scripts
.htw     ; Maps to webhits.dll, part of Index Server
.ida     ; Maps to idq.dll, part of Index Server
.idq     ; Maps to idq.dll, part of Index Server
.htr     ; Maps to ism.dll, a legacy administrative tool
.idc     ; Maps to httpodbc.dll, a legacy database access tool
.shtm    ; Maps to ssinc.dll, for Server Side Includes
.shtml   ; Maps to ssinc.dll, for Server Side Includes
.stm     ; Maps to ssinc.dll, for Server Side Includes
.printer ; Maps to msw3prt.dll, for Internet Printing Services

; Deny various static files
.ini     ; Configuration files
.log     ; Log files
.pol     ; Policy files
.dat     ; Configuration files
.config  ; Configuration files

[AlwaysAllowedUrls]
;
; URLs listed here will always be explicitly allowed by UrlScan
; and will bypass all UrlScan checks.  URLs must be listed
; with a leading '/' character.  For example:
;
;   /SampleURL.htm
;

[DenyUrlSequences]
;
; If any character sequences listed here appear in the URL for
; any request, that request will be rejected.
;

..  ; Don't allow directory traversals
./  ; Don't allow trailing dot on a directory name
\   ; Don't allow backslashes in URL
:   ; Don't allow alternate stream access
%   ; Don't allow escaping after normalization
&   ; Don't allow multiple CGI processes to run on a single request

[AlwaysAllowedQueryStrings]
;
; Query strings listed here will always be explicitly allowed by
; UrlScan and will bypass all query string based checks.
;


[DenyQueryStringSequences]
;
; If any character sequences listed here appear in the query
; string for any request, that request will be rejected.
;

<   ; Commonly used by script injection attacks
>   ; Commonly used by script injection attacks

lunes, 9 de noviembre de 2015

NullReferenceException en método AjaxRequestProcessor.Run() con AjaxPro 5.7.22.2

Antecedentes


Alguien se le ocurrió una vez, que la mejor forma de conectarse de forma asíncronica desde Javascript (no existía jQuery como ahora) al servidor Web era usando AjaxPro (Ajax.dll). Bueno, eso funcionó, pero trajo un problema arrastrado por años, ahora que me dijeron "resuélvelo de una vea papá". Bueno, el error lo logré resolver luego de semanas cabezeandome. El mensaje en el Visor de Eventos decía en resumen:

NullReferenceException en Ajax.AjaxRequestProcessor.Run()

Antecedentes productivos:

  • Se usaron dos tipos de servidores, web primero Windows Server 2003 32 bit y luego Windows Server 2008 64 bit.
  • El ambiente productivo usa Framework 2.0 y Visual Basic .NET.
  • La arquitectura es un balanceador con 4 servidores Web y un servidor de base de datos.
  • Son 4 aplicaciones Web, las 4 usan 4 AppPool, 3 AppPool tienen 3 worker process y uno tiene 7. Todos tienen diferente tiempo de reciclaje.
  • El ambiente tiene mediana carga 2000 RPM (Request por Minuto) según la fabulosa herramienta Newrelic.
  • Las 4 aplicaciones usan Ajax.dll 5.7.22.2 liberada por el año 2005.

Antecedentes del error:

  • El error da irregularmente, sin un horario dado.
  • El problema puede dejar de dar durante días.
  • El error no daba en ambiente de desarrollo (lo típico).

El error registrado en el visor de Eventos es:

Event code: 3005 
Event message: An unhandled exception has occurred. 
Event time: 10/18/2015 3:12:47 PM 
Event time (UTC): 10/18/2015 6:12:47 PM 
Event ID: 305bbfa9f184433986b6655786e66f72 
Event sequence: 50 
Event occurrence: 1 
Event detail code: 0 
 
Application information: 
    Application domain: /LM/W3SVC/416041936/Root/SitioTest-1-130896653636875000 
    Trust level: Full 
    Application Virtual Path: /SitioTest 
    Application Path: C:\Inetpub\wwwroot\Test\Sitio\1.0\SitioTest\ 
    Machine name: Servidor-1 
 
Process information: 
    Process ID: 6948 
    Process name: w3wp.exe 
    Account name: NT AUTHORITY\NETWORK SERVICE 
 
Exception information: 
    Exception type: NullReferenceException 
    Exception message: Object reference not set to an instance of an object. 
 
Request information: 
    Request URL: http://sitio.cl/folder/ajax/SitioWebDotNet.namespace_clase,App_Web_datos.aspx.fe7917e2.ashx 
    Request path: /sitio/ajax/SitioWebDotNet.sitio_namespace_sitio,App_Web_datos.aspx.fe7917e2.ashx 
    User host address: X.X.X.X 
    User:  
    Is authenticated: False 
    Authentication Type:  
    Thread account name: NT AUTHORITY\NETWORK SERVICE 
 
Thread information: 
    Thread ID: 1 
    Thread account name: NT AUTHORITY\NETWORK SERVICE 
    Is impersonating: True 
    Stack trace:    at Ajax.AjaxRequestProcessor.Run()
   at Ajax.AjaxHandler.ProcessRequest(HttpContext context)
   at System.Web.HttpApplication.CallHandlerExecutionStep.System.Web.HttpApplication.IExecutionStep.Execute()
   at System.Web.HttpApplication.ExecuteStep(IExecutionStep step, Boolean& completedSynchronously)
 
 
Custom event details: 

For more information, see Help and Support Center at http://go.microsoft.com/fwlink/events.asp.

Si uno miraba el contenido del ashx indicado en el error: http://sitio.cl/folder/ajax/SitioWebDotNet.namespace_clase,App_Web_datos.aspx.fe7917e2.ashx, no había nada raro.

Lo que intenté, sin que funcionara

  • Cambiar la DLL 5.7.22.2 a la última versión 9.2.17.1. No fue posible por que la versión que uso no usa JSON y la nueva usa JSON, cambiar todo el código, validar las entradas y salidas era mucho trabajo, quizá de meses.
  • Dejar sólo un servidor, sacándolo del balanceador de carga.
  • Revisar el código fuente: JS, código VB.Net, no encontré errores.
  • Dejar todos los Pool con 3 WP y el mismo tiempo de reciclaje.
  • Evalué pasar de AjaxPro a Ajax 1.0 de Microsoft, pero la cantidad de código era inmensa.
  • El error no da cuando se recicla un pool, era una teoría que tenía
  • Revisé los LOGS de IIS de los 4 servidores y no había algún de patrón del error, que fallara en todos a una hora, o que luego de un reciclaje de los pool fallara.

Solución


La solución fue para cada uno de los 4 Pool de las aplicaciones bajar todos los Worker Process a 1. De pasada dejarles el mismo tiempo de reciclaje. Me acordé que en el blog de Google se hablaba que tenía problemas en balanceo, y quizá pensé el error se da en Web Farm y Web Garden. Y era eso, ¡toque y fama!. Luego de monitorear los errores del visor de eventos durante 2 semanas observé que el problema desapareció. Recomiendo usar para visualizar de mejor forma los Event Logs la herramienta Event Log Explorer, que tiene una licencia Free.

Referencias



Bueno, espero que les sirva de ayuda, en caso que tengan este engorroso error.

lunes, 28 de septiembre de 2015

Combo autocomplete VS 2005 Framework 2.0 VB.Net con jQueryUI

Hola, a todos los geeeks. El otro día me pidieron hacer combo autocomplete para una aplicación Web en Visual Studio 2005, Framework 2.0 y además con VB.NET. Tarea que no fue de media hora como pensé.

Primero en aplicación Web se llaman DropDownList y no Combobox como en las aplicaciones de Escritorio.

Vamos a usar jQuery y jQueryUI.

Lo otro importante es que no implementaremos un combo propiamente autocomplete, sino que haremos un textbox autocomplete, ese el truco. Se verá como un combo pero no lo es.

Código de ASPX

<%@ Page Language="vb" AutoEventWireup="false" CodeBehind="Default.aspx.vb" Inherits="TestCombo._Default" %>

<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml" >
<head runat="server">
    <title>Demo autocomplete textbox VB.Net 2005 jQuery</title>    
    <link rel="stylesheet" href="css/jquery-ui.css"> 
    <script type="text/javascript" src="js/jquery-1.11.0.min.js"></script> 
    <script src="js/jquery-ui-1.10.3.custom.js"></script>
</head>
<body>

    <% =Autocompletar()%>    

    <script type="text/javascript">
        $(document).ready(function() {
         Autocompletar();       
        });
    </script>

    <form id="form1" runat="server">        
        <input type="text" name="combo" id="combo" class="combo" placeholder="Escriba dos letras" maxlength="50" size="50" />        
        <br />
        Valores posibles:<br /><br />
        Hamburguesa, Hamburguesa c/ queso, Quesadillas, Torta de pierna, Torta mixta, Torta ahogada, Paquete Torta con refresco, Refresco, Café, Té 
</form>
</body>
</html>

Código de VB.NET

Partial Public Class _Default
    Inherits System.Web.UI.Page

    Protected Sub Page_Load(ByVal sender As Object, ByVal e As System.EventArgs) Handles Me.Load

    End Sub

    Public Function Autocompletar() As String
        Dim tags As String

        tags = "'Hamburguesa','Hamburguesa c/ queso','Quesadillas','Torta de pierna','Torta mixta','Torta ahogada','Paquete Torta con refresco','Refresco','Café','Té'"

        Autocompletar = ""

    End Function

End Class
La Solución se ve así:

Probando

Si seleccionamos un elemento se ve el combo:

Luego si usamos la consola de Chrome con F12 y ejecutamos

$('#combo').val()

La consola entrega: "Paquete Torta con refresco"

lunes, 20 de julio de 2015

Forms TimeOut vs sessionState Timeout vs ExecutionTimeout

En este artículo hablaremos de los timeout que se configuran en el web.config en aplicaciones .Net:

  • timeout del tag Form
  • timeout del tag sessionState
  • ExecutionTimeout del tag httpRuntime

Ejemplo de una configuración del web.config:


      






timeout del tag Form


Por defecto 30 minutos. En el ejemplo está en 40 minutos. Es el tiempo en minutos que el ticket de autenticación es válida (no de la cookie donde esté almacenada el ticket que puede tener otro tiempo o no tener por lo que caduca cuando se cierra el navegador), pasado este tiempo el ticket expira, y al siguiente request te tira al sitio de inicio/login deforma automática. En caso de que la cookies es persistente, el tiempo de timeout es el mismo entre el ticket y la cookie donde está la cookie.

slidingExpiration: Por defecto True en todos los Frameworks, aunque en MSDN hay un error ya que en algunas referencias dice que es False para Framework 2.0 pero en los mismos comentarios MSDN dicen que es True, por ejemplo aquí indica que es True: https://msdn.microsoft.com/en-us/library/system.web.security.formsauthentication.slidingexpiration(v=vs.80).aspx y aquí dice que por defecto es False para Framework 2.0: https://msdn.microsoft.com/es-cl/library/1d3t3c61(v=vs.80).aspx. Bueno oficialmente por defecto es True para todos los Frameworks.

En modo false es lo que es conocido como absolute expiration, si el tiempo timeout es 30 (30 min) si un usuario hace un request a las 13 hrs, luego si a las 13:31 hace un nuevo request, este no se procesará y en cambio, le dejará automáticamente en el login.

En modo true, sliding expiration (expiración corrida), la cookie y el ticket se actualizan si el usuario hace un request luego que haya pasado la mitad del tiempo. Por ejemplo si el timeout es 20 y usas sliding expiration en true, y el usuario visita el sitio a las 14 hrs, el usuario recibirá una cookie que expirará a las 14:20 hrs. La cookie se actualiza solo si el usuario visita el sitio luego de las 14:10 hrs.
Si el usuario hace una visita a las 14:12 hrs, la cookie se actualiza y ahora expirará a las 14:32 hrs.
Si el usuario visita el sitio a las 14:09 hrs, la cookie no se actualizará y si se da que el usuario visita el sitio a las 14:21 hrs, el ticket estará expirado y lo dejará en el home de forma automática.

Cuando ticket expira, da el famoso error de Ticket expirado:

Event code: 4005
Event message: Forms authentication failed for the request. Reason: The ticket supplied has expired.


Si no usa el tag authentication en el web.config, por defecto el Framework lo setea así (en Framework 2.0): https://msdn.microsoft.com/en-us/library/532aee0e(v=vs.80).aspx para v2.0, o https://msdn.microsoft.com/en-us/library/532aee0e(v=vs.100).aspx para v4.0.


Este modo en el web.config es solo para decirle a .NET como manejar la autenticación no tiene que ver con IIS o el servidor.
De echo, si miras IIS en la sección Autenticación, solo quedará el modo anónimo que es el modo que por defecto trae habilitado. Si quieres modo Windows Authentication y que pida el usuario y clave, debes cambiar en el IIS en la sección Autenticación, y dejar modo Anónimo desactivado y Windows modo activado. Y al ejecutar el sitio pedirá un usuario y clave de Windows.

Si se cambia en el IIS 7 esta configuración, estás afectando a la sección de tu web.config:
system.webServer/security/authentication o applicationHost.config
Y no a <authentication> de <system.web>
En IIS 6 este cambio afecta a la metabase y no a un web.config.

Si no usa el tag forms dentro de authentication en el web.config, por defecto el Framework lo setea así (en Framework 2.0): https://msdn.microsoft.com/es-cl/library/1d3t3c61(v=vs.80).aspx

  


timeout del tag sessionState


Por defecto es 20 minutos. Es el tiempo en minutos que estarán los datos en memoria para una sesión en particular. Por ejemplo, si guardas un objeto en la clase Session se pierde la información a los 480 minutos. El time out se resetea en cada request.
Solo válido si se usa sessionState, es decir que no está como:

Como dato, por defecto el sessionState es InProc:


Si no usa el tag sessionState en el web.config, por defecto el Framework lo setea así en Framework 2.0: https://msdn.microsoft.com/en-us/library/h6bb9cz9(v=vs.80).aspx

   
      
   


executionTimeout del tag httpRuntime


Por defecto es 110 segundos. En el ejemplo tiene 36000 (10 hrs). Es el número máximo de segundos que se permite que se ejecute una solicitud antes de que ASP.NET la concluya automáticamente y quede como failed o de el famoso error The request has been aborted. Request timed out.

Este tiempo de espera solo se aplica si el atributo de depuración del elemento
 
está establecido en false. Así que executionTimeout sólo lo toma funciona en modo release, de lo contrario toma el valor por defecto, 110 seg.

Si estas haciendo una aplicación que sube un archivo pesado, por ejemplo de 50 MG, este valor debe ser por ejemplo 3600 (1 hr) para que .NET no la cierre automáticamente.

Tamaño de archivos de subida: maxRequestLength y maxAllowedContentLength

Si quieres permitir que el peso máximo de un archivo a subir sea de 200 MB debes setear dos tag maxRequestLength y maxAllowedContentLength ambos deben ir en el web.config.

El tag maxRequestLength está en KB e indica el tamaño máximo de un archivo soportado por ASP.NET. Si son 200 MB el valor sería 2048000. Por defecto está en 4096 es decir 4 MB.

Si no se usa el tag httpRuntime en el web.config, por defecto el Framework lo setea así en Framework 2.0: https://msdn.microsoft.com/en-us/library/e1f13641(v=vs.80).aspx


En IIS 7 además se debe setear el atributo maxAllowedContentLength que está en Bytes, indica el tamaño máximo de un archivo a subir soportado por IIS. Aquí los 200 MB serían 2097152000. El valor por defecto es 30000000, lo que equivale a 28.6 MB.
Ojo si que en IIS 7 este valor va en una estructura aparte:

    
      
        
      
    
  

NOTA: Si estás con IIS 7 o superior, debes setear AMBOS TAG en el mismo valor, sino tomará el menor de ellos como el valor con prioridad.

Referencias