-
Notifications
You must be signed in to change notification settings - Fork 497
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Browse files
Browse the repository at this point in the history
* Implement RabbitMQ logging Add an EventSource listener to listen to the RabbitMQ event source which logs info, warn, and error messages. Forward these messages to an ILogger. Fix #564 * Stop coding like a dinosaur.
- Loading branch information
Showing
4 changed files
with
233 additions
and
2 deletions.
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
9 changes: 9 additions & 0 deletions
9
src/Components/Aspire.RabbitMQ.Client/ConfigurationSchema.json
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
218 changes: 218 additions & 0 deletions
218
src/Components/Aspire.RabbitMQ.Client/RabbitMQEventSourceLogForwarder.cs
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,218 @@ | ||
// Licensed to the .NET Foundation under one or more agreements. | ||
// The .NET Foundation licenses this file to you under the MIT license. | ||
|
||
using System.Collections; | ||
using System.Diagnostics; | ||
using System.Diagnostics.Tracing; | ||
using Microsoft.Extensions.Logging; | ||
|
||
namespace Aspire.RabbitMQ.Client; | ||
|
||
internal sealed class RabbitMQEventSourceLogForwarder : IDisposable | ||
{ | ||
private static readonly Func<ErrorEventSourceEvent, Exception?, string> s_formatErrorEvent = FormatErrorEvent; | ||
private static readonly Func<EventSourceEvent, Exception?, string> s_formatEvent = FormatEvent; | ||
|
||
private readonly ILogger _logger; | ||
private RabbitMQEventSourceListener? _listener; | ||
|
||
public RabbitMQEventSourceLogForwarder(ILoggerFactory loggerFactory) | ||
{ | ||
_logger = loggerFactory.CreateLogger("RabbitMQ.Client"); | ||
} | ||
|
||
/// <summary> | ||
/// Initiates the log forwarding from the RabbitMQ event sources to a provided <see cref="ILoggerFactory"/>, call <see cref="Dispose"/> to stop forwarding. | ||
/// </summary> | ||
public void Start() | ||
{ | ||
_listener ??= new RabbitMQEventSourceListener(LogEvent, EventLevel.Verbose); | ||
} | ||
|
||
private void LogEvent(EventWrittenEventArgs eventData) | ||
{ | ||
var level = MapLevel(eventData.Level); | ||
var eventId = new EventId(eventData.EventId, eventData.EventName); | ||
|
||
// Special case the Error event so the Exception Details are written correctly | ||
if (eventData.EventId == 3 && | ||
eventData.EventName == "Error" && | ||
eventData.PayloadNames?.Count == 2 && | ||
eventData.Payload?.Count == 2 && | ||
eventData.PayloadNames[0] == "message" && | ||
eventData.PayloadNames[1] == "ex") | ||
{ | ||
_logger.Log(level, eventId, new ErrorEventSourceEvent(eventData), null, s_formatErrorEvent); | ||
} | ||
else | ||
{ | ||
Debug.Assert( | ||
(eventData.EventId == 1 && eventData.EventName == "Info") || | ||
(eventData.EventId == 2 && eventData.EventName == "Warn")); | ||
|
||
_logger.Log(level, eventId, new EventSourceEvent(eventData), null, s_formatEvent); | ||
} | ||
} | ||
|
||
private static string FormatErrorEvent(ErrorEventSourceEvent eventSourceEvent, Exception? ex) => | ||
eventSourceEvent.EventData.Payload?[0]?.ToString() ?? "<empty>"; | ||
|
||
private static string FormatEvent(EventSourceEvent eventSourceEvent, Exception? ex) => | ||
eventSourceEvent.EventData.Payload?[0]?.ToString() ?? "<empty>"; | ||
|
||
public void Dispose() => _listener?.Dispose(); | ||
|
||
private static LogLevel MapLevel(EventLevel level) => level switch | ||
{ | ||
EventLevel.Critical => LogLevel.Critical, | ||
EventLevel.Error => LogLevel.Error, | ||
EventLevel.Informational => LogLevel.Information, | ||
EventLevel.Verbose => LogLevel.Debug, | ||
EventLevel.Warning => LogLevel.Warning, | ||
EventLevel.LogAlways => LogLevel.Information, | ||
_ => throw new ArgumentOutOfRangeException(nameof(level), level, null), | ||
}; | ||
|
||
private readonly struct EventSourceEvent : IReadOnlyList<KeyValuePair<string, object?>> | ||
{ | ||
public EventWrittenEventArgs EventData { get; } | ||
|
||
public EventSourceEvent(EventWrittenEventArgs eventData) | ||
{ | ||
// only Info and Warn events are expected, which always have 'message' as the only payload | ||
Debug.Assert(eventData.PayloadNames?.Count == 1 && eventData.PayloadNames[0] == "message"); | ||
|
||
EventData = eventData; | ||
} | ||
|
||
public IEnumerator<KeyValuePair<string, object?>> GetEnumerator() | ||
{ | ||
for (var i = 0; i < Count; i++) | ||
{ | ||
yield return this[i]; | ||
} | ||
} | ||
|
||
IEnumerator IEnumerable.GetEnumerator() | ||
{ | ||
return GetEnumerator(); | ||
} | ||
|
||
public int Count => EventData.PayloadNames?.Count ?? 0; | ||
|
||
public KeyValuePair<string, object?> this[int index] => new(EventData.PayloadNames![index], EventData.Payload![index]); | ||
} | ||
|
||
private readonly struct ErrorEventSourceEvent : IReadOnlyList<KeyValuePair<string, object?>> | ||
{ | ||
public EventWrittenEventArgs EventData { get; } | ||
|
||
public ErrorEventSourceEvent(EventWrittenEventArgs eventData) | ||
{ | ||
EventData = eventData; | ||
} | ||
|
||
public IEnumerator<KeyValuePair<string, object?>> GetEnumerator() | ||
{ | ||
for (var i = 0; i < Count; i++) | ||
{ | ||
yield return this[i]; | ||
} | ||
} | ||
|
||
IEnumerator IEnumerable.GetEnumerator() | ||
{ | ||
return GetEnumerator(); | ||
} | ||
|
||
public int Count => 5; | ||
|
||
public KeyValuePair<string, object?> this[int index] | ||
{ | ||
get | ||
{ | ||
Debug.Assert(EventData.PayloadNames?.Count == 2 && EventData.Payload?.Count == 2); | ||
Debug.Assert(EventData.PayloadNames[0] == "message"); | ||
Debug.Assert(EventData.PayloadNames[1] == "ex"); | ||
|
||
ArgumentOutOfRangeException.ThrowIfGreaterThanOrEqual(index, 5); | ||
|
||
return index switch | ||
{ | ||
0 => new(EventData.PayloadNames[0], EventData.Payload[0]), | ||
< 5 => GetExData(EventData, index), | ||
_ => throw new UnreachableException() | ||
}; | ||
|
||
static KeyValuePair<string, object?> GetExData(EventWrittenEventArgs eventData, int index) | ||
{ | ||
Debug.Assert(index >= 1 && index <= 4); | ||
Debug.Assert(eventData.Payload?.Count == 2); | ||
var exData = eventData.Payload[1] as IDictionary<string, object?>; | ||
Debug.Assert(exData is not null && exData.Count == 4); | ||
|
||
return index switch | ||
{ | ||
1 => new("ex.Type", exData["Type"]), | ||
2 => new("ex.Message", exData["Message"]), | ||
3 => new("ex.StackTrace", exData["StackTrace"]), | ||
4 => new("ex.InnerException", exData["InnerException"]), | ||
_ => throw new UnreachableException() | ||
}; | ||
} | ||
} | ||
} | ||
} | ||
|
||
/// <summary> | ||
/// Implementation of <see cref="EventListener"/> that listens to events produced by the RabbitMQ.Client library. | ||
/// </summary> | ||
private sealed class RabbitMQEventSourceListener : EventListener | ||
{ | ||
private readonly List<EventSource> _eventSources = new List<EventSource>(); | ||
|
||
private readonly Action<EventWrittenEventArgs> _log; | ||
private readonly EventLevel _level; | ||
|
||
public RabbitMQEventSourceListener(Action<EventWrittenEventArgs> log, EventLevel level) | ||
{ | ||
_log = log; | ||
_level = level; | ||
|
||
foreach (EventSource eventSource in _eventSources) | ||
{ | ||
OnEventSourceCreated(eventSource); | ||
} | ||
|
||
_eventSources.Clear(); | ||
} | ||
|
||
protected sealed override void OnEventSourceCreated(EventSource eventSource) | ||
{ | ||
base.OnEventSourceCreated(eventSource); | ||
|
||
if (_log == null) | ||
{ | ||
_eventSources.Add(eventSource); | ||
} | ||
|
||
if (eventSource.Name == "rabbitmq-dotnet-client" || eventSource.Name == "rabbitmq-client") | ||
{ | ||
EnableEvents(eventSource, _level); | ||
} | ||
} | ||
|
||
protected sealed override void OnEventWritten(EventWrittenEventArgs eventData) | ||
{ | ||
// Workaround https://github.com/dotnet/corefx/issues/42600 | ||
if (eventData.EventId == -1) | ||
{ | ||
return; | ||
} | ||
|
||
// There is a very tight race during the listener creation where EnableEvents was called | ||
// and the thread producing events not observing the `_log` field assignment | ||
_log?.Invoke(eventData); | ||
} | ||
} | ||
} |
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