Click here to Skip to main content
15,860,972 members
Articles / Web Development

Elmah for Glimpse – Best of Both Worlds

Rate me:
Please Sign up or sign in to vote.
4.65/5 (16 votes)
5 Jun 2011Apache9 min read 51.8K   33   11
Creating an Elmah plugin for Glimpse

Introducing Elmah

We all know Elmah right? It’s one of the top packages in NuGet. For those who don’t really know or use Elmah, I’ll give a brief explanation of the functionality it provides and how you can use it in your application.

Elmah stands for Error Logging Modules and Handlers. It’s an application-wide error logging library which logs nearly all unhandled exceptions. It provides you with a web page which enables you to (remotely) view the log of recorded exceptions. This log doesn’t only contain the error message; it also includes the entire stack trace and the values of all server variables at the time of the error.

Using Elmah with ASP.NET MVC

Since we have NuGet, setting up Elmah with an ASP.NET MVC application has become really easy. Right click your web application and select ‘Add Library Package Reference…’ to pop up the NuGet Package Manager. Search for Elmah and install it. After installing, Elmah will have been added as a project reference.

Image 1

The next step is to configure Elmah. I won’t go into too much detail here. We’ll just set up Elmah to log all errors to an in-memory repository and to expose those errors through a web page. First let’s add the following code to the <configSections> node to make Elmah read it’s configuration from your web.config.

XML
<sectiongroup name="elmah">
  <section name="errorLog" type="Elmah.ErrorLogSectionHandler, Elmah" 
	requirepermission="false">
</sectiongroup>

Elmah allows you to log errors to several storages. They currently support Microsoft SQL Server, Oracle, SQLite, Access, XML files, in-memory, SQL CE, MySQL and PostgreSQL. Insert the code below in your web.config to log error to an in-memory storage. When using an in-memory repository, all errors that got recorded will vanish when the application restarts.

XML
<elmah>
  <errorlog type="Elmah.MemoryErrorLog, Elmah">
</elmah>

We still have to do two things to get everything up and running:

  1. Make Elmah catch and log all unhandled exceptions.
  2. Expose a web page to allow the user to view everything that got caught by Elmah

To achieve this, we have to add an HTTP module and an HTTP handler. The module is responsible for logging the exceptions while the HTTP handler will render a page with a list of errors. You’ll have to add these sections to both the <system.web> node and the <system.webServer> node in order to support IIS running in Classic mode as well as Integrated mode. More information in this can be found on MSDN – How to: Register HTPP Handlers.

XML
<system.web>
  <httphandlers>
    <add type="Elmah.ErrorLogPageFactory, Elmah" 
       verb="POST,GET,HEAD" path="elmah.axd">
  </httphandlers>
 
  <httpmodules>
    <add name="ErrorLog" type="Elmah.ErrorLogModule, Elmah">
  </httpmodules>
</system.web>

<system.webserver>   
  <modules runallmanagedmodulesforallrequests="true">
    <add name="ErrorLog" type="Elmah.ErrorLogModule, Elmah">
  </modules>
 
  <handlers>
    <add name="Elmah" type="Elmah.ErrorLogPageFactory, Elmah" 
       verb="POST,GET,HEAD" path="elmah.axd" 
       precondition="integratedMode">
  </handlers>
</system.webserver>

Let’s take a closer look at the registration of the HTTP handler. We specified a path ‘elmah.axd’, this means that the handler will only kick in when we browse to that specific path. So, let’s launch our application and take a look. If I browse to http://localhost/elmah.axd I get this:

Image 2

And oh, it even gets better. Let's click on the ‘details’ link next to the error message. This is where the magic happens:

Image 3

This is really neat; we get a ton of information on what was going on when the error was thrown. If an application error occurred, we can almost immediately pin-point the problem.

Obviously, you don’t want to give access to this log to everyone that visits your website, as user might be able to spot vital problems in your code and exploit them. You can prevent access to this log in various ways (no remote access, through ASP.NET authorization …). More information on this topic can be found here: Elmah: Securing Error Log Pages.

Introducing Glimpse

Most of you probably know Firebug. And actually, Glimpse is a lot like Firebug, except it’s implemented in JavaScript on the client side with hooks in to ASP.NET on the server side. What Firebug is for the client, Glimpse does for the server... in other words, a client side Glimpse into what’s going on in your server.

Currently Glimpse can only be used within web applications that target .NET 4.0; however the Glimpse team is currently working on a build that supports.NET 3.5 as well. As a matter of fact, Glimpse is not just a .NET only tool; currently it only works with ASP.NET MVC and Web Forms but eventually it will support Ruby on Rails, PHP and others.

For ASP.NET MVC 3, it currently already has a bunch of nice features, like:

  • Ajax call tracing
  • Route debugging
  • Request inspection
  • Server variables inspection
  • Session inspection
  • .NET Tracing (no more need for the trace.axd)

