The @microsoft/agents-telemetry package provides OpenTelemetry instrumentation primitives used across the Microsoft Agents SDK. It enables distributed tracing, structured log emission, and stable metric naming so you can monitor agent interactions, connector calls, storage operations, authentication flows, and turn processing.
The package exports:
trace — span creation with record tracking, custom actions, and lifecycle hooksdebug — loggers that bridge the debug module with OpenTelemetry log recordsmetric — histogram and counter creation via the OpenTelemetry Meter APISpanNames / MetricNames — stable name constants for observability across the SDKThis package supports both ESM and CommonJS consumers and requires Node.js 20 or later.
This section covers how to enable and configure telemetry in your agent application. The SDK automatically instruments its components — you only need to wire up OpenTelemetry exporters and, optionally, tune which span categories are active.
Install the OpenTelemetry API packages in your agent for full integration:
npm install @opentelemetry/api @opentelemetry/api-logs
Both are optional. If missing, the SDK falls back to noop implementations and logs a warning. Installing the official packages is recommended for full tracing, metrics, and log export.
For traces to be exported, configure the OpenTelemetry SDK with an exporter. Console exporter for development:
import { NodeSDK } from '@opentelemetry/sdk-node'
import { ConsoleSpanExporter } from '@opentelemetry/sdk-trace-node'
const sdk = new NodeSDK({
traceExporter: new ConsoleSpanExporter(),
serviceName: 'my-agent-service',
})
sdk.start()
Once configured, all SDK components that use trace(...) will emit spans through the global tracer provider.
To export metrics referenced by MetricNames, configure a metric reader/exporter in your OpenTelemetry SDK as well.
For production, use an OTLP exporter to send traces to your observability backend (Azure Monitor, Jaeger, Grafana, etc.):
import { NodeSDK } from '@opentelemetry/sdk-node'
import { OTLPTraceExporter } from '@opentelemetry/exporter-trace-otlp-http'
const sdk = new NodeSDK({
traceExporter: new OTLPTraceExporter({
url: 'http://localhost:4318/v1/traces', // Your OTLP endpoint
}),
serviceName: 'my-agent-service',
})
sdk.start()
To export logs produced by debug loggers to an OTLP backend, add a logs processor during startup:
import { NodeSDK } from '@opentelemetry/sdk-node'
import { resourceFromAttributes } from '@opentelemetry/resources'
import { ATTR_SERVICE_NAME, ATTR_SERVICE_VERSION } from '@opentelemetry/semantic-conventions'
import { OTLPLogExporter } from '@opentelemetry/exporter-logs-otlp-grpc'
import { BatchLogRecordProcessor } from '@opentelemetry/sdk-logs'
const otlpEndpoint = process.env.OTEL_EXPORTER_OTLP_ENDPOINT || 'http://localhost:4317'
const sdk = new NodeSDK({
resource: resourceFromAttributes({
[ATTR_SERVICE_NAME]: 'my-agent-service',
[ATTR_SERVICE_VERSION]: '1.0.0',
}),
logRecordProcessors: [
new BatchLogRecordProcessor(new OTLPLogExporter({ url: otlpEndpoint })),
],
})
sdk.start()
All built-in span categories are enabled by default. To disable one or more categories without changing code, set AGENTS_TELEMETRY_DISABLED_SPAN_CATEGORIES in your environment:
AGENTS_TELEMETRY_DISABLED_SPAN_CATEGORIES=STORAGE,AUTHORIZATION
# or
AGENTS_TELEMETRY_DISABLED_SPAN_CATEGORIES=STORAGE AUTHORIZATION
Valid category names are:
STORAGEAUTHENTICATIONAUTHORIZATIONDIALOGSWhen a span category is disabled, instrumented code still runs normally with a noop context — no telemetry is emitted for those spans.
The SDK uses predefined constants for all span and metric names:
import { SpanNames, MetricNames } from '@microsoft/agents-telemetry'
// Example span names:
SpanNames.ADAPTER_PROCESS // 'agents.adapter.process'
SpanNames.AGENTS_APP_RUN // 'agents.app.run'
SpanNames.STORAGE_READ // 'agents.storage.read'
SpanNames.CONNECTOR_REPLY_TO_ACTIVITY // 'agents.connector.reply_to_activity'
// Example metric names:
MetricNames.ACTIVITIES_RECEIVED // 'agents.activities.received'
MetricNames.CONNECTOR_REQUEST_DURATION // 'agents.connector.request.duration'
MetricNames.TURN_DURATION // 'agents.turn.duration'
MetricNames.AUTH_TOKEN_REQUEST_COUNT // 'agents.auth.token.request.count'
// ... and more
This section covers internal APIs used to instrument SDK components. If you are contributing to the Agents SDK packages, use trace, debug, and metric to add observability to your code.
Although this package is published, it primarily serves as a shared telemetry layer for other Agents SDK packages. Some exported TypeScript types intentionally depend on OpenTelemetry packages even though runtime behavior still supports noop fallback when those optional peer dependencies are not installed.
Use trace to create validated spans. Every definition requires a name from SpanNames, a record object for tracking state, and an end hook called when the span finishes.
The span is started and ended automatically around the callback:
import { SpanNames, trace } from '@microsoft/agents-telemetry'
const result = await trace(
{
name: SpanNames.AGENTS_APP_ROUTE_HANDLER,
record: { conversationId: '', messageCount: 0 },
actions: ({ span }) => ({
setAttribute: (key: string, value: string) => span.setAttribute(key, value),
}),
end: ({ span, record, duration, error }) => {
span.setAttribute('conversation.id', record.conversationId)
span.setAttribute('message.count', record.messageCount)
},
},
async ({ record, actions }) => {
record({ conversationId })
actions.setAttribute('code.function', 'handleMessage')
const result = await service.handleMessage(conversationId, text)
record({ messageCount: result.count })
return result
}
)
For long-lived operations where you need manual control over the span lifecycle:
import { SpanNames, trace } from '@microsoft/agents-telemetry'
const { record, actions, end, fail } = trace({
name: SpanNames.ADAPTER_PROCESS,
record: { status: 'pending' },
end: ({ span, record, duration }) => {
span.setAttribute('status', record.status)
},
})
try {
record({ status: 'processing' })
await doWork()
record({ status: 'done' })
end()
} catch (error) {
fail(error)
end()
throw error
}
Use trace.define to declare a trace definition with full type inference, without starting a span:
const storageReadTrace = trace.define({
name: SpanNames.STORAGE_READ,
record: { keys: 0 },
end: ({ span, record }) => {
span.setAttribute('storage.keys', record.keys)
},
})
// Later:
trace(storageReadTrace, ({ record }) => {
record({ keys: 5 })
return storage.read(keys)
})
OK.ERROR, and the error is rethrown.end hook receives { span, record, duration, error? }.Create loggers that emit to both the debug module and OpenTelemetry logs:
import { debug } from '@microsoft/agents-telemetry'
const logger = debug('agents:mycomponent')
logger.info('Processing request', { requestId })
logger.error('Failed to connect', error)
When @opentelemetry/api-logs is available, log records are emitted with severity and namespace attributes. Otherwise, only the debug output is produced.
Create OpenTelemetry instruments for recording measurements:
import { metric, MetricNames } from '@microsoft/agents-telemetry'
const duration = metric.histogram(MetricNames.TURN_DURATION)
const counter = metric.counter(MetricNames.ACTIVITIES_RECEIVED)
duration.record(elapsed, { 'activity.type': 'message' })
counter.add(1, { 'activity.type': 'message' })
When @opentelemetry/api is not available, the instruments are safe noops.
ERROR statusend hookactions factorytrace(definition) — Managed modeStarts a span and returns a managed context with record, actions, end, and fail methods.
| Parameter | Type | Description |
|---|---|---|
definition |
TraceDefinition |
Trace definition object (see below). |
Returns: { record, actions, end, fail }
trace(definition, callback) — Callback modeRuns callback inside an active span. The span is ended automatically when the callback returns or throws.
| Parameter | Type | Description |
|---|---|---|
definition |
TraceDefinition |
Trace definition object (see below). |
callback |
(context) => TReturn |
Callback receiving { record, actions }. Return value (including a Promise) is forwarded. |
Returns: Whatever callback returns.
TraceDefinition| Property | Type | Description |
|---|---|---|
name |
SpanName |
Span name. Must be one of SpanNames values. |
record |
object |
Initial record state. Updated via record() during the trace. |
actions |
(ctx: { span }) => object |
(Optional) Factory that receives the span and returns action helpers. |
end |
(ctx: { span, record, duration, error? }) => void |
Hook called when the span ends. Use it to set final attributes or record metrics. |
trace.define(definition)Returns the definition object as-is. Useful for declaring reusable trace definitions with full type inference.
debug(namespace)Creates a logger for the given namespace. When @opentelemetry/api-logs is available, the logger bridges the debug module with OpenTelemetry log emission. Otherwise, it creates a debug-only logger.
| Parameter | Type | Description |
|---|---|---|
namespace |
string |
Logger namespace (e.g. 'agents:hosting'). |
Returns: { info, warn, error, debug } — each method accepts a message string and optional arguments.
metricProvides factory methods for creating OpenTelemetry instruments. When @opentelemetry/api is not available, returns noop instruments.
| Method | Description |
|---|---|
metric.histogram(name) |
Creates a histogram instrument. |
metric.counter(name) |
Creates a counter instrument. |
SpanNamesProvides stable span name constants used by Agents SDK components, including adapter, application, connector, storage, authentication, authorization, user token client, turn-processing, and Copilot Studio client operations.
MetricNamesProvides stable metric name constants for counters and histograms used by Agents SDK components, including activity counts, request counts, turn counts, and duration metrics.
MIT