-
Notifications
You must be signed in to change notification settings - Fork 778
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
* Add LogEmitter API. * Added LogEmitter tests. * Updated to NOT push ILogger scopes onto LogRecords created through LogEmitter. * Timestamp tests. * LogRecordAttributeList tests + bug fix. * Warning cleanup. Co-authored-by: Cijo Thomas <cithomas@microsoft.com>
- Loading branch information
1 parent
85ba904
commit be588bd
Showing
7 changed files
with
695 additions
and
1 deletion.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,67 @@ | ||
// <copyright file="LogEmitter.cs" company="OpenTelemetry Authors"> | ||
// Copyright The OpenTelemetry Authors | ||
// | ||
// Licensed under the Apache License, Version 2.0 (the "License"); | ||
// you may not use this file except in compliance with the License. | ||
// You may obtain a copy of the License at | ||
// | ||
// http://www.apache.org/licenses/LICENSE-2.0 | ||
// | ||
// Unless required by applicable law or agreed to in writing, software | ||
// distributed under the License is distributed on an "AS IS" BASIS, | ||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||
// See the License for the specific language governing permissions and | ||
// limitations under the License. | ||
// </copyright> | ||
|
||
#nullable enable | ||
|
||
using OpenTelemetry.Internal; | ||
|
||
namespace OpenTelemetry.Logs | ||
{ | ||
/// <summary> | ||
/// LogEmitter implementation. | ||
/// </summary> | ||
/// <remarks> | ||
/// Spec reference: <a href="https://github.com/open-telemetry/opentelemetry-specification/blob/main/specification/logs/logging-library-sdk.md#logemitter">LogEmitter</a>. | ||
/// </remarks> | ||
internal sealed class LogEmitter | ||
{ | ||
private readonly OpenTelemetryLoggerProvider loggerProvider; | ||
|
||
internal LogEmitter(OpenTelemetryLoggerProvider loggerProvider) | ||
{ | ||
Guard.ThrowIfNull(loggerProvider); | ||
|
||
this.loggerProvider = loggerProvider; | ||
} | ||
|
||
/// <summary> | ||
/// Emit a <see cref="LogRecord"/>. | ||
/// </summary> | ||
/// <param name="data"><see cref="LogRecordData"/>.</param> | ||
/// <param name="attributes"><see cref="LogRecordAttributeList"/>.</param> | ||
public void Log(in LogRecordData data, in LogRecordAttributeList attributes = default) | ||
{ | ||
var provider = this.loggerProvider; | ||
var processor = provider.Processor; | ||
if (processor != null) | ||
{ | ||
var pool = provider.LogRecordPool; | ||
|
||
var logRecord = pool.Rent(); | ||
|
||
logRecord.Data = data; | ||
|
||
attributes.ApplyToLogRecord(logRecord); | ||
|
||
processor.OnEnd(logRecord); | ||
|
||
// Attempt to return the LogRecord to the pool. This will no-op | ||
// if a batch exporter has added a reference. | ||
pool.Return(logRecord); | ||
} | ||
} | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,326 @@ | ||
// <copyright file="LogRecordAttributeList.cs" company="OpenTelemetry Authors"> | ||
// Copyright The OpenTelemetry Authors | ||
// | ||
// Licensed under the Apache License, Version 2.0 (the "License"); | ||
// you may not use this file except in compliance with the License. | ||
// You may obtain a copy of the License at | ||
// | ||
// http://www.apache.org/licenses/LICENSE-2.0 | ||
// | ||
// Unless required by applicable law or agreed to in writing, software | ||
// distributed under the License is distributed on an "AS IS" BASIS, | ||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||
// See the License for the specific language governing permissions and | ||
// limitations under the License. | ||
// </copyright> | ||
|
||
#nullable enable | ||
|
||
using System; | ||
using System.Collections; | ||
using System.Collections.Generic; | ||
using System.ComponentModel; | ||
using System.Diagnostics; | ||
using OpenTelemetry.Internal; | ||
|
||
namespace OpenTelemetry.Logs | ||
{ | ||
/// <summary> | ||
/// Stores attributes to be added to a log message. | ||
/// </summary> | ||
internal struct LogRecordAttributeList : IReadOnlyList<KeyValuePair<string, object?>> | ||
{ | ||
internal const int OverflowAdditionalCapacity = 8; | ||
internal List<KeyValuePair<string, object?>>? OverflowAttributes; | ||
private KeyValuePair<string, object?> attribute1; | ||
private KeyValuePair<string, object?> attribute2; | ||
private KeyValuePair<string, object?> attribute3; | ||
private KeyValuePair<string, object?> attribute4; | ||
private KeyValuePair<string, object?> attribute5; | ||
private KeyValuePair<string, object?> attribute6; | ||
private KeyValuePair<string, object?> attribute7; | ||
private KeyValuePair<string, object?> attribute8; | ||
private int count; | ||
|
||
/// <inheritdoc/> | ||
public readonly int Count => this.count; | ||
|
||
/// <inheritdoc/> | ||
public KeyValuePair<string, object?> this[int index] | ||
{ | ||
readonly get | ||
{ | ||
if (this.OverflowAttributes is not null) | ||
{ | ||
Debug.Assert(index < this.OverflowAttributes.Count, "Invalid index accessed."); | ||
return this.OverflowAttributes[index]; | ||
} | ||
|
||
if ((uint)index >= (uint)this.count) | ||
{ | ||
throw new ArgumentOutOfRangeException(nameof(index)); | ||
} | ||
|
||
return index switch | ||
{ | ||
0 => this.attribute1, | ||
1 => this.attribute2, | ||
2 => this.attribute3, | ||
3 => this.attribute4, | ||
4 => this.attribute5, | ||
5 => this.attribute6, | ||
6 => this.attribute7, | ||
7 => this.attribute8, | ||
_ => default, // we shouldn't come here anyway. | ||
}; | ||
} | ||
|
||
set | ||
{ | ||
if (this.OverflowAttributes is not null) | ||
{ | ||
Debug.Assert(index < this.OverflowAttributes.Count, "Invalid index accessed."); | ||
this.OverflowAttributes[index] = value; | ||
return; | ||
} | ||
|
||
if ((uint)index >= (uint)this.count) | ||
{ | ||
throw new ArgumentOutOfRangeException(nameof(index)); | ||
} | ||
|
||
switch (index) | ||
{ | ||
case 0: this.attribute1 = value; break; | ||
case 1: this.attribute2 = value; break; | ||
case 2: this.attribute3 = value; break; | ||
case 3: this.attribute4 = value; break; | ||
case 4: this.attribute5 = value; break; | ||
case 5: this.attribute6 = value; break; | ||
case 6: this.attribute7 = value; break; | ||
case 7: this.attribute8 = value; break; | ||
default: | ||
Debug.Assert(false, "Unreachable code executed."); | ||
break; | ||
} | ||
} | ||
} | ||
|
||
/// <summary> | ||
/// Add an attribute. | ||
/// </summary> | ||
/// <param name="key">Attribute name.</param> | ||
/// <returns>Attribute value.</returns> | ||
[EditorBrowsable(EditorBrowsableState.Never)] | ||
public object? this[string key] | ||
{ | ||
// Note: This only exists to enable collection initializer syntax | ||
// like { ["key"] = value }. | ||
set => this.Add(new KeyValuePair<string, object?>(key, value)); | ||
} | ||
|
||
/// <summary> | ||
/// Create a <see cref="LogRecordAttributeList"/> collection from an enumerable. | ||
/// </summary> | ||
/// <param name="attributes">Source attributes.</param> | ||
/// <returns><see cref="LogRecordAttributeList"/>.</returns> | ||
public static LogRecordAttributeList CreateFromEnumerable(IEnumerable<KeyValuePair<string, object?>> attributes) | ||
{ | ||
Guard.ThrowIfNull(attributes); | ||
|
||
LogRecordAttributeList logRecordAttributes = default; | ||
logRecordAttributes.OverflowAttributes = new(attributes); | ||
logRecordAttributes.count = logRecordAttributes.OverflowAttributes.Count; | ||
return logRecordAttributes; | ||
} | ||
|
||
/// <summary> | ||
/// Add an attribute. | ||
/// </summary> | ||
/// <param name="key">Attribute name.</param> | ||
/// <param name="value">Attribute value.</param> | ||
public void Add(string key, object? value) | ||
=> this.Add(new KeyValuePair<string, object?>(key, value)); | ||
|
||
/// <summary> | ||
/// Add an attribute. | ||
/// </summary> | ||
/// <param name="attribute">Attribute.</param> | ||
public void Add(KeyValuePair<string, object?> attribute) | ||
{ | ||
if (this.OverflowAttributes is not null) | ||
{ | ||
this.OverflowAttributes.Add(attribute); | ||
this.count++; | ||
return; | ||
} | ||
|
||
Debug.Assert(this.count <= 8, "Item added beyond struct capacity."); | ||
|
||
switch (this.count) | ||
{ | ||
case 0: this.attribute1 = attribute; break; | ||
case 1: this.attribute2 = attribute; break; | ||
case 2: this.attribute3 = attribute; break; | ||
case 3: this.attribute4 = attribute; break; | ||
case 4: this.attribute5 = attribute; break; | ||
case 5: this.attribute6 = attribute; break; | ||
case 6: this.attribute7 = attribute; break; | ||
case 7: this.attribute8 = attribute; break; | ||
case 8: | ||
Debug.Assert(this.OverflowAttributes is null, "Overflow attributes already created."); | ||
this.MoveAttributesToTheOverflowList(); | ||
Debug.Assert(this.OverflowAttributes is not null, "Overflow attributes creation failure."); | ||
this.OverflowAttributes!.Add(attribute); | ||
break; | ||
default: | ||
// We shouldn't come here. | ||
Debug.Assert(this.OverflowAttributes is null, "Unreachable code executed."); | ||
return; | ||
} | ||
|
||
this.count++; | ||
} | ||
|
||
/// <summary> | ||
/// Returns an enumerator that iterates through the <see cref="LogRecordAttributeList"/>. | ||
/// </summary> | ||
/// <returns><see cref="Enumerator"/>.</returns> | ||
public readonly Enumerator GetEnumerator() | ||
=> new(in this); | ||
|
||
/// <inheritdoc/> | ||
readonly IEnumerator<KeyValuePair<string, object?>> IEnumerable<KeyValuePair<string, object?>>.GetEnumerator() => this.GetEnumerator(); | ||
|
||
/// <inheritdoc/> | ||
readonly IEnumerator IEnumerable.GetEnumerator() => this.GetEnumerator(); | ||
|
||
internal readonly void ApplyToLogRecord(LogRecord logRecord) | ||
{ | ||
int count = this.count; | ||
if (count <= 0) | ||
{ | ||
logRecord.StateValues = null; | ||
return; | ||
} | ||
|
||
var overflowAttributes = this.OverflowAttributes; | ||
if (overflowAttributes != null) | ||
{ | ||
// An allocation has already occurred, just use the buffer. | ||
logRecord.StateValues = overflowAttributes; | ||
return; | ||
} | ||
|
||
Debug.Assert(count <= 8, "Invalid size detected."); | ||
|
||
var attributeStorage = logRecord.AttributeStorage ??= new List<KeyValuePair<string, object?>>(OverflowAdditionalCapacity); | ||
|
||
try | ||
{ | ||
// TODO: Perf test this, adjust as needed. | ||
|
||
attributeStorage.Add(this.attribute1); | ||
if (count == 1) | ||
{ | ||
return; | ||
} | ||
|
||
attributeStorage.Add(this.attribute2); | ||
if (count == 2) | ||
{ | ||
return; | ||
} | ||
|
||
attributeStorage.Add(this.attribute3); | ||
if (count == 3) | ||
{ | ||
return; | ||
} | ||
|
||
attributeStorage.Add(this.attribute4); | ||
if (count == 4) | ||
{ | ||
return; | ||
} | ||
|
||
attributeStorage.Add(this.attribute5); | ||
if (count == 5) | ||
{ | ||
return; | ||
} | ||
|
||
attributeStorage.Add(this.attribute6); | ||
if (count == 6) | ||
{ | ||
return; | ||
} | ||
|
||
attributeStorage.Add(this.attribute7); | ||
if (count == 7) | ||
{ | ||
return; | ||
} | ||
|
||
attributeStorage.Add(this.attribute8); | ||
} | ||
finally | ||
{ | ||
logRecord.StateValues = attributeStorage; | ||
} | ||
} | ||
|
||
private void MoveAttributesToTheOverflowList() | ||
{ | ||
this.OverflowAttributes = new(16) | ||
{ | ||
{ this.attribute1 }, | ||
{ this.attribute2 }, | ||
{ this.attribute3 }, | ||
{ this.attribute4 }, | ||
{ this.attribute5 }, | ||
{ this.attribute6 }, | ||
{ this.attribute7 }, | ||
{ this.attribute8 }, | ||
}; | ||
} | ||
|
||
/// <summary> | ||
/// Enumerates the elements of a <see cref="LogRecordAttributeList"/>. | ||
/// </summary> | ||
public struct Enumerator : IEnumerator<KeyValuePair<string, object?>>, IEnumerator | ||
{ | ||
private LogRecordAttributeList attributes; | ||
private int index; | ||
|
||
internal Enumerator(in LogRecordAttributeList attributes) | ||
{ | ||
this.index = -1; | ||
this.attributes = attributes; | ||
} | ||
|
||
/// <inheritdoc/> | ||
public readonly KeyValuePair<string, object?> Current | ||
=> this.attributes[this.index]; | ||
|
||
/// <inheritdoc/> | ||
readonly object IEnumerator.Current => this.Current; | ||
|
||
/// <inheritdoc/> | ||
public bool MoveNext() | ||
{ | ||
this.index++; | ||
return this.index < this.attributes.Count; | ||
} | ||
|
||
/// <inheritdoc/> | ||
public readonly void Dispose() | ||
{ | ||
} | ||
|
||
/// <inheritdoc/> | ||
readonly void IEnumerator.Reset() | ||
=> throw new NotSupportedException(); | ||
} | ||
} | ||
} |
Oops, something went wrong.