Using Glimpse with ASP.NET MVC

Integrating Glimpse with ASP.NET MVC is very easy. It’s available as a NuGet package so installing it literally takes about 30 seconds. Let’s have a look … Right click your web application and select ‘Add Library Package Reference…’ to pop up the NuGet Package Manager. Search for Glimpse and install it:

Image 4

The Glimpse NuGet package will automatically reference the right assembly and it will add a configuration section to your web.config. The only thing you have to do to get it up and running is the following:

  • Launch your application
  • Browse to: http://localhost/Glimpse/Config
  • Click ‘Turn Glimpse On’.

    Image 5

  • Go back to your application, you should now see something like this:

    Image 6

What these Glimpse guys developed is really mind blowing. Currently Glimpse is still beta, but in my opinion it’s stable enough to integrate with all of your current ASP.NET projects.

Elmah for Glimpse - Best of Both Worlds

Scott Hanselman, unofficially titled the funniest guy at Microsoft, recently featured Glimpse as ‘NuGet Package of the Week’. At the end of the article, he says:

‘Glimpse, along with ELMAH, is officially my favorite add-on to ASP.NET MVC. I'll be using it every day and I recommend you do as well.’

So … wouldn’t it be cool to integrate Elmah with Glimpse?

Plugin System

Glimpse has a really clean plugin system. Internally, they use MEF to discover extensions. The best of all is that writing a Glimpse plugin is really easy. All you have to do is:

  • Implement the IGlimpsePlugin interface
  • Decorate your class with the GlimpsePlugin attribute

Let’s take a look at my Elmah plugin for Glimpse:

C#
[GlimpsePlugin]
public class ElmahPlugin : IGlimpsePlugin
{
  public string Name
  {
    get { return "Elmah"; }
  }
 
  public object GetData(HttpApplication application)
  {
    if (application == null || application.Context == null)
      return null;
 
    var errorList = new List<errorlogentry>();
    var errorCount = ErrorLog
          .GetDefault(application.Context)
          .GetErrors(0, 15, errorList);

    if (errorCount == 0)
      return null;
 
    var data = new List<object[]>
    {
      new object[]
      {
         "Host", "Code", "Type", 
         "Error", "User", "Date", "Time"
      }
    };
 
    data.AddRange(
      errorList.Select(
        errorEntry => new object[]
    {
          errorEntry.Error.HostName,
      errorEntry.Error.StatusCode,
      errorEntry.Error.Type,
      errorEntry.Error.Message,
      errorEntry.Error.User,
      errorEntry.Error.Time.Date.ToShortDateString(),
      errorEntry.Error.Time.ToShortTimeString()
        }));
 
    return data;
  }
 
  public void SetupInit(HttpApplication application)
  {
    throw new NotImplementedException();
  }
}

In the code sample above, you see I implement the IGlimpsePlugin interface. The interface has the following methods:

  • A getter for the Name of the plugin: this property will be used as the name of the tab in the Glimpse user interface.
  • GetData method: accepts an HttpApplication parameter and returns an object. The object you return will be serialized into JSON, sent to the client and rendered by the Glimpse UI. How it will be rendered depends on the kind of object you return. More details about this can be found on the Glimpse Protocol page.
  • SetupInit method: This is the place where you perform any initialization that needs to be done. This method will run once (when your plugin is loaded for the first time). If you want this method to be called by Glimpse, you will need to set the ShouldSetupInInit property on the GlimpsePlugin attribute to true: [GlimpsePlugin(ShouldSetupInInit=true)]

The GetData method is the place where you should make any decisions about what should be rendered and how you want to render it. If this method returns null, then your tab will be disabled in the Glimpse UI.

In the above implementation, I return null if no HTTP context is available or when no errors have been logged yet. In case errors have been logged, I return them as a list of objects. The first entry in the list will be used as the title row; all subsequent rows are data rows. Currently only the last 15 errors are returned to the user.

Update 05/31/2011: The above code sample and the explanation that comes with it are not entirely valid anymore, but for the purpose of the document I'll leave it as it initially was.

NuGet Package

The Elmah plugin for Glimpse is available as a NuGet package. To install the package, right click your web application and select ‘Add Library Package Reference …’ search for either ‘Glimpse’, ‘Elmah’ or ‘Glimpse.Elmah’ and the Package Manager will come up with the ‘Elmah plugin for Glimpse’.

The package has dependencies to Glimpse and Elmah, so if you haven’t got these package installed yet, then NuGet will get them automatically as well. If you didn’t have Elmah before, you might still have to configure it.

Image 7

After installing the package, you still have to include the Elmah for Glimpse client side script on your pages, preferable in your main master page.

