A Serilog sink that writes events to Google Cloud PubSub.
Branch | AppVeyor |
---|---|
dev | |
master |
- Based on [Serilog.Sinks.Seq] (https://github.com/serilog/serilog-sinks-seq) and [Serilog.Sinks.Elasticsearch] (https://github.com/serilog/serilog-sinks-elasticsearch)
- Internally uses performing grpc communication via [Google.Pubsub.V1] (https://github.com/GoogleCloudPlatform/google-cloud-dotnet/tree/master/apis/Google.Pubsub.V1)
- Internally uses durable sink configuration via [Serilog.Sinks.RollingFile] (https://github.com/serilog/serilog-sinks-rollingfile/) .
- PeriodicBatching not implemented yet.
It is used the authentication specified by the [Google.Pubsub.V1] (https://github.com/GoogleCloudPlatform/google-cloud-dotnet/tree/master/apis/Google.Pubsub.V1) package. Please refer to its documentation to see how it works.
The most common method to authenticate is to define the environment variable GOOGLE_APPLICATION_CREDENTIALS to be the location of the key (a .json file). This environment variable can be set for a machine, for a user or also can be set into code (for the process). For example: set GOOGLE_APPLICATION_CREDENTIALS=/path/to/my/key.json
Install the Serilog.Sinks.GoogleCloudPubSub package from NuGet:
Install-Package Serilog.Sinks.GoogleCloudPubSub
To configure the sink in C# code, call WriteTo.GoogleCloudPubSub()
during logger configuration passing an object with the options you need:
var log = new LoggerConfiguration()
.WriteTo.GoogleCloudPubSub(new GoogleCloudPubSubSinkOptions("_project_", "_topic_")
{
BufferBaseFilename = "\\path\\buffer-file-name"
})
.CreateLogger();
You have to provide the PubSub project and topic to the constructor of the object with the options (GoogleCloudPubSubSinkOptions type) and also you have to set the path and name of the file that will be used as buffer on disk (without extension). The other configuration options are all optional and if are not set then default values will be used.
You can also instantiate and initialize the durable sink outside the logger configuration and then provide the instance:
DurableGoogleCloudPubSubSink mySinkPubSub = new DurableGoogleCloudPubSubSink(new GoogleCloudPubSubSinkOptions("_project_", "_topic_")
{
BufferBaseFilename = "\\path\\buffer-file-name"
});
var log = new LoggerConfiguration()
.WriteTo.GoogleCloudPubSub(mySinkPubSub)
.CreateLogger();
In both cases an object of type GoogleCloudPubSubSinkOptions is used to set the configuration options of the sink, which you can instantiate separately too.
GoogleCloudPubSubSinkOptions mySinkPubSubOptions = new GoogleCloudPubSubSinkOptions("_project_", "_topic_")
{
BufferBaseFilename = "\\path\\buffer-file-name"
};
var log = new LoggerConfiguration()
.WriteTo.GoogleCloudPubSub(mySinkPubSubOptions)
.CreateLogger();
And you can also provide the configuration values directly while configurating the logger:
var log = new LoggerConfiguration()
.WriteTo.GoogleCloudPubSub("_project_", "_topic_", "\\path\\buffer-file-name")
.CreateLogger();
In all cases the project, the topic and the path for buffer file on disk are mandatory. The other configuration options are all optional and if are not set then default values will be used.
To send an event just call the method associated with the level you desire:
Log.Information("This will be sent to Google PubSub");
The default formatter (GoogleCloudPubSubRawFormatter) will send to PubSub the messageTemplate you define and will ignore any parameter you set for the event.
Google PubSub project ID (mandatory).
Google PubSub topic ID (mandatory).
Path (mandatory) for the buffer file on disk. It must not contain any rolling specifier neither the file extension.
The maximum number of events to post to PubSub in a single batch. The default value is 50.
The maximum size, in bytes, to post to PubSub in a single batch. By default no limit will be applied.
The minimum log event level required in order to write an event to the sink.
Customizes the formatter used when converting events into data to send to PubSub.
The interval (miliseconds) between checking the buffer files. The default value is 2 seconds.
Extension for the buffer files (will be added to the given BufferBaseFilename). The default value is ".swap".
Rolling specifier: {Date}, {Hour} or {HalfHour}. The default one is {Hour}.
The maximum size, in bytes, to which the buffer file for the specifier will be allowed to grow. By default no limit will be applied.
The maximum number of buffer files that will be retained, including the current buffer file. For unlimited retention, pass null. The default value is 31.
If set to 'true' then data on PubSub messages is converted to Base64. By default it is true.
Fields separator in event data.
If given indicates that the PubSub message has to contain an attribute that is obtained as the MIN value for a concret field in the event dada. This value has to be the field position (0 base), the separator "#" and the name to give to the PubSub message attribute. It is mandatory to specify the fields separador with the property EventFieldSeparator. If there is any problem then no attribute will be added to the message. The field from where to get the MIN value will be treated as an string. Null values will be omitted. Example: "3#timestamp"
If given then in each message to PubSub will be added as many attributes as elements has de dictionary, where the key corresponds to an attribute name and the value corresponds to its value to set.
Path that can be used as a log for storing internal errors and debuf information. If set then it means we want to store errors and/or debug information. It can be used the same path as the buffer log (BufferBaseFilename) but the file name can't start with the same string. It must not contain any rolling specifier: it will use the same one set for the buffer file. By default just errors information is stored.
The maximum size, in bytes, to which the error/debug file for the specifier will be allowed to grow. By default no limit will be applied.
Rolling specifier: {Date}, {Hour} or {HalfHour}. The default one is {Date}.
If set to 'true' then events related to any error will be saved to the error file (after the error message). By default it is false.
If set to 'true' then overflows when creating batch posts will be stored (overflows for BatchPostingLimit and also for BatchSizeLimitBytes). By default it is false.
If set to 'true' then skiped events (greater than the BatchSizeLimitBytes) will be stored. By default it is false.
If set to 'true' then all file actions (move forward, delete, ...) will me stored. By default it is false.
If set to 'true' then ALL debug data will be stored. If set to 'false' then each type of debug data will be stored depending on its own switch. By default it is false.
The maximum number of error log files that will be retained, including the current error file. For unlimited retention, pass null. The default value is 31.
By default, this sink will flush each event written through it to the buffer file on disk and it will be sent to PubSub from disk when a timer expires.
The buffered: true
functionality of the underlying RollingFile is not exposed on this sink as it doesn't ensure that each event is completely written to disk (they can be written partialy).
The Serilog.Sinks.Async package can be used to wrap this sink and perform all disk writings on a background worker thread.
DurableGoogleCloudPubSubSink mySinkPubSub = new DurableGoogleCloudPubSubSink(new GoogleCloudPubSubSinkOptions("_project_", "_topic_")
{
BufferBaseFilename = "\\path\\buffer-file-name"
});
var log = new LoggerConfiguration()
.WriteTo.Async(a => a.GoogleCloudPubSub(mySinkPubSub))
.CreateLogger();
GoogleCloudPubSubSinkOptions mySinkPubSubOptions = new GoogleCloudPubSubSinkOptions("_project_", "_topic_")
{
BufferBaseFilename = "\\path\\buffer-file-name"
};
var log = new LoggerConfiguration()
.WriteTo.Async(a => a.GoogleCloudPubSub(mySinkPubSubOptions))
.CreateLogger();
Serilog works with structured log events and the messageTemplate is examined char by char to find parameter references. This means that if we use a large messageTemplate then it will be spent some time looking for parameters. To avoid to spend this time we can use a short messageTemplate and then define a parameter with the real data to send to PubSub: in this case the default formatter (GoogleCloudPubSubRawFormatter) will not do the proper work and we have to use the GoogleCloudPubSubParamFormatter formatter.
DurableGoogleCloudPubSubSink mySinkPubSub = new DurableGoogleCloudPubSubSink(new GoogleCloudPubSubSinkOptions("_project_", "_topic_")
{
BufferBaseFilename = "\\path\\buffer-file-name",
CustomFormatter = new GoogleCloudPubSubParamFormatter()
});
var log = new LoggerConfiguration()
.WriteTo.Async(a => a.GoogleCloudPubSub(mySinkPubSub))
.CreateLogger();
GoogleCloudPubSubSinkOptions mySinkPubSubOptions = new GoogleCloudPubSubSinkOptions("_project_", "_topic_")
{
BufferBaseFilename = "\\path\\buffer-file-name",
CustomFormatter = new GoogleCloudPubSubParamFormatter()
};
var log = new LoggerConfiguration()
.WriteTo.Async(a => a.GoogleCloudPubSub(mySinkPubSubOptions))
.CreateLogger();
And then:
//Avoiding to spend time with a large messageTemplate: we set the message as a parameter.
Log.Information("{mt}", "This will be sent to Google PubSub");
For compatibility: with GoogleCloudPubSubParamFormatter formatter we can also log events using directly the messageTemplate as the message and not setting parameters.
//This also works with GoogleCloudPubSubParamFormatter
//but we will spend time searching for not defined parameter references.
Log.Information("This will be sent to Google PubSub");
-
2.0.5
If the Error/Debug file is enabled then it is registered the 'Dispose' action. -
2.0.4
Support for NetStandard.
New configuration option: ErrorRetainedFileCountLimit. -
2.0.3
Bug fix.
New configuration options: ErrorRollingSpecifier and DebugStoreFileAction.