Skip to content

Latest commit

 

History

History
289 lines (222 loc) · 13.3 KB

README.md

File metadata and controls

289 lines (222 loc) · 13.3 KB

License Build Nuget

Elmah.AspNetCore

ELMAH (Error Logging Middleware and Handlers) for ASP.NET Core.

Features include:

  • Logging of unhandled exceptions
  • Friendly UI to view captured errors along with contextual information
  • Hooks to include handled exceptions and additional contextual information
  • Various methods to persist error logs
  • Supports securing UI via built-in ASP.Net Core functionality
  • Notifications of errors through email or custom notifiers
  • Integration with Microsoft.Extensions.Logging to capture logs made during a request
  • Supports .NET 6+

alt text

This is a fork of ElmahCore which is itself a fork of the original Elmah library. Credit goes to the owners and contributors of those libraries. This fork attempts to catch up with features added to .NET.

Basic usage

First, install the Elmah.AspNetCore Nuget package into your app.

dotnet add package Elmah.AspNetCore

Next, in your application's Program.cs file, configure Elmah:

using Elmah.AspNetCore;

var builder = WebApplication.CreateBuilder(args);

builder.Host.UseElmah(); // <- Add this to configure Elmah

var app = builder.Build();

app.UseExceptionHandler();
app.UseElmahMiddleware(); // <- Add this to register middleware

app.MapElmah(); // <- Add this to register Elmah endpoints

builder.Host.UseElmah() registers and configures the Elmah services. An overload which accepts an action is available to modify the configuration.

app.UseElmahMiddleware() registers the middleware used by Elmah to start capturing errors and contextual information. Only middleware registered after the Elmah middleware will be included in the error capturing. It is recommended that this is included before most other middleware. For best results, call after the built-in UseExceptionHandler().

app.MapElmah() registers the routes used to serve content for the Elmah UI. By default these will be under /elmah, but the method includes an overload which allows overriding the root path.

Elmah Options

Option Type Default Description
ApplicationName string ApplicationName from IHostEnvironment Application name captured in error log
Filters IErrorFilter[] empty A collection of IErrorFilter instances
FiltersConfig string null Path to XML for filter configuration
LogRequestBody bool true Logs the body of the request
LogRequestCookies bool true Logs the cookie values for the request
LogRequestForm bool true Logs the form values for the request
LogSqlQueries bool true Logs SQL queries using "SqlClientDiagnosticListener"
LogSqlQueryParameters bool true Logs parameter values for the SQL queries captured by LogSqlQueries
Notifiers IErrorNotifier[] empty A collection of IErrorNotifier instances to send notifications on errors
OnError Func<HttpContext, Error, Task> empty Callback that is executed before error is logged. Consumer can add or remove content to be logged in this callback.
ShowElmahErrorPage bool false Displays the Elmah UI when an error is captured
SourcePaths string[] empty Paths to source code to enrich stack traces

TIP: ℹ️ Elmah options work well with environment specific appsettings files. A Configure method exists on the builder to enable binding configuration to Elmah options.

{
    "Elmah": {
        "LogRequestCookies": false,
        "ShowElmahErrorPage": true
    }
}
builder.Host.UseElmah((builderContext, elmah) =>
{
    elmah.Configure(builderContext.Configuration.GetSection("Elmah"));
});

Restrict access to the Elmah UI

The MapElmah() method registers the Elmah endpoints as regular endpoints in the application. As such, it can accept authorization policies just like any other endpoints in the application. Metadata can be applied to the returned endpoint collection.

// allow all users to access UI
app.MapElmah().AllowAnonymous();

// or require authenticated user
app.MapElmah().RequireAuthorization();

See .NET documentation for Authorization for additional details.

Error Persistence

The following persistence options are built into the core package:

  • MemoryErrorLog – store errors in memory (default)
  • XmlFileErrorLog – store errors in XML files
using Elmah.AspNetCore;

builder.Host.UseElmah((builderContext, elmah) =>
{
    elmah.PersistToFile("~/log"; /* OR "с:\errors" */);
});
using Elmah.AspNetCore;

builder.Host.UseElmah((builderContext, elmah) =>
{
    elmah.PersistToSql(options =>
    {
        options.ConnectionString = "connection_string";
        options.SqlServerDatabaseSchemaName = "Errors"; //Defaults to dbo if not set
        options.SqlServerDatabaseTableName = "ElmahError"; //Defaults to ELMAH_Error if not set
    });
});
  • RedisErrorLog - store errors in Redis (add reference to Elmah.AspNetCore.Redis and use PersistToRedis method)
using Elmah.AspNetCore;

builder.Host.UseElmah((builderContext, elmah) =>
{
    elmah.PersistToRedis(options =>
    {
        // Defaults
        options.RedisListKeyPrefix = "urn:elmah:error_list:";
        options.RedisKeyPrefix = "urn:elmah:error:";
        options.MaximumSize = 200; // (FIFO)
    });
});

You can create implement your own error log persistence by implementing the abstract class Elmah.ErrorLog and registered it using the builder method elmah.PersistTo<YourErrorLog>() (or one of the other PersistTo overloads).

Using UseElmahExceptionPage

Use UseElmahExceptionPage (Or the ShowElmahErrorPage in Elmah options) to automatically display the Elmah UI diagnostics page when an uncaught exception occurs.

builder.Host.UseElmah((builderContext, elmah) =>
{
    if (builderContext.HostingEnvironment.IsDevelopment())
    {
        elmah.UseElmahExceptionPage();
    }
});

⚠️ The Elmah diagnostics page can leak sensitive environmental details. Consider limiting the page to development environments or placing security on the Elmah endpoints.

Using Notifiers

You can create your own notifiers by implementing IErrorNotifier interface and add notifier to Elmah options. Each notifier must have unique name. (A notifier which generates emails is build into the library.)

using Elmah.AspNetCore;
using Elmah.AspNetCore.Notifiers;

builder.Host.UseElmah((builderContext, elmah) =>
{
    elmah.Configure(options =>
    {
        options.Notifiers.Add(new ErrorMailNotifier("Email", emailOptions));
    });
});

Using Filters

You can use Elmah XML filter configuration in separate file or define them in code. Implement IErrorFilter to define custom filters in code. Filtered errors will be logged, but will not be sent.

using Elmah.AspNetCore;

builder.Host.UseElmah((builderContext, elmah) =>
{
    elmah.Configure(options =>
    {
        // Path to filters defined in XML
        options.FiltersConfig = "elmah.xml";

        // Add filters defined in code
        options.Filters.Add(new MyFilter());
    });
})

XML filter config example:

<?xml version="1.0" encoding="utf-8" ?>
<elmah>
  <errorFilter>
    <notifiers>
      <notifier name="Email"/>
    </notifiers>
    <test>
      <and>
        <greater binding="HttpStatusCode" value="399" type="Int32" />
        <lesser binding="HttpStatusCode" value="500" type="Int32" />
      </and> 
    </test>
  </errorFilter>
</elmah>

See more details in original documentation.

Extensions

Raise an Exception

To log a handled exception, use the RaiseErrorAsync extension method.

using Elmah.AspNetCore;

public async Task<IActionResult> Test()
{
    await HttpContext.RaiseErrorAsync(new InvalidOperationException("Test"));
}

Logging method parameters

using Elmah.AspNetCore;

public void TestMethod(string p1, int p2)
{
    // Logging method parameters
    HttpContext.LogParamsToElmah(this, p1, p2);
}

Serilog support

If you use Serilog for logs and would like these logs to be included as context when Elmah captures errors, the Serilog.Sinks.Elmah.AspNetCore package can be used can be used. This will add a ILogEventSink to DI which will be picked up by Serilog when the option to read configuration from services is used.

using Elmah.AspNetCore;

builder.Host.UseElmah((builderContext, elmah) =>
{
    elmah.CaptureSerilogMessages();
});

Migrating from ElmahCore

Although this is a fork of ElmahCore, it is not quite a drop-in replacement. The core functionality is all here and largely unchanged, however there has been a lot of refactoring. There are numerous breaking changes, including the following. (This is not an exhaustive list.)

  • Most bootstrapping is now handled through a UseElmah method added to the WebBuilder.Host and a callback which exposes fluent builder methods.
  • Remove custom security through OnPermissionCheck option. Instead endpoint routing and standard authorization built into ASP.Net Core are leveraged. This now requires calling MapElmah during bootstrapping.
  • With these changes, the middleware no longer handles the UI. The middleware just handles capturing errors. For this reason, it no longer needs to be after authentication/authorization middleware.
  • Dropping support for .NET 5 and previous versions. (Feel free to open a ticket if this does become an obstacle for adoption and support will be considered if there is enough demand. Several newer .NET features have been leveraged which would make back-porting difficult but not impossible.)
  • Removing sync-over-async. ErrorLog and other exposed methods are async when anything within the call stack is async.
  • More options were added to omit logging sensitive details like cookies or SQL queries.

Running Source Locally

The Elmah application contains a small Vue frontend which is bundled and embedded into the application when packaged. The source for the frontend is in the ui folder. The bundled content is not included in source control. Run npm install and then npm run build to generate the bundled content locally, which will place the bundled application into the wwwroot folder of the Elmah.AspNetCore project. Building the Elmah project will then embed the bundled content.

Running build.ps1 from the root of the repository will run all of these steps and generate the packages locally.