Skip to content
This repository was archived by the owner on Jun 1, 2024. It is now read-only.

Commit fa5b097

Browse files
nenadvicenticNenad Vicentic
and
Nenad Vicentic
authored
Automatically handle TypeName parameter for different versions of Elasticsearch (<7, 7 or higher) (#462)
* fix file references in the Visual Studio Solution file. * Do not set `TypeName` by default any more. * last version of Elasticsearch that supported user-defined `_type` field - v6.8.x is running out of support on 2022-02-10 (https://www.elastic.co/support/eol) * Automatically handle `ElasticsearchSinkOptions.TypeName` for different versions Elasticsearch (<7, 7+) when `ElasticsearchSinkOptions.DetectElasticsearchVersion` is enabled. * Add unit test for automatic settings of `TypeName` when `DetectElasticsearchVersion` is set to `true`. - Two methods used - instantiate `ElasticsearchSinkState` directly and via `LoggerConfiguration` * Add Elasticsearch v8 template + parsing of Elasticsearch major version to `int` + decision branch for which version of index-template API to use + removal of obsolete `ElasticsearchTemplateProvider.GetTemplate(...)` method overload. * Upgrade to .NET 6.0 and update test-frameworks related NuGet pacakges * Upgrade to Elasticsearch.NET 7.17.5 + handle new "preflight" request https://discuss.elastic.co/t/the-client-is-unable-to-verify-that-the-server-is-elasticsearch-due-to-an-unsuccessful-product-check-call-some-functionality-may-not-be-compatible-if-the-server-is-running-an-unsupported-product/310969/9 "The 7.16 client performs a pre-flight GET request to the root URL of the server before the first request.". * Make `ConnectionStub` a bit more robust . * Use `System.Version` to parse Elasticsearch server version number (similar to what `Elasticsearch.Net` does) * Update NuGet packages * Replace obsolete NuGet package `Serilog.Sinks.ColoredConsole` with `Serilog.Sinks.Console` * Update `Serilog.Sinks.PeriodicBatching` package and reimplent `ElasticsearchSink` so that it does not use obsolete `PeriodicBatchingSink` constructor. * Better handling of Elasticsearch server version number in mocked `ConnectionStub` * Cleanup: refactor to use single `JsonEquals` method. * Turn on `DetectElasticSearchVersion` option by default. Assume version 7 on fallback. * Cleanup: remove unused namespaces * Cleanup: move `ElasticsearchSinkTestsBase` into `Stubs` subfolder. * Refactor: extract `ConnectionStub` into a separate file. * Fix: json comparison in .NET Framework 4.6+ * Run unit-tests on multiple .NET frameworks. * Cleanup: remove unused NUnit runner package. * Use newer, built-in compilation constants. https://learn.microsoft.com/en-us/dotnet/csharp/language-reference/preprocessor-directives#conditional-compilation * Use standard MSBuild property `IsPackable` for clarity. * Cleanup: remove unused package refrence. * Update GitHub actions * docs: updated documentation to reflect changes in behavior of the sink. Co-authored-by: Nenad Vicentic <vicentic@immounited.com>
1 parent 306a3aa commit fa5b097

File tree

61 files changed

+1002
-788
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

61 files changed

+1002
-788
lines changed

CHANGES.md

+20
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,26 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
66

77
## [Unreleased]
88

9+
## [9.0.0] - 2023-01-23
10+
11+
### Added
12+
- PR #462
13+
14+
### Major Changes
15+
- `DetectElasticsearchVersion` is set to `true` by default.
16+
- When `DetectElasticsearchVersion` is set to `false` Elasticsearch version 7 is assumed (as it has broadest wire-compatibility at the moment - v7 and v8)
17+
- When `DetectElasticsearchVersion` is set to `true`, `TypeName` is handled automatically across different versions of Elasticserach (6.x to 8.x). For example, user-defined name will NOT be used on v7 and v8. Also, correct templates endpoint will be picked up.
18+
- Elasticsearch 8.x endpoint for templates is supported (`_index_template`)
19+
- Internal class `ElasticsearchVersionManager` has been added, mainly to handle situations where detection of version fails or when it is disabled. In the case of fallback, sink will assume "default" version 7.
20+
- Elasticsearch.NET client version 7.15.2 (latest version 7, until new `Elastic.Clients.Elasticsearch` 8.x catches up functional parity with 7.x)
21+
22+
### Other Changes
23+
- Nuget pacakges have been updated (except for the Elasticsearch integration-tests related packages)
24+
- Most of the `ElasticserachSink` functionality has been moved into internal `BatchedElasticsearchSink` class that inherits from `IBatchedLogEventSink`, so it complies with new recommended way of integration with `PeriodicBatchingSink` and we don't use obsolete constructors.
25+
- `ConnectionStub` was moved out of `ElasticsearchSinkTestsBase` and extended. Both are now in `/Stubs` subfolder. Newer versions of Elasticsearch.NET client are now using "pre-flight" request to determine if endpoint is Elasticsearch and if it is indeed between 6.x and 8.x. `ConnectionStub` had to accommodate for that.
26+
- Unit tests have been fixed/added accordingly, running on multiple target frameworks (`net6`, `net7` and `net48`).
27+
- Built-in .NET SDK conditional compilation symbols are now used (e.g NETFRAMEWORK).
28+
929
## [9.0.0] - 2022-04-29
1030

1131
### Fixed

README.md

+68-2
Original file line numberDiff line numberDiff line change
@@ -30,6 +30,7 @@ The Serilog Elasticsearch sink project is a sink (basically a writer) for the Se
3030
* Starting from version 3, compatible with Elasticsearch 2.
3131
* Version 6.x supports the new Elasticsearch.net version 6.x library.
3232
* From version 8.x there is support for Elasticsearch.net version 7.
33+
* From version 9.x there is support for Elasticsearch.net version 8. Version detection is enabled by default, in which case `TypeName` is handled automatically across major versions 6, 7 and 8.
3334

3435

3536
## Quick start
@@ -40,16 +41,80 @@ The Serilog Elasticsearch sink project is a sink (basically a writer) for the Se
4041
Install-Package serilog.sinks.elasticsearch
4142
```
4243

43-
Register the sink in code or using the appSettings reader (from v2.0.42+) as shown below. Make sure to specify the version of ES you are targeting. Be aware that the AutoRegisterTemplate option will not overwrite an existing template.
44+
Simplest way to register this sink is to use default configuration:
45+
46+
```csharp
47+
var loggerConfig = new LoggerConfiguration()
48+
.WriteTo.Elasticsearch(new ElasticsearchSinkOptions(new Uri("http://localhost:9200"));
49+
```
50+
51+
Or, if using .NET Core and `Serilog.Settings.Configuration` Nuget package and `appsettings.json`, default configuration would look like this:
52+
53+
```json
54+
{
55+
"Serilog": {
56+
"Using": [ "Serilog.Sinks.Elasticsearch" ],
57+
"MinimumLevel": "Warning",
58+
"WriteTo": [
59+
{
60+
"Name": "Elasticsearch",
61+
"Args": {
62+
"nodeUris": "http://localhost:9200"
63+
}
64+
}
65+
],
66+
"Enrich": [ "FromLogContext", "WithMachineName" ],
67+
"Properties": {
68+
"Application": "ImmoValuation.Swv - Web"
69+
}
70+
}
71+
}
72+
```
73+
74+
More elaborate configuration, using additional Nuget packages (e.g. `Serilog.Enrichers.Environment`) would look like:
75+
76+
```json
77+
{
78+
"Serilog": {
79+
"Using": [ "Serilog.Sinks.Elasticsearch" ],
80+
"MinimumLevel": "Warning",
81+
"WriteTo": [
82+
{
83+
"Name": "Elasticsearch",
84+
"Args": {
85+
"nodeUris": "http://localhost:9200"
86+
}
87+
}
88+
],
89+
"Enrich": [ "FromLogContext", "WithMachineName" ],
90+
"Properties": {
91+
"Application": "My app"
92+
}
93+
}
94+
}
95+
```
96+
97+
This way the sink will detect version of Elasticsearch server (`DetectElasticsearchVersion` is set to `true` by default) and handle `TypeName` behavior correctly, based on the server version (6.x, 7.x or 8.x).
98+
99+
### Disable detection of Elasticsearch server version
100+
101+
Alternatively, `DetectElasticsearchVersion` can be set to `false` and certain option can be configured manually. In that case, the sink will assume version 7 of Elasticsearch, but options will be ignored due to a potential version incompatibility.
102+
103+
For example, you can configure the sink to force registeration of v6 index template. Be aware that the AutoRegisterTemplate option will not overwrite an existing template.
44104

45105
```csharp
46106
var loggerConfig = new LoggerConfiguration()
47107
.WriteTo.Elasticsearch(new ElasticsearchSinkOptions(new Uri("http://localhost:9200") ){
108+
DetectElasticsearchVersion = false,
48109
AutoRegisterTemplate = true,
49110
AutoRegisterTemplateVersion = AutoRegisterTemplateVersion.ESv6
50111
});
51112
```
52113

114+
### Configurable properties
115+
116+
Besides a registration of the sink in the code, it is possible to register it using appSettings reader (from v2.0.42+) reader (from v2.0.42+) as shown below.
117+
53118
This example shows the options that are currently available when using the appSettings reader.
54119

55120
```xml
@@ -75,7 +140,8 @@ This example shows the options that are currently available when using the appSe
75140
<add key="serilog:write-to:Elasticsearch.emitEventFailure" value="WriteToSelfLog" />
76141
<add key="serilog:write-to:Elasticsearch.queueSizeLimit" value="100000" />
77142
<add key="serilog:write-to:Elasticsearch.autoRegisterTemplate" value="true" />
78-
<add key="serilog:write-to:Elasticsearch.autoRegisterTemplateVersion" value="ESv2" />
143+
<add key="serilog:write-to:Elasticsearch.autoRegisterTemplateVersion" value="ESv7" />
144+
<add key="serilog:write-to:Elasticsearch.detectElasticsearchVersion" value="false" /><!-- `true` is default -->
79145
<add key="serilog:write-to:Elasticsearch.overwriteTemplate" value="false" />
80146
<add key="serilog:write-to:Elasticsearch.registerTemplateFailure" value="IndexAnyway" />
81147
<add key="serilog:write-to:Elasticsearch.deadLetterIndexName" value="deadletter-{0:yyyy.MM}" />

sample/Serilog.Sinks.Elasticsearch.Sample/Serilog.Sinks.Elasticsearch.Sample.csproj

+3-3
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@
22

33
<PropertyGroup>
44
<OutputType>Exe</OutputType>
5-
<TargetFramework>netcoreapp3.1</TargetFramework>
5+
<TargetFramework>net6.0</TargetFramework>
66
<IsPackable>false</IsPackable>
77
</PropertyGroup>
88

@@ -11,8 +11,8 @@
1111
<PackageReference Include="Microsoft.Extensions.Configuration.EnvironmentVariables" Version="6.0.1" />
1212
<PackageReference Include="Microsoft.Extensions.Configuration.FileExtensions" Version="6.0.0" />
1313
<PackageReference Include="Microsoft.Extensions.Configuration.Json" Version="6.0.0" />
14-
<PackageReference Include="Serilog" Version="2.11.0" />
15-
<PackageReference Include="Serilog.Sinks.Console" Version="4.0.1" />
14+
<PackageReference Include="Serilog" Version="2.12.0" />
15+
<PackageReference Include="Serilog.Sinks.Console" Version="4.1.0" />
1616
</ItemGroup>
1717

1818
<ItemGroup>

src/Serilog.Formatting.Elasticsearch/Serilog.Formatting.Elasticsearch.csproj

+1-1
Original file line numberDiff line numberDiff line change
@@ -28,7 +28,7 @@
2828
</PropertyGroup>
2929

3030
<ItemGroup>
31-
<PackageReference Include="Serilog" Version="2.11.0" />
31+
<PackageReference Include="Serilog" Version="2.12.0" />
3232
</ItemGroup>
3333

3434
<ItemGroup>

src/Serilog.Sinks.Elasticsearch/LoggerConfigurationElasticSearchExtensions.cs

+7-5
Original file line numberDiff line numberDiff line change
@@ -143,6 +143,7 @@ public static LoggerConfiguration Elasticsearch(
143143
/// <param name="failureSink">Sink to use when Elasticsearch is unable to accept the events. This is optionally and depends on the EmitEventFailure setting.</param>
144144
/// <param name="singleEventSizePostingLimit"><see cref="ElasticsearchSinkOptions.SingleEventSizePostingLimit"/>The maximum length of an event allowed to be posted to Elasticsearch.default null</param>
145145
/// <param name="templateCustomSettings">Add custom elasticsearch settings to the template</param>
146+
/// <param name="detectElasticsearchVersion">Turns on detection of elasticsearch version via background HTTP call. This will also set `TypeName` automatically, according to the version of Elasticsearch.</param>
146147
/// <param name="batchAction">Configures the OpType being used when inserting document in batch. Must be set to create for data streams.</param>
147148
/// <returns>LoggerConfiguration object</returns>
148149
/// <exception cref="ArgumentNullException"><paramref name="nodeUris"/> is <see langword="null" />.</exception>
@@ -151,7 +152,7 @@ public static LoggerConfiguration Elasticsearch(
151152
string nodeUris,
152153
string indexFormat = null,
153154
string templateName = null,
154-
string typeName = "logevent",
155+
string typeName = null,
155156
int batchPostingLimit = 50,
156157
int period = 2,
157158
bool inlineFields = false,
@@ -166,7 +167,7 @@ public static LoggerConfiguration Elasticsearch(
166167
int queueSizeLimit = 100000,
167168
string pipelineName = null,
168169
bool autoRegisterTemplate = false,
169-
AutoRegisterTemplateVersion autoRegisterTemplateVersion = AutoRegisterTemplateVersion.ESv2,
170+
AutoRegisterTemplateVersion? autoRegisterTemplateVersion = null,
170171
bool overwriteTemplate = false,
171172
RegisterTemplateRecovery registerTemplateFailure = RegisterTemplateRecovery.IndexAnyway,
172173
string deadLetterIndexName = null,
@@ -182,7 +183,8 @@ public static LoggerConfiguration Elasticsearch(
182183
long? singleEventSizePostingLimit = null,
183184
int? bufferFileCountLimit = null,
184185
Dictionary<string,string> templateCustomSettings = null,
185-
ElasticOpType batchAction = ElasticOpType.Index)
186+
ElasticOpType batchAction = ElasticOpType.Index,
187+
bool detectElasticsearchVersion = true)
186188
{
187189
if (string.IsNullOrEmpty(nodeUris))
188190
throw new ArgumentNullException(nameof(nodeUris), "No Elasticsearch node(s) specified.");
@@ -204,8 +206,6 @@ public static LoggerConfiguration Elasticsearch(
204206
options.TemplateName = templateName;
205207
}
206208

207-
options.TypeName = !string.IsNullOrWhiteSpace(typeName) ? typeName : null;
208-
209209
options.BatchPostingLimit = batchPostingLimit;
210210
options.BatchAction = batchAction;
211211
options.SingleEventSizePostingLimit = singleEventSizePostingLimit;
@@ -272,6 +272,8 @@ public static LoggerConfiguration Elasticsearch(
272272

273273
options.TemplateCustomSettings = templateCustomSettings;
274274

275+
options.DetectElasticsearchVersion = detectElasticsearchVersion;
276+
275277
return Elasticsearch(loggerSinkConfiguration, options);
276278
}
277279
}

src/Serilog.Sinks.Elasticsearch/Properties/AssemblyInfo.cs

+1-1
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,7 @@
55
[assembly: AssemblyDescription("Serilog sink for Elasticsearch")]
66
[assembly: AssemblyCopyright("Copyright © Serilog Contributors 2018")]
77

8-
[assembly: InternalsVisibleTo("Serilog.Tests, PublicKey=0024000004800000940000000602000000240000525341310004000001000100fb8d13fd344a1c" +
8+
[assembly: InternalsVisibleTo("Serilog.Sinks.Elasticsearch.Tests, PublicKey=0024000004800000940000000602000000240000525341310004000001000100fb8d13fd344a1c" +
99
"6fe0fe83ef33c1080bf30690765bc6eb0df26ebfdf8f21670c64265b30db09f73a0dea5b3db4c9" +
1010
"d18dbf6d5a25af5ce9016f281014d79dc3b4201ac646c451830fc7e61a2dfd633d34c39f87b818" +
1111
"94191652df5ac63cc40c77f3542f702bda692e6e8a9158353df189007a49da0f3cfd55eb250066" +

src/Serilog.Sinks.Elasticsearch/Serilog.Sinks.Elasticsearch.csproj

+4-3
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@
44
<Authors>Michiel van Oudheusden, Martijn Laarman, Mogens Heller Grabe, Serilog Contributors</Authors>
55
<TargetFrameworks>netstandard2.0</TargetFrameworks>
66
<TreatWarningsAsErrors>true</TreatWarningsAsErrors>
7+
<LangVersion>latest</LangVersion>
78
<GenerateDocumentationFile>true</GenerateDocumentationFile>
89
<AssemblyName>Serilog.Sinks.Elasticsearch</AssemblyName>
910
<AssemblyOriginatorKeyFile>../../assets/Serilog.snk</AssemblyOriginatorKeyFile>
@@ -35,11 +36,11 @@
3536

3637
<ProjectReference Include="..\Serilog.Formatting.Elasticsearch\Serilog.Formatting.Elasticsearch.csproj" />
3738

38-
<PackageReference Include="Serilog" Version="2.11.0" />
39+
<PackageReference Include="Serilog" Version="2.12.0" />
3940
<PackageReference Include="Serilog.Formatting.Compact" Version="1.1.0" />
4041
<PackageReference Include="Serilog.Sinks.File" Version="5.0.0" />
41-
<PackageReference Include="Serilog.Sinks.PeriodicBatching" Version="2.1.1" />
42-
<PackageReference Include="Elasticsearch.Net" Version="7.8.1" />
42+
<PackageReference Include="Serilog.Sinks.PeriodicBatching" Version="3.1.0" />
43+
<PackageReference Include="Elasticsearch.Net" Version="7.17.5" />
4344
<PackageReference Include="System.Diagnostics.DiagnosticSource" Version="6.0.0" />
4445
</ItemGroup>
4546

src/Serilog.Sinks.Elasticsearch/Sinks/ElasticSearch/Durable/Elasticsearch/ElasticsearchLogClient.cs

+1-1
Original file line numberDiff line numberDiff line change
@@ -86,7 +86,7 @@ private InvalidResult GetInvalidPayloadAsync(DynamicResponse baseResult, List<st
8686
bool hasErrors = false;
8787
foreach (dynamic item in items)
8888
{
89-
var itemIndex = item?[ElasticsearchSink.BulkAction(_elasticOpType)];
89+
var itemIndex = item?[BatchedElasticsearchSink.BulkAction(_elasticOpType)];
9090
long? status = itemIndex?["status"];
9191
i++;
9292
if (!status.HasValue || status < 300)

src/Serilog.Sinks.Elasticsearch/Sinks/ElasticSearch/Durable/Elasticsearch/ElasticsearchPayloadReader.cs

+1-1
Original file line numberDiff line numberDiff line change
@@ -92,7 +92,7 @@ protected override List<string> FinishPayLoad()
9292
protected override void AddToPayLoad(string nextLine)
9393
{
9494
var indexName = _getIndexForEvent(nextLine, _date);
95-
var action = ElasticsearchSink.CreateElasticAction(
95+
var action = BatchedElasticsearchSink.CreateElasticAction(
9696
opType: _elasticOpType,
9797
indexName: indexName, pipelineName: _pipelineName,
9898
id: _count + "_" + Guid.NewGuid(),

src/Serilog.Sinks.Elasticsearch/Sinks/ElasticSearch/ElasticSearchSink.cs

+31-40
Original file line numberDiff line numberDiff line change
@@ -25,26 +25,39 @@
2525

2626
namespace Serilog.Sinks.Elasticsearch
2727
{
28+
public sealed class ElasticsearchSink : PeriodicBatchingSink
29+
{
30+
public ElasticsearchSink(ElasticsearchSinkOptions options)
31+
: this(
32+
new BatchedElasticsearchSink(options),
33+
new PeriodicBatchingSinkOptions
34+
{
35+
BatchSizeLimit = options.BatchPostingLimit,
36+
Period = options.Period,
37+
EagerlyEmitFirstEvent = true,
38+
QueueLimit = (options.QueueSizeLimit == -1) ? null : new int?(options.QueueSizeLimit)
39+
}
40+
)
41+
{
42+
}
43+
44+
private ElasticsearchSink(IBatchedLogEventSink batchedSink, PeriodicBatchingSinkOptions options)
45+
: base(batchedSink, options)
46+
{
47+
}
48+
}
2849
/// <summary>
2950
/// Writes log events as documents to ElasticSearch.
3051
/// </summary>
31-
public class ElasticsearchSink : PeriodicBatchingSink
52+
internal sealed class BatchedElasticsearchSink : IBatchedLogEventSink
3253
{
33-
3454
private readonly ElasticsearchSinkState _state;
3555

36-
/// <summary>
37-
/// Creates a new ElasticsearchSink instance with the provided options
38-
/// </summary>
39-
/// <param name="options">Options configuring how the sink behaves, may NOT be null</param>
40-
public ElasticsearchSink(ElasticsearchSinkOptions options)
41-
: base(options.BatchPostingLimit, options.Period, options.QueueSizeLimit)
56+
public BatchedElasticsearchSink(ElasticsearchSinkOptions options)
4257
{
4358
_state = ElasticsearchSinkState.Create(options);
44-
_state.DiscoverClusterVersion();
4559
_state.RegisterTemplateIfNeeded();
4660
}
47-
4861
/// <summary>
4962
/// Emit a batch of log events, running to completion synchronously.
5063
/// </summary>
@@ -54,7 +67,7 @@ public ElasticsearchSink(ElasticsearchSinkOptions options)
5467
/// or <see cref="M:Serilog.Sinks.PeriodicBatching.PeriodicBatchingSink.EmitBatchAsync(System.Collections.Generic.IEnumerable{Serilog.Events.LogEvent})" />,
5568
/// not both.
5669
/// </remarks>
57-
protected override async Task EmitBatchAsync(IEnumerable<LogEvent> events)
70+
public async Task EmitBatchAsync(IEnumerable<LogEvent> events)
5871
{
5972
DynamicResponse result;
6073

@@ -71,42 +84,32 @@ protected override async Task EmitBatchAsync(IEnumerable<LogEvent> events)
7184
HandleResponse(events, result);
7285
}
7386

74-
/// <summary>
75-
/// Emit a batch of log events, running to completion synchronously.
76-
/// </summary>
77-
/// <param name="events">The events to emit.</param>
78-
/// <returns>Response from Elasticsearch</returns>
79-
protected virtual Task<T> EmitBatchCheckedAsync<T>(IEnumerable<LogEvent> events) where T : class, IElasticsearchResponse, new()
87+
public Task OnEmptyBatchAsync()
8088
{
81-
// ReSharper disable PossibleMultipleEnumeration
82-
if (events == null || !events.Any())
83-
return Task.FromResult<T>(default(T));
84-
85-
var payload = CreatePayload(events);
86-
return _state.Client.BulkAsync<T>(PostData.MultiJson(payload));
89+
return Task.CompletedTask;
8790
}
8891

8992
/// <summary>
9093
/// Emit a batch of log events, running to completion synchronously.
9194
/// </summary>
9295
/// <param name="events">The events to emit.</param>
9396
/// <returns>Response from Elasticsearch</returns>
94-
protected virtual T EmitBatchChecked<T>(IEnumerable<LogEvent> events) where T : class, IElasticsearchResponse, new()
97+
private Task<T> EmitBatchCheckedAsync<T>(IEnumerable<LogEvent> events) where T : class, IElasticsearchResponse, new()
9598
{
9699
// ReSharper disable PossibleMultipleEnumeration
97100
if (events == null || !events.Any())
98-
return null;
101+
return Task.FromResult<T>(default(T));
99102

100103
var payload = CreatePayload(events);
101-
return _state.Client.Bulk<T>(PostData.MultiJson(payload));
104+
return _state.Client.BulkAsync<T>(PostData.MultiJson(payload));
102105
}
103106

104107
/// <summary>
105108
/// Handles the exceptions.
106109
/// </summary>
107110
/// <param name="ex"></param>
108111
/// <param name="events"></param>
109-
protected virtual void HandleException(Exception ex, IEnumerable<LogEvent> events)
112+
private void HandleException(Exception ex, IEnumerable<LogEvent> events)
110113
{
111114
if (_state.Options.EmitEventFailure.HasFlag(EmitEventFailureHandling.WriteToSelfLog))
112115
{
@@ -153,18 +156,6 @@ protected virtual void HandleException(Exception ex, IEnumerable<LogEvent> event
153156
throw ex;
154157
}
155158

156-
// Helper function: checks if a given dynamic member / dictionary key exists at runtime
157-
private static bool HasProperty(dynamic settings, string name)
158-
{
159-
if (settings is IDictionary<string, object>)
160-
return ((IDictionary<string, object>)settings).ContainsKey(name);
161-
162-
if (settings is System.Dynamic.DynamicObject)
163-
return ((System.Dynamic.DynamicObject)settings).GetDynamicMemberNames().Contains(name);
164-
165-
return settings.GetType().GetProperty(name) != null;
166-
}
167-
168159
private IEnumerable<string> CreatePayload(IEnumerable<LogEvent> events)
169160
{
170161
if (!_state.TemplateRegistrationSuccess && _state.Options.RegisterTemplateFailure == RegisterTemplateRecovery.FailSink)
@@ -280,7 +271,7 @@ internal static object CreateElasticAction(ElasticOpType opType, string indexNam
280271
: new ElasticIndexAction(actionPayload);
281272
return action;
282273
}
283-
274+
284275
sealed class ElasticCreateAction
285276
{
286277
public ElasticCreateAction(ElasticActionPayload payload)

0 commit comments

Comments
 (0)