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+
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.
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.
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"));
});
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.
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" */);
});
- SqlErrorLog - store errors in MS SQL (add reference to Elmah.AspNetCore.MsSql and use
PersistToSql
method) - MysqlErrorLog - store errors in MySQL (add reference to Elmah.AspNetCore.MySql and use
PersistToMySql
method) - PgsqlErrorLog - store errors in PostgreSQL (add reference to Elmah.AspNetCore.PostgreSql and use
PersistToPgsql
method)
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).
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.
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));
});
});
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.
To log a handled exception, use the RaiseErrorAsync
extension method.
using Elmah.AspNetCore;
public async Task<IActionResult> Test()
{
await HttpContext.RaiseErrorAsync(new InvalidOperationException("Test"));
}
using Elmah.AspNetCore;
public void TestMethod(string p1, int p2)
{
// Logging method parameters
HttpContext.LogParamsToElmah(this, p1, p2);
}
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();
});
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 callingMapElmah
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.
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.