This is a Serilog enricher that can mask sensitive data from a LogEvent
message template and its properties. Currently this supports e-mail addresses and IBAN numbers but could easily be extended to other types of data.
There are two ways of using this enricher:
- Always mask sensitive data (default behaviour)
- Mask data in sensitive areas only
See Usage below on how to configure this.
Let's say you have written a request/response logging middleware for ASP.Net Core that outputs:
Request start {method} {url}
End {method} {status_code} {url} {duration}
Here you have the potential that the url
property contains sensitive data because someone might do GET /api/users/[email protected]
.
Of course you can write your logging middleware to capture this and that may be the best place in this situation. However there might be cases where you don't know this is likely to happen and then you end up with the e-mail address in your logging platform.
When using this enricher what you will get is that the log message that used to be:
Request start GET /api/users/[email protected]
will be:
Request start GET /api/users/?email=***MASKED***
Even though that you know the sensitive data will be masked, it is good practice to not log sensitive data at all.
The good thing is that with the masking applied you can add an alert to your logging platform that scans for ***MASKED***
and gives you feedback when sensitive data has been detected. That allows you to fix the problem where it originates (the logging middleware).
Configure your logger with the enricher:
var logger = new LoggerConfiguration()
.Enrich.WithSensitiveDataMasking()
.WriteTo.Console()
.CreateLogger();
If you then have a log message that contains sensitive data:
logger.Information("This is a sensitive {Email}", "[email protected]");
the rendered message will be logged as:
This is a sensitive ***MASKED***
the structured log event will look like (abbreviated):
{
"RenderedMessage": "This is a sensitive ***MASKED***",
"message": "This is a sensitive {Email}",
"Properties.Email": "***MASKED***"
}
Configure your logger with the enricher:
var logger = new LoggerConfiguration()
.Enrich.WithSensitiveDataMaskingInArea()
.WriteTo.Console()
.CreateLogger();
in your application you can then define a sensitive area:
using(logger.EnterSensitiveArea())
{
logger.Information("This is a sensitive {Email}", "[email protected]");
}
The effect is that the log message will be rendered as:
This is a sensitive ***MASKED***
See the Serilog.Enrichers.Sensitive.Demo app for a code example of the above.
By default the enricher uses the following masking operators:
- EmailAddressMaskingOperator
- IbanMaskingOperator
- CreditCardMaskingOperator
It's good practice to only configure the masking operators that are applicable for your application. For example:
new LoggerConfiguration()
.Enrich
.WithSensitiveDataMasking(
options =>
{
options.MaskingOperators = new List<IMaskingOperator>
{
new EmailAddressMaskingOperator(),
new IbanMaskingOperator()
// etc etc
};
});
It is also possible to not use any masking operators but instead mask based on property names. In that case you can configure the enricher to not use any masking operators at all:
new LoggerConfiguration()
.Enrich
.WithSensitiveDataMasking(
options =>
{
options.MaskingOperators.Clear();
});
In case the default mask value ***MASKED***
is not what you want, you can supply your own mask value:
var logger = new LoggerConfiguration()
.Enrich.WithSensitiveDataMasking(options => options.MaskValue = "**")
.WriteTo.Console()
.CreateLogger();
A example rendered message would then look like:
This is a sensitive value: **
You can specify any mask string as long as it's non-null or an empty string.
In situations where you want to change the mask and have it include parts of the matched value you can override the PreprocessMask
method that takes both mask
and match
parameters. This allows you to perform more masks that are more dynamic.
For example: mask only the "user" part of an e-mail address.
public class CustomizedEmailAddressMaskingOperator : EmailAddressMaskingOperator
{
protected override string PreprocessMask(string mask, Match match)
{
var parts = match.Value.Split('@');
return mask + "@" + parts[1];
}
}
When the mask is ***MASKED***
and we pass in [email protected]
the result will be ***MASKED***@universalexports.co.uk
.
Note that this example uses EmailAddressMaskingOperator
which has a fairly complex regular expression. If possible change your regular expression to have match groups so you can more easily access them through the match
parameter.
It may be that you always want to mask the value of a property regardless of whether it matches a pattern for any of the masking operators. In that case you can specify that the property is always masked:
var logger = new LoggerConfiguration()
.Enrich.WithSensitiveDataMasking(options => options.MaskProperties.Add("email"))
.WriteTo.Console()
.CreateLogger();
Note: The property names are treated case-insensitive. If you specify
When you log any message with an email
property it will be masked:
logger.Information("This is a sensitive {Email}", "this doesn't match the regex at all");
the rendered log message comes out as: "This is a sensitive ***MASKED***"
It may be that you never want to mask the value of a property regardless of whether it matches a pattern for any of the masking operators. In that case you can specify that the property is never masked:
var logger = new LoggerConfiguration()
.Enrich.WithSensitiveDataMasking(options => options.ExcludeProperties.Add("email"))
.WriteTo.Console()
.CreateLogger();
Note: The property names are treated case-insensitive. If you specify
When you log any message with an email
property it will not be masked:
logger.Information("This is a sensitive {Email}", "[email protected]");
the rendered log message comes out as: "This is a sensitive [email protected]"
Depending on the type of masking operation you want to perform, the RegexMaskingOperator
base class is most likely your best starting point. It provides a number of extension points:
Method | Purpose |
---|---|
ShouldMaskInput | Indicate whether the operator should continue with masking the input |
PreprocessInput | Perform any operations on the input value before masking the input |
PreprocessMask | Perform any operations on the mask before masking the matched value |
ShouldMaskMatch | Indicate whether the operator should continue with masking the matched value from the input |
To implement your own masking operator, inherit from RegexMaskingOperator
, supply the regex through the base constructor and where necessary override any of the above extension points.
Then, when configuring your logger, pass your new encricher in the collection of masking operators:
var logger = new LoggerConfiguration()
.Enrich.WithSensitiveDataMasking(options => {
// Add your masking operator:
options.MaskingOperators.Add(new YourMaskingOperator());
})
.WriteTo.Console()
.CreateLogger();
If you are configuring your logger through appsettings.json
, you can configure the enricher too. You will have to add a Using
section if it doesn't exist already and include the Serilog.Enrichers.Sensitive
assembly name there, otherwise configuration will silently fail.
{
"Serilog": {
"Using": [
"Serilog.Enrichers.Sensitive"
],
"Enrich": [
{
"Name": "WithSensitiveDataMasking",
"Args": {
"options": {
"MaskValue": "CUSTOM_MASK_FROM_JSON",
"ExcludeProperties": [
"email"
],
"Mode": "Globally"
}
}
}
]
}
}
Note that options
is the argument name of the WithSensitiveDataMasking
extension method and must match exactly.
To configure masking operators you will need to specify the fully qualified name of the masking operator type. For example: MyApplication.Logging.Serilog.MyCustomMaskingOperator, MyAppliation.Logging
for the type MyCustomMaskingOperator
in the MyApplication.Logging
assembly.
An example config file:
{
"Serilog": {
"Using": [
"Serilog.Enrichers.Sensitive"
],
"Enrich": [
{
"Name": "WithSensitiveDataMasking",
"Args": {
"options": {
"MaskValue": "CUSTOM_MASK_FROM_JSON",
"Operators": [ "MyApplication.Logging.Serilog.MyCustomMaskingOperator, MyAppliation.Logging" ]
}
}
}
]
}
}
Warning: Contrary to what you might expect, for JSON configuration
Operators
should be used instead ofMaskingOperators
.