ELMAH(ErrorLoggingModulesandHandlers)錯誤日志記錄模塊和處理程序,是一種應用廣泛的錯誤日志工具是完全可插拔。它可以動態添加到一個正在運行的asp.netWeb應用程序,甚至是一臺機器上的所有ASP.NETWeb應用程序,而無需重新編譯或重新部署。
ELMAH既支持ASP.NETWebForms又支持ASP.NETMVC。你可以對ELMAH進行配置來存儲各種不同的錯誤(xml文件,事件日志,access數據庫,SQL數據庫,Oracle數據庫,或者計算機RAM。)你還可以讓ELMAH在錯誤發生的時候,把錯誤信息email給你。
在默認情況下,在一個已經安裝ELMAH的網站中,你可以通過請求的elmah.axd頁面的方式來訪問ELMAH。
本篇來嘗試Elmah在Asp.netMVC 5使用.
首先Build空的Asp.netMVC 5 PRoject:
添加Elmah引用:
Elmah組建已經配置成功.其實這個過程做了兩件事:
在webConfig文件中添加如下內容:
<configSections> <sectionGroup name="elmah"> <section name="security" requirePermission="false" type="Elmah.SecuritySectionHandler, Elmah" /> <section name="errorLog" requirePermission="false" type="Elmah.ErrorLogSectionHandler, Elmah" /> <section name="errorMail" requirePermission="false" type="Elmah.ErrorMailSectionHandler, Elmah" /> <section name="errorFilter" requirePermission="false" type="Elmah.ErrorFilterSectionHandler, Elmah" /> </sectionGroup> </configSections><elmah> <!-- See http://code.google.com/p/elmah/wiki/SecuringErrorLogPages for more information on remote access and securing ELMAH. --> <security allowRemoteAccess="false" /> </elmah> <location path="elmah.axd" inheritInChildapplications="false"> <system.web> <httpHandlers> <add verb="POST,GET,HEAD" path="elmah.axd" type="Elmah.ErrorLogPageFactory, Elmah" /> </httpHandlers> <!-- See http://code.google.com/p/elmah/wiki/SecuringErrorLogPages for more information on using ASP.NET authorization securing ELMAH. <authorization> <allow roles="admin" /> <deny users="*" /> </authorization> --> </system.web> <system.webServer> <handlers> <add name="ELMAH" verb="POST,GET,HEAD" path="elmah.axd" type="Elmah.ErrorLogPageFactory, Elmah" preCondition="integratedMode" /> </handlers> </system.webServer> </location>
HomeController.cs
using System;using System.Collections.Generic;using System.Linq;using System.Web;using System.Web.Mvc;namespace Elmah.Demo.Controllers{ public class HomeController : Controller { // // GET: /Home/ public ActionResult Index() { return View(); } [HttpPost] public ActionResult GenerateError(string error) { throw new ApplicationException(error); } }}
index.cshtml
@{ Layout = null; ViewBag.Title = "Index";}<div> <input type="text" id="ErrorMsg" /> <button id="GenerateError">生成錯誤日志</button> <a href="/elmah.axd" target="_blank">在elmah中查看錯誤日志</a></div><script src="~/Scripts/jquery-1.10.2.js"></script><script type="text/javascript"> $("#GenerateError").click(function () { $.post("/Home/GenerateError?error=" + $("#ErrorMsg").val()); });</script>
運行效果如下:
如果不是Post方式,會報黃頁,如:
來看看Elmah是否記錄本次執行過程中出現的異常:
可以看到Elmah已經如期的撲捉到當前應用程序的異常.ELMAH在后臺記錄了錯誤信息,并為我們提供了查詢錯誤日志信息的界面,只需要簡單的操作,就完成了基本的需求.
有人可能會問,上面的自動配置中,并沒有指定存儲日志的方式?。ó斎贿@里還沒介紹如何配置,但是從上面配置中,似乎也看不到有哪里指定了存儲方 式),那這些數據存儲在哪里了呢?答案是,NuGet安裝ELMAH后,它是沒有指定任何存儲方式。而ELMAH認為,如果沒有指定存儲方式,那么就采用 默認的內存存儲方式(也可以顯式的指定)。但是這種存儲方式只能作為調試階段使用,生產環境下不應使用此方式,具體的缺點請看下面對內存存儲方式的介紹。
接下來就具體介紹各種存儲方式,分別以數據庫存儲、文件存儲和內存存儲為例,需要強調一點,ELMAH目前只支持一下三種方式中的任意一種,不支持同時采用多種記錄方式。(想必也沒這個必要)
內存存儲,顧名思義,將日志記錄于操作系統分配給應用程序的內存中。應用程序的內存是與應用程序域相關的,這可以保證每個應用程序只能獲取和記錄屬于自己 的日志信息。但是,一旦應用程序重啟,之前記錄的信息將會消失。最簡單的例子,如果你用這種方式調試呢,默認是用ASP.NET Development Server作為web服務器,如果這時停止此服務器,則就滿足上述條件了(如下圖)。另外,斷電,發布后IIS的重啟等問題,都會導致記錄的信息丟失。 因此,這種方式只能用于測試用。
其實Elmah處理原理.當我們請求頁面報錯時.在返回黃頁錯誤時首先被 httpModules中名為ErrorLog模塊進行攔截. 該模塊將本次請求出錯的信息保存起來.-默認是放置在內存中.便于即時調試.但用戶輸入elmah.axd要查看日志信息時. 首先httpHandlers捕獲到該請求.并交給專門處理elmah.axd的處理程序.該模塊把錯誤日志View返回給用戶.可見Elmah核心技術 還是基于HttpModules和HttpHandlers來實現的.
文件存儲實際上ELMAH提供了xml文件的存儲方式,每一個報錯日志信息生成一個xml文件。配置相當簡單:
<elmah> <!-- See http://code.google.com/p/elmah/wiki/SecuringErrorLogPages for more information on remote access and securing ELMAH. --> <security allowRemoteAccess="false" /> <!--只有這一句就行了,其中logPath用于指定記錄日志的文件夾位置--> <errorLog type="Elmah.XmlFileErrorLog, Elmah" logPath="~/Static/Log/" /> </elmah>
該配置必需確認LogPath路徑目錄是完整存在的.測試會發現在本地文件中(/Elmah.Demo/Static/Log)會出現一個XML文件:
3. 數據庫存儲方式
在數據可視化和管理上數據庫依然是最理想的選擇.這里采用SQlServer2008 版本測試.在構建Elmah支持SQLServer數據支持需要如下三個操作:
其中a和b步驟需要在web.config中指定,c則需要在數據庫中添加相關對象。
web.config配置如下(httpModules以及httpHandlers就不貼了,這里只給出ELMAH記錄日志于sqlserver數據庫的配置):
<connectionStrings> <add name="elmah-sqlserver" connectionString="server=.;database=MvcTest;user id=sa;passWord=111111@a" providerName="System.Data.SqlClient" /> </connectionStrings> <elmah> <!-- See http://code.google.com/p/elmah/wiki/SecuringErrorLogPages for more information on remote access and securing ELMAH. --> <security allowRemoteAccess="false" /> <!--只有這一句就行了,其中logPath用于指定記錄日志的文件夾位置--> <!--<errorLog type="Elmah.XmlFileErrorLog, Elmah" logPath="~/Static/Log/" />--> <!-- 告訴elmah,我要采用sqlserver來記錄我的日志,連接那個數據庫的字符串名為myconnectionString。--> <errorLog type="Elmah.SqlErrorLog, Elmah" connectionStringName="elmah-sqlserver" /> </elmah>
創建數據庫,在該數據執行如下SQL語句.請參考官方的連接.
Elmah SQL Server Script File:http://code.google.com/p/elmah/source/browse/src/Elmah/SQLServer.sql
腳本:
CREATE TABLE dbo.ELMAH_Error( ErrorId UNIQUEIDENTIFIER NOT NULL, Application NVARCHAR(60) COLLATE SQL_Latin1_General_CP1_CI_AS NOT NULL, Host NVARCHAR(50) COLLATE SQL_Latin1_General_CP1_CI_AS NOT NULL, Type NVARCHAR(100) COLLATE SQL_Latin1_General_CP1_CI_AS NOT NULL, Source NVARCHAR(60) COLLATE SQL_Latin1_General_CP1_CI_AS NOT NULL, Message NVARCHAR(500) COLLATE SQL_Latin1_General_CP1_CI_AS NOT NULL, [User] NVARCHAR(50) COLLATE SQL_Latin1_General_CP1_CI_AS NOT NULL, StatusCode INT NOT NULL, TimeUtc DATETIME NOT NULL, Sequence INT IDENTITY (1, 1) NOT NULL, AllXml NTEXT COLLATE SQL_Latin1_General_CP1_CI_AS NOT NULL ) ON [PRIMARY] TEXTIMAGE_ON [PRIMARY]GOALTER TABLE dbo.ELMAH_Error WITH NOCHECK ADD CONSTRAINT PK_ELMAH_Error PRIMARY KEY NONCLUSTERED ( ErrorId ) ON [PRIMARY] GOALTER TABLE dbo.ELMAH_Error ADD CONSTRAINT DF_ELMAH_Error_ErrorId DEFAULT (newid()) FOR [ErrorId]GOCREATE NONCLUSTERED INDEX IX_ELMAH_Error_App_Time_Seq ON dbo.ELMAH_Error( [Application] ASC, [TimeUtc] DESC, [Sequence] DESC) ON [PRIMARY]GOSET QUOTED_IDENTIFIER ON GOSET ANSI_NULLS ON GOCREATE PROCEDURE dbo.ELMAH_GetErrorXml( @Application NVARCHAR(60), @ErrorId UNIQUEIDENTIFIER)ASSET NOCOUNT ONSELECT AllXmlFROM ELMA
新聞熱點
疑難解答