| | | 1 | | using System.Diagnostics; |
| | | 2 | | using System.Diagnostics.Metrics; |
| | | 3 | | using AsiBackbone.Core.Emissions; |
| | | 4 | | |
| | | 5 | | namespace AsiBackbone.OpenTelemetry; |
| | | 6 | | |
| | | 7 | | /// <summary> |
| | | 8 | | /// Emits provider-neutral governance envelopes through OpenTelemetry-friendly .NET diagnostics primitives. |
| | | 9 | | /// </summary> |
| | | 10 | | /// <remarks> |
| | | 11 | | /// This emitter records activity events, activity tags, and low-cardinality metrics. It does not configure exporters or |
| | | 12 | | /// </remarks> |
| | | 13 | | public sealed class OpenTelemetryGovernanceEmitter : IAsiBackboneGovernanceEmitter |
| | | 14 | | { |
| | 2 | 15 | | private static readonly ActivitySource ActivitySource = new(OpenTelemetryGovernanceInstrumentation.ActivitySourceNam |
| | 2 | 16 | | private static readonly Meter Meter = new(OpenTelemetryGovernanceInstrumentation.MeterName); |
| | 2 | 17 | | private static readonly Counter<long> EmissionsCounter = Meter.CreateCounter<long>( |
| | 2 | 18 | | OpenTelemetryGovernanceInstrumentation.EmissionsCounterName, |
| | 2 | 19 | | description: "Counts AsiBackbone governance emission attempts accepted by the OpenTelemetry diagnostics provider |
| | 2 | 20 | | private static readonly Counter<long> EmissionFailuresCounter = Meter.CreateCounter<long>( |
| | 2 | 21 | | OpenTelemetryGovernanceInstrumentation.EmissionFailuresCounterName, |
| | 2 | 22 | | description: "Counts AsiBackbone governance emission failures normalized by the OpenTelemetry diagnostics provid |
| | 2 | 23 | | private static readonly Histogram<double> EmissionLatencyHistogram = Meter.CreateHistogram<double>( |
| | 2 | 24 | | OpenTelemetryGovernanceInstrumentation.EmissionLatencyHistogramName, |
| | 2 | 25 | | unit: "ms", |
| | 2 | 26 | | description: "Measures local OpenTelemetry governance emission latency in milliseconds."); |
| | | 27 | | |
| | | 28 | | private readonly OpenTelemetryGovernanceEmitterOptions options; |
| | | 29 | | |
| | | 30 | | /// <summary> |
| | | 31 | | /// Initializes a new instance of the <see cref="OpenTelemetryGovernanceEmitter" /> class using default options. |
| | | 32 | | /// </summary> |
| | | 33 | | public OpenTelemetryGovernanceEmitter() |
| | 4 | 34 | | : this(new OpenTelemetryGovernanceEmitterOptions()) |
| | | 35 | | { |
| | 4 | 36 | | } |
| | | 37 | | |
| | | 38 | | /// <summary> |
| | | 39 | | /// Initializes a new instance of the <see cref="OpenTelemetryGovernanceEmitter" /> class using host-owned options. |
| | | 40 | | /// </summary> |
| | | 41 | | /// <param name="options">The OpenTelemetry governance emitter options.</param> |
| | 6 | 42 | | public OpenTelemetryGovernanceEmitter(OpenTelemetryGovernanceEmitterOptions options) |
| | | 43 | | { |
| | 6 | 44 | | ArgumentNullException.ThrowIfNull(options); |
| | 6 | 45 | | options.Validate(); |
| | | 46 | | |
| | 6 | 47 | | this.options = options; |
| | 6 | 48 | | } |
| | | 49 | | |
| | | 50 | | /// <inheritdoc /> |
| | | 51 | | public async ValueTask<GovernanceEmissionResult> EmitAsync( |
| | | 52 | | GovernanceEmissionEnvelope envelope, |
| | | 53 | | CancellationToken cancellationToken = default) |
| | | 54 | | { |
| | 6 | 55 | | ArgumentNullException.ThrowIfNull(envelope); |
| | 6 | 56 | | cancellationToken.ThrowIfCancellationRequested(); |
| | | 57 | | |
| | 6 | 58 | | var stopwatch = Stopwatch.StartNew(); |
| | | 59 | | |
| | | 60 | | try |
| | | 61 | | { |
| | 6 | 62 | | if (options.BeforeEmitAsync is not null) |
| | | 63 | | { |
| | 2 | 64 | | await options.BeforeEmitAsync(envelope, cancellationToken).ConfigureAwait(false); |
| | | 65 | | } |
| | | 66 | | |
| | 4 | 67 | | cancellationToken.ThrowIfCancellationRequested(); |
| | 4 | 68 | | double latencyMs = stopwatch.Elapsed.TotalMilliseconds; |
| | 4 | 69 | | string eventName = GetEventName(envelope); |
| | 4 | 70 | | string providerName = options.ProviderName.Trim(); |
| | | 71 | | |
| | 4 | 72 | | if (options.EmitActivityEvents) |
| | | 73 | | { |
| | 4 | 74 | | EmitActivity(envelope, eventName, providerName, options.DefaultActivityName, latencyMs); |
| | | 75 | | } |
| | | 76 | | |
| | 4 | 77 | | if (options.EmitMetrics) |
| | | 78 | | { |
| | 4 | 79 | | EmitDeliveredMetrics(envelope, providerName, latencyMs); |
| | | 80 | | } |
| | | 81 | | |
| | 4 | 82 | | return GovernanceEmissionResult.Delivered( |
| | 4 | 83 | | providerName, |
| | 4 | 84 | | envelope.EnvelopeId, |
| | 4 | 85 | | new Dictionary<string, string>(StringComparer.Ordinal) |
| | 4 | 86 | | { |
| | 4 | 87 | | ["opentelemetry.activity_source"] = OpenTelemetryGovernanceInstrumentation.ActivitySourceName, |
| | 4 | 88 | | ["opentelemetry.meter"] = OpenTelemetryGovernanceInstrumentation.MeterName, |
| | 4 | 89 | | ["opentelemetry.event_name"] = eventName, |
| | 4 | 90 | | ["opentelemetry.emission_latency_ms"] = latencyMs.ToString("0.###", System.Globalization.CultureInfo |
| | 4 | 91 | | }); |
| | | 92 | | } |
| | 0 | 93 | | catch (OperationCanceledException) when (cancellationToken.IsCancellationRequested) |
| | | 94 | | { |
| | 0 | 95 | | throw; |
| | | 96 | | } |
| | 2 | 97 | | catch (Exception ex) |
| | | 98 | | { |
| | 2 | 99 | | double latencyMs = stopwatch.Elapsed.TotalMilliseconds; |
| | 2 | 100 | | var error = GovernanceEmissionError.Create( |
| | 2 | 101 | | "opentelemetry.emission.exception", |
| | 2 | 102 | | $"OpenTelemetry governance emission failed with {ex.GetType().Name}.", |
| | 2 | 103 | | isRetryable: true, |
| | 2 | 104 | | providerName: options.ProviderName, |
| | 2 | 105 | | providerErrorCode: ex.GetType().FullName); |
| | | 106 | | |
| | 2 | 107 | | if (options.EmitMetrics) |
| | | 108 | | { |
| | 2 | 109 | | EmitFailureMetrics(envelope, error, latencyMs); |
| | | 110 | | } |
| | | 111 | | |
| | 2 | 112 | | return GovernanceEmissionResult.RetryableFailure( |
| | 2 | 113 | | error, |
| | 2 | 114 | | providerName: options.ProviderName, |
| | 2 | 115 | | metadata: new Dictionary<string, string>(StringComparer.Ordinal) |
| | 2 | 116 | | { |
| | 2 | 117 | | ["opentelemetry.activity_source"] = OpenTelemetryGovernanceInstrumentation.ActivitySourceName, |
| | 2 | 118 | | ["opentelemetry.meter"] = OpenTelemetryGovernanceInstrumentation.MeterName, |
| | 2 | 119 | | ["opentelemetry.emission_latency_ms"] = latencyMs.ToString("0.###", System.Globalization.CultureInfo |
| | 2 | 120 | | }); |
| | | 121 | | } |
| | 6 | 122 | | } |
| | | 123 | | |
| | | 124 | | private static void EmitActivity( |
| | | 125 | | GovernanceEmissionEnvelope envelope, |
| | | 126 | | string eventName, |
| | | 127 | | string providerName, |
| | | 128 | | string defaultActivityName, |
| | | 129 | | double latencyMs) |
| | | 130 | | { |
| | 4 | 131 | | ActivityTagsCollection tags = BuildTags(envelope, providerName, GovernanceEmissionStatus.Delivered.ToString(), l |
| | 4 | 132 | | string activityName = string.IsNullOrWhiteSpace(envelope.OperationName) |
| | 4 | 133 | | ? defaultActivityName |
| | 4 | 134 | | : envelope.OperationName; |
| | | 135 | | |
| | 4 | 136 | | using Activity? activity = ActivitySource.StartActivity(activityName, ActivityKind.Internal); |
| | | 137 | | |
| | 4 | 138 | | if (activity is null) |
| | | 139 | | { |
| | 2 | 140 | | return; |
| | | 141 | | } |
| | | 142 | | |
| | 108 | 143 | | foreach (KeyValuePair<string, object?> tag in tags) |
| | | 144 | | { |
| | 52 | 145 | | _ = activity.SetTag(tag.Key, tag.Value); |
| | | 146 | | } |
| | | 147 | | |
| | 2 | 148 | | _ = activity.AddEvent(new ActivityEvent(eventName, envelope.OccurredUtc, tags)); |
| | 6 | 149 | | } |
| | | 150 | | |
| | | 151 | | private static ActivityTagsCollection BuildTags( |
| | | 152 | | GovernanceEmissionEnvelope envelope, |
| | | 153 | | string providerName, |
| | | 154 | | string result, |
| | | 155 | | double latencyMs) |
| | | 156 | | { |
| | 4 | 157 | | ActivityTagsCollection tags = []; |
| | | 158 | | |
| | 4 | 159 | | AddTag(tags, OpenTelemetryGovernanceAttributes.EnvelopeId, envelope.EnvelopeId); |
| | 4 | 160 | | AddTag(tags, OpenTelemetryGovernanceAttributes.SchemaVersion, envelope.SchemaVersion); |
| | 4 | 161 | | AddTag(tags, OpenTelemetryGovernanceAttributes.EventType, envelope.EventType.ToString()); |
| | 4 | 162 | | AddTag(tags, OpenTelemetryGovernanceAttributes.EventId, envelope.EventId); |
| | 4 | 163 | | AddTag(tags, OpenTelemetryGovernanceAttributes.CorrelationId, envelope.CorrelationId); |
| | 4 | 164 | | AddTag(tags, OpenTelemetryGovernanceAttributes.AuditResidueId, envelope.AuditResidueId); |
| | 4 | 165 | | AddTag(tags, OpenTelemetryGovernanceAttributes.TraceId, envelope.TraceId); |
| | 4 | 166 | | AddTag(tags, OpenTelemetryGovernanceAttributes.SpanId, envelope.SpanId); |
| | 4 | 167 | | AddTag(tags, OpenTelemetryGovernanceAttributes.ParentSpanId, envelope.ParentSpanId); |
| | 4 | 168 | | AddTag(tags, OpenTelemetryGovernanceAttributes.DecisionOutcome, envelope.Outcome); |
| | 4 | 169 | | AddTag(tags, OpenTelemetryGovernanceAttributes.DecisionStage, envelope.DecisionStage); |
| | 4 | 170 | | AddTag(tags, OpenTelemetryGovernanceAttributes.PolicyVersion, envelope.PolicyVersion); |
| | 4 | 171 | | AddTag(tags, OpenTelemetryGovernanceAttributes.PolicyHash, envelope.PolicyHash); |
| | 4 | 172 | | AddTag(tags, OpenTelemetryGovernanceAttributes.LifecycleStage, envelope.LifecycleStage?.ToString()); |
| | 4 | 173 | | AddTag(tags, OpenTelemetryGovernanceAttributes.LifecycleStageSequence, envelope.LifecycleStageSequence); |
| | 4 | 174 | | AddTag(tags, OpenTelemetryGovernanceAttributes.GatewayExecutionId, envelope.GatewayExecutionId); |
| | 4 | 175 | | AddTag(tags, OpenTelemetryGovernanceAttributes.OutboxSequence, envelope.OutboxSequence); |
| | 4 | 176 | | AddTag(tags, OpenTelemetryGovernanceAttributes.EmitterProvider, providerName); |
| | 4 | 177 | | AddTag(tags, OpenTelemetryGovernanceAttributes.EmitterStatus, envelope.EmitterStatus); |
| | 4 | 178 | | AddTag(tags, OpenTelemetryGovernanceAttributes.EmitterResult, result); |
| | 4 | 179 | | AddTag(tags, OpenTelemetryGovernanceAttributes.EmissionLatencyMs, latencyMs); |
| | | 180 | | |
| | 4 | 181 | | if (envelope.Payload is not null) |
| | | 182 | | { |
| | 4 | 183 | | AddTag(tags, OpenTelemetryGovernanceAttributes.PayloadType, envelope.Payload.PayloadType); |
| | 4 | 184 | | AddTag(tags, OpenTelemetryGovernanceAttributes.PayloadSchemaVersion, envelope.Payload.SchemaVersion); |
| | 4 | 185 | | AddTag(tags, OpenTelemetryGovernanceAttributes.PayloadContentType, envelope.Payload.ContentType); |
| | 4 | 186 | | AddTag(tags, OpenTelemetryGovernanceAttributes.PayloadContentHash, envelope.Payload.ContentHash); |
| | 4 | 187 | | AddTag(tags, OpenTelemetryGovernanceAttributes.PayloadSizeBytes, envelope.Payload.SizeBytes); |
| | | 188 | | } |
| | | 189 | | |
| | 4 | 190 | | return tags; |
| | | 191 | | } |
| | | 192 | | |
| | | 193 | | private static void EmitDeliveredMetrics( |
| | | 194 | | GovernanceEmissionEnvelope envelope, |
| | | 195 | | string providerName, |
| | | 196 | | double latencyMs) |
| | | 197 | | { |
| | 4 | 198 | | TagList tags = BuildMetricTags(envelope, providerName, GovernanceEmissionStatus.Delivered.ToString()); |
| | 4 | 199 | | EmissionsCounter.Add(1, tags); |
| | 4 | 200 | | EmissionLatencyHistogram.Record(latencyMs, tags); |
| | 4 | 201 | | } |
| | | 202 | | |
| | | 203 | | private static void EmitFailureMetrics( |
| | | 204 | | GovernanceEmissionEnvelope envelope, |
| | | 205 | | GovernanceEmissionError error, |
| | | 206 | | double latencyMs) |
| | | 207 | | { |
| | 2 | 208 | | TagList tags = BuildMetricTags(envelope, error.ProviderName ?? OpenTelemetryGovernanceInstrumentation.ProviderNa |
| | 2 | 209 | | tags.Add(OpenTelemetryGovernanceAttributes.MetricFailureCode, error.Code); |
| | 2 | 210 | | tags.Add(OpenTelemetryGovernanceAttributes.MetricRetryable, error.IsRetryable); |
| | | 211 | | |
| | 2 | 212 | | EmissionFailuresCounter.Add(1, tags); |
| | 2 | 213 | | EmissionLatencyHistogram.Record(latencyMs, tags); |
| | 2 | 214 | | } |
| | | 215 | | |
| | | 216 | | private static TagList BuildMetricTags( |
| | | 217 | | GovernanceEmissionEnvelope envelope, |
| | | 218 | | string providerName, |
| | | 219 | | string result) |
| | | 220 | | { |
| | 6 | 221 | | TagList tags = new() |
| | 6 | 222 | | { |
| | 6 | 223 | | { OpenTelemetryGovernanceAttributes.MetricEventType, envelope.EventType.ToString() }, |
| | 6 | 224 | | { OpenTelemetryGovernanceAttributes.MetricResult, result }, |
| | 6 | 225 | | { OpenTelemetryGovernanceAttributes.MetricProvider, providerName } |
| | 6 | 226 | | }; |
| | | 227 | | |
| | 6 | 228 | | return tags; |
| | | 229 | | } |
| | | 230 | | |
| | | 231 | | private static void AddTag(ActivityTagsCollection tags, string key, string? value) |
| | | 232 | | { |
| | 88 | 233 | | if (!string.IsNullOrWhiteSpace(value)) |
| | | 234 | | { |
| | 88 | 235 | | tags.Add(key, value); |
| | | 236 | | } |
| | 88 | 237 | | } |
| | | 238 | | |
| | | 239 | | private static void AddTag(ActivityTagsCollection tags, string key, long? value) |
| | | 240 | | { |
| | 8 | 241 | | if (value.HasValue) |
| | | 242 | | { |
| | 8 | 243 | | tags.Add(key, value.Value); |
| | | 244 | | } |
| | 8 | 245 | | } |
| | | 246 | | |
| | | 247 | | private static void AddTag(ActivityTagsCollection tags, string key, int? value) |
| | | 248 | | { |
| | 4 | 249 | | if (value.HasValue) |
| | | 250 | | { |
| | 4 | 251 | | tags.Add(key, value.Value); |
| | | 252 | | } |
| | 4 | 253 | | } |
| | | 254 | | |
| | | 255 | | private static void AddTag(ActivityTagsCollection tags, string key, double value) |
| | | 256 | | { |
| | 4 | 257 | | tags.Add(key, value); |
| | 4 | 258 | | } |
| | | 259 | | |
| | | 260 | | private static string GetEventName(GovernanceEmissionEnvelope envelope) |
| | | 261 | | { |
| | 4 | 262 | | return envelope.EventType switch |
| | 4 | 263 | | { |
| | 4 | 264 | | GovernanceEmissionEventType.Decision => OpenTelemetryGovernanceInstrumentation.DecisionEvaluatedEventName, |
| | 0 | 265 | | GovernanceEmissionEventType.Acknowledgment => OpenTelemetryGovernanceInstrumentation.AcknowledgmentRecordedE |
| | 0 | 266 | | GovernanceEmissionEventType.CapabilityToken => OpenTelemetryGovernanceInstrumentation.CapabilityTokenIssuedE |
| | 0 | 267 | | GovernanceEmissionEventType.Gateway => OpenTelemetryGovernanceInstrumentation.GatewayCompletedEventName, |
| | 0 | 268 | | GovernanceEmissionEventType.AuditResidue => OpenTelemetryGovernanceInstrumentation.AuditResidueCreatedEventN |
| | 0 | 269 | | GovernanceEmissionEventType.AuditLifecycle => OpenTelemetryGovernanceInstrumentation.LifecycleRecordedEventN |
| | 0 | 270 | | GovernanceEmissionEventType.Outbox => OpenTelemetryGovernanceInstrumentation.OutboxUpdatedEventName, |
| | 0 | 271 | | GovernanceEmissionEventType.ProviderEmission => IsFailureStatus(envelope.EmitterStatus) |
| | 0 | 272 | | ? OpenTelemetryGovernanceInstrumentation.EmissionFailedEventName |
| | 0 | 273 | | : OpenTelemetryGovernanceInstrumentation.EmissionDeliveredEventName, |
| | 0 | 274 | | _ => OpenTelemetryGovernanceInstrumentation.GenericGovernanceEventName |
| | 4 | 275 | | }; |
| | | 276 | | } |
| | | 277 | | |
| | | 278 | | private static bool IsFailureStatus(string? status) |
| | | 279 | | { |
| | 0 | 280 | | return status is not null |
| | 0 | 281 | | && (status.Contains("fail", StringComparison.OrdinalIgnoreCase) |
| | 0 | 282 | | || status.Contains("dead", StringComparison.OrdinalIgnoreCase) |
| | 0 | 283 | | || status.Contains("retry", StringComparison.OrdinalIgnoreCase)); |
| | | 284 | | } |
| | | 285 | | } |