Skip to content

Commit

Permalink
Logs: Add LogEmitter API (#3423)
Browse files Browse the repository at this point in the history
* 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
CodeBlanch and cijothomas authored Jul 11, 2022
1 parent 85ba904 commit be588bd
Show file tree
Hide file tree
Showing 7 changed files with 695 additions and 1 deletion.
67 changes: 67 additions & 0 deletions src/OpenTelemetry/Logs/LogEmitter.cs
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);
}
}
}
}
326 changes: 326 additions & 0 deletions src/OpenTelemetry/Logs/LogRecordAttributeList.cs
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();
}
}
}
Loading

0 comments on commit be588bd

Please sign in to comment.