ASP.NET
<script src="<%: Url.Content("~/Glimpse/Resource/?resource=Pager") %>" 
type="text/javascript"></script>

Let’s run our application again and see what happens. Our Elmah plugin has been added as a reference to the project so it will be discovered by the Glimpse plugin system. Glimpse will load it into its UI and voila, there we have it:

Image 8

NuGet Sample Package

The easiest to get started is by getting the ‘Elmah plugin for Glimpse Sample’ package, which is also available through NuGet.

Just create a new empty ASP.NET MVC 3 application and add a library package reference to the Elmah plugin for Glimpse Sample package. The package will automatically:

  • Install Glimpse
  • Install Elmah
  • Install the Elmah plugin for Glimpse
  • Install jQuery
  • Configure everything
  • Create a new controller, model and view so you can throw some random exceptions.

After installing the package, you’ve got yourself an application to throw exceptions! Cool huh!? OK, so let’s run the application, throw some exceptions and see what happens:

Image 9

If you try the sample yourself, but you don’t see the Glimpse plugin you might still have to enable it. Please take a look at the ‘Using Glimpse with ASP.NET MVC’ section on how to do this.

Future

If there's any feature you would like to see implemented, you can always submit a feature request on the project home page at CodePlex. If you want to implement a feature yourself, you can fork the project and submit a pull request.

Things I’m considering for the next version are:

  • Adding some kind of configuration (page size, ...)
  • Automatically include the client script

Release Notes

  • 06/05/2011 - 0.9.3.0
    • Implemented the IProvideGlimpseHelp interface
    • Changed default page size to 10
    • Fixed bug in pager script
  • 06/03/2011 - 0.9.2.0
    • Updated to version 0.82 of Glimpse
    • Updated to version 1.2.0.1 of Elmah
  • 05/31/2011 - 0.9.1.0
    • Added details link
    • Added pager

Image 10

  • 05/18/2011 - 0.9.0.0
    • Initial version

License

This article, along with any associated source code and files, is licensed under The Apache License, Version 2.0


Written By
Software Developer Cronos - QFrame
Belgium Belgium
This member has not yet provided a Biography. Assume it's interesting and varied, and probably something to do with programming.

Comments and Discussions

 
GeneralMy vote of 5 Pin
Gaston Verelst14-Oct-13 1:11
Gaston Verelst14-Oct-13 1:11 
QuestionDetails link missing Pin
cocowalla25-Apr-13 3:41
cocowalla25-Apr-13 3:41 
QuestionGliimpse for Elmah nuget uses non-mvc glimpse Pin
Noyabronok28-Dec-11 10:14
Noyabronok28-Dec-11 10:14 
AnswerRe: Gliimpse for Elmah nuget uses non-mvc glimpse Pin
stevenlauwers225-Jan-12 2:29
stevenlauwers225-Jan-12 2:29 
Hey, I was able to reproduce the problem. There seems to be a problem when you remove the Glimpse or Glimpse.Elmah package and immediately add it again (if you restart VS before adding it again it should work fine). I will discuss it with Anthony and Nik of the Glimpse project and we'll try to fix it in the next release.

If you do it like this, everything should be ok:
- install the Glimpse for ASP.NET MVC 3 package (this will automatically install the Glimpse for ASP.NET pacakge as well)
- install the Glimpse for Elmah package (this will automatically install the Elmah package too)

If you have installed the Glimpse for ASP.NET package before installing the MVC3 package you don't first need to remove it, you can just install the MVC3 package, it will notice the ASP.NET package is already there.
Generalnice tool Pin
saxenaabhi618-May-11 18:10
saxenaabhi618-May-11 18:10 
GeneralMy vote of 3 Pin
Andrey Mazoulnitsyn18-May-11 8:04
Andrey Mazoulnitsyn18-May-11 8:04 
GeneralRe: My vote of 3 Pin
thatraja14-Jun-11 4:47
professionalthatraja14-Jun-11 4:47 
GeneralGood Pin
BillW3318-May-11 5:55
professionalBillW3318-May-11 5:55 
GeneralRe: Good Pin
stevenlauwers2218-May-11 7:30
stevenlauwers2218-May-11 7:30 
GeneralGood one Pin
Shahriar Iqbal Chowdhury/Galib18-May-11 5:38
professionalShahriar Iqbal Chowdhury/Galib18-May-11 5:38 
GeneralRe: Good one Pin
stevenlauwers2218-May-11 7:30
stevenlauwers2218-May-11 7:30 

General General    News News    Suggestion Suggestion    Question Question    Bug Bug    Answer Answer    Joke Joke    Praise Praise    Rant Rant    Admin Admin   

Use Ctrl+Left/Right to switch messages, Ctrl+Up/Down to switch threads, Ctrl+Shift+Left/Right to switch pages.