Skip to content

Commit

Permalink
Merge branch 'main' into instrumentation-sqlclient-drop-db.statement_…
Browse files Browse the repository at this point in the history
…type
  • Loading branch information
Kielek authored Feb 1, 2024
2 parents 6344e5d + 9fd01f7 commit fbf5ccf
Show file tree
Hide file tree
Showing 5 changed files with 149 additions and 8 deletions.
66 changes: 58 additions & 8 deletions docs/metrics/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,8 @@
* [Pre-Aggregation](#pre-aggregation)
* [Cardinality Limits](#cardinality-limits)
* [Memory Preallocation](#memory-preallocation)
* [Metrics Correlation](#metrics-correlation)
* [Metrics Enrichment](#metrics-enrichment)

</details>
<!-- markdownlint-enable MD033 -->
Expand Down Expand Up @@ -138,9 +140,9 @@ Here is the rule of thumb:

> [!NOTE]
> When reporting measurements with more than 8 tags, the API allocates memory on
the hot-path. You SHOULD try to keep the number of tags less than or equal to 8.
If you are exceeding this, check if you can model some of the tags as Resource,
as [shown here](#modeling-static-tags-as-resource).
the hot-path. You SHOULD try to keep the number of tags less than or equal to 8.
If you are exceeding this, check if you can model some of the tags as Resource,
as [shown here](#metrics-enrichment).

## MeterProvider Management

Expand Down Expand Up @@ -397,12 +399,60 @@ SDK to reclaim unused metric points.

### Memory Preallocation

### Modeling static tags as Resource
OpenTelemetry .NET SDK aims to avoid memory allocation on the hot code path.
When this is combined with [proper use of Metrics API](#metrics-api), heap
allocation can be avoided on the hot code path. Refer to the [metrics benchmark
results](../../test/Benchmarks/Metrics/MetricsBenchmarks.cs) to learn more.

:heavy_check_mark: You should measure memory allocation on hot code path, and
ideally avoid any heap allocation while using the metrics API and SDK,
especially when you use metrics to measure the performance of your application
(for example, you do not want to spend 2 seconds doing [garbage
collection](https://learn.microsoft.com/dotnet/standard/garbage-collection/)
while measuring an operation which normally takes 10 milliseconds).

## Metrics Correlation

In OpenTelemetry, metrics can be correlated to [traces](../trace/README.md) via
[exemplars](https://github.com/open-telemetry/opentelemetry-specification/blob/main/specification/metrics/sdk.md#exemplar).
Check the [Exemplars](./exemplars/README.md) tutorial to learn more.

## Metrics Enrichment

When the metrics are being collected, they normally get stored in a [time series
database](https://en.wikipedia.org/wiki/Time_series_database). From storage and
consumption perspective, metrics can be multi-dimensional. Taking the [fruit
example](#example), there are two dimensions - "name" and "color". For basic
scenarios, all the dimensions can be reported during the [Metrics
API](#metrics-api) invocation, however, for less trivial scenarios, the
dimensions can come from different sources:

* [Measurements](https://github.com/open-telemetry/opentelemetry-specification/blob/main/specification/metrics/api.md#measurement)
reported via the [Metrics API](#metrics-api).
* [Resources](https://github.com/open-telemetry/opentelemetry-specification/blob/main/specification/resource/sdk.md)
configured at the `MeterProvider` level. Refer to this
[doc](./customizing-the-sdk/README.md#resource) for details and examples.
* Additional attributes provided by the exporter or collector. For example,
[jobs and instances](https://prometheus.io/docs/concepts/jobs_instances/) in
Prometheus.

Here is the rule of thumb when modeling the dimensions:

* If the dimension value is static throughout the process lifetime (e.g. the
name of the machine, data center), model it as Resource, or even better, let
the collector add these dimensions if feasible (e.g. a collector running in
the same data center should know the name of the data center, rather than
relying on / trusting each service instance to report the data center name).
* If the dimension value is dynamic, report it via the [Metrics
API](#metrics-api).

Tags such as `MachineName`, `Environment` etc. which are static throughout the
process lifetime should be be modeled as `Resource`, instead of adding them to
each metric measurement. Refer to this
[doc](./customizing-the-sdk/README.md#resource) for details and examples.
> [!NOTE]
> There were discussions around adding a new concept called
`MeasurementProcessor`, which allows dimensions to be added to / removed from
measurements dynamically. This idea did not get traction due to the complexity
and performance implications, refer to this [pull
request](https://github.com/open-telemetry/opentelemetry-specification/pull/1938)
for more context.

## Common issues that lead to missing metrics

Expand Down
20 changes: 20 additions & 0 deletions src/OpenTelemetry.Exporter.InMemory/AssemblyInfo.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
// Copyright The OpenTelemetry Authors
// SPDX-License-Identifier: Apache-2.0

#if !EXPOSE_EXPERIMENTAL_FEATURES
using System.Runtime.CompilerServices;

[assembly: InternalsVisibleTo("OpenTelemetry.Exporter.OpenTelemetryProtocol.Tests" + AssemblyInfo.PublicKey)]
#endif

#if SIGNED
file static class AssemblyInfo
{
public const string PublicKey = ", PublicKey=002400000480000094000000060200000024000052534131000400000100010051C1562A090FB0C9F391012A32198B5E5D9A60E9B80FA2D7B434C9E5CCB7259BD606E66F9660676AFC6692B8CDC6793D190904551D2103B7B22FA636DCBB8208839785BA402EA08FC00C8F1500CCEF28BBF599AA64FFB1E1D5DC1BF3420A3777BADFE697856E9D52070A50C3EA5821C80BEF17CA3ACFFA28F89DD413F096F898";
}
#else
file static class AssemblyInfo
{
public const string PublicKey = "";
}
#endif
6 changes: 6 additions & 0 deletions src/OpenTelemetry.Exporter.OpenTelemetryProtocol/CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,12 @@
will be automatically included in exports.
([#5258](https://github.com/open-telemetry/opentelemetry-dotnet/pull/5258))

* Updated `OtlpLogExporter` to set `body` on the data model from
`LogRecord.Body` if `{OriginalFormat}` attribute is NOT found and
`FormattedMessage` is `null`. This is typically the case when using the
experimental Logs Bridge API.
([#5268](https://github.com/open-telemetry/opentelemetry-dotnet/pull/5268))

## 1.7.0

Released 2023-Dec-08
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -170,6 +170,14 @@ internal OtlpLogs.LogRecord ToOtlpLog(LogRecord logRecord)
AddAttribute(otlpLogRecord, result, attributeCountLimit);
}
}

// Supports setting Body directly on LogRecord for the Logs Bridge API.
if (otlpLogRecord.Body == null && logRecord.Body != null)
{
// If {OriginalFormat} is not present in the attributes,
// use logRecord.Body if it is set.
otlpLogRecord.Body = new OtlpCommon.AnyValue { StringValue = logRecord.Body };
}
}

if (logRecord.TraceId != default && logRecord.SpanId != default)
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -589,6 +589,63 @@ public void CheckToOtlpLogRecordBodyIsPopulated(bool includeFormattedMessage)
Assert.Equal("state", otlpLogRecord.Body.StringValue);
}

[Theory]
[InlineData(true)]
[InlineData(false)]
public void LogRecordBodyIsExportedWhenUsingBridgeApi(bool isBodySet)
{
LogRecordAttributeList attributes = default;
attributes.Add("name", "tomato");
attributes.Add("price", 2.99);
attributes.Add("{OriginalFormat}", "Hello from {name} {price}.");

var logRecords = new List<LogRecord>();

using (var loggerProvider = Sdk.CreateLoggerProviderBuilder()
.AddInMemoryExporter(logRecords)
.Build())
{
var logger = loggerProvider.GetLogger();

logger.EmitLog(new LogRecordData()
{
Body = isBodySet ? "Hello world" : null,
});

logger.EmitLog(new LogRecordData(), attributes);
}

Assert.Equal(2, logRecords.Count);

var otlpLogRecordTransformer = new OtlpLogRecordTransformer(DefaultSdkLimitOptions, new());

var otlpLogRecord = otlpLogRecordTransformer.ToOtlpLog(logRecords[0]);

if (isBodySet)
{
Assert.Equal("Hello world", otlpLogRecord.Body?.StringValue);
}
else
{
Assert.Null(otlpLogRecord.Body);
}

otlpLogRecord = otlpLogRecordTransformer.ToOtlpLog(logRecords[1]);

Assert.Equal(2, otlpLogRecord.Attributes.Count);

var index = 0;
var attribute = otlpLogRecord.Attributes[index];
Assert.Equal("name", attribute.Key);
Assert.Equal("tomato", attribute.Value.StringValue);

attribute = otlpLogRecord.Attributes[++index];
Assert.Equal("price", attribute.Key);
Assert.Equal(2.99, attribute.Value.DoubleValue);

Assert.Equal("Hello from {name} {price}.", otlpLogRecord.Body.StringValue);
}

[Fact]
public void CheckToOtlpLogRecordExceptionAttributes()
{
Expand Down

0 comments on commit fbf5ccf

Please sign in to comment.