-
Notifications
You must be signed in to change notification settings - Fork 145
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Add support for RabbitMQ v7 #6479
base: master
Are you sure you want to change the base?
Conversation
Execution-Time Benchmarks Report ⏱️Execution-time results for samples comparing the following branches/commits: Execution-time benchmarks measure the whole time it takes to execute a program. And are intended to measure the one-off costs. Cases where the execution time results for the PR are worse than latest master results are shown in red. The following thresholds were used for comparing the execution times:
Note that these results are based on a single point-in-time result for each branch. For full results, see the dashboard. Graphs show the p99 interval based on the mean and StdDev of the test run, as well as the mean value of the run (shown as a diamond below the graph). gantt
title Execution time (ms) FakeDbCommand (.NET Framework 4.6.2)
dateFormat X
axisFormat %s
todayMarker off
section Baseline
This PR (6479) - mean (69ms) : 66, 73
. : milestone, 69,
master - mean (69ms) : 65, 73
. : milestone, 69,
section CallTarget+Inlining+NGEN
This PR (6479) - mean (985ms) : 960, 1009
. : milestone, 985,
master - mean (987ms) : 962, 1013
. : milestone, 987,
gantt
title Execution time (ms) FakeDbCommand (.NET Core 3.1)
dateFormat X
axisFormat %s
todayMarker off
section Baseline
This PR (6479) - mean (108ms) : 104, 111
. : milestone, 108,
master - mean (107ms) : 105, 109
. : milestone, 107,
section CallTarget+Inlining+NGEN
This PR (6479) - mean (684ms) : 666, 701
. : milestone, 684,
master - mean (681ms) : 668, 695
. : milestone, 681,
gantt
title Execution time (ms) FakeDbCommand (.NET 6)
dateFormat X
axisFormat %s
todayMarker off
section Baseline
This PR (6479) - mean (91ms) : 89, 94
. : milestone, 91,
master - mean (91ms) : 89, 93
. : milestone, 91,
section CallTarget+Inlining+NGEN
This PR (6479) - mean (637ms) : 620, 654
. : milestone, 637,
master - mean (635ms) : 620, 650
. : milestone, 635,
gantt
title Execution time (ms) HttpMessageHandler (.NET Framework 4.6.2)
dateFormat X
axisFormat %s
todayMarker off
section Baseline
This PR (6479) - mean (191ms) : 187, 195
. : milestone, 191,
master - mean (190ms) : 186, 195
. : milestone, 190,
section CallTarget+Inlining+NGEN
This PR (6479) - mean (1,096ms) : 1060, 1132
. : milestone, 1096,
master - mean (1,087ms) : 1060, 1114
. : milestone, 1087,
gantt
title Execution time (ms) HttpMessageHandler (.NET Core 3.1)
dateFormat X
axisFormat %s
todayMarker off
section Baseline
This PR (6479) - mean (278ms) : 273, 283
. : milestone, 278,
master - mean (276ms) : 270, 281
. : milestone, 276,
section CallTarget+Inlining+NGEN
This PR (6479) - mean (871ms) : 837, 905
. : milestone, 871,
master - mean (863ms) : 839, 886
. : milestone, 863,
gantt
title Execution time (ms) HttpMessageHandler (.NET 6)
dateFormat X
axisFormat %s
todayMarker off
section Baseline
This PR (6479) - mean (265ms) : 260, 270
. : milestone, 265,
master - mean (264ms) : 260, 269
. : milestone, 264,
section CallTarget+Inlining+NGEN
This PR (6479) - mean (846ms) : 819, 874
. : milestone, 846,
master - mean (847ms) : 808, 886
. : milestone, 847,
|
Datadog ReportBranch report: ✅ 0 Failed, 562809 Passed, 4632 Skipped, 46h 23m 23.01s Total Time |
Benchmarks Report for tracer 🐌Benchmarks for #6479 compared to master:
The following thresholds were used for comparing the benchmark speeds:
Allocation changes below 0.5% are ignored. Benchmark detailsBenchmarks.Trace.ActivityBenchmark - Same speed ✔️ Same allocations ✔️Raw results
Benchmarks.Trace.AgentWriterBenchmark - Same speed ✔️ Same allocations ✔️Raw results
Benchmarks.Trace.AspNetCoreBenchmark - Same speed ✔️ Same allocations ✔️Raw results
Benchmarks.Trace.CIVisibilityProtocolWriterBenchmark - Same speed ✔️ Same allocations ✔️Raw results
Benchmarks.Trace.DbCommandBenchmark - Same speed ✔️ Same allocations ✔️Raw results
Benchmarks.Trace.ElasticsearchBenchmark - Same speed ✔️ Same allocations ✔️Raw results
Benchmarks.Trace.GraphQLBenchmark - Same speed ✔️ Same allocations ✔️Raw results
Benchmarks.Trace.HttpClientBenchmark - Same speed ✔️ Same allocations ✔️Raw results
Benchmarks.Trace.ILoggerBenchmark - Same speed ✔️ Same allocations ✔️Raw results
Benchmarks.Trace.Log4netBenchmark - Same speed ✔️ Same allocations ✔️Raw results
Benchmarks.Trace.NLogBenchmark - Same speed ✔️ Same allocations ✔️Raw results
Benchmarks.Trace.RedisBenchmark - Same speed ✔️ Same allocations ✔️Raw results
Benchmarks.Trace.SerilogBenchmark - Same speed ✔️ Same allocations ✔️Raw results
Benchmarks.Trace.SpanBenchmark - Slower
|
Benchmark | diff/base | Base Median (ns) | Diff Median (ns) | Modality |
---|---|---|---|---|
Benchmarks.Trace.SpanBenchmark.StartFinishSpan‑net6.0 | 1.193 | 397.43 | 474.03 |
Raw results
Branch | Method | Toolchain | Mean | StdError | StdDev | Gen 0 | Gen 1 | Gen 2 | Allocated |
---|---|---|---|---|---|---|---|---|---|
master | StartFinishSpan |
net6.0 | 398ns | 0.155ns | 0.602ns | 0.00819 | 0 | 0 | 576 B |
master | StartFinishSpan |
netcoreapp3.1 | 615ns | 0.451ns | 1.75ns | 0.00767 | 0 | 0 | 576 B |
master | StartFinishSpan |
net472 | 700ns | 0.374ns | 1.4ns | 0.0917 | 0 | 0 | 578 B |
master | StartFinishScope |
net6.0 | 542ns | 0.325ns | 1.22ns | 0.00975 | 0 | 0 | 696 B |
master | StartFinishScope |
netcoreapp3.1 | 718ns | 0.412ns | 1.49ns | 0.00946 | 0 | 0 | 696 B |
master | StartFinishScope |
net472 | 884ns | 0.661ns | 2.56ns | 0.104 | 0 | 0 | 658 B |
#6479 | StartFinishSpan |
net6.0 | 474ns | 0.28ns | 1.08ns | 0.00808 | 0 | 0 | 576 B |
#6479 | StartFinishSpan |
netcoreapp3.1 | 572ns | 0.484ns | 1.87ns | 0.00768 | 0 | 0 | 576 B |
#6479 | StartFinishSpan |
net472 | 722ns | 0.79ns | 2.95ns | 0.0916 | 0 | 0 | 578 B |
#6479 | StartFinishScope |
net6.0 | 516ns | 0.297ns | 1.07ns | 0.00967 | 0 | 0 | 696 B |
#6479 | StartFinishScope |
netcoreapp3.1 | 713ns | 0.502ns | 1.95ns | 0.00931 | 0 | 0 | 696 B |
#6479 | StartFinishScope |
net472 | 929ns | 0.819ns | 3.17ns | 0.104 | 0 | 0 | 658 B |
Benchmarks.Trace.TraceAnnotationsBenchmark - Same speed ✔️ Same allocations ✔️
Raw results
Branch | Method | Toolchain | Mean | StdError | StdDev | Gen 0 | Gen 1 | Gen 2 | Allocated |
---|---|---|---|---|---|---|---|---|---|
master | RunOnMethodBegin |
net6.0 | 667ns | 0.353ns | 1.32ns | 0.00969 | 0 | 0 | 696 B |
master | RunOnMethodBegin |
netcoreapp3.1 | 876ns | 0.706ns | 2.74ns | 0.00922 | 0 | 0 | 696 B |
master | RunOnMethodBegin |
net472 | 1.07μs | 0.659ns | 2.55ns | 0.104 | 0 | 0 | 658 B |
#6479 | RunOnMethodBegin |
net6.0 | 665ns | 0.363ns | 1.41ns | 0.00965 | 0 | 0 | 696 B |
#6479 | RunOnMethodBegin |
netcoreapp3.1 | 884ns | 0.34ns | 1.32ns | 0.00926 | 0 | 0 | 696 B |
#6479 | RunOnMethodBegin |
net472 | 1.04μs | 0.709ns | 2.75ns | 0.105 | 0 | 0 | 658 B |
c47a319
to
c2b1675
Compare
ffc922e
to
457a93f
Compare
c2b1675
to
288fb51
Compare
457a93f
to
6c8dbe5
Compare
288fb51
to
882ea95
Compare
6c8dbe5
to
a73c2c7
Compare
882ea95
to
8443b2e
Compare
…#6480) ## Summary of changes Add support for correct `CallTarget` instrumentation of methods that return `ValueTask` in < .NET Core 3.1 or .NET Framework ## Reason for change We already support instrumenting methods that returns `ValueTask` in .NET Core 3.1. However, in .NET Framework or .NET Standard 2.0, this support is [provided by a package](https://www.nuget.org/packages/System.Threading.Tasks.Extensions), and we currently _don't_ support instrumenting these methods. Or rather, we just ignore the `OnAsyncMethodEnd` in integrations in these cases. ## Implementation details We already support `ValueTask` in more recent frameworks, and the support is very similar to our `Task` support. Unfortunately, in .NET FX we can't reference the `ValueTask` type itself. To work around this, we do the following: - Detect that the return type is `ValueTask` or `ValueTask<T>` either by loading the type directly (.NET Core) or checking the type name (.NET Framework) - Duck-type the `ValueTask` to read the `IsCompletedSuccessfully` value. - If this is true, the task is already completed synchronously, and we can simply call the target method. - For `ValueTask<T>` we duck type `Result` and read that directly too. - If it _hasn't_ completed, we need to extract the `Task` from it. - Duck typing is used again to extract the `Task()` for uncompleted `ValueTask` - At this point, it's mostly a copy-paste of the existing `Task` integrations - Once the integration returns, we need to create a "new" `ValueTask` instead from the previous one - The semantics of `ValueTask` _require_ that we create a "fresh" one, we can't just "reuse" the one we got originally, because we've already retrieved the result/ awaited the inner task - Have to use `Activator`/`DynamicMethod` for this > [!WARNING] > The existing `ValueTask`/`ValueTask<>` and `Task`/`Task<>` integrations are written quite differently, and I'm not entirely sure why 🤔 Given the `ContinuationAction()` methods for both these cases operate on `Task`, I based on the new `ValueTask` integrations on the `Task` integrations, but if anyone has reasons why it shouldn't be, I'm all ears! ## Test coverage - Added `Task` and `ValueTask` tests to the `CallTargetNativeTests` integration tests. Previously we were only testing a single `Task` example, and that was somewhat insufficient - Update the `CallTargetNativeTests` to explicitly assert that the `OnAsyncMethodEnd` methods are called for `Task` / `ValueTask` method integrations. As we provide _both_ methods in our target integration, we were silently calling the wrong one for .NET FX - Prior to the fix in this PR, these updated tests would fail on < .NET Core 3.0 and .NET FX - Run the `ValueTaskAsyncContinutationGenerator` unit tests on all TFMs, not just .NET Core 3.1 - Unit tests for the `ValueTaskHelper` for checking if a type is a `ValueTask` - Unit tests for the `ValueTaskActivator` for creating a `ValueTask` from a `Task` or `Task<T>` - Verified it fixes the issues I was seeing in the RabbitMQ integration - #6479 ## Other details Required for - #6479
a3bd662
to
3059d67
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Took a bit to review but thanks LGTM
Just one comment on the ValueTask
addition/change
....Trace/ClrProfiler/AutoInstrumentation/RabbitMQ/BasicPublishAsyncCachedStringsIntegration.cs
Outdated
Show resolved
Hide resolved
127cf20
to
09e7eb4
Compare
…tMQ/BasicPublishAsyncCachedStringsIntegration.cs
09e7eb4
to
fca9828
Compare
Summary of changes
Adds support for RabbitMQ v7
Reason for change
We want to support the latest versions of packages. 7.x.x was released ~6 weeks ago, with significant changes to both the public API and the internal API (which is probably why it hasn't been explicitly requested yet).
Implementation details
This one took a lot of work. In summary:
BasicProperties
because nowIReadOnlyBasicProperties
is null, which means we can't easily add the headers if they're not provided (to propagate context).#if RABBITMQ_7_0
my way in the existing methods, but was impossible to follow, so split into -pre and -post v7 methods instead.Test coverage
We get the same overall snapshots for v7 as we do for v6, so 🤞 Did an "all TFMs, all versions" run here (and a previous one for DSM)
Other details
Stacked on:
[DuckPropertyOrField]
#6463ValueTask
in .NET FX and < .NET Core 3.1 #6480Note
This PR actually highlighted a current limitation in our CallTarget instrumentation. If an API uses
ValueTask
, which comes from a library rather than being built-in (i.e. it's netfx or < .NET Core 3.1) then we don't instrument it correctly. Originally I worked around it, but there's a "proper" fix in #6480. Assuming that actually works and is merged, then we don't need the workaround, otherwise I can re-enable it.