Telemetry
Brighter provides comprehensive OpenTelemetry integration for distributed tracing across message boundaries, enabling end-to-end observability in distributed systems.
OpenTelemetry Semantic Conventions
V10 introduces support for OpenTelemetry Semantic Conventions, replacing the custom conventions used in V9.
OTel Semantic Conventions for Messaging: Standard span names and attributes for messaging operations
W3C TraceContext: Standard context propagation across service boundaries
CloudEvents Integration: Trace propagation via CloudEvents
traceparentandtracestateheadersConfigurable Instrumentation: Fine-grained control over what attributes are recorded
Comprehensive Coverage: Tracing for Command Processor, Dispatcher, Outbox, Inbox, and Transform pipelines
Configuring OpenTelemetry
The OpenTelemetry SDK can be configured to listen to Activities emitted by Brighter. For more information, see OpenTelemetry Tracing in .NET.
Activity Source
Brighter emits traces using the following Activity Source:
Source Name:
paramore.brighterVersion: Includes the Brighter version number
Basic Configuration
The following code configures OpenTelemetry to:
Enable tracing
Set the service name
Listen to Brighter and Microsoft sources
Export traces to Jaeger
Configuration with Different Backends
Jaeger
Zipkin
OTLP (OpenTelemetry Protocol)
Azure Monitor / Application Insights
Configurable Instrumentation
V10 provides fine-grained control over which attributes are recorded to optimize performance and reduce costs.
Instrumentation Options
Configure instrumentation using BrighterInstrumentation:
Best Practice: Only enable RecordRequestBody and RecordMessageBody in development or debugging scenarios, as they can significantly increase trace size and cost.
Command Processor Spans
When Brighter operates as a Command Processor, it creates spans for each operation:
Span Names and Operations
send
<request type> send
Internal
Command routed to single handler
publish
<request type> publish
Internal
Event routed to multiple handlers
deposit
<request type> deposit
Internal
Request transformed and stored in Outbox
clear
clear
Internal
Messages dispatched from Outbox to broker
create
<channel> create
Producer
Single message sent to broker
publish (messaging)
<channel> publish
Producer
Batch of messages sent to broker
Example: Send Operation
Command Processor Attributes
paramore.brighter.requestid
string
Request ID
"1234-5678-9012-3456"
paramore.brighter.requestids
string
Batch: comma-separated IDs
"1234..., 2345..."
paramore.brighter.requesttype
string
Full type name
"MyNamespace.MyCommand"
paramore.brighter.request_body
string
Request as JSON
{"orderId": 123}
paramore.brighter.operation
string
Operation performed
"send"
paramore.brighter.spancontext.*
varies
Custom context attributes
spancontext.userid: "1234"
Adding Custom Span Attributes
You can add custom attributes via the Request Context:
Any context bag entries starting with paramore.brighter.spancontext. will be added as span attributes.
Handler Pipeline Events
Brighter records an event for each handler entered in the pipeline:
paramore.brighter.handlername
string
Full handler type name
"MyNamespace.MyHandler"
paramore.brighter.handlertype
string
Sync or async
"async"
paramore.brighter.is_sink
bool
Final handler in chain
true
Dispatcher (Consumer) Spans
When Brighter operates as a Dispatcher (message consumer), it creates spans for each message received:
Span Names
Pull-based (Kafka)
<channel> receive
Consumer
Message pulled from broker
Push-based (RabbitMQ)
<channel> process
Consumer
Message pushed by broker
Example Flow
Message Attributes
messaging.system
string
Broker type
"rabbitmq", "kafka"
messaging.destination
string
Channel name
"task.commands"
messaging.operation
string
Operation type
"receive", "process"
messaging.message_id
string
Message ID
"msg-1234"
messaging.destination.partition.id
string
Partition ID (Kafka)
"0"
messaging.message.body.size
int
Payload size in bytes
1024
server.address
string
Broker address
"localhost:5672"
Outbox Tracing
Outbox operations create child spans for database operations:
Deposit Operation
Clear Operation
Database Span Attributes
Outbox and Inbox database operations follow OTel Database Semantic Conventions:
db.system
Database type
"mysql", "postgresql"
db.name
Database name
"myapp"
db.operation
Operation type
"outbox_add", "outbox_get"
Inbox Tracing
Inbox operations create child spans for deduplication checks:
Inbox Operations
Check
inbox_check
Check if message already processed
Add
inbox_add
Record message as processed
Transform Pipeline Tracing
Transform operations (Claim Check, Compression, Encryption) create child spans for external calls:
Claim Check (S3 Example)
Retrieve Claim
External call spans follow their respective OTel conventions:
W3C TraceContext Propagation
Brighter automatically propagates trace context across service boundaries using W3C TraceContext headers.
How It Works
Producer: Brighter injects
traceparentandtracestateinto message headersConsumer: Brighter extracts
traceparentandtracestateto continue the trace
Message Headers
Integration with ASP.NET
Brighter participates in existing traces. When called from an ASP.NET controller, the Command Processor span becomes a child of the ASP.NET request span:
CloudEvents Integration
When using CloudEvents, Brighter propagates trace context via the CloudEvents Distributed Tracing Extension.
CloudEvents Attributes
CloudEvents adds alternative attribute names following CloudEvents Semantic Conventions:
messaging.message_id
cloudevents.event_id
Message ID
messaging.destination
cloudevents.event_source
Event source
N/A
cloudevents.event_type
Event type
Enabling CloudEvents Conventions
You can enable both conventions simultaneously, and both sets of attributes will be recorded.
Complete Configuration Example
Producer Service
Consumer Service (Dispatcher)
Distributed Tracing Example
A complete distributed trace across services:
Best Practices
Start with Minimal Instrumentation: Enable
RecordRequestInformationandRecordMessageInformation, disable expensive options likeRecordRequestBodyUse Sampling: Configure sampling in production to reduce costs:
Add Custom Attributes Judiciously: Only add context attributes that are essential for debugging and analysis
Monitor Trace Costs: Large payloads and high cardinality attributes can significantly increase observability costs
Use Structured Logging: Combine tracing with structured logging for comprehensive observability
Enable CloudEvents for Cross-Organization Tracing: If exchanging messages with external systems, use CloudEvents for standard trace propagation
Configure Appropriate Exporters: Use OTLP for flexibility, or native exporters for specific backends
Test Trace Propagation: Verify that traces flow correctly across service boundaries in development
Migration from V9
Changed Span Names
Custom handler names
<request type> <operation>
Outbox.Add
Follows database conventions
Transport-specific names
<channel> create/publish/receive/process
Changed Attributes
V9 used custom attribute names. V10 uses OTel standard conventions:
Custom attributes
paramore.brighter.* and OTel standard attributes
No standard messaging attributes
messaging.* attributes following OTel conventions
Action Required
Update Dashboards: Update queries and visualizations to use V10 span names and attributes
Update Alerts: Update alert rules based on new span structure
Review Instrumentation Options: Configure which attributes to record based on your needs
Test Trace Propagation: Verify distributed traces work correctly with V10
Troubleshooting
Traces Not Appearing
Problem: No traces appear in your observability backend.
Solutions:
Verify Activity Source is registered:
.AddSource("paramore.brighter")Check exporter configuration and endpoint
Ensure services can reach the exporter endpoint
Check firewall rules
Incomplete Traces
Problem: Traces are missing child spans or appear disconnected.
Solutions:
Verify
traceparentheader is being propagatedCheck that all services have OpenTelemetry configured
Ensure consistent trace propagation format (W3C TraceContext)
Review CloudEvents configuration if using CloudEvents
High Trace Costs
Problem: Observability costs are too high.
Solutions:
Disable
RecordRequestBodyandRecordMessageBodyReduce sampling rate:
.SetSampler(new TraceIdRatioBasedSampler(0.1))Disable unnecessary attribute collection
Use tail-based sampling to only keep interesting traces
Missing Attributes
Problem: Expected attributes are not appearing on spans.
Solutions:
Check
Activity.IsAllDataRequestedis true (controlled by sampling)Verify instrumentation options are configured correctly
Ensure custom context attributes start with
paramore.brighter.spancontext.
Additional Resources
Last updated
Was this helpful